1 ------------------------------------------------------------------------------
3 -- GNAT COMPILER COMPONENTS --
5 -- G N A T . D Y N A M I C _ T A B L E S --
11 -- Copyright (C) 2000-2001 Ada Core Technologies, Inc. --
13 -- GNAT is free software; you can redistribute it and/or modify it under --
14 -- terms of the GNU General Public License as published by the Free Soft- --
15 -- ware Foundation; either version 2, or (at your option) any later ver- --
16 -- sion. GNAT is distributed in the hope that it will be useful, but WITH- --
17 -- OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY --
18 -- or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License --
19 -- for more details. You should have received a copy of the GNU General --
20 -- Public License distributed with GNAT; see file COPYING. If not, write --
21 -- to the Free Software Foundation, 59 Temple Place - Suite 330, Boston, --
22 -- MA 02111-1307, USA. --
24 -- As a special exception, if other files instantiate generics from this --
25 -- unit, or you link this unit with other files to produce an executable, --
26 -- this unit does not by itself cause the resulting executable to be --
27 -- covered by the GNU General Public License. This exception does not --
28 -- however invalidate any other reasons why the executable file might be --
29 -- covered by the GNU Public License. --
31 -- GNAT is maintained by Ada Core Technologies Inc (http://www.gnat.com). --
33 ------------------------------------------------------------------------------
35 with System
; use System
;
37 package body GNAT
.Dynamic_Tables
is
39 Min
: constant Integer := Integer (Table_Low_Bound
);
40 -- Subscript of the minimum entry in the currently allocated table
42 type size_t
is new Integer;
44 -----------------------
45 -- Local Subprograms --
46 -----------------------
48 procedure Reallocate
(T
: in out Instance
);
49 -- Reallocate the existing table according to the current value stored
50 -- in Max. Works correctly to do an initial allocation if the table
62 T
.P
.Last_Val
:= T
.P
.Last_Val
+ Num
;
64 if T
.P
.Last_Val
> T
.P
.Max
then
73 procedure Append
(T
: in out Instance
; New_Val
: Table_Component_Type
) is
76 T
.Table
(Table_Index_Type
(T
.P
.Last_Val
)) := New_Val
;
83 procedure Decrement_Last
(T
: in out Instance
) is
85 T
.P
.Last_Val
:= T
.P
.Last_Val
- 1;
92 procedure Free
(T
: in out Instance
) is
93 procedure free
(T
: Table_Ptr
);
94 pragma Import
(C
, free
);
106 procedure Increment_Last
(T
: in out Instance
) is
108 T
.P
.Last_Val
:= T
.P
.Last_Val
+ 1;
110 if T
.P
.Last_Val
> T
.P
.Max
then
119 procedure Init
(T
: in out Instance
) is
120 Old_Length
: constant Integer := T
.P
.Length
;
123 T
.P
.Last_Val
:= Min
- 1;
124 T
.P
.Max
:= Min
+ Table_Initial
- 1;
125 T
.P
.Length
:= T
.P
.Max
- Min
+ 1;
127 -- If table is same size as before (happens when table is never
128 -- expanded which is a common case), then simply reuse it. Note
129 -- that this also means that an explicit Init call right after
130 -- the implicit one in the package body is harmless.
132 if Old_Length
= T
.P
.Length
then
135 -- Otherwise we can use Reallocate to get a table of the right size.
136 -- Note that Reallocate works fine to allocate a table of the right
137 -- initial size when it is first allocated.
148 function Last
(T
: in Instance
) return Table_Index_Type
is
150 return Table_Index_Type
(T
.P
.Last_Val
);
157 procedure Reallocate
(T
: in out Instance
) is
160 (memblock
: Table_Ptr
;
163 pragma Import
(C
, realloc
);
168 pragma Import
(C
, malloc
);
173 if T
.P
.Max
< T
.P
.Last_Val
then
174 while T
.P
.Max
< T
.P
.Last_Val
loop
175 T
.P
.Length
:= T
.P
.Length
* (100 + Table_Increment
) / 100;
176 T
.P
.Max
:= Min
+ T
.P
.Length
- 1;
181 size_t
((T
.P
.Max
- Min
+ 1) *
182 (Table_Type
'Component_Size / Storage_Unit
));
184 if T
.Table
= null then
185 T
.Table
:= malloc
(New_Size
);
187 elsif New_Size
> 0 then
190 (memblock
=> T
.Table
,
194 if T
.P
.Length
/= 0 and then T
.Table
= null then
204 procedure Release
(T
: in out Instance
) is
206 T
.P
.Length
:= T
.P
.Last_Val
- Integer (Table_Low_Bound
) + 1;
207 T
.P
.Max
:= T
.P
.Last_Val
;
216 (T
: in out Instance
;
217 Index
: Table_Index_Type
;
218 Item
: Table_Component_Type
)
221 if Integer (Index
) > T
.P
.Max
then
225 T
.Table
(Index
) := Item
;
232 procedure Set_Last
(T
: in out Instance
; New_Val
: Table_Index_Type
) is
234 if Integer (New_Val
) < T
.P
.Last_Val
then
235 T
.P
.Last_Val
:= Integer (New_Val
);
238 T
.P
.Last_Val
:= Integer (New_Val
);
240 if T
.P
.Last_Val
> T
.P
.Max
then
246 end GNAT
.Dynamic_Tables
;