summaryrefslogtreecommitdiff
path: root/gcc/ada/a-fihema.ads
blob: e3f412f91e4b3b1d50aeef1d02c7f0e95d4b74ae (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
------------------------------------------------------------------------------
--                                                                          --
--                         GNAT COMPILER COMPONENTS                         --
--                                                                          --
--     A D A . F I N A L I Z A T I O N . H E A P _ M A N A G E M E N T      --
--                                                                          --
--                                S p e c                                   --
--                                                                          --
--          Copyright (C) 2008-2011, Free Software Foundation, Inc.         --
--                                                                          --
-- GNAT is free software;  you can  redistribute it  and/or modify it under --
-- terms of the  GNU General Public License as published  by the Free Soft- --
-- ware  Foundation;  either version 3,  or (at your option) any later ver- --
-- sion.  GNAT is distributed in the hope that it will be useful, but WITH- --
-- OUT ANY WARRANTY;  without even the  implied warranty of MERCHANTABILITY --
-- or FITNESS FOR A PARTICULAR PURPOSE.                                     --
--                                                                          --
-- As a special exception under Section 7 of GPL version 3, you are granted --
-- additional permissions described in the GCC Runtime Library Exception,   --
-- version 3.1, as published by the Free Software Foundation.               --
--                                                                          --
-- You should have received a copy of the GNU General Public License and    --
-- a copy of the GCC Runtime Library Exception along with this program;     --
-- see the files COPYING3 and COPYING.RUNTIME respectively.  If not, see    --
-- <http://www.gnu.org/licenses/>.                                          --
--                                                                          --
-- GNAT was originally developed  by the GNAT team at  New York University. --
-- Extensive contributions were provided by Ada Core Technologies Inc.      --
--                                                                          --
------------------------------------------------------------------------------

with System;
with System.Storage_Elements;
with System.Storage_Pools;

package Ada.Finalization.Heap_Management is

   --  A reference to any derivation of Root_Storage_Pool. Since this type may
   --  not be used to allocate objects, its storage size is zero.

   type Any_Storage_Pool_Ptr is
     access System.Storage_Pools.Root_Storage_Pool'Class;
   for Any_Storage_Pool_Ptr'Storage_Size use 0;

   --  ??? Comment needed on overall mechanism

   type Finalization_Collection is
     new Ada.Finalization.Limited_Controlled with private;

   type Finalization_Collection_Ptr is access all Finalization_Collection;
   for Finalization_Collection_Ptr'Storage_Size use 0;

   --  A reference used to describe primitive Finalize_Address

   type Finalize_Address_Ptr is access procedure (Obj : System.Address);

   --  Since RTSfind cannot contain names of the form RE_"+", the following
   --  routine serves as a wrapper around System.Storage_Elements."+".

   function Add_Offset_To_Address
     (Addr   : System.Address;
      Offset : System.Storage_Elements.Storage_Offset) return System.Address;

   procedure Allocate
     (Collection   : in out Finalization_Collection;
      Addr         : out System.Address;
      Storage_Size : System.Storage_Elements.Storage_Count;
      Alignment    : System.Storage_Elements.Storage_Count;
      Needs_Header : Boolean := True);
   --  Allocate a chunk of memory described by Storage_Size and Alignment on
   --  Collection's underlying storage pool. Return the address of the chunk.
   --  The routine creates a list header which precedes the chunk of memory if
   --  Needs_Header is True. If allocated, the header is attached to the
   --  Collection's objects. The interface to this routine is provided by
   --  Build_Allocate_Deallocate_Proc.

   function Base_Pool
     (Collection : Finalization_Collection) return Any_Storage_Pool_Ptr;
   --  Return a reference to the underlying storage pool of Collection

   procedure Deallocate
     (Collection   : in out Finalization_Collection;
      Addr         : System.Address;
      Storage_Size : System.Storage_Elements.Storage_Count;
      Alignment    : System.Storage_Elements.Storage_Count;
      Has_Header   : Boolean := True);
   --  Deallocate a chunk of memory described by Storage_Size and Alignment
   --  from Collection's underlying storage pool. The beginning of the memory
   --  chunk is designated by Addr. The routine detaches and destroys the
   --  preceding list header if flag Has_Header is set. The interface to this
   --  routine is provided by Build_Allocate_Deallocate_Proc.

   overriding procedure Finalize
     (Collection : in out Finalization_Collection);
   --  Traverse objects of Collection, invoking Finalize_Address on each one

   overriding procedure Initialize
     (Collection : in out Finalization_Collection);
   --  Initialize the finalization list to empty

   procedure Set_Finalize_Address_Ptr
     (Collection : in out Finalization_Collection;
      Proc_Ptr   : Finalize_Address_Ptr);
   --  Set the finalization address routine of a finalization collection

   procedure Set_Storage_Pool_Ptr
     (Collection : in out Finalization_Collection;
      Pool_Ptr   : Any_Storage_Pool_Ptr);
   --  Set the underlying storage pool of a finalization collection

private
   --  Homogeneous collection types

   type Node;
   type Node_Ptr is access all Node;
   pragma No_Strict_Aliasing (Node_Ptr);

   --  The following record type should really be limited, but we can see the
   --  full view of Limited_Controlled, which is NOT limited. Note that default
   --  initialization does not happen for this type (the pointers will not be
   --  automatically set to null), because of the games we're playing with
   --  address arithmetic.

   type Node is record
      Prev : Node_Ptr;
      Next : Node_Ptr;
   end record;

   type Finalization_Collection is
     new Ada.Finalization.Limited_Controlled with
   record
      Base_Pool : Any_Storage_Pool_Ptr;
      --  All objects and node headers are allocated on this underlying pool;
      --  the collection is simply a wrapper around it.

      Objects : aliased Node;
      --  The head of a doubly linked list containing all allocated objects
      --  with controlled parts that still exist (Unchecked_Deallocation has
      --  not been done on them).

      Finalize_Address : Finalize_Address_Ptr;
      --  A reference to a routine that finalizes an object denoted by its
      --  address. The collection must be homogeneous since the same routine
      --  will be invoked for every allocated object when the pool is
      --  finalized.

      Finalization_Started : Boolean := False;
      pragma Atomic (Finalization_Started);
      --  When the finalization of a collection takes place, any allocations of
      --  objects with controlled or protected parts on the same collection are
      --  prohibited and the action must raise Program_Error. This needs to be
      --  atomic, because it is accessed without Lock_Task/Unlock_Task. See
      --  RM-4.8(10.2/2).
   end record;

   procedure pcol (Collection : Finalization_Collection);
   --  Output the contents of a collection in a readable form. Intended for
   --  debugging purposes.

end Ada.Finalization.Heap_Management;