-- C761003.A
--
--                             Grant of Unlimited Rights
--
--     Under contracts F33600-87-D-0337, F33600-84-D-0280, MDA903-79-C-0687,
--     F08630-91-C-0015, and DCA100-97-D-0025, the U.S. Government obtained 
--     unlimited rights in the software and documentation contained herein.
--     Unlimited rights are defined in DFAR 252.227-7013(a)(19).  By making 
--     this public release, the Government intends to confer upon all 
--     recipients unlimited rights  equal to those held by the Government.  
--     These rights include rights to use, duplicate, release or disclose the 
--     released technical data and computer software in whole or in part, in 
--     any manner and for any purpose whatsoever, and to have or permit others 
--     to do so.
--
--                                    DISCLAIMER
--
--     ALL MATERIALS OR INFORMATION HEREIN RELEASED, MADE AVAILABLE OR
--     DISCLOSED ARE AS IS.  THE GOVERNMENT MAKES NO EXPRESS OR IMPLIED 
--     WARRANTY AS TO ANY MATTER WHATSOEVER, INCLUDING THE CONDITIONS OF THE
--     SOFTWARE, DOCUMENTATION OR OTHER INFORMATION RELEASED, MADE AVAILABLE 
--     OR DISCLOSED, OR THE OWNERSHIP, MERCHANTABILITY, OR FITNESS FOR A
--     PARTICULAR PURPOSE OF SAID MATERIAL.
--*
--
-- OBJECTIVE:
--      Check that an object of a controlled type is finalized when the
--      enclosing master is complete.
--      Check this for controlled types where the derived type has a
--      discriminant.
--      Check this for subprograms of abstract types derived from the
--      types in Ada.Finalization.
--
--      Check that finalization of controlled objects is
--      performed in the correct order.  In particular, check that if
--      multiple objects of controlled types are declared immediately
--      within the same declarative part then type are finalized in the
--      reverse order of their creation.
--
-- TEST DESCRIPTION:
--      This test checks these conditions for subprograms and
--      block statements; both variables and constants of controlled
--      types; cases of a controlled component of a record type, as
--      well as an array with controlled components.
--
--      The base controlled types used for the test are defined
--      with a character discriminant.  The initialize procedure for
--      the types will record the order of creation in a globally
--      accessible array, the finalize procedure for the types will call
--      TCTouch with that tag character.  The test can then check that
--      the order of finalization is indeed the reverse of the order of
--      creation (assuming that the implementation calls Initialize in
--      the order that the objects are created).
--
--
-- CHANGE HISTORY:
--      06 Dec 94   SAIC    ACVC 2.0
--      02 Nov 95   SAIC    ACVC 2.0.1
--
--!
------------------------------------------------------------ C761003_Support
package C761003_Support is
  
  function Pick_Char return Character;
  -- successive calls to Pick_Char return distinct characters which may
  -- be assigned to objects to track an order sequence.  These characters
  -- are then used in calls to TCTouch.Touch.
  procedure Validate(Initcount   : Natural;
                     Testnumber  : Natural;
                     Check_Order : Boolean := True);
  -- does a little extra processing prior to calling TCTouch.Validate,
  -- specifically, it reverses the stored string of characters, and checks
  -- for a correct count.
  Inits_Order  : String(1..255);
  Inits_Called : Natural := 0;
