-- C974005.A
|
-- C974005.A
|
--
|
--
|
-- Grant of Unlimited Rights
|
-- Grant of Unlimited Rights
|
--
|
--
|
-- Under contracts F33600-87-D-0337, F33600-84-D-0280, MDA903-79-C-0687,
|
-- 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
|
-- 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 in the software and documentation contained herein.
|
-- Unlimited rights are defined in DFAR 252.227-7013(a)(19). By making
|
-- Unlimited rights are defined in DFAR 252.227-7013(a)(19). By making
|
-- this public release, the Government intends to confer upon all
|
-- this public release, the Government intends to confer upon all
|
-- recipients unlimited rights equal to those held by the Government.
|
-- recipients unlimited rights equal to those held by the Government.
|
-- These rights include rights to use, duplicate, release or disclose the
|
-- These rights include rights to use, duplicate, release or disclose the
|
-- released technical data and computer software in whole or in part, in
|
-- 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
|
-- any manner and for any purpose whatsoever, and to have or permit others
|
-- to do so.
|
-- to do so.
|
--
|
--
|
-- DISCLAIMER
|
-- DISCLAIMER
|
--
|
--
|
-- ALL MATERIALS OR INFORMATION HEREIN RELEASED, MADE AVAILABLE OR
|
-- ALL MATERIALS OR INFORMATION HEREIN RELEASED, MADE AVAILABLE OR
|
-- DISCLOSED ARE AS IS. THE GOVERNMENT MAKES NO EXPRESS OR IMPLIED
|
-- DISCLOSED ARE AS IS. THE GOVERNMENT MAKES NO EXPRESS OR IMPLIED
|
-- WARRANTY AS TO ANY MATTER WHATSOEVER, INCLUDING THE CONDITIONS OF THE
|
-- WARRANTY AS TO ANY MATTER WHATSOEVER, INCLUDING THE CONDITIONS OF THE
|
-- SOFTWARE, DOCUMENTATION OR OTHER INFORMATION RELEASED, MADE AVAILABLE
|
-- SOFTWARE, DOCUMENTATION OR OTHER INFORMATION RELEASED, MADE AVAILABLE
|
-- OR DISCLOSED, OR THE OWNERSHIP, MERCHANTABILITY, OR FITNESS FOR A
|
-- OR DISCLOSED, OR THE OWNERSHIP, MERCHANTABILITY, OR FITNESS FOR A
|
-- PARTICULAR PURPOSE OF SAID MATERIAL.
|
-- PARTICULAR PURPOSE OF SAID MATERIAL.
|
--*
|
--*
|
--
|
--
|
-- OBJECTIVE:
|
-- OBJECTIVE:
|
-- Check that Tasking_Error is raised at the point of an entry call
|
-- Check that Tasking_Error is raised at the point of an entry call
|
-- which is the triggering statement of an asynchronous select, if
|
-- which is the triggering statement of an asynchronous select, if
|
-- the entry call is queued, but the task containing the entry completes
|
-- the entry call is queued, but the task containing the entry completes
|
-- before it can be accepted or canceled.
|
-- before it can be accepted or canceled.
|
--
|
--
|
-- Check that the abortable part is aborted if it does not complete
|
-- Check that the abortable part is aborted if it does not complete
|
-- before the triggering statement completes.
|
-- before the triggering statement completes.
|
--
|
--
|
-- Check that the sequence of statements of the triggering alternative
|
-- Check that the sequence of statements of the triggering alternative
|
-- is not executed.
|
-- is not executed.
|
--
|
--
|
-- TEST DESCRIPTION:
|
-- TEST DESCRIPTION:
|
-- Declare a main procedure containing an asynchronous select with a task
|
-- Declare a main procedure containing an asynchronous select with a task
|
-- entry call as triggering statement. Force the entry call to be
|
-- entry call as triggering statement. Force the entry call to be
|
-- queued by having the task call a procedure, prior to the corresponding
|
-- queued by having the task call a procedure, prior to the corresponding
|
-- accept statement, which simulates a routine waiting for user input
|
-- accept statement, which simulates a routine waiting for user input
|
-- (with a delay).
|
-- (with a delay).
|
--
|
--
|
-- Simulate a time-consuming routine in the abortable part by calling a
|
-- Simulate a time-consuming routine in the abortable part by calling a
|
-- procedure containing an infinite loop. Meanwhile, simulate input by
|
-- procedure containing an infinite loop. Meanwhile, simulate input by
|
-- the user (the delay expires) which is NOT the input expected by the
|
-- the user (the delay expires) which is NOT the input expected by the
|
-- guard on the accept statement. The entry remains closed, and the
|
-- guard on the accept statement. The entry remains closed, and the
|
-- task completes its execution. Since the entry was not accepted before
|
-- task completes its execution. Since the entry was not accepted before
|
-- its task completed, Tasking_Error is raised at the point of the entry
|
-- its task completed, Tasking_Error is raised at the point of the entry
|
-- call.
|
-- call.
|
--
|
--
|
--
|
--
|
-- CHANGE HISTORY:
|
-- CHANGE HISTORY:
|
-- 06 Dec 94 SAIC ACVC 2.0
|
-- 06 Dec 94 SAIC ACVC 2.0
|
--
|
--
|
--!
|
--!
|
|
|
package C974005_0 is -- Automated teller machine abstraction.
|
package C974005_0 is -- Automated teller machine abstraction.
|
|
|
|
|
-- Flags for testing purposes:
|
-- Flags for testing purposes:
|
|
|
Count : Integer := 1234;
|
Count : Integer := 1234;
|
|
|
type Key_Enum is (None, Cancel, Deposit, Withdraw);
|
type Key_Enum is (None, Cancel, Deposit, Withdraw);
|
|
|
type Card_Number_Type is private;
|
type Card_Number_Type is private;
|
type Card_PIN_Type is private;
|
type Card_PIN_Type is private;
|
type ATM_Card_Type is private;
|
type ATM_Card_Type is private;
|
|
|
|
|
Transaction_Canceled : exception;
|
Transaction_Canceled : exception;
|
|
|
|
|
task type ATM_Keyboard_Task is
|
task type ATM_Keyboard_Task is
|
entry Cancel_Pressed;
|
entry Cancel_Pressed;
|
end ATM_Keyboard_Task;
|
end ATM_Keyboard_Task;
|
|
|
|
|
procedure Read_Card (Card : in out ATM_Card_Type);
|
procedure Read_Card (Card : in out ATM_Card_Type);
|
|
|
procedure Validate_Card (Card : in ATM_Card_Type);
|
procedure Validate_Card (Card : in ATM_Card_Type);
|
|
|
procedure Perform_Transaction (Card : in ATM_Card_Type);
|
procedure Perform_Transaction (Card : in ATM_Card_Type);
|
|
|
private
|
private
|
|
|
type Card_Number_Type is range 1 .. 9999;
|
type Card_Number_Type is range 1 .. 9999;
|
type Card_PIN_Type is range 100 .. 999;
|
type Card_PIN_Type is range 100 .. 999;
|
|
|
type ATM_Card_Type is record
|
type ATM_Card_Type is record
|
Number : Card_Number_Type;
|
Number : Card_Number_Type;
|
PIN : Card_PIN_Type;
|
PIN : Card_PIN_Type;
|
end record;
|
end record;
|
|
|
end C974005_0;
|
end C974005_0;
|
|
|
|
|
--==================================================================--
|
--==================================================================--
|
|
|
|
|
with Report;
|
with Report;
|
with ImpDef;
|
with ImpDef;
|
|
|
package body C974005_0 is
|
package body C974005_0 is
|
|
|
|
|
procedure Listen_For_Input (Key : out Key_Enum) is
|
procedure Listen_For_Input (Key : out Key_Enum) is
|
begin
|
begin
|
-- Simulate the situation where a user waits a bit for the card to
|
-- Simulate the situation where a user waits a bit for the card to
|
-- be validated, then presses a transaction key (NOT Cancel).
|
-- be validated, then presses a transaction key (NOT Cancel).
|
|
|
-- Delay long enough to force queuing of Keyboard.Cancel_Pressed.
|
-- Delay long enough to force queuing of Keyboard.Cancel_Pressed.
|
delay ImpDef.Clear_Ready_Queue;
|
delay ImpDef.Clear_Ready_Queue;
|
|
|
if Report.Equal (3, 3) then -- Always true.
|
if Report.Equal (3, 3) then -- Always true.
|
Key := Deposit; -- Cancel is NOT pressed.
|
Key := Deposit; -- Cancel is NOT pressed.
|
end if;
|
end if;
|
end Listen_For_Input;
|
end Listen_For_Input;
|
|
|
|
|
task body ATM_Keyboard_Task is
|
task body ATM_Keyboard_Task is
|
Key_Pressed : Key_Enum := None;
|
Key_Pressed : Key_Enum := None;
|
begin
|
begin
|
|
|
-- Note: no loop. If the user does not press Cancel, the task completes.
|
-- Note: no loop. If the user does not press Cancel, the task completes.
|
-- In this model of the keyboard monitor, the user only gets one chance
|
-- In this model of the keyboard monitor, the user only gets one chance
|
-- to cancel the card validation.
|
-- to cancel the card validation.
|
-- Force entry
|
-- Force entry
|
Listen_For_Input (Key_Pressed); -- calls to be
|
Listen_For_Input (Key_Pressed); -- calls to be
|
-- queued, but do
|
-- queued, but do
|
-- NOT set guard
|
-- NOT set guard
|
-- to true.
|
-- to true.
|
select
|
select
|
when (Key_Pressed = Cancel) => -- Guard is false,
|
when (Key_Pressed = Cancel) => -- Guard is false,
|
accept Cancel_Pressed do -- so entry call
|
accept Cancel_Pressed do -- so entry call
|
Report.Failed ("Accept statement executed"); -- remains queued.
|
Report.Failed ("Accept statement executed"); -- remains queued.
|
end Cancel_Pressed;
|
end Cancel_Pressed;
|
else -- Else alternative
|
else -- Else alternative
|
Key_Pressed := None; -- executed, then
|
Key_Pressed := None; -- executed, then
|
end select; -- task ends.
|
end select; -- task ends.
|
exception
|
exception
|
when others =>
|
when others =>
|
Report.Failed ("Unexpected exception in ATM_Keyboard_Task");
|
Report.Failed ("Unexpected exception in ATM_Keyboard_Task");
|
end ATM_Keyboard_Task;
|
end ATM_Keyboard_Task;
|
|
|
|
|
|
|
procedure Read_Card (Card : in out ATM_Card_Type) is
|
procedure Read_Card (Card : in out ATM_Card_Type) is
|
begin
|
begin
|
Card.Number := 9999;
|
Card.Number := 9999;
|
Card.PIN := 111;
|
Card.PIN := 111;
|
end Read_Card;
|
end Read_Card;
|
|
|
|
|
procedure Validate_Card (Card : in ATM_Card_Type) is
|
procedure Validate_Card (Card : in ATM_Card_Type) is
|
begin
|
begin
|
-- Simulate an exceedingly long validation activity.
|
-- Simulate an exceedingly long validation activity.
|
loop -- Infinite loop.
|
loop -- Infinite loop.
|
Count := (Count + 1) mod Integer (Card.PIN);
|
Count := (Count + 1) mod Integer (Card.PIN);
|
|
|
-- Synch Point to allow transfer of control to Keyboard task
|
-- Synch Point to allow transfer of control to Keyboard task
|
-- during this simulation
|
-- during this simulation
|
delay ImpDef.Minimum_Task_Switch;
|
delay ImpDef.Minimum_Task_Switch;
|
|
|
exit when not Report.Equal (Count, Count); -- Always false.
|
exit when not Report.Equal (Count, Count); -- Always false.
|
end loop;
|
end loop;
|
end Validate_Card;
|
end Validate_Card;
|
|
|
|
|
procedure Perform_Transaction (Card : in ATM_Card_Type) is
|
procedure Perform_Transaction (Card : in ATM_Card_Type) is
|
begin
|
begin
|
Report.Failed ("Exception not re-raised immediately following " &
|
Report.Failed ("Exception not re-raised immediately following " &
|
"asynchronous select");
|
"asynchronous select");
|
if Count = 1234 then
|
if Count = 1234 then
|
-- Additional analysis added to aid developers
|
-- Additional analysis added to aid developers
|
Report.Failed ("Abortable part did not execute");
|
Report.Failed ("Abortable part did not execute");
|
end if;
|
end if;
|
end Perform_Transaction;
|
end Perform_Transaction;
|
|
|
|
|
end C974005_0;
|
end C974005_0;
|
|
|
|
|
--==================================================================--
|
--==================================================================--
|
|
|
|
|
with Report;
|
with Report;
|
|
|
with C974005_0; -- Automated teller machine abstraction.
|
with C974005_0; -- Automated teller machine abstraction.
|
use C974005_0;
|
use C974005_0;
|
|
|
procedure C974005 is
|
procedure C974005 is
|
|
|
Card_Data : ATM_Card_Type;
|
Card_Data : ATM_Card_Type;
|
|
|
begin -- Main program.
|
begin -- Main program.
|
|
|
Report.Test ("C974005", "ATC: trigger is queued but task terminates" &
|
Report.Test ("C974005", "ATC: trigger is queued but task terminates" &
|
" before call is serviced");
|
" before call is serviced");
|
|
|
Read_Card (Card_Data);
|
Read_Card (Card_Data);
|
|
|
begin
|
begin
|
|
|
declare
|
declare
|
Keyboard : C974005_0.ATM_Keyboard_Task;
|
Keyboard : C974005_0.ATM_Keyboard_Task;
|
begin
|
begin
|
|
|
-- --
|
-- --
|
-- Asynchronous select is tested here --
|
-- Asynchronous select is tested here --
|
-- --
|
-- --
|
|
|
select
|
select
|
Keyboard.Cancel_Pressed; -- Entry call initially queued, so
|
Keyboard.Cancel_Pressed; -- Entry call initially queued, so
|
-- abortable part starts.
|
-- abortable part starts.
|
|
|
-- Tasking_Error raised here when
|
-- Tasking_Error raised here when
|
-- Keyboard completes before entry
|
-- Keyboard completes before entry
|
-- call can be accepted, and before
|
-- call can be accepted, and before
|
-- abortable part completes.
|
-- abortable part completes.
|
|
|
raise Transaction_Canceled; -- Should not be executed.
|
raise Transaction_Canceled; -- Should not be executed.
|
then abort
|
then abort
|
Validate_Card (Card_Data); -- Keyboard task completes before
|
Validate_Card (Card_Data); -- Keyboard task completes before
|
-- Keyboard.Cancel_Pressed is
|
-- Keyboard.Cancel_Pressed is
|
-- accepted, and before this call
|
-- accepted, and before this call
|
-- finishes. Tasking_Error is raised
|
-- finishes. Tasking_Error is raised
|
-- at the point of the entry call,
|
-- at the point of the entry call,
|
-- and this call is aborted.
|
-- and this call is aborted.
|
-- Check that the whole of the abortable part is aborted, not just
|
-- Check that the whole of the abortable part is aborted, not just
|
-- the statement in the abortable part that was executing at
|
-- the statement in the abortable part that was executing at
|
-- the time
|
-- the time
|
Report.Failed ("Abortable part not aborted");
|
Report.Failed ("Abortable part not aborted");
|
end select;
|
end select;
|
Perform_Transaction (Card_Data); -- Should not be reached.
|
Perform_Transaction (Card_Data); -- Should not be reached.
|
exception
|
exception
|
when Transaction_Canceled =>
|
when Transaction_Canceled =>
|
Report.Failed ("Triggering alternative sequence of statements " &
|
Report.Failed ("Triggering alternative sequence of statements " &
|
"executed");
|
"executed");
|
when Tasking_Error =>
|
when Tasking_Error =>
|
if Count = 1234 then
|
if Count = 1234 then
|
Report.Failed ("Abortable part did not execute");
|
Report.Failed ("Abortable part did not execute");
|
end if;
|
end if;
|
when others =>
|
when others =>
|
Report.Failed ("Wrong exception raised");
|
Report.Failed ("Wrong exception raised");
|
end;
|
end;
|
|
|
exception
|
exception
|
when Tasking_Error =>
|
when Tasking_Error =>
|
Report.Failed ("Correct exception raised at wrong level");
|
Report.Failed ("Correct exception raised at wrong level");
|
when others =>
|
when others =>
|
Report.Failed ("Wrong exception raised at wrong level");
|
Report.Failed ("Wrong exception raised at wrong level");
|
end;
|
end;
|
|
|
Report.Result;
|
Report.Result;
|
|
|
end C974005;
|
end C974005;
|
|
|