OSDN Git Service

Update FSF address
[pf3gnuchains/gcc-fork.git] / gcc / ada / s-taprop-posix.adb
1 ------------------------------------------------------------------------------
2 --                                                                          --
3 --                 GNAT RUN-TIME LIBRARY (GNARL) COMPONENTS                 --
4 --                                                                          --
5 --     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    --
6 --                                                                          --
7 --                                  B o d y                                 --
8 --                                                                          --
9 --         Copyright (C) 1992-2005, Free Software Foundation, Inc.          --
10 --                                                                          --
11 -- GNARL is free software; you can  redistribute it  and/or modify it under --
12 -- terms of the  GNU General Public License as published  by the Free Soft- --
13 -- ware  Foundation;  either version 2,  or (at your option) any later ver- --
14 -- sion. GNARL is distributed in the hope that it will be useful, but WITH- --
15 -- OUT ANY WARRANTY;  without even the  implied warranty of MERCHANTABILITY --
16 -- or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License --
17 -- for  more details.  You should have  received  a copy of the GNU General --
18 -- Public License  distributed with GNARL; see file COPYING.  If not, write --
19 -- to  the  Free Software Foundation,  51  Franklin  Street,  Fifth  Floor, --
20 -- Boston, MA 02110-1301, USA.                                              --
21 --                                                                          --
22 -- As a special exception,  if other files  instantiate  generics from this --
23 -- unit, or you link  this unit with other files  to produce an executable, --
24 -- this  unit  does not  by itself cause  the resulting  executable  to  be --
25 -- covered  by the  GNU  General  Public  License.  This exception does not --
26 -- however invalidate  any other reasons why  the executable file  might be --
27 -- covered by the  GNU Public License.                                      --
28 --                                                                          --
29 -- GNARL was developed by the GNARL team at Florida State University.       --
30 -- Extensive contributions were provided by Ada Core Technologies, Inc.     --
31 --                                                                          --
32 ------------------------------------------------------------------------------
33
34 --  This is a POSIX-like version of this package
35
36 --  This package contains all the GNULL primitives that interface directly
37 --  with the underlying OS.
38
39 --  Note: this file can only be used for POSIX compliant systems that
40 --  implement SCHED_FIFO and Ceiling Locking correctly.
41
42 --  For configurations where SCHED_FIFO and priority ceiling are not a
43 --  requirement, this file can also be used (e.g AiX threads)
44
45 pragma Polling (Off);
46 --  Turn off polling, we do not want ATC polling to take place during
47 --  tasking operations. It causes infinite loops and other problems.
48
49 with System.Tasking.Debug;
50 --  used for Known_Tasks
51
52 with System.Task_Info;
53 --  used for Task_Info_Type
54
55 with Interfaces.C;
56 --  used for int
57 --           size_t
58
59 with System.Interrupt_Management;
60 --  used for Keep_Unmasked
61 --           Abort_Task_Interrupt
62 --           Interrupt_ID
63
64 with System.Parameters;
65 --  used for Size_Type
66
67 with System.Tasking;
68 --  used for Ada_Task_Control_Block
69 --           Task_Id
70
71 with System.Soft_Links;
72 --  used for Defer/Undefer_Abort
73
74 --  Note that we do not use System.Tasking.Initialization directly since
75 --  this is a higher level package that we shouldn't depend on. For example
76 --  when using the restricted run time, it is replaced by
77 --  System.Tasking.Restricted.Stages.
78
79 with System.OS_Primitives;
80 --  used for Delay_Modes
81
82 with Unchecked_Conversion;
83 with Unchecked_Deallocation;
84
85 package body System.Task_Primitives.Operations is
86
87    use System.Tasking.Debug;
88    use System.Tasking;
89    use Interfaces.C;
90    use System.OS_Interface;
91    use System.Parameters;
92    use System.OS_Primitives;
93
94    package SSL renames System.Soft_Links;
95
96    ----------------
97    -- Local Data --
98    ----------------
99
100    --  The followings are logically constants, but need to be initialized
101    --  at run time.
102
103    Single_RTS_Lock : aliased RTS_Lock;
104    --  This is a lock to allow only one thread of control in the RTS at
105    --  a time; it is used to execute in mutual exclusion from all other tasks.
106    --  Used mainly in Single_Lock mode, but also to protect All_Tasks_List
107
108    ATCB_Key : aliased pthread_key_t;
109    --  Key used to find the Ada Task_Id associated with a thread
110
111    Environment_Task_Id : Task_Id;
112    --  A variable to hold Task_Id for the environment task.
113
114    Locking_Policy : Character;
115    pragma Import (C, Locking_Policy, "__gl_locking_policy");
116    --  Value of the pragma Locking_Policy:
117    --    'C' for Ceiling_Locking
118    --    'I' for Inherit_Locking
119    --    ' ' for none.
120
121    Unblocked_Signal_Mask : aliased sigset_t;
122    --  The set of signals that should unblocked in all tasks
123
124    --  The followings are internal configuration constants needed.
125
126    Next_Serial_Number : Task_Serial_Number := 100;
127    --  We start at 100, to reserve some special values for
128    --  using in error checking.
129
130    Time_Slice_Val : Integer;
131    pragma Import (C, Time_Slice_Val, "__gl_time_slice_val");
132
133    Dispatching_Policy : Character;
134    pragma Import (C, Dispatching_Policy, "__gl_task_dispatching_policy");
135
136    FIFO_Within_Priorities : constant Boolean := Dispatching_Policy = 'F';
137    --  Indicates whether FIFO_Within_Priorities is set.
138
139    Foreign_Task_Elaborated : aliased Boolean := True;
140    --  Used to identified fake tasks (i.e., non-Ada Threads).
141
142    --------------------
143    -- Local Packages --
144    --------------------
145
146    package Specific is
147
148       procedure Initialize (Environment_Task : Task_Id);
149       pragma Inline (Initialize);
150       --  Initialize various data needed by this package.
151
152       function Is_Valid_Task return Boolean;
153       pragma Inline (Is_Valid_Task);
154       --  Does executing thread have a TCB?
155
156       procedure Set (Self_Id : Task_Id);
157       pragma Inline (Set);
158       --  Set the self id for the current task.
159
160       function Self return Task_Id;
161       pragma Inline (Self);
162       --  Return a pointer to the Ada Task Control Block of the calling task.
163
164    end Specific;
165
166    package body Specific is separate;
167    --  The body of this package is target specific.
168
169    ---------------------------------
170    -- Support for foreign threads --
171    ---------------------------------
172
173    function Register_Foreign_Thread (Thread : Thread_Id) return Task_Id;
174    --  Allocate and Initialize a new ATCB for the current Thread.
175
176    function Register_Foreign_Thread
177      (Thread : Thread_Id) return Task_Id is separate;
178
179    -----------------------
180    -- Local Subprograms --
181    -----------------------
182
183    procedure Abort_Handler (Sig : Signal);
184    --  Signal handler used to implement asynchronous abort.
185    --  See also comment before body, below.
186
187    function To_Address is new Unchecked_Conversion (Task_Id, System.Address);
188
189    -------------------
190    -- Abort_Handler --
191    -------------------
192
193    --  Target-dependent binding of inter-thread Abort signal to
194    --  the raising of the Abort_Signal exception.
195
196    --  The technical issues and alternatives here are essentially
197    --  the same as for raising exceptions in response to other
198    --  signals (e.g. Storage_Error). See code and comments in
199    --  the package body System.Interrupt_Management.
200
201    --  Some implementations may not allow an exception to be propagated
202    --  out of a handler, and others might leave the signal or
203    --  interrupt that invoked this handler masked after the exceptional
204    --  return to the application code.
205
206    --  GNAT exceptions are originally implemented using setjmp()/longjmp().
207    --  On most UNIX systems, this will allow transfer out of a signal handler,
208    --  which is usually the only mechanism available for implementing
209    --  asynchronous handlers of this kind. However, some
210    --  systems do not restore the signal mask on longjmp(), leaving the
211    --  abort signal masked.
212
213    procedure Abort_Handler (Sig : Signal) is
214       pragma Warnings (Off, Sig);
215
216       T       : constant Task_Id := Self;
217       Result  : Interfaces.C.int;
218       Old_Set : aliased sigset_t;
219
220    begin
221       --  It is not safe to raise an exception when using ZCX and the GCC
222       --  exception handling mechanism.
223
224       if ZCX_By_Default and then GCC_ZCX_Support then
225          return;
226       end if;
227
228       if T.Deferral_Level = 0
229         and then T.Pending_ATC_Level < T.ATC_Nesting_Level and then
230         not T.Aborting
231       then
232          T.Aborting := True;
233
234          --  Make sure signals used for RTS internal purpose are unmasked
235
236          Result := pthread_sigmask (SIG_UNBLOCK,
237            Unblocked_Signal_Mask'Unchecked_Access, Old_Set'Unchecked_Access);
238          pragma Assert (Result = 0);
239
240          raise Standard'Abort_Signal;
241       end if;
242    end Abort_Handler;
243
244    -----------------
245    -- Stack_Guard --
246    -----------------
247
248    procedure Stack_Guard (T : ST.Task_Id; On : Boolean) is
249       Stack_Base : constant Address := Get_Stack_Base (T.Common.LL.Thread);
250       Guard_Page_Address : Address;
251
252       Res : Interfaces.C.int;
253
254    begin
255       if Stack_Base_Available then
256
257          --  Compute the guard page address
258
259          Guard_Page_Address :=
260            Stack_Base - (Stack_Base mod Get_Page_Size) + Get_Page_Size;
261
262          if On then
263             Res := mprotect (Guard_Page_Address, Get_Page_Size, PROT_ON);
264          else
265             Res := mprotect (Guard_Page_Address, Get_Page_Size, PROT_OFF);
266          end if;
267
268          pragma Assert (Res = 0);
269       end if;
270    end Stack_Guard;
271
272    --------------------
273    -- Get_Thread_Id  --
274    --------------------
275
276    function Get_Thread_Id (T : ST.Task_Id) return OSI.Thread_Id is
277    begin
278       return T.Common.LL.Thread;
279    end Get_Thread_Id;
280
281    ----------
282    -- Self --
283    ----------
284
285    function Self return Task_Id renames Specific.Self;
286
287    ---------------------
288    -- Initialize_Lock --
289    ---------------------
290
291    --  Note: mutexes and cond_variables needed per-task basis are
292    --        initialized in Intialize_TCB and the Storage_Error is
293    --        handled. Other mutexes (such as RTS_Lock, Memory_Lock...)
294    --        used in RTS is initialized before any status change of RTS.
295    --        Therefore rasing Storage_Error in the following routines
296    --        should be able to be handled safely.
297
298    procedure Initialize_Lock
299      (Prio : System.Any_Priority;
300       L    : access Lock)
301    is
302       Attributes : aliased pthread_mutexattr_t;
303       Result : Interfaces.C.int;
304
305    begin
306       Result := pthread_mutexattr_init (Attributes'Access);
307       pragma Assert (Result = 0 or else Result = ENOMEM);
308
309       if Result = ENOMEM then
310          raise Storage_Error;
311       end if;
312
313       if Locking_Policy = 'C' then
314          Result := pthread_mutexattr_setprotocol
315            (Attributes'Access, PTHREAD_PRIO_PROTECT);
316          pragma Assert (Result = 0);
317
318          Result := pthread_mutexattr_setprioceiling
319             (Attributes'Access, Interfaces.C.int (Prio));
320          pragma Assert (Result = 0);
321
322       elsif Locking_Policy = 'I' then
323          Result := pthread_mutexattr_setprotocol
324            (Attributes'Access, PTHREAD_PRIO_INHERIT);
325          pragma Assert (Result = 0);
326       end if;
327
328       Result := pthread_mutex_init (L, Attributes'Access);
329       pragma Assert (Result = 0 or else Result = ENOMEM);
330
331       if Result = ENOMEM then
332          raise Storage_Error;
333       end if;
334
335       Result := pthread_mutexattr_destroy (Attributes'Access);
336       pragma Assert (Result = 0);
337    end Initialize_Lock;
338
339    procedure Initialize_Lock (L : access RTS_Lock; Level : Lock_Level) is
340       pragma Warnings (Off, Level);
341
342       Attributes : aliased pthread_mutexattr_t;
343       Result     : Interfaces.C.int;
344
345    begin
346       Result := pthread_mutexattr_init (Attributes'Access);
347       pragma Assert (Result = 0 or else Result = ENOMEM);
348
349       if Result = ENOMEM then
350          raise Storage_Error;
351       end if;
352
353       if Locking_Policy = 'C' then
354          Result := pthread_mutexattr_setprotocol
355            (Attributes'Access, PTHREAD_PRIO_PROTECT);
356          pragma Assert (Result = 0);
357
358          Result := pthread_mutexattr_setprioceiling
359             (Attributes'Access, Interfaces.C.int (System.Any_Priority'Last));
360          pragma Assert (Result = 0);
361
362       elsif Locking_Policy = 'I' then
363          Result := pthread_mutexattr_setprotocol
364            (Attributes'Access, PTHREAD_PRIO_INHERIT);
365          pragma Assert (Result = 0);
366       end if;
367
368       Result := pthread_mutex_init (L, Attributes'Access);
369       pragma Assert (Result = 0 or else Result = ENOMEM);
370
371       if Result = ENOMEM then
372          Result := pthread_mutexattr_destroy (Attributes'Access);
373          raise Storage_Error;
374       end if;
375
376       Result := pthread_mutexattr_destroy (Attributes'Access);
377       pragma Assert (Result = 0);
378    end Initialize_Lock;
379
380    -------------------
381    -- Finalize_Lock --
382    -------------------
383
384    procedure Finalize_Lock (L : access Lock) is
385       Result : Interfaces.C.int;
386
387    begin
388       Result := pthread_mutex_destroy (L);
389       pragma Assert (Result = 0);
390    end Finalize_Lock;
391
392    procedure Finalize_Lock (L : access RTS_Lock) is
393       Result : Interfaces.C.int;
394
395    begin
396       Result := pthread_mutex_destroy (L);
397       pragma Assert (Result = 0);
398    end Finalize_Lock;
399
400    ----------------
401    -- Write_Lock --
402    ----------------
403
404    procedure Write_Lock (L : access Lock; Ceiling_Violation : out Boolean) is
405       Result : Interfaces.C.int;
406
407    begin
408       Result := pthread_mutex_lock (L);
409
410       --  Assume that the cause of EINVAL is a priority ceiling violation
411
412       Ceiling_Violation := (Result = EINVAL);
413       pragma Assert (Result = 0 or else Result = EINVAL);
414    end Write_Lock;
415
416    procedure Write_Lock
417      (L           : access RTS_Lock;
418       Global_Lock : Boolean := False)
419    is
420       Result : Interfaces.C.int;
421
422    begin
423       if not Single_Lock or else Global_Lock then
424          Result := pthread_mutex_lock (L);
425          pragma Assert (Result = 0);
426       end if;
427    end Write_Lock;
428
429    procedure Write_Lock (T : Task_Id) is
430       Result : Interfaces.C.int;
431
432    begin
433       if not Single_Lock then
434          Result := pthread_mutex_lock (T.Common.LL.L'Access);
435          pragma Assert (Result = 0);
436       end if;
437    end Write_Lock;
438
439    ---------------
440    -- Read_Lock --
441    ---------------
442
443    procedure Read_Lock (L : access Lock; Ceiling_Violation : out Boolean) is
444    begin
445       Write_Lock (L, Ceiling_Violation);
446    end Read_Lock;
447
448    ------------
449    -- Unlock --
450    ------------
451
452    procedure Unlock (L : access Lock) is
453       Result : Interfaces.C.int;
454
455    begin
456       Result := pthread_mutex_unlock (L);
457       pragma Assert (Result = 0);
458    end Unlock;
459
460    procedure Unlock (L : access RTS_Lock; Global_Lock : Boolean := False) is
461       Result : Interfaces.C.int;
462
463    begin
464       if not Single_Lock or else Global_Lock then
465          Result := pthread_mutex_unlock (L);
466          pragma Assert (Result = 0);
467       end if;
468    end Unlock;
469
470    procedure Unlock (T : Task_Id) is
471       Result : Interfaces.C.int;
472
473    begin
474       if not Single_Lock then
475          Result := pthread_mutex_unlock (T.Common.LL.L'Access);
476          pragma Assert (Result = 0);
477       end if;
478    end Unlock;
479
480    -----------
481    -- Sleep --
482    -----------
483
484    procedure Sleep
485      (Self_ID : Task_Id;
486       Reason   : System.Tasking.Task_States)
487    is
488       pragma Warnings (Off, Reason);
489
490       Result : Interfaces.C.int;
491
492    begin
493       if Single_Lock then
494          Result := pthread_cond_wait
495            (Self_ID.Common.LL.CV'Access, Single_RTS_Lock'Access);
496       else
497          Result := pthread_cond_wait
498            (Self_ID.Common.LL.CV'Access, Self_ID.Common.LL.L'Access);
499       end if;
500
501       --  EINTR is not considered a failure.
502
503       pragma Assert (Result = 0 or else Result = EINTR);
504    end Sleep;
505
506    -----------------
507    -- Timed_Sleep --
508    -----------------
509
510    --  This is for use within the run-time system, so abort is
511    --  assumed to be already deferred, and the caller should be
512    --  holding its own ATCB lock.
513
514    procedure Timed_Sleep
515      (Self_ID  : Task_Id;
516       Time     : Duration;
517       Mode     : ST.Delay_Modes;
518       Reason   : Task_States;
519       Timedout : out Boolean;
520       Yielded  : out Boolean)
521    is
522       pragma Warnings (Off, Reason);
523
524       Check_Time : constant Duration := Monotonic_Clock;
525       Rel_Time   : Duration;
526       Abs_Time   : Duration;
527       Request    : aliased timespec;
528       Result     : Interfaces.C.int;
529
530    begin
531       Timedout := True;
532       Yielded := False;
533
534       if Mode = Relative then
535          Abs_Time := Duration'Min (Time, Max_Sensible_Delay) + Check_Time;
536
537          if Relative_Timed_Wait then
538             Rel_Time := Duration'Min (Max_Sensible_Delay, Time);
539          end if;
540
541       else
542          Abs_Time := Duration'Min (Check_Time + Max_Sensible_Delay, Time);
543
544          if Relative_Timed_Wait then
545             Rel_Time := Duration'Min (Max_Sensible_Delay, Time - Check_Time);
546          end if;
547       end if;
548
549       if Abs_Time > Check_Time then
550          if Relative_Timed_Wait then
551             Request := To_Timespec (Rel_Time);
552          else
553             Request := To_Timespec (Abs_Time);
554          end if;
555
556          loop
557             exit when Self_ID.Pending_ATC_Level < Self_ID.ATC_Nesting_Level
558               or else Self_ID.Pending_Priority_Change;
559
560             if Single_Lock then
561                Result := pthread_cond_timedwait
562                  (Self_ID.Common.LL.CV'Access, Single_RTS_Lock'Access,
563                   Request'Access);
564
565             else
566                Result := pthread_cond_timedwait
567                  (Self_ID.Common.LL.CV'Access, Self_ID.Common.LL.L'Access,
568                   Request'Access);
569             end if;
570
571             exit when Abs_Time <= Monotonic_Clock;
572
573             if Result = 0 or Result = EINTR then
574
575                --  Somebody may have called Wakeup for us
576
577                Timedout := False;
578                exit;
579             end if;
580
581             pragma Assert (Result = ETIMEDOUT);
582          end loop;
583       end if;
584    end Timed_Sleep;
585
586    -----------------
587    -- Timed_Delay --
588    -----------------
589
590    --  This is for use in implementing delay statements, so
591    --  we assume the caller is abort-deferred but is holding
592    --  no locks.
593
594    procedure Timed_Delay
595      (Self_ID  : Task_Id;
596       Time     : Duration;
597       Mode     : ST.Delay_Modes)
598    is
599       Check_Time : constant Duration := Monotonic_Clock;
600       Abs_Time   : Duration;
601       Rel_Time   : Duration;
602       Request    : aliased timespec;
603       Result     : Interfaces.C.int;
604
605    begin
606       --  Only the little window between deferring abort and
607       --  locking Self_ID is the reason we need to
608       --  check for pending abort and priority change below! :(
609
610       SSL.Abort_Defer.all;
611
612       if Single_Lock then
613          Lock_RTS;
614       end if;
615
616       Write_Lock (Self_ID);
617
618       if Mode = Relative then
619          Abs_Time := Duration'Min (Time, Max_Sensible_Delay) + Check_Time;
620
621          if Relative_Timed_Wait then
622             Rel_Time := Duration'Min (Max_Sensible_Delay, Time);
623          end if;
624
625       else
626          Abs_Time := Duration'Min (Check_Time + Max_Sensible_Delay, Time);
627
628          if Relative_Timed_Wait then
629             Rel_Time := Duration'Min (Max_Sensible_Delay, Time - Check_Time);
630          end if;
631       end if;
632
633       if Abs_Time > Check_Time then
634          if Relative_Timed_Wait then
635             Request := To_Timespec (Rel_Time);
636          else
637             Request := To_Timespec (Abs_Time);
638          end if;
639
640          Self_ID.Common.State := Delay_Sleep;
641
642          loop
643             if Self_ID.Pending_Priority_Change then
644                Self_ID.Pending_Priority_Change := False;
645                Self_ID.Common.Base_Priority := Self_ID.New_Base_Priority;
646                Set_Priority (Self_ID, Self_ID.Common.Base_Priority);
647             end if;
648
649             exit when Self_ID.Pending_ATC_Level < Self_ID.ATC_Nesting_Level;
650
651             if Single_Lock then
652                Result := pthread_cond_timedwait (Self_ID.Common.LL.CV'Access,
653                  Single_RTS_Lock'Access, Request'Access);
654             else
655                Result := pthread_cond_timedwait (Self_ID.Common.LL.CV'Access,
656                  Self_ID.Common.LL.L'Access, Request'Access);
657             end if;
658
659             exit when Abs_Time <= Monotonic_Clock;
660
661             pragma Assert (Result = 0
662                              or else Result = ETIMEDOUT
663                              or else Result = EINTR);
664          end loop;
665
666          Self_ID.Common.State := Runnable;
667       end if;
668
669       Unlock (Self_ID);
670
671       if Single_Lock then
672          Unlock_RTS;
673       end if;
674
675       Result := sched_yield;
676       SSL.Abort_Undefer.all;
677    end Timed_Delay;
678
679    ---------------------
680    -- Monotonic_Clock --
681    ---------------------
682
683    function Monotonic_Clock return Duration is
684       TS     : aliased timespec;
685       Result : Interfaces.C.int;
686    begin
687       Result := clock_gettime
688         (clock_id => CLOCK_REALTIME, tp => TS'Unchecked_Access);
689       pragma Assert (Result = 0);
690       return To_Duration (TS);
691    end Monotonic_Clock;
692
693    -------------------
694    -- RT_Resolution --
695    -------------------
696
697    function RT_Resolution return Duration is
698    begin
699       return 10#1.0#E-6;
700    end RT_Resolution;
701
702    ------------
703    -- Wakeup --
704    ------------
705
706    procedure Wakeup (T : Task_Id; Reason : System.Tasking.Task_States) is
707       pragma Warnings (Off, Reason);
708       Result : Interfaces.C.int;
709    begin
710       Result := pthread_cond_signal (T.Common.LL.CV'Access);
711       pragma Assert (Result = 0);
712    end Wakeup;
713
714    -----------
715    -- Yield --
716    -----------
717
718    procedure Yield (Do_Yield : Boolean := True) is
719       Result : Interfaces.C.int;
720       pragma Unreferenced (Result);
721    begin
722       if Do_Yield then
723          Result := sched_yield;
724       end if;
725    end Yield;
726
727    ------------------
728    -- Set_Priority --
729    ------------------
730
731    procedure Set_Priority
732      (T                   : Task_Id;
733       Prio                : System.Any_Priority;
734       Loss_Of_Inheritance : Boolean := False)
735    is
736       pragma Warnings (Off, Loss_Of_Inheritance);
737
738       Result : Interfaces.C.int;
739       Param  : aliased struct_sched_param;
740
741    begin
742       T.Common.Current_Priority := Prio;
743       Param.sched_priority := Interfaces.C.int (Prio);
744
745       if Time_Slice_Supported and then Time_Slice_Val > 0 then
746          Result := pthread_setschedparam
747            (T.Common.LL.Thread, SCHED_RR, Param'Access);
748
749       elsif FIFO_Within_Priorities or else Time_Slice_Val = 0 then
750          Result := pthread_setschedparam
751            (T.Common.LL.Thread, SCHED_FIFO, Param'Access);
752
753       else
754          Result := pthread_setschedparam
755            (T.Common.LL.Thread, SCHED_OTHER, Param'Access);
756       end if;
757
758       pragma Assert (Result = 0);
759    end Set_Priority;
760
761    ------------------
762    -- Get_Priority --
763    ------------------
764
765    function Get_Priority (T : Task_Id) return System.Any_Priority is
766    begin
767       return T.Common.Current_Priority;
768    end Get_Priority;
769
770    ----------------
771    -- Enter_Task --
772    ----------------
773
774    procedure Enter_Task (Self_ID : Task_Id) is
775    begin
776       Self_ID.Common.LL.Thread := pthread_self;
777       Self_ID.Common.LL.LWP := lwp_self;
778
779       Specific.Set (Self_ID);
780
781       Lock_RTS;
782
783       for J in Known_Tasks'Range loop
784          if Known_Tasks (J) = null then
785             Known_Tasks (J) := Self_ID;
786             Self_ID.Known_Tasks_Index := J;
787             exit;
788          end if;
789       end loop;
790
791       Unlock_RTS;
792    end Enter_Task;
793
794    --------------
795    -- New_ATCB --
796    --------------
797
798    function New_ATCB (Entry_Num : Task_Entry_Index) return Task_Id is
799    begin
800       return new Ada_Task_Control_Block (Entry_Num);
801    end New_ATCB;
802
803    -------------------
804    -- Is_Valid_Task --
805    -------------------
806
807    function Is_Valid_Task return Boolean renames Specific.Is_Valid_Task;
808
809    -----------------------------
810    -- Register_Foreign_Thread --
811    -----------------------------
812
813    function Register_Foreign_Thread return Task_Id is
814    begin
815       if Is_Valid_Task then
816          return Self;
817       else
818          return Register_Foreign_Thread (pthread_self);
819       end if;
820    end Register_Foreign_Thread;
821
822    --------------------
823    -- Initialize_TCB --
824    --------------------
825
826    procedure Initialize_TCB (Self_ID : Task_Id; Succeeded : out Boolean) is
827       Mutex_Attr : aliased pthread_mutexattr_t;
828       Result     : Interfaces.C.int;
829       Cond_Attr  : aliased pthread_condattr_t;
830
831    begin
832       --  Give the task a unique serial number.
833
834       Self_ID.Serial_Number := Next_Serial_Number;
835       Next_Serial_Number := Next_Serial_Number + 1;
836       pragma Assert (Next_Serial_Number /= 0);
837
838       if not Single_Lock then
839          Result := pthread_mutexattr_init (Mutex_Attr'Access);
840          pragma Assert (Result = 0 or else Result = ENOMEM);
841
842          if Result = 0 then
843             if Locking_Policy = 'C' then
844                Result := pthread_mutexattr_setprotocol
845                  (Mutex_Attr'Access, PTHREAD_PRIO_PROTECT);
846                pragma Assert (Result = 0);
847
848                Result := pthread_mutexattr_setprioceiling
849                   (Mutex_Attr'Access,
850                    Interfaces.C.int (System.Any_Priority'Last));
851                pragma Assert (Result = 0);
852
853             elsif Locking_Policy = 'I' then
854                Result := pthread_mutexattr_setprotocol
855                  (Mutex_Attr'Access, PTHREAD_PRIO_INHERIT);
856                pragma Assert (Result = 0);
857             end if;
858
859             Result := pthread_mutex_init (Self_ID.Common.LL.L'Access,
860               Mutex_Attr'Access);
861             pragma Assert (Result = 0 or else Result = ENOMEM);
862          end if;
863
864          if Result /= 0 then
865             Succeeded := False;
866             return;
867          end if;
868
869          Result := pthread_mutexattr_destroy (Mutex_Attr'Access);
870          pragma Assert (Result = 0);
871       end if;
872
873       Result := pthread_condattr_init (Cond_Attr'Access);
874       pragma Assert (Result = 0 or else Result = ENOMEM);
875
876       if Result = 0 then
877          Result := pthread_cond_init (Self_ID.Common.LL.CV'Access,
878            Cond_Attr'Access);
879          pragma Assert (Result = 0 or else Result = ENOMEM);
880       end if;
881
882       if Result = 0 then
883          Succeeded := True;
884       else
885          if not Single_Lock then
886             Result := pthread_mutex_destroy (Self_ID.Common.LL.L'Access);
887             pragma Assert (Result = 0);
888          end if;
889
890          Succeeded := False;
891       end if;
892
893       Result := pthread_condattr_destroy (Cond_Attr'Access);
894       pragma Assert (Result = 0);
895    end Initialize_TCB;
896
897    -----------------
898    -- Create_Task --
899    -----------------
900
901    procedure Create_Task
902      (T          : Task_Id;
903       Wrapper    : System.Address;
904       Stack_Size : System.Parameters.Size_Type;
905       Priority   : System.Any_Priority;
906       Succeeded  : out Boolean)
907    is
908       Attributes          : aliased pthread_attr_t;
909       Adjusted_Stack_Size : Interfaces.C.size_t;
910       Result              : Interfaces.C.int;
911
912       function Thread_Body_Access is new
913         Unchecked_Conversion (System.Address, Thread_Body);
914
915       use System.Task_Info;
916
917    begin
918       if Stack_Size = Unspecified_Size then
919          Adjusted_Stack_Size := Interfaces.C.size_t (Default_Stack_Size);
920
921       elsif Stack_Size < Minimum_Stack_Size then
922          Adjusted_Stack_Size := Interfaces.C.size_t (Minimum_Stack_Size);
923
924       else
925          Adjusted_Stack_Size := Interfaces.C.size_t (Stack_Size);
926       end if;
927
928       if Stack_Base_Available then
929          --  If Stack Checking is supported then allocate 2 additional pages:
930          --
931          --  In the worst case, stack is allocated at something like
932          --  N * Get_Page_Size - epsilon, we need to add the size for 2 pages
933          --  to be sure the effective stack size is greater than what
934          --  has been asked.
935
936          Adjusted_Stack_Size := Adjusted_Stack_Size + 2 * Get_Page_Size;
937       end if;
938
939       Result := pthread_attr_init (Attributes'Access);
940       pragma Assert (Result = 0 or else Result = ENOMEM);
941
942       if Result /= 0 then
943          Succeeded := False;
944          return;
945       end if;
946
947       Result := pthread_attr_setdetachstate
948         (Attributes'Access, PTHREAD_CREATE_DETACHED);
949       pragma Assert (Result = 0);
950
951       Result := pthread_attr_setstacksize
952         (Attributes'Access, Adjusted_Stack_Size);
953       pragma Assert (Result = 0);
954
955       if T.Common.Task_Info /= Default_Scope then
956
957          --  We are assuming that Scope_Type has the same values than the
958          --  corresponding C macros
959
960          Result := pthread_attr_setscope
961            (Attributes'Access, Task_Info_Type'Pos (T.Common.Task_Info));
962          pragma Assert (Result = 0);
963       end if;
964
965       --  Since the initial signal mask of a thread is inherited from the
966       --  creator, and the Environment task has all its signals masked, we
967       --  do not need to manipulate caller's signal mask at this point.
968       --  All tasks in RTS will have All_Tasks_Mask initially.
969
970       Result := pthread_create
971         (T.Common.LL.Thread'Access,
972          Attributes'Access,
973          Thread_Body_Access (Wrapper),
974          To_Address (T));
975       pragma Assert (Result = 0 or else Result = EAGAIN);
976
977       Succeeded := Result = 0;
978
979       Result := pthread_attr_destroy (Attributes'Access);
980       pragma Assert (Result = 0);
981
982       Set_Priority (T, Priority);
983    end Create_Task;
984
985    ------------------
986    -- Finalize_TCB --
987    ------------------
988
989    procedure Finalize_TCB (T : Task_Id) is
990       Result  : Interfaces.C.int;
991       Tmp     : Task_Id := T;
992       Is_Self : constant Boolean := T = Self;
993
994       procedure Free is new
995         Unchecked_Deallocation (Ada_Task_Control_Block, Task_Id);
996
997    begin
998       if not Single_Lock then
999          Result := pthread_mutex_destroy (T.Common.LL.L'Access);
1000          pragma Assert (Result = 0);
1001       end if;
1002
1003       Result := pthread_cond_destroy (T.Common.LL.CV'Access);
1004       pragma Assert (Result = 0);
1005
1006       if T.Known_Tasks_Index /= -1 then
1007          Known_Tasks (T.Known_Tasks_Index) := null;
1008       end if;
1009
1010       Free (Tmp);
1011
1012       if Is_Self then
1013          Specific.Set (null);
1014       end if;
1015    end Finalize_TCB;
1016
1017    ---------------
1018    -- Exit_Task --
1019    ---------------
1020
1021    procedure Exit_Task is
1022    begin
1023       --  Mark this task as unknown, so that if Self is called, it won't
1024       --  return a dangling pointer.
1025
1026       Specific.Set (null);
1027    end Exit_Task;
1028
1029    ----------------
1030    -- Abort_Task --
1031    ----------------
1032
1033    procedure Abort_Task (T : Task_Id) is
1034       Result : Interfaces.C.int;
1035    begin
1036       Result := pthread_kill (T.Common.LL.Thread,
1037         Signal (System.Interrupt_Management.Abort_Task_Interrupt));
1038       pragma Assert (Result = 0);
1039    end Abort_Task;
1040
1041       ----------------
1042    -- Initialize --
1043    ----------------
1044
1045    procedure Initialize (S : in out Suspension_Object) is
1046       Mutex_Attr : aliased pthread_mutexattr_t;
1047       Cond_Attr  : aliased pthread_condattr_t;
1048       Result     : Interfaces.C.int;
1049    begin
1050       --  Initialize internal state. It is always initialized to False (ARM
1051       --  D.10 par. 6).
1052
1053       S.State := False;
1054       S.Waiting := False;
1055
1056       --  Initialize internal mutex
1057
1058       Result := pthread_mutexattr_init (Mutex_Attr'Access);
1059       pragma Assert (Result = 0 or else Result = ENOMEM);
1060
1061       if Result = ENOMEM then
1062          raise Storage_Error;
1063       end if;
1064
1065       Result := pthread_mutex_init (S.L'Access, Mutex_Attr'Access);
1066       pragma Assert (Result = 0 or else Result = ENOMEM);
1067
1068       if Result = ENOMEM then
1069          Result := pthread_mutexattr_destroy (Mutex_Attr'Access);
1070          pragma Assert (Result = 0);
1071
1072          raise Storage_Error;
1073       end if;
1074
1075       Result := pthread_mutexattr_destroy (Mutex_Attr'Access);
1076       pragma Assert (Result = 0);
1077
1078       --  Initialize internal condition variable
1079
1080       Result := pthread_condattr_init (Cond_Attr'Access);
1081       pragma Assert (Result = 0 or else Result = ENOMEM);
1082
1083       if Result /= 0 then
1084          Result := pthread_mutex_destroy (S.L'Access);
1085          pragma Assert (Result = 0);
1086
1087          if Result = ENOMEM then
1088             raise Storage_Error;
1089          end if;
1090       end if;
1091
1092       Result := pthread_cond_init (S.CV'Access, Cond_Attr'Access);
1093       pragma Assert (Result = 0 or else Result = ENOMEM);
1094
1095       if Result /= 0 then
1096          Result := pthread_mutex_destroy (S.L'Access);
1097          pragma Assert (Result = 0);
1098
1099          if Result = ENOMEM then
1100             Result := pthread_condattr_destroy (Cond_Attr'Access);
1101             pragma Assert (Result = 0);
1102
1103             raise Storage_Error;
1104          end if;
1105       end if;
1106
1107       Result := pthread_condattr_destroy (Cond_Attr'Access);
1108       pragma Assert (Result = 0);
1109    end Initialize;
1110
1111    --------------
1112    -- Finalize --
1113    --------------
1114
1115    procedure Finalize (S : in out Suspension_Object) is
1116       Result  : Interfaces.C.int;
1117    begin
1118       --  Destroy internal mutex
1119
1120       Result := pthread_mutex_destroy (S.L'Access);
1121       pragma Assert (Result = 0);
1122
1123       --  Destroy internal condition variable
1124
1125       Result := pthread_cond_destroy (S.CV'Access);
1126       pragma Assert (Result = 0);
1127    end Finalize;
1128
1129    -------------------
1130    -- Current_State --
1131    -------------------
1132
1133    function Current_State (S : Suspension_Object) return Boolean is
1134    begin
1135       --  We do not want to use lock on this read operation. State is marked
1136       --  as Atomic so that we ensure that the value retrieved is correct.
1137
1138       return S.State;
1139    end Current_State;
1140
1141    ---------------
1142    -- Set_False --
1143    ---------------
1144
1145    procedure Set_False (S : in out Suspension_Object) is
1146       Result  : Interfaces.C.int;
1147    begin
1148       Result := pthread_mutex_lock (S.L'Access);
1149       pragma Assert (Result = 0);
1150
1151       S.State := False;
1152
1153       Result := pthread_mutex_unlock (S.L'Access);
1154       pragma Assert (Result = 0);
1155    end Set_False;
1156
1157    --------------
1158    -- Set_True --
1159    --------------
1160
1161    procedure Set_True (S : in out Suspension_Object) is
1162       Result : Interfaces.C.int;
1163    begin
1164       Result := pthread_mutex_lock (S.L'Access);
1165       pragma Assert (Result = 0);
1166
1167       --  If there is already a task waiting on this suspension object then
1168       --  we resume it, leaving the state of the suspension object to False,
1169       --  as it is specified in ARM D.10 par. 9. Otherwise, it just leaves
1170       --  the state to True.
1171
1172       if S.Waiting then
1173          S.Waiting := False;
1174          S.State := False;
1175
1176          Result := pthread_cond_signal (S.CV'Access);
1177          pragma Assert (Result = 0);
1178       else
1179          S.State := True;
1180       end if;
1181
1182       Result := pthread_mutex_unlock (S.L'Access);
1183       pragma Assert (Result = 0);
1184    end Set_True;
1185
1186    ------------------------
1187    -- Suspend_Until_True --
1188    ------------------------
1189
1190    procedure Suspend_Until_True (S : in out Suspension_Object) is
1191       Result : Interfaces.C.int;
1192    begin
1193       Result := pthread_mutex_lock (S.L'Access);
1194       pragma Assert (Result = 0);
1195
1196       if S.Waiting then
1197          --  Program_Error must be raised upon calling Suspend_Until_True
1198          --  if another task is already waiting on that suspension object
1199          --  (ARM D.10 par. 10).
1200
1201          Result := pthread_mutex_unlock (S.L'Access);
1202          pragma Assert (Result = 0);
1203
1204          raise Program_Error;
1205       else
1206          --  Suspend the task if the state is False. Otherwise, the task
1207          --  continues its execution, and the state of the suspension object
1208          --  is set to False (ARM D.10 par. 9).
1209
1210          if S.State then
1211             S.State := False;
1212          else
1213             S.Waiting := True;
1214             Result := pthread_cond_wait (S.CV'Access, S.L'Access);
1215          end if;
1216       end if;
1217
1218       Result := pthread_mutex_unlock (S.L'Access);
1219       pragma Assert (Result = 0);
1220    end Suspend_Until_True;
1221
1222    ----------------
1223    -- Check_Exit --
1224    ----------------
1225
1226    --  Dummy version
1227
1228    function Check_Exit (Self_ID : ST.Task_Id) return Boolean is
1229       pragma Warnings (Off, Self_ID);
1230    begin
1231       return True;
1232    end Check_Exit;
1233
1234    --------------------
1235    -- Check_No_Locks --
1236    --------------------
1237
1238    function Check_No_Locks (Self_ID : ST.Task_Id) return Boolean is
1239       pragma Warnings (Off, Self_ID);
1240    begin
1241       return True;
1242    end Check_No_Locks;
1243
1244    ----------------------
1245    -- Environment_Task --
1246    ----------------------
1247
1248    function Environment_Task return Task_Id is
1249    begin
1250       return Environment_Task_Id;
1251    end Environment_Task;
1252
1253    --------------
1254    -- Lock_RTS --
1255    --------------
1256
1257    procedure Lock_RTS is
1258    begin
1259       Write_Lock (Single_RTS_Lock'Access, Global_Lock => True);
1260    end Lock_RTS;
1261
1262    ----------------
1263    -- Unlock_RTS --
1264    ----------------
1265
1266    procedure Unlock_RTS is
1267    begin
1268       Unlock (Single_RTS_Lock'Access, Global_Lock => True);
1269    end Unlock_RTS;
1270
1271    ------------------
1272    -- Suspend_Task --
1273    ------------------
1274
1275    function Suspend_Task
1276      (T           : ST.Task_Id;
1277       Thread_Self : Thread_Id) return Boolean
1278    is
1279       pragma Warnings (Off, T);
1280       pragma Warnings (Off, Thread_Self);
1281    begin
1282       return False;
1283    end Suspend_Task;
1284
1285    -----------------
1286    -- Resume_Task --
1287    -----------------
1288
1289    function Resume_Task
1290      (T           : ST.Task_Id;
1291       Thread_Self : Thread_Id) return Boolean
1292    is
1293       pragma Warnings (Off, T);
1294       pragma Warnings (Off, Thread_Self);
1295    begin
1296       return False;
1297    end Resume_Task;
1298
1299    ----------------
1300    -- Initialize --
1301    ----------------
1302
1303    procedure Initialize (Environment_Task : Task_Id) is
1304       act     : aliased struct_sigaction;
1305       old_act : aliased struct_sigaction;
1306       Tmp_Set : aliased sigset_t;
1307       Result  : Interfaces.C.int;
1308
1309       function State
1310         (Int : System.Interrupt_Management.Interrupt_ID) return Character;
1311       pragma Import (C, State, "__gnat_get_interrupt_state");
1312       --  Get interrupt state.  Defined in a-init.c
1313       --  The input argument is the interrupt number,
1314       --  and the result is one of the following:
1315
1316       Default : constant Character := 's';
1317       --    'n'   this interrupt not set by any Interrupt_State pragma
1318       --    'u'   Interrupt_State pragma set state to User
1319       --    'r'   Interrupt_State pragma set state to Runtime
1320       --    's'   Interrupt_State pragma set state to System (use "default"
1321       --           system handler)
1322
1323    begin
1324       Environment_Task_Id := Environment_Task;
1325
1326       --  Initialize the lock used to synchronize chain of all ATCBs.
1327
1328       Initialize_Lock (Single_RTS_Lock'Access, RTS_Lock_Level);
1329
1330       Specific.Initialize (Environment_Task);
1331
1332       Enter_Task (Environment_Task);
1333
1334       --  Install the abort-signal handler
1335
1336       if State (System.Interrupt_Management.Abort_Task_Interrupt)
1337         /= Default
1338       then
1339          act.sa_flags := 0;
1340          act.sa_handler := Abort_Handler'Address;
1341
1342          Result := sigemptyset (Tmp_Set'Access);
1343          pragma Assert (Result = 0);
1344          act.sa_mask := Tmp_Set;
1345
1346          Result :=
1347            sigaction
1348            (Signal (System.Interrupt_Management.Abort_Task_Interrupt),
1349             act'Unchecked_Access,
1350             old_act'Unchecked_Access);
1351          pragma Assert (Result = 0);
1352       end if;
1353    end Initialize;
1354
1355 begin
1356    declare
1357       Result : Interfaces.C.int;
1358    begin
1359       --  Prepare the set of signals that should unblocked in all tasks
1360
1361       Result := sigemptyset (Unblocked_Signal_Mask'Access);
1362       pragma Assert (Result = 0);
1363
1364       for J in Interrupt_Management.Interrupt_ID loop
1365          if System.Interrupt_Management.Keep_Unmasked (J) then
1366             Result := sigaddset (Unblocked_Signal_Mask'Access, Signal (J));
1367             pragma Assert (Result = 0);
1368          end if;
1369       end loop;
1370    end;
1371 end System.Task_Primitives.Operations;