end C761003_Support;
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
with Report;
with TCTouch;
package body C761003_Support is
  type Pick_Rotation is mod 52;
  type Pick_String is array(Pick_Rotation) of Character;
  From : constant Pick_String  := "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
                                & "abcdefghijklmnopqrstuvwxyz";
  Recent_Pick : Pick_Rotation := Pick_Rotation'Last;
  function Pick_Char return Character is
  begin
    Recent_Pick := Recent_Pick +1;
    return From(Recent_Pick);
  end Pick_Char;
  function Invert(S:String) return String is
    T: String(1..S'Length);
  begin
    for SI in reverse S'Range loop
      T(S'Last - SI + 1) := S(SI);
    end loop;
    return T;
  end Invert;
  procedure Validate(Initcount   : Natural;
                     Testnumber  : Natural;
                     Check_Order : Boolean := True) is
    Number : constant String := Natural'Image(Testnumber);
  begin
    if Inits_Called /= Initcount then
      Report.Failed("Got" & Natural'Image(Inits_Called) & " inits, expected"
                    & Natural'Image(Initcount) & ", Subtest " & Number);
      TCTouch.Flush;
    else
      TCTouch.Validate(
        Invert(Inits_Order(1..Inits_Called)),
               "Subtest " & Number, Order_Meaningful => Check_Order );
    end if;
    Inits_Called := 0;  -- reset for the next batch
  end Validate;
end C761003_Support;
------------------------------------------------------------------ C761003_0
with Ada.Finalization;
package C761003_0 is
  type Global(Tag: Character) is new Ada.Finalization.Controlled
    with null record;
  procedure Initialize( It: in out Global );
  procedure Finalize  ( It: in out Global );
  Null_Global : Global('1') := (Ada.Finalization.Controlled with Tag => '1');
  type Second(Tag: Character) is new Ada.Finalization.Limited_Controlled
    with null record;
  procedure Initialize( It: in out Second );
  procedure Finalize  ( It: in out Second );
end C761003_0;
------------------------------------------------------------------ C761003_1
with Ada.Finalization;
package C761003_1 is
  type Global is abstract new Ada.Finalization.Controlled with record
    Tag: Character;
  end record;
  procedure Initialize( It: in out Global );
  procedure Finalize  ( It: in out Global );
  type Second is abstract new Ada.Finalization.Limited_Controlled with record
    Tag: Character;
  end record;
  procedure Initialize( It: in out Second );
  procedure Finalize  ( It: in out Second );
end C761003_1;
------------------------------------------------------------------ C761003_2
with C761003_1;
package C761003_2 is
  type Global is new C761003_1.Global with null record;
  -- inherits Initialize and Finalize
  type Second is new C761003_1.Second with null record;
  -- inherits Initialize and Finalize
end C761003_2;
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --  C761003_0
with TCTouch;
with C761003_Support;
package body C761003_0 is
  package Sup renames C761003_Support;
  procedure Initialize( It: in out Global ) is
  begin
    Sup.Inits_Called := Sup.Inits_Called +1;
    Sup.Inits_Order(Sup.Inits_Called) := It.Tag;
  end Initialize;
  procedure Finalize( It: in out Global ) is
  begin
    TCTouch.Touch(It.Tag);  --------------------------------------------- Tag
  end Finalize;
  procedure Initialize( It: in out Second ) is
  begin
    Sup.Inits_Called := Sup.Inits_Called +1;
    Sup.Inits_Order(Sup.Inits_Called) := It.Tag;
  end Initialize;
  procedure Finalize( It: in out Second ) is
  begin
    TCTouch.Touch(It.Tag);  --------------------------------------------- Tag
  end Finalize;
end C761003_0;
-- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --  C761003_1
with TCTouch;
with C761003_Support;
package body C761003_1 is
  package Sup renames C761003_Support;
  procedure Initialize( It: in out Global ) is
  begin
    Sup.Inits_Called := Sup.Inits_Called +1;
    It.Tag := Sup.Pick_Char;
    Sup.Inits_Order(Sup.Inits_Called) := It.Tag;
  end Initialize;
  procedure Finalize( It: in out Global ) is
  begin
    TCTouch.Touch(It.Tag);  --------------------------------------------- Tag
  end Finalize;
  procedure Initialize( It: in out Second ) is
  begin
    Sup.Inits_Called := Sup.Inits_Called +1;
    It.Tag := Sup.Pick_Char;
    Sup.Inits_Order(Sup.Inits_Called) := It.Tag;
  end Initialize;
  procedure Finalize( It: in out Second ) is
  begin
    TCTouch.Touch(It.Tag);  --------------------------------------------- Tag
  end Finalize;
end C761003_1;
-------------------------------------------------------------------- C761003
with Report;
with TCTouch;
with C761003_0;
with C761003_2;
with C761003_Support;
procedure C761003 is
  package Sup renames C761003_Support;
---------------------------------------------------------------- Subtest_1
  Subtest_1_Inits_Expected : constant := 5;  -- includes 1 previous
  procedure Subtest_1 is
    -- the constant will take its constraint from the value.
    -- must be declared first to be finalized last (and take the
    -- initialize from before calling subtest_1)
    Item_1 : constant C761003_0.Global := C761003_0.Null_Global;
    -- Item_2, declared second, should be finalized second to last.
    Item_2 : C761003_0.Global(Sup.Pick_Char);
    -- Item_3 and Item_4 will be created in the order of the
    -- list.
    Item_3, Item_4 : C761003_0.Global(Sup.Pick_Char);
   -- Item_5 will be finalized first.
    Item_5 : C761003_0.Second(Sup.Pick_Char);
  begin
    if Item_3.Tag >= Item_4.Tag then
      Report.Failed("Controlled objects created by list in wrong order");
    end if;
    -- check that nothing has happened yet!
    TCTouch.Validate("","Subtest 1 body");
  end Subtest_1;
---------------------------------------------------------------- Subtest_2
  -- These declarations should cause calls to initialize and
  -- finalize.  The expected operations are the subprograms associated
  -- with the abstract types.  Note that for these objects, the
  -- Initialize and Finalize are visible only by inheritance.
  Subtest_2_Inits_Expected : constant := 4;
  procedure Subtest_2 is
    Item_1 : C761003_2.Global;
    Item_2, Item_3 : C761003_2.Global;
    Item_4 : C761003_2.Second;
  begin
    -- check that nothing has happened yet!
    TCTouch.Validate("","Subtest 2 body");
  end Subtest_2;
---------------------------------------------------------------- Subtest_3
  -- Test for controlled objects embedded in arrays.  Using structures
  -- that will cause a checkable order.
  Subtest_3_Inits_Expected : constant := 8;
  procedure Subtest_3 is
    type Global_List is array(Natural range <>)
                          of C761003_0.Global(Sup.Pick_Char);
    Items : Global_List(1..4);  -- components have the same tag
    type Second_List is array(Natural range <>)
                          of C761003_0.Second(Sup.Pick_Char);
    Second_Items : Second_List(1..4);  -- components have the same tag,
                                       -- distinct from the tag used in Items
  begin
    -- check that nothing has happened yet!
    TCTouch.Validate("","Subtest 3 body");
  end Subtest_3;
---------------------------------------------------------------- Subtest_4
  -- These declarations should cause dispatching calls to initialize and
  -- finalize.  The expected operations are the subprograms associated
  -- with the abstract types.
  Subtest_4_Inits_Expected : constant := 2;
  procedure Subtest_4 is
    type Global_Rec is record
      Item1: C761003_0.Global(Sup.Pick_Char);
    end record;
    type Second_Rec is record
      Item2: C761003_2.Second;
    end record;
    G : Global_Rec;
    S : Second_Rec;
  begin
    -- check that nothing has happened yet!
    TCTouch.Validate("","Subtest 4 body");
  end Subtest_4;
---------------------------------------------------------------- Subtest_5
  -- Test for controlled objects embedded in arrays.  In these cases, the
  -- order of the finalization of the components is not defined by the
  -- language.
  Subtest_5_Inits_Expected : constant := 8;
  procedure Subtest_5 is
    type Another_Global_List is array(Natural range <>)
                          of C761003_2.Global;
    More_Items : Another_Global_List(1..4);
    type Another_Second_List is array(Natural range <>)
                          of C761003_2.Second;
    Second_More_Items : Another_Second_List(1..4);
  begin
    -- check that nothing has happened yet!
    TCTouch.Validate("","Subtest 5 body");
  end Subtest_5;
---------------------------------------------------------------- Subtest_6
  -- These declarations should cause dispatching calls to initialize and
  -- finalize.  The expected operations are the subprograms associated
  -- with the abstract types.
  Subtest_6_Inits_Expected : constant := 2;
  procedure Subtest_6 is
    type Global_Rec is record
     Item2: C761003_2.Global;
    end record;
    type Second_Rec is record
      Item1: C761003_0.Second(Sup.Pick_Char);
   end record;
    G : Global_Rec;
    S : Second_Rec;
  begin
    -- check that nothing has happened yet!
    TCTouch.Validate("","Subtest 6 body");
  end Subtest_6;
begin  -- Main test procedure.
  Report.Test ("C761003", "Check that an object of a controlled type "
                        & "is finalized when the enclosing master is "
                        & "complete, left by a transfer of control, "
                        & "and performed in the correct order" );
  -- adjust for optional adjusts and initializes for C761003_0.Null_Global
  TCTouch.Flush; -- clear the optional adjust
  if Sup.Inits_Called /= 1 then
    -- C761003_0.Null_Global did not get "initialized"
    C761003_0.Initialize(C761003_0.Null_Global);  -- prime the pump
  end if;
  Subtest_1;
  Sup.Validate(Subtest_1_Inits_Expected, 1);
  Subtest_2;
  Sup.Validate(Subtest_2_Inits_Expected, 2);
  Subtest_3;
  Sup.Validate(Subtest_3_Inits_Expected, 3);
  Subtest_4;
  Sup.Validate(Subtest_4_Inits_Expected, 4);
  Subtest_5;
  Sup.Validate(Subtest_5_Inits_Expected, 5, Check_Order => False);
  Subtest_6;
  Sup.Validate(Subtest_6_Inits_Expected, 6);
  Report.Result;
end C761003;