Library Design: Strategies, Adapters, and Concurrency
Let's design a library of reusable data structure components that we can
mix and match into any combination.
The first thing to decide is what the dimensions of the library are:
o A "bounded" component as allocated statically. It is implemented
using an array, and the client specifies what the maximum number of
items is.
o An "unbounded" component allocates nodes from a storage pool. It
grows and shrinks in size as the number of items changes.
o A "static" storage pool is like a heap with a fixed size, and the
client specifies what the maximum size is.
o An "dynamic" storage pool allocates new nodes off the heap, and caches
unused nodes.
o A "sequential" component can only be accessed only by a single thread.
o A "concurrent" component can safely be accessed by multiple threads.
o A "controlled" component is sequential, but with a concurrent storage
pool.
These dimensions were largely inspired by the Ada83 Booch Components.
Now let's figure out what the size of the design space is:
o A bounded component can be sequential or concurrent. It doesn't use a
storage pool. That's two possibilities.
o An unbounded component uses a static or a dynamic storage pool. For
the concurrency behavior:
1) both the component and the pool can be sequential; or,
2) just the pool can be concurrent ("controlled"); or,
3) both the component and the pool are concurrent.
That's six possibilities.
A brute-force implementation of this library would provide separate
implementations for each kind of data structure. So for example there
would be eight versions of a stack.
The problem is that if you add another memory management strategy (say),
then you have to go back and add the new implementation for every data
structure. Or you might decide you want to have guarded data structures
too. You end up with a "combinatorial explosion" of possibilities that
makes the library large and unwieldy.
It would be better if we could simply parameterize a component with a
memory management strategy, or adapt a sequential component to allow
concurrent access. The client can just plug in the specific behavior he
requires, and we can keep the library small.
In general, when building reusable component library, it's best to
provide simple primitives that can be used to construct more complex
abstractions.
Implementation of the Sequential Forms
We're going to build a small library that can be used to construct stack
abstractions. The library must provide separate implementations for
bounded stack and an unbounded stack, because they have fundamentally
different representations.
We've seen the bounded stack already in previous articles:
generic
type Item_Type is private;
Max_Depth : in Positive;
package ACL.Stacks_Bounded is
type Stack_Type is tagged limited private;
procedure Push
(Item : in Item_Type;
On : in out Stack_Type);
procedure Pop
(Stack : in out Stack_Type);
...
private
...
type Stack_Type is
tagged limited record
Items : Item_Array;
Top : Natural := 0;
end record;
end ACL.Stacks_Bounded;
A bounded sequential stack has a very simple instantiation:
with ACL.Stacks_Bounded;
package Integer_Stacks_Bounded is
new ACL.Stacks_Bounded (Integer, Max_Depth => 10);
The unbounded form is more interesting, because we're going to
parameterize the stack with a strategy for storage management. Here's
the basic template first:
generic
type Item_Type is private;
...
package ACL.Stacks_Unbounded is
type Stack_Type is tagged limited private;
procedure Push
(Item : in Item_Type;
On : in out Stack_Type);
procedure Pop
(Stack : in out Stack_Type);
...
private
...
type Stack_Type is
tagged limited record
Top : Storage_Node_Access;
Control : Stack_Control (Stack_Type'Access);
end record;
end ACL.Stacks_Unbounded;
The unbounded form implements the stack as a linked list of storage
nodes (containing an item and a next pointer), together with a control
component to reclaim storage when the stack object is finalized.
In order to implement Push, we need to allocate a new storage node from
a storage pool:
procedure Push
(Item : in Item_Type;
On : in out Stack_Type) is
Node : Storage_Node_Access;
begin
Allocate (Storage, Node); <--
Node.Item := Item;
Node.Next := On.Top;
On.Top := Node;
end Push;
To implement Pop, we need to deallocate the top node:
procedure Pop
(Stack : in out Stack_Type) is
Node : Storage_Node_Access := Stack.Top;
begin
Stack.Top := Stack.Top.Next;
Deallocate (Storage, Node); <--
end Pop;
We want to keep the representation of the storage pool separate from the
unbounded stacks. Some clients (say, in the high-integrity domain)
can't use heap, and would use a fixed-size static pool. Other clients
will choose to allocate nodes off the heap, from a dynamic pool.
In order to allow the client to choose the strategy for storage pool
management (we shouldn't choose for him), the unbounded stack package
imports the storage pool as a generic formal object:
generic
type Item_Type is private;
...
Storage : in out Item_Storage_Type; <--
package ACL.Stacks_Unbounded is
There's another way to do it, but in this implementation
Item_Storage_Type is a type in a storage pool class with operations to
Allocate and Deallocate storage nodes:
generic
...
package ACL.Storage is
type Root_Storage_Type is abstract tagged limited null record;
procedure Allocate
(Storage : in out Root_Storage_Type;
Node : out Storage_Node_Access) is abstract;
procedure Deallocate
(Storage : in out Root_Storage_Type;
Node : in out Storage_Node_Access) is abstract;
end ACL.Storage;
The unbounded stack component accepts an instantiation of the storage
package as a generic formal package (since that's what defines the
class), and imports the specific storage pool type:
generic
type Item_Type is private;
...
with package Item_Storage is <--
new Storage ...;
type Item_Storage_Type (<>) is
new Item_Storage.Root_Storage_Type with private; <--
Storage : in out Item_Storage_Type;
package ACL.Stacks_Unbounded is
Because both the storage pool and the unbounded stack (and indeed, any
other kind of data structure) need to understand the representation of a
storage node, we declare it as a separate component:
generic
type Item_Type is limited private;
package ACL.Storage_Nodes is
type Storage_Node;
type Storage_Node_Access is access all Storage_Node;
type Storage_Node is
limited record
Item : aliased Item_Type;
Next : Storage_Node_Access;
Prev : Storage_Node_Access;
end record;
end ACL.Storage_Nodes;
An instantiation of node package is imported by the unbounded stacks
package and the storage type package:
generic
type Item_Type is private;
with package Item_Nodes is <--
new Storage_Nodes (Item_Type);
with package Item_Storage is
new Storage (Item_Nodes); <--
type Item_Storage_Type (<>) is
new Item_Storage.Root_Storage_Type with private;
Storage : in out Item_Storage_Type;
package ACL.Stacks_Unbounded is ...;
Now we need to implement some storage pools. We use a bounded storage
pool to implement a fixed-size, statically-allocated heap:
generic
Size : in Positive;
package ACL.Storage.Bounded_G is
type Bounded_Storage_Type is
new Root_Storage_Type with private;
procedure Allocate ...;
procedure Deallocate ...;
private
...
end ACL.Storage.Bounded_G;
The bounded pool is implemented as an array of nodes with a pointer to
the first unused node:
type Bounded_Storage_Type is
new Root_Storage_Type with record
Nodes : Storage_Node_Array (1 .. Size);
Head : Storage_Node_Access :=
Initialize (Bounded_Storage_Type'Access);
end record;
During elaboration, a bounded storage pool object initializes the free
list by setting each node to point to the next one in the array:
function Initialize
(Storage : access Bounded_Storage_Type)
return Storage_Node_Access is
begin
for I in Positive range 1 .. Size - 1 loop
Storage.Nodes (I).Next :=
Storage.Nodes (I+1)'Access;
end loop;
return Storage.Nodes (1)'Access;
end Initialize;
Note that we implement initialization of the storage pool using a
function called during elaboration, because this is less expensive than
using Ada.Finalization.
OK, now that we have a storage pool type, we can finally implement an
actual stack. What we'll end up with will look something like this:
package Integer_Stacks_Unbounded_Static is
...
type Integer_Stack is
new Stack_Type with null record;
end Integer_Stacks_Unbounded_Static;
Before we can instantiate anything else, we need storage nodes:
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
We use the storage node type to instantiate the storage pool class:
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
This gives us the abstract root type in the class. We get the static
heap type by instantiating the bounded pool package:
package Integer_Storage_Bounded is
new Integer_Storage.Bounded_G (Size => 25);
With a storage pool type, we can declare a storage pool object:
Storage : Bounded_Storage_Type;
We now have all the components we need to instantiate the stack package:
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Bounded_Storage_Type,
Storage);
The final step is to make one more derivation, so that stack operations
are directly visible in the outer package:
type Integer_Stack is
new Stack_Types.Stack_Type with null record;
We're done. We now have a stack for integers, and each instance of the
stack has unbounded size. Storage nodes for stack instances are drawn
from a pool that has has a bounded size. No heap has been used, which
satisfies the needs of developers of safety-critical software.
Desktop systems can be written with heap, so we also provide an
unbounded pool:
generic
package ACL.Storage.Unbounded_G is
type Unbounded_Storage_Type is
new Root_Storage_Type with private;
procedure Allocate ...;
procedure Deallocate ...;
private
type Unbounded_Storage_Type is
new Root_Storage_Type with record
Head : Storage_Node_Access;
end record;
end ACL.Storage.Unbounded_G;
This is a very simple implementation that maintains a linked list of
unused nodes. If the free list is empty, then it allocates a new node
off the heap; otherwise, it just returns a node off the free list.
We create an unbounded stack with a dynamic storage pool the same way we
did before, except that we use an unbounded pool:
package Integer_Storage_Unbounded is
new Integer_Storage.Unbounded_G;
Storage : Unbounded_Storage_Type;
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Unbounded_Storage_Type, <--
Storage);
We have created two different kinds of unbounded stacks, just by
parameterizing the package with a different storage pool strategy.
Implementation of the Concurrent Forms
In an earlier article we implemented a concurrent stack with a
semaphore, and used a controlled object to make sure the semaphore was
always released.
A more efficient technique is to implement the stack as a protected type
directly. Protected subprograms are used to synchronize access to stack
data, instead of using an entry queue (Seize) to serialize threads.
We use a type adapter to import a sequential stack as a generic formal
type, and implement the concurrent stack as a protected type with the
imported stack as protected data:
generic
type Item_Type is private;
type Stack_Rep is limited private; <--
with procedure Push
(Item : in Item_Type;
On : in out Stack_Rep) is <>;
with procedure Pop
(Stack : in out Stack_Rep) is <>;
...
package ACL.Stacks_Concurrent is
type Stack_Type is limited private;
procedure Push
(Item : in Item_Type;
On : in out Stack_Type);
procedure Pop
(Stack : in out Stack_Type);
...
private
protected type Stack_Type is
procedure Push (Item : in Item_Type);
procedure Pop;
...
private
Rep : Stack_Rep; <--
end Stack_Type;
end ACL.Stacks_Concurrent;
Note that it doesn't matter what kind of (sequential) stack is imported;
it could be bounded or unbounded. All the concurrent stack adapter
cares about is that the rep stack has the indicated operations.
I have chosen here to hide the fact that the concurrent stack is a
protected type, and have implemented public operations as call-throughs
to the underlying type:
procedure Push
(Item : in Item_Type;
On : in out Stack_Type) is
begin
On.Push (Item);
end;
procedure Pop
(Stack : in out Stack_Type) is
begin
Stack.Pop;
end;
The protected type passes the calls down to its representation stack:
protected body Stack_Type is
procedure Push (Item : in Item_Type) is
begin
Push (Item, On => Rep);
end;
procedure Pop is
begin
Pop (Rep);
end;
...
end Stack_Type;
Is this the best way to do it? I don't know. Tucker Taft and Mark
Gerhardt have both made the observation that you should advertise the
concurrent semantics of an abstraction, as this facilitates formal
analysis.
In that case, you might decide to just publicly declare the concurrent
stack as a protected type:
generic
...
package ACL.Stacks_Concurrent is
protected type Stack_Type is
procedure Push (Item : in Item_Type);
procedure Pop;
function Get_Top return Item_Type;
procedure Set_Top (Item : in Item_Type);
function Is_Empty return Boolean;
private
Rep : Stack_Rep;
end Stack_Type;
end ACL.Stacks_Concurrent;
Now that we have an adapter, we can start implementing concurrent
stacks. Let's begin by implementing a concurrent bounded stack.
First we declare a bounded sequential stack, like we did before:
package Stack_Types is
new ACL.Stacks_Bounded (Integer, Max_Depth => 10);
Then we pass the sequential type to the concurrent stack adapter:
use Stack_Types;
package Concurrent_Stacks is
new ACL.Stacks_Concurrent (Integer, Stack_Type);
Note that we used a use clause to make Stack_Type operations directly
visible; this simplifies the instantiation of the concurrent stacks
package, because all the formal subprograms take default values.
Finally, we make one more derivation, to effect transitivity of
visibility:
type Integer_Stack is
new Concurrent_Stacks.Stack_Type;
The completed spec looks like this:
package Integer_Stacks_Bounded_Concurrent is
package Stack_Types is
new ACL.Stacks_Bounded (Integer, Max_Depth => 10);
use Stack_Types;
package Concurrent_Stacks is
new ACL.Stacks_Concurrent (Integer, Stack_Type);
type Integer_Stack is
new Concurrent_Stacks.Stack_Type;
end Integer_Stacks_Bounded_Concurrent;
In order to implement concurrent unbounded stacks, we're going to have
to create another concurrent adapter for storage pools. Like the
adapter for stacks, this also uses a protected type, to synchronize
access to the storage pool:
generic
type Storage_Type is new Root_Storage_Type with private;
package ACL.Storage.Concurrent_G is
type Concurrent_Storage_Type is
new Root_Storage_Type with private;
procedure Allocate ...;
procedure Deallocate ...;
private
protected type Synchronization_Type is
procedure Allocate (Node : in out Storage_Node_Access);
procedure Deallocate (Node : in out Storage_Node_Access);
private
Storage : Storage_Type;
end Synchronization_Type;
type Concurrent_Storage_Type is
new Root_Storage_Type with record
Synchronization : Synchronization_Type;
end record;
end ACL.Storage.Concurrent_G;
Note that we don't have to import generic formal operations for the
sequential storage type, because we're importing it as a formal tagged
type in the Root_Storage_Type class, and therefore we already know what
its operations are.
Let's take our sequential unbounded stack with the dynamic pool, and
make the storage pool concurrent. As before, we start with an unbounded
pool:
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
package Integer_Storage_Unbounded is
new Integer_Storage.Unbounded_G;
Next we adapt the sequential pool, to make it concurrent:
package Integer_Storage_Unbounded_Concurrent is
new Integer_Storage.Concurrent_G (Unbounded_Storage_Type);
Now that we have a concurrent storage type, we can declare a concurrent
storage pool object:
Storage : Concurrent_Storage_Type;
We pass this concurrent storage pool object to the instantiation of the
(sequential) stack package. The completed spec looks like this:
package Integer_Stacks_Unbounded_Dynamic_Controlled is
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
package Integer_Storage_Unbounded is
new Integer_Storage.Unbounded_G;
package Integer_Storage_Unbounded_Concurrent is
new Integer_Storage.Concurrent_G
(Integer_Storage_Unbounded.Unbounded_Storage_Type);
use Integer_Storage_Unbounded_Concurrent;
Storage : Concurrent_Storage_Type;
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Concurrent_Storage_Type,
Storage);
type Integer_Stack is
new Stack_Types.Stack_Type with null record;
end Integer_Stacks_Unbounded_Dynamic_Controlled;
This gives us controlled unbounded stacks. We can create concurrent
unbounded stacks by taking what we have above, and just adapting the
sequential stack:
package Integer_Stacks_Unbounded_Dynamic_Concurrent is
...
use Stack_Types;
package Concurrent_Stack_Types is
new ACL.Stacks_Concurrent (Integer, Stack_Type);
type Integer_Stack is
new Concurrent_Stack_Types.Stack_Type;
end Integer_Stacks_Unbounded_Dynamic_Concurrent;
Implementations of concurrent unbounded stacks that use a static heap
are implemented similarly.
--STX
package body ACL.Stacks_Bounded is
procedure Push
(Item : in Item_Type;
On : in out Stack_Type) is
begin
On.Top := On.Top + 1;
On.Items (On.Top) := Item;
end;
procedure Pop
(Stack : in out Stack_Type) is
begin
Stack.Top := Stack.Top - 1;
end;
function Get_Top
(Stack : Stack_Type) return Item_Type is
begin
return Stack.Items (Stack.Top);
end;
procedure Set_Top
(Stack : in out Stack_Type;
Item : in Item_Type) is
begin
Stack.Items (Stack.Top) := Item;
end;
function Is_Empty (Stack : Stack_Type) return Boolean is
begin
return Stack.Top = 0;
end;
procedure For_Every_Item (Stack : in out Stack_Type'Class) is
Done : Boolean := False;
begin
for I in reverse 1 .. Stack.Top loop
Process (Stack.Items (I), Done);
exit when Done;
end loop;
end For_Every_Item;
end ACL.Stacks_Bounded;
generic
type Item_Type is private;
Max_Depth : in Positive;
package ACL.Stacks_Bounded is
pragma Preelaborate;
type Stack_Type is tagged limited private;
procedure Push
(Item : in Item_Type;
On : in out Stack_Type);
procedure Pop
(Stack : in out Stack_Type);
function Get_Top
(Stack : Stack_Type) return Item_Type;
procedure Set_Top
(Stack : in out Stack_Type;
Item : in Item_Type);
function Is_Empty (Stack : Stack_Type) return Boolean;
generic
with procedure Process
(Item : in out Item_Type;
Done : in out Boolean);
procedure For_Every_Item (Stack : in out Stack_Type'Class);
private
subtype Item_Array_Range is Positive range 1 .. Max_Depth;
type Item_Array is array (Item_Array_Range) of Item_Type;
type Stack_Type is
tagged limited record
Items : Item_Array;
Top : Natural := 0;
end record;
end ACL.Stacks_Bounded;
package body ACL.Stacks_Concurrent is
procedure Push
(Item : in Item_Type;
On : in out Stack_Type) is
begin
On.Push (Item);
end;
procedure Pop
(Stack : in out Stack_Type) is
begin
Stack.Pop;
end;
function Get_Top
(Stack : Stack_Type) return Item_Type is
begin
return Stack.Get_Top;
end;
procedure Set_Top
(Stack : in out Stack_Type;
Item : in Item_Type) is
begin
Stack.Set_Top (Item);
end;
function Is_Empty
(Stack : Stack_Type) return Boolean is
begin
return Stack.Is_Empty;
end;
protected body Stack_Type is
procedure Push (Item : in Item_Type) is
begin
Push (Item, On => Rep);
end;
procedure Pop is
begin
Pop (Rep);
end;
function Get_Top return Item_Type is
begin
return Get_Top (Rep);
end;
procedure Set_Top (Item : in Item_Type) is
begin
Set_Top (Rep, Item);
end;
function Is_Empty return Boolean is
begin
return Is_Empty (Rep);
end;
end Stack_Type;
end ACL.Stacks_Concurrent;
generic
type Item_Type is private;
type Stack_Rep is limited private;
with procedure Push
(Item : in Item_Type;
On : in out Stack_Rep) is <>;
with procedure Pop
(Stack : in out Stack_Rep) is <>;
with function Get_Top
(Stack : Stack_Rep) return Item_Type is <>;
with procedure Set_Top
(Stack : in out Stack_Rep;
Item : in Item_Type) is <>;
with function Is_Empty
(Stack : Stack_Rep) return Boolean is <>;
package ACL.Stacks_Concurrent is
type Stack_Type is limited private;
procedure Push
(Item : in Item_Type;
On : in out Stack_Type);
procedure Pop
(Stack : in out Stack_Type);
function Get_Top
(Stack : Stack_Type) return Item_Type;
procedure Set_Top
(Stack : in out Stack_Type;
Item : in Item_Type);
function Is_Empty
(Stack : Stack_Type) return Boolean;
private
protected type Stack_Type is
procedure Push (Item : in Item_Type);
procedure Pop;
function Get_Top return Item_Type;
procedure Set_Top (Item : in Item_Type);
function Is_Empty return Boolean;
private
Rep : Stack_Rep;
end Stack_Type;
end ACL.Stacks_Concurrent;
package body ACL.Stacks_Unbounded is
procedure Finalize (Control : in out Stack_Control) is
Top : Storage_Node_Access renames Control.Stack.Top;
Node : Storage_Node_Access;
begin
while Top /= null loop
Node := Top;
Top := Top.Next;
Deallocate (Storage, Node);
end loop;
end Finalize;
procedure Push
(Item : in Item_Type;
On : in out Stack_Type) is
Node : Storage_Node_Access;
begin
Allocate (Storage, Node);
Node.Item := Item;
Node.Next := On.Top;
On.Top := Node;
end Push;
procedure Pop
(Stack : in out Stack_Type) is
Node : Storage_Node_Access := Stack.Top;
begin
Stack.Top := Stack.Top.Next;
Deallocate (Storage, Node);
end Pop;
function Get_Top
(Stack : Stack_Type) return Item_Type is
begin
return Stack.Top.Item;
end;
procedure Set_Top
(Stack : in out Stack_Type;
Item : in Item_Type) is
begin
Stack.Top.Item := Item;
end;
function Is_Empty (Stack : Stack_Type) return Boolean is
begin
return Stack.Top = null;
end;
procedure For_Every_Item (Stack : in out Stack_Type'Class) is
Node : Storage_Node_Access := Stack.Top;
Done : Boolean := False;
begin
while Node /= null loop
Process (Node.Item, Done);
exit when Done;
Node := Node.Next;
end loop;
end For_Every_Item;
end ACL.Stacks_Unbounded;
with ACL.Storage_Nodes;
with ACL.Storage;
with Ada.Finalization;
generic
type Item_Type is private;
with package Item_Nodes is
new Storage_Nodes (Item_Type);
with package Item_Storage is
new Storage (Item_Nodes);
type Item_Storage_Type (<>) is
new Item_Storage.Root_Storage_Type with private;
Storage : in out Item_Storage_Type;
package ACL.Stacks_Unbounded is
type Stack_Type is tagged limited private;
procedure Push
(Item : in Item_Type;
On : in out Stack_Type);
procedure Pop
(Stack : in out Stack_Type);
function Get_Top
(Stack : Stack_Type) return Item_Type;
procedure Set_Top
(Stack : in out Stack_Type;
Item : in Item_Type);
function Is_Empty (Stack : Stack_Type) return Boolean;
generic
with procedure Process
(Item : in out Item_Type;
Done : in out Boolean);
procedure For_Every_Item (Stack : in out Stack_Type'Class);
private
use Item_Nodes;
use Ada.Finalization;
type Stack_Control (Stack : access Stack_Type) is
new Limited_Controlled with null record;
procedure Finalize (Control : in out Stack_Control);
type Stack_Type is
tagged limited record
Top : Storage_Node_Access;
Control : Stack_Control (Stack_Type'Access);
end record;
end ACL.Stacks_Unbounded;
package body ACL.Storage.Bounded_G is
procedure Allocate
(Storage : in out Bounded_Storage_Type;
Node : out Storage_Node_Access) is
begin
if Storage.Head = null then
raise Storage_Error;
end if;
Node := Storage.Head;
Storage.Head := Storage.Head.Next;
Node.Next := null;
end Allocate;
procedure Deallocate
(Storage : in out Bounded_Storage_Type;
Node : in out Storage_Node_Access) is
begin
if Node /= null then
Node.Next := Storage.Head;
Storage.Head := Node;
end if;
Node := null;
end Deallocate;
function Initialize
(Storage : access Bounded_Storage_Type)
return Storage_Node_Access is
begin
for I in Positive range 1 .. Size - 1 loop
Storage.Nodes (I).Next :=
Storage.Nodes (I+1)'Access;
end loop;
return Storage.Nodes (1)'Access;
end Initialize;
end ACL.Storage.Bounded_G;
generic
Size : in Positive;
package ACL.Storage.Bounded_G is
type Bounded_Storage_Type is
new Root_Storage_Type with private;
procedure Allocate
(Storage : in out Bounded_Storage_Type;
Node : out Storage_Node_Access);
procedure Deallocate
(Storage : in out Bounded_Storage_Type;
Node : in out Storage_Node_Access);
private
type Storage_Node_Array is
array (Positive range <>) of aliased Storage_Node;
function Initialize
(Storage : access Bounded_Storage_Type)
return Storage_Node_Access;
type Bounded_Storage_Type is
new Root_Storage_Type with record
Nodes : Storage_Node_Array (1 .. Size);
Head : Storage_Node_Access :=
Initialize (Bounded_Storage_Type'Access);
end record;
end ACL.Storage.Bounded_G;
package body ACL.Storage.Concurrent_G is
procedure Allocate
(Storage : in out Concurrent_Storage_Type;
Node : out Storage_Node_Access) is
begin
Storage.Synchronization.Allocate (Node);
end;
procedure Deallocate
(Storage : in out Concurrent_Storage_Type;
Node : in out Storage_Node_Access) is
begin
Storage.Synchronization.Deallocate (Node);
end;
protected body Synchronization_Type is
procedure Allocate (Node : in out Storage_Node_Access) is
begin
Allocate (Storage, Node);
end;
procedure Deallocate (Node : in out Storage_Node_Access) is
begin
Deallocate (Storage, Node);
end;
end Synchronization_Type;
end ACL.Storage.Concurrent_G;
generic
type Storage_Type is new Root_Storage_Type with private;
package ACL.Storage.Concurrent_G is
type Concurrent_Storage_Type is
new Root_Storage_Type with private;
procedure Allocate
(Storage : in out Concurrent_Storage_Type;
Node : out Storage_Node_Access);
procedure Deallocate
(Storage : in out Concurrent_Storage_Type;
Node : in out Storage_Node_Access);
private
protected type Synchronization_Type is
procedure Allocate (Node : in out Storage_Node_Access);
procedure Deallocate (Node : in out Storage_Node_Access);
private
Storage : Storage_Type;
end Synchronization_Type;
type Concurrent_Storage_Type is
new Root_Storage_Type with record
Synchronization : Synchronization_Type;
end record;
end ACL.Storage.Concurrent_G;
package body ACL.Storage.Unbounded_G is
procedure Allocate
(Storage : in out Unbounded_Storage_Type;
Node : out Storage_Node_Access) is
begin
if Storage.Head = null then
Node := new Storage_Node;
else
Node := Storage.Head;
Storage.Head := Storage.Head.Next;
Node.Next := null;
end if;
end Allocate;
procedure Deallocate
(Storage : in out Unbounded_Storage_Type;
Node : in out Storage_Node_Access) is
begin
if Node /= null then
Node.Next := Storage.Head;
Storage.Head := Node;
end if;
Node := null;
end Deallocate;
end ACL.Storage.Unbounded_G;
generic
package ACL.Storage.Unbounded_G is
type Unbounded_Storage_Type is
new Root_Storage_Type with private;
procedure Allocate
(Storage : in out Unbounded_Storage_Type;
Node : out Storage_Node_Access);
procedure Deallocate
(Storage : in out Unbounded_Storage_Type;
Node : in out Storage_Node_Access);
private
type Unbounded_Storage_Type is
new Root_Storage_Type with record
Head : Storage_Node_Access;
end record;
end ACL.Storage.Unbounded_G;
with ACL.Storage_Nodes;
generic
with package Nodes is new Storage_Nodes (<>);
package ACL.Storage is
use Nodes;
type Root_Storage_Type is abstract tagged limited null record;
procedure Allocate
(Storage : in out Root_Storage_Type;
Node : out Storage_Node_Access) is abstract;
procedure Deallocate
(Storage : in out Root_Storage_Type;
Node : in out Storage_Node_Access) is abstract;
end ACL.Storage;
generic
type Item_Type is limited private;
package ACL.Storage_Nodes is
type Storage_Node;
type Storage_Node_Access is access all Storage_Node;
type Storage_Node is
limited record
Item : aliased Item_Type;
Next : Storage_Node_Access;
Prev : Storage_Node_Access;
end record;
end ACL.Storage_Nodes;
package ACL is
pragma Pure;
end ACL;
with ACL.Stacks_Bounded;
package Integer_Stacks_Bounded is
new ACL.Stacks_Bounded (Integer, Max_Depth => 10);
with ACL.Stacks_Bounded;
with ACL.Stacks_Concurrent;
pragma Elaborate (ACL.Stacks_Bounded);
pragma Elaborate (ACL.Stacks_Concurrent);
package Integer_Stacks_Bounded_Concurrent is
package Stack_Types is
new ACL.Stacks_Bounded (Integer, Max_Depth => 10);
use Stack_Types;
package Concurrent_Stacks is
new ACL.Stacks_Concurrent (Integer, Stack_Type);
type Integer_Stack is
new Concurrent_Stacks.Stack_Type;
end Integer_Stacks_Bounded_Concurrent;
with ACL.Storage_Nodes;
with ACL.Storage.Unbounded_G;
with ACL.Stacks_Unbounded;
pragma Elaborate (ACL.Storage_Nodes);
pragma Elaborate (ACL.Storage.Unbounded_G);
pragma Elaborate (ACL.Stacks_Unbounded);
package Integer_Stacks_Unbounded_Dynamic is
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
package Integer_Storage_Unbounded is
new Integer_Storage.Unbounded_G;
use Integer_Storage_Unbounded;
Storage : Unbounded_Storage_Type;
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Unbounded_Storage_Type,
Storage);
type Integer_Stack is
new Stack_Types.Stack_Type with null record;
end Integer_Stacks_Unbounded_Dynamic;
with ACL.Storage_Nodes;
with ACL.Storage.Unbounded_G;
with ACL.Storage.Concurrent_G;
with ACL.Stacks_Unbounded;
with ACL.Stacks_Concurrent;
pragma Elaborate (ACL.Storage_Nodes);
pragma Elaborate (ACL.Storage.Unbounded_G);
pragma Elaborate (ACL.Storage.Concurrent_G);
pragma Elaborate (ACL.Stacks_Unbounded);
pragma Elaborate (ACL.Stacks_Concurrent);
package Integer_Stacks_Unbounded_Dynamic_Concurrent is
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
package Integer_Storage_Unbounded is
new Integer_Storage.Unbounded_G;
package Integer_Storage_Unbounded_Concurrent is
new Integer_Storage.Concurrent_G
(Integer_Storage_Unbounded.Unbounded_Storage_Type);
use Integer_Storage_Unbounded_Concurrent;
Storage : Concurrent_Storage_Type;
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Concurrent_Storage_Type,
Storage);
use Stack_Types;
package Concurrent_Stack_Types is
new ACL.Stacks_Concurrent (Integer, Stack_Type);
type Integer_Stack is
new Concurrent_Stack_Types.Stack_Type;
end Integer_Stacks_Unbounded_Dynamic_Concurrent;
with ACL.Storage_Nodes;
with ACL.Storage.Unbounded_G;
with ACL.Storage.Concurrent_G;
with ACL.Stacks_Unbounded;
pragma Elaborate (ACL.Storage_Nodes);
pragma Elaborate (ACL.Storage.Unbounded_G);
pragma Elaborate (ACL.Storage.Concurrent_G);
pragma Elaborate (ACL.Stacks_Unbounded);
package Integer_Stacks_Unbounded_Dynamic_Controlled is
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
package Integer_Storage_Unbounded is
new Integer_Storage.Unbounded_G;
package Integer_Storage_Unbounded_Concurrent is
new Integer_Storage.Concurrent_G
(Integer_Storage_Unbounded.Unbounded_Storage_Type);
use Integer_Storage_Unbounded_Concurrent;
Storage : Concurrent_Storage_Type;
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Concurrent_Storage_Type,
Storage);
type Integer_Stack is
new Stack_Types.Stack_Type with null record;
end Integer_Stacks_Unbounded_Dynamic_Controlled;
with ACL.Storage_Nodes;
with ACL.Storage.Bounded_G;
with ACL.Stacks_Unbounded;
pragma Elaborate (ACL.Storage_Nodes);
pragma Elaborate (ACL.Storage.Bounded_G);
pragma Elaborate (ACL.Stacks_Unbounded);
package Integer_Stacks_Unbounded_Static is
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
package Integer_Storage_Bounded is
new Integer_Storage.Bounded_G (Size => 25);
use Integer_Storage_Bounded;
Storage : Bounded_Storage_Type;
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Bounded_Storage_Type,
Storage);
type Integer_Stack is
new Stack_Types.Stack_Type with null record;
end Integer_Stacks_Unbounded_Static;
with ACL.Storage_Nodes;
with ACL.Storage.Bounded_G;
with ACL.Storage.Concurrent_G;
with ACL.Stacks_Unbounded;
with ACL.Stacks_Concurrent;
pragma Elaborate (ACL.Storage_Nodes);
pragma Elaborate (ACL.Storage.Bounded_G);
pragma Elaborate (ACL.Storage.Concurrent_G);
pragma Elaborate (ACL.Stacks_Unbounded);
pragma Elaborate (ACL.Stacks_Concurrent);
package Integer_Stacks_Unbounded_Static_Concurrent is
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
package Integer_Storage_Bounded is
new Integer_Storage.Bounded_G (Size => 25);
package Integer_Storage_Bounded_Concurrent is
new Integer_Storage.Concurrent_G
(Integer_Storage_Bounded.Bounded_Storage_Type);
use Integer_Storage_Bounded_Concurrent;
Storage : Concurrent_Storage_Type;
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Concurrent_Storage_Type,
Storage);
use Stack_Types;
package Concurrent_Stack_Types is
new ACL.Stacks_Concurrent (Integer, Stack_Type);
type Integer_Stack is
new Concurrent_Stack_Types.Stack_Type;
end Integer_Stacks_Unbounded_Static_Concurrent;
with ACL.Storage_Nodes;
with ACL.Storage.Bounded_G;
with ACL.Storage.Concurrent_G;
with ACL.Stacks_Unbounded;
pragma Elaborate (ACL.Storage_Nodes);
pragma Elaborate (ACL.Storage.Bounded_G);
pragma Elaborate (ACL.Storage.Concurrent_G);
pragma Elaborate (ACL.Stacks_Unbounded);
package Integer_Stacks_Unbounded_Static_Controlled is
package Integer_Nodes is
new ACL.Storage_Nodes (Integer);
package Integer_Storage is
new ACL.Storage (Integer_Nodes);
package Integer_Storage_Bounded is
new Integer_Storage.Bounded_G (Size => 25);
package Integer_Storage_Bounded_Concurrent is
new Integer_Storage.Concurrent_G
(Integer_Storage_Bounded.Bounded_Storage_Type);
use Integer_Storage_Bounded_Concurrent;
Storage : Concurrent_Storage_Type;
package Stack_Types is
new ACL.Stacks_Unbounded
(Integer,
Integer_Nodes,
Integer_Storage,
Concurrent_Storage_Type,
Storage);
type Integer_Stack is
new Stack_Types.Stack_Type with null record;
end Integer_Stacks_Unbounded_Static_Controlled;
with Integer_Stacks_Bounded;
with Integer_Stacks_Bounded_Concurrent;
with Integer_Stacks_Unbounded_Static;
with Integer_Stacks_Unbounded_Static_Controlled;
with Integer_Stacks_Unbounded_Static_Concurrent;
with Integer_Stacks_Unbounded_Dynamic;
with Integer_Stacks_Unbounded_Dynamic_Controlled;
with Integer_Stacks_Unbounded_Dynamic_Concurrent;
procedure Test_Stacks is
begin
null;
end Test_Stacks;
Contributed by: Matthew Heaney
Contributed on: May 24, 1999
License: Public Domain
Back