File: 5otaprop.adb

package info (click to toggle)
gnat 3.10p-3
  • links: PTS
  • area: main
  • in suites: hamm, slink
  • size: 49,492 kB
  • ctags: 33,976
  • sloc: ansic: 347,844; ada: 227,415; sh: 8,759; yacc: 7,861; asm: 5,252; makefile: 3,632; objc: 475; cpp: 400; sed: 261; pascal: 95
file content (632 lines) | stat: -rw-r--r-- 21,663 bytes parent folder | download
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
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
------------------------------------------------------------------------------
--                                                                          --
--                 GNU ADA RUNTIME LIBRARY (GNARL) COMPONENTS               --
--                                                                          --
--                 S Y S T E M . T A S K _ P R I M I T I V E S .            --
--                           O P E R A T I O N S                            --
--                                                                          --
--                                  B o d y                                 --
--                         (Version for new GNARL)                          --
--                                                                          --
--                             $Revision: 1.2 $                            --
--                                                                          --
--   Copyright (C) 1991,1992,1993,1994,1995,1996 Florida State University   --
--                                                                          --
-- GNARL 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 2,  or (at your option) any later ver- --
-- sion. GNARL 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.  See the GNU General Public License --
-- for  more details.  You should have  received  a copy of the GNU General --
-- Public License  distributed with GNARL; see file COPYING.  If not, write --
-- to  the Free Software Foundation,  59 Temple Place - Suite 330,  Boston, --
-- MA 02111-1307, USA.                                                      --
--                                                                          --
-- As a special exception,  if other files  instantiate  generics from this --
-- unit, or you link  this unit with other files  to produce an executable, --
-- this  unit  does not  by itself cause  the resulting  executable  to  be --
-- covered  by the  GNU  General  Public  License.  This exception does not --
-- however invalidate  any other reasons why  the executable file  might be --
-- covered by the  GNU Public License.                                      --
--                                                                          --
-- GNARL was developed by the GNARL team at Florida State University. It is --
-- now maintained by Ada Core Technologies Inc. in cooperation with Florida --
-- State University (http://www.gnat.com).                                  --
--                                                                          --
------------------------------------------------------------------------------

--  .... This is a preliminary draft.
--  Lines containing .... contain notes about details
--  that still need attention. -- Ted Baker

--  This is an OS/2 version of this package.

with Interfaces.C;
--  used for size_t

with Interfaces.OS2Lib.Threads;
with Interfaces.OS2Lib.Synchronization;
with Interfaces.OS2Lib.Errors;

with System.Parameters;
--  used for Size_Type

with System.Tasking;
--  used for Task_ID

with System.Error_Reporting;
--  used for Shutdown

with System.Parameters;
--  used for Size_Type

with System.Storage_Elements; use System.Storage_Elements;

with Unchecked_Conversion;
with Unchecked_Deallocation;

package body System.Task_Primitives.Operations is

   package IC renames Interfaces.C;
   use Interfaces.OS2Lib;
   use Interfaces.OS2Lib.Errors;
   use Interfaces.OS2Lib.Threads;
   use Interfaces.OS2Lib.Synchronization;
   use System.Tasking;
   use System.Error_Reporting;

   ------------------
   --  Local Types --
   ------------------

   type Microseconds is new IC.long;

   subtype Fractional_Second is Microseconds range
     0 .. 10#1#E6; --  less than or equal to one second

   type time_t is new IC.int;

   type struct_timeval is record
      tv_sec : time_t;
      tv_usec : Fractional_Second;
   end record;

   type timezone is record
      tz_minuteswest : IC.int;   -- of GMT
      tz_dsttime     : IC.int;   -- type of dst correction to apply
   end record;

   ------------------
   --  Local Data  --
   ------------------

   Clock_Delay_Correction : constant := 0.0;
   --  to allow for early wakeup due to truncation of time values
   --  .... set this to an appropriate value for OS/2

   --  The OS/2 DosAllocThreadLocalMemory API is used to allocate our TCB_Ptr.

   --  This API reserves a small range of virtual addresses that is backed
   --  by different physical memory for each running thread. In this case we
   --  create a pointer at a fixed address that points to the TCB_Ptr for the
   --  running thread. So all threads will be able to query and update their
   --  own TCB_Ptr without destroying the TCB_Ptr of other threads.

   type Thread_Local_Data is record
      Self_ID : Task_ID; --  ID of the current thread
      --  ... room for expansion here, if we decide to make access to
      --  jump-buffer and exception stack more efficient in future
   end record;
   type Access_Thread_Local_Data is access all Thread_Local_Data;
   Thread_Local_Data_Ptr : aliased Access_Thread_Local_Data;
   --  Pointer to Thread Local Data

   type PPTLD is access all Access_Thread_Local_Data;
   type PPVOID is access all PVOID;

   -----------------------
   -- Local Subprograms --
   -----------------------

   function To_PPVOID is new Unchecked_Conversion (PPTLD, PPVOID);
   function To_Address is new Unchecked_Conversion (Task_ID, System.Address);
   function To_PFNTHREAD is
     new Unchecked_Conversion (System.Address, PFNTHREAD);

   function To_Duration (TV : struct_timeval) return Duration;
   function To_MS (D : Duration) return IC.int;
   function To_MS (D : Duration) return ULONG;
   function To_Timeval (D : Duration) return struct_timeval;

   --  OS/2 has the gettimeofday() function in the EMX library

   function gettimeofday
     (tp : access struct_timeval;
      tzp : access timezone)
      return IC.int;
   pragma Import (C, gettimeofday);

   function To_Duration (TV : struct_timeval) return Duration is
   begin
      return Duration (TV.tv_sec) + Duration (TV.tv_usec) / 10#1#E6;
   end To_Duration;

   function To_Timeval (D : Duration) return struct_timeval is
      S : time_t;
      F : Duration;
   begin
      S := time_t (D);
      F := D - Duration (S);

      --  If F has negative value due to a round-up, adjust for positive F
      --  value.
      if F < 0.0 then S := S - 1; F := F + 1.0; end if;
      return struct_timeval' (tv_sec => S,
        tv_usec => Microseconds (F * 10#1#E6));
   end To_Timeval;

   function To_MS (D : Duration) return ULONG is
   begin
      return ULONG (D * 10#1#E3);
   end To_MS;

   function To_MS (D : Duration) return IC.int is
   begin
      return IC.int (D * 10#1#E3);
   end To_MS;

   -------------------
   -- Abort_Handler --
   -------------------

   --  Apparently, OS/2 does not support per-thread asynchronous signals,
   --  so we have no support for ATC or asynchronous task abort.

   ----------
   -- Self --
   ----------

   function Self return Task_ID is
   begin
      return Thread_Local_Data_Ptr.Self_ID;
   end Self;

   ---------------------
   -- Initialize_Lock --
   ---------------------

   procedure Initialize_Lock
     (Prio : System.Any_Priority;
      L    : access Lock)
   is
   begin
      if DosCreateMutexSem
        (ICS.Null_Ptr, L.Mutex'Unchecked_Access, 0, False32) /= NO_ERROR then
         raise Storage_Error;
      end if;
      L.Priority := Prio;
   end Initialize_Lock;

   procedure Initialize_Lock (L : access RTS_Lock) is
   begin
      if DosCreateMutexSem
        (ICS.Null_Ptr, L.Mutex'Unchecked_Access, 0, False32) /= NO_ERROR then
         raise Storage_Error;
      end if;
      --  Is this the right priority, or do we want System.Priority'Last??????
      L.Priority := System.Any_Priority'Last;
   end Initialize_Lock;

   -------------------
   -- Finalize_Lock --
   -------------------

   procedure Finalize_Lock (L : access Lock) is
   begin
      Must_Not_Fail (DosCloseMutexSem (L.Mutex));
   end Finalize_Lock;

   procedure Finalize_Lock (L : access RTS_Lock) is
   begin
      Must_Not_Fail (DosCloseMutexSem (L.Mutex));
   end Finalize_Lock;

   ----------------
   -- Write_Lock --
   ----------------

   procedure Write_Lock (L : access Lock; Ceiling_Violation : out Boolean) is
      Self_ID : constant Task_ID := Thread_Local_Data_Ptr.Self_ID;
   begin
      L.Owner_Priority := Self_ID.LL.Active_Priority;
      if L.Priority < L.Owner_Priority then
         Ceiling_Violation := True;
         return;
      end if;
      Must_Not_Fail (DosRequestMutexSem (L.Mutex, SEM_INDEFINITE_WAIT));
      Ceiling_Violation := False;
      if L.Priority > L.Owner_Priority then
         Self_ID.LL.Active_Priority := L.Priority;
      end if;
   end Write_Lock;

   procedure Write_Lock (L : access RTS_Lock) is
      Self_ID : constant Task_ID := Thread_Local_Data_Ptr.Self_ID;
   begin
      L.Owner_Priority := Self_ID.LL.Active_Priority;
      Must_Not_Fail (DosRequestMutexSem (L.Mutex, SEM_INDEFINITE_WAIT));
      if L.Priority > L.Owner_Priority then
         Set_Priority (Self_ID, L.Priority);
      end if;
   end Write_Lock;

   procedure Write_Lock (T : Task_ID) is
   begin
      T.LL.L.Owner_Priority := T.LL.Active_Priority;
      Must_Not_Fail (DosRequestMutexSem (T.LL.L.Mutex, SEM_INDEFINITE_WAIT));
      if T.LL.L.Priority > T.LL.L.Owner_Priority then
         Set_Priority (T, T.LL.L.Priority);
      end if;
   end Write_Lock;

   ---------------
   -- Read_Lock --
   ---------------

   procedure Read_Lock (L : access Lock; Ceiling_Violation : out Boolean)
      renames Write_Lock;

   ------------
   -- Unlock --
   ------------

   procedure Unlock (L : access Lock) is
      Self_ID : constant Task_ID := Thread_Local_Data_Ptr.Self_ID;
   begin
      if L.Owner_Priority /= L.Priority then
         Set_Priority (Self_ID, L.Owner_Priority);
      end if;
      Must_Not_Fail (DosReleaseMutexSem (L.Mutex));
   end Unlock;

   procedure Unlock (L : access RTS_Lock) is
      Self_ID : constant Task_ID := Thread_Local_Data_Ptr.Self_ID;
   begin
      if L.Owner_Priority /= L.Priority then
         Set_Priority (Self_ID, L.Owner_Priority);
      end if;
      Must_Not_Fail (DosReleaseMutexSem (L.Mutex));
   end Unlock;

   procedure Unlock (T : Task_ID) is
   begin
      if T.LL.L.Owner_Priority /= T.LL.L.Priority then
         T.LL.Active_Priority := T.LL.L.Owner_Priority;
      end if;
      Must_Not_Fail (DosReleaseMutexSem (T.LL.L.Mutex));
   end Unlock;

   -----------
   -- Sleep --
   -----------

   procedure Sleep (Self_ID : Task_ID) is
      Count : aliased ULONG; -- Unused
   begin
      --  Must reset Cond BEFORE L is unlocked.
      Must_Not_Fail (DosResetEventSem (Self_ID.LL.CV, Count'Unchecked_Access));
      Unlock (Self_ID);
      --  No problem if we are interrupted here.
      --  If the condition is signaled, DosWaitEventSem will simply not block.
      Must_Not_Fail (DosWaitEventSem (Self_ID.LL.CV, SEM_INDEFINITE_WAIT));
      --  Since L was previously accquired, lock operation should not fail.
      Write_Lock (Self_ID);
   end Sleep;

   ---------------
   -- Sleep_For --
   ---------------

   procedure Sleep_For
     (Self_ID : Task_ID; Rel_Time : Duration; Timedout : out Boolean) is
      Count : aliased ULONG; -- Unused
   begin
--  ***** We need tO set the value for Timedout somewhere below!!!!!

      --  Must reset Cond BEFORE L is unlocked.
      Must_Not_Fail (DosResetEventSem (Self_ID.LL.CV, Count'Unchecked_Access));
      Unlock (Self_ID);
      if Rel_Time > 0.0 then
         Must_Not_Fail (DosWaitEventSem (Self_ID.LL.CV,
--  .... check all L and CV ref's to see whether we need 'access.
           To_MS (Rel_Time + Clock_Delay_Correction)));
         --  No problem if we are interrupted here.
         --  If the condition is signaled,
         --  DosWaitEventSem will simply not block.
         --  .... How about early return?  If that is possible, we need to
         --  loop, possibly checking the clock.
         --  .... How about overflow on To_MS? Can it happen?
      else
         --  Must reset Cond BEFORE L is unlocked.
         Must_Not_Fail (DosResetEventSem
           (Self_ID.LL.CV, Count'Unchecked_Access));
         Unlock (Self_ID);
         --  No problem if we are interrupted here.
         --  If the condition is signaled,
         --  DosWaitEventSem will simply not block.
         Must_Not_Fail (DosWaitEventSem
           (Self_ID.LL.CV, SEM_INDEFINITE_WAIT));
         --  Since L was previously accquired,
         --  lock operation should not fail.
      end if;
      Write_Lock (Self_ID);
   end Sleep_For;

   -----------------
   -- Sleep_Until --
   -----------------

   procedure Sleep_Until
     (Self_ID : Task_ID; Abs_Time : Duration; Timedout : out Boolean) is
      Rel_Time : Duration;
   begin
      --  Change Abs_time to a relative delay.
      --  Be careful not to reintroduce the race condition that gave birth
      --  to delay until.
      Must_Not_Fail (DosEnterCritSec);
      Rel_Time := Abs_Time - Clock + Clock_Delay_Correction;
      Must_Not_Fail (DosExitCritSec);
      Sleep_For (Self_ID, Rel_Time, Timedout);
   end Sleep_Until;

   ------------
   -- Wakeup --
   ------------

   procedure Wakeup (T : Task_ID) is
   begin
      Must_Not_Fail (DosPostEventSem (T.LL.CV));
   end Wakeup;

   ------------------
   -- Set_Priority --
   ------------------

   procedure Set_Priority (T : Task_ID; Prio : System.Any_Priority) is
   begin
      Must_Not_Fail
        (DosSetPriority (Scope   => PRTYS_THREAD,
                         Class   => PRTYC_NOCHANGE,
                         Delta_P => IC.long (Prio - T.LL.Active_Priority),
                         PorTid  => T.LL.Thread));
      T.LL.Active_Priority := Prio;
   end Set_Priority;

   ------------------
   -- Get_Priority --
   ------------------

   function Get_Priority (T : Task_ID) return System.Any_Priority is
   begin
      return System.Any_Priority (T.LL.Active_Priority);
   end Get_Priority;

   ----------------
   -- Enter_Task --
   ----------------

   procedure Enter_Task (Self_ID : Task_ID) is
   begin
      Thread_Local_Data_Ptr.Self_ID := Self_ID;
      --  For OS/2, we can set Self_ID.LL.Thread in
      --  Create_Task, since the thread is created suspended.
      --  That is, there is no danger of the thread racing ahead
      --  and trying to reference Self_ID.LL.Thread before it
      --  has been initialized.
      --  .... Do we need to do anything with signals for OS/2?
   end Enter_Task;

   ----------------------
   --  Initialize_TCB  --
   ----------------------

   procedure Initialize_TCB (Self_ID : Task_ID; Succeeded : out Boolean) is
   begin
      Unlock (Self_ID);
      if DosCreateMutexSem
        (ICS.Null_Ptr, Self_ID.LL.L.Mutex'Unchecked_Access, 0, False32)
          = NO_ERROR then
         --  Is this the right priority,
         --  or do we want System.Priority'Last??????
         Self_ID.LL.L.Priority := System.Any_Priority'Last;
         if DosCreateEventSem (ICS.Null_Ptr,
              Self_ID.LL.CV'Unchecked_Access, 0, True32)
           /= NO_ERROR then
            Must_Not_Fail (DosCloseMutexSem (Self_ID.LL.L.Mutex));
            Succeeded := False;
         else Succeeded := True;
         end if;
      else Succeeded := False;
      end if;
      --  assumes any failure must be due to insufficient resources
   end Initialize_TCB;

   -----------------
   -- Create_Task --
   -----------------

   procedure Create_Task
     (T          : Task_ID;
      Wrapper    : System.Address;
      Stack_Size : System.Parameters.Size_Type;
      Priority   : System.Any_Priority;
      Succeeded  : out Boolean)
   is
      Result  : APIRET;
      Success : Boolean;
      Adjusted_Stack_Size : IC.size_t;
      use System.Parameters;
   begin
      if Stack_Size = Unspecified_Size then
         Adjusted_Stack_Size :=
           IC.size_t (2 * Default_Stack_Size);
      --  .... If OS/@ has a way to make a thread's stack size extensible,
      --  arrange to make the stack extensible.
      --  if Stack_Size = Unspecified_Size then
      --  .... Round up the stack size, if it is below the minimum size
      --  required for OS/2 threads.
      else
         if Stack_Size < Parameters.Minimum_Stack_Size then
            Adjusted_Stack_Size :=
              IC.size_t (Stack_Size + Minimum_Stack_Size);
         else
            Adjusted_Stack_Size := IC.size_t (Stack_Size);
         end if;
      end if;

      --  .... Add some head-space to the stack size, if there is a known
      --  amount needed by OS/2.

      Initialize_TCB (T, Success);
      if Success then
         --  create the thread, in blocked mode
         Result := DosCreateThread
                   (F_ptid   => T.LL.Thread'Unchecked_Access,
                    pfn      => To_PFNTHREAD (Wrapper),
                    param    => To_Address (T),
                    flag     => 1, -- Block_child + No_commit_stack,
                    cbStack  => ULONG (Adjusted_Stack_Size));
         Succeeded := Result /= NO_ERROR;
         --  set the new thread's priority
         --  (child has inherited priority from parent)
         Must_Not_Fail (DosSetPriority
           (Scope   => PRTYS_THREAD,
            Class   => PRTYC_NOCHANGE,
            Delta_P => IC.long (Priority - Get_Priority (Self)),
            PorTid  => T.LL.Thread));
         --  start the thread executing
         Must_Not_Fail (DosResumeThread (T.LL.Thread));
      else Succeeded := False;
      end if;
   end Create_Task;

   ------------------
   -- Finalize_TCB --
   ------------------

   procedure Finalize_TCB (T : Task_ID) is
   begin
      Must_Not_Fail (DosCloseMutexSem (T.LL.L.Mutex));
      Must_Not_Fail (DosCloseEventSem (T.LL.CV));
      --  Do not deallocate TCB here.
      --  GNARL layer is responsible for that.
   end Finalize_TCB;

   ---------------
   -- Exit_Task --
   ---------------

   procedure Exit_Task is
   begin
      DosExit (EXIT_THREAD, 0);
      --  Do not finalize TCB here.
      --  GNARL layer is responsible for that.
   end Exit_Task;

   ----------------
   -- Abort_Task --
   ----------------

   procedure Abort_Task (T : Task_ID) is
   begin
      null;
      --  .... verify this
      --  OS/2 apparently has no per-thread signal capability,
      --  so we can't support ATC.
   end Abort_Task;

   -----------
   -- Clock --
   -----------

   function Clock return Duration is
      TV     : aliased struct_timeval;
      Result : IC.int;
      use type IC.int;
   begin
      Result := gettimeofday (TV'Access, null);
      pragma Assert (Result = 0
        or else Shutdown ("GNULLI failure---gettimeofday"));
      return To_Duration (TV);
   exception
   when others =>
      pragma Assert (Shutdown ("exception in Clock"));
      return 0.0; --  to avert warning for no return
   end Clock;

   ---------------
   -- Delay_For --
   ---------------

   --  ???? We need some more thought here about the correct treatment of
   --  early wakeup due to a signal.  If the signal is for task abortion we
   --  need to check for it and proceed accordingly.
   --  This may require checking all the points where these are called.

   procedure Delay_For (Rel_Time : Duration) is
   begin
      if Rel_Time > 0.0 then
         DosSleep (To_MS (Rel_Time + Clock_Delay_Correction));
         --  .... Can this return early?
         --  If that is possible, we need to loop, checking the clock.
      else
         --  .... What is the correct OS/2 way to yield
         --  to equal-priority threads?
         DosSleep (0);
      end if;
   end Delay_For;

   -----------------
   -- Delay_Until --
   -----------------

   procedure Delay_Until (Abs_Time : Duration) is
      Rel_Time : Duration;
   begin
      --  Change Abs_time to a relative delay.
      --  Be careful not to reintroduce the race condition that gave birth
      --  to delay until.
      Must_Not_Fail (DosEnterCritSec);
      Rel_Time := Abs_Time - Clock + Clock_Delay_Correction;
      Must_Not_Fail (DosExitCritSec);
      Delay_For (Rel_Time);
   end Delay_Until;

   procedure Initialize (Environment_Task : Task_ID) is
      Succeeded : Boolean;
   begin
      --  Initialize pointer to task local data.
      --  This is done once, for all tasks.
      Must_Not_Fail (DosAllocThreadLocalMemory
        (1, To_PPVOID (Thread_Local_Data_Ptr'Access)));
      --  Set ID of environment task.
      Environment_Task.LL.Thread := 1; --  By definition
      --  Initialize TCB for this task.
      --  This includes all the normal task-external initialization.
      Initialize_TCB (Environment_Task, Succeeded);
      pragma Assert (Succeeded
        or else Shutdown ("GNULLI failure---Initialize_TCB"));
      --  Consider raising Storage_Error, if propagation can be tolerated????
      --  Do normal task-internal initialization,
      --  which depends on an initialized TCB.
      Enter_Task (Environment_Task);
      --  Insert here any other special
      --  initialization needed for the environment task.
   end Initialize;

--  Please, put no initialization code in the body!
--  All global initializations for this package belong
--  in procedure Initialize (above).

end System.Task_Primitives.Operations;