1 ------------------------------------------------------------------------------
3 -- GNAT RUN-TIME LIBRARY (GNARL) COMPONENTS --
5 -- S Y S T E M . T A S K I N G . P R O T E C T E D _ O B J E C T S . --
10 -- Copyright (C) 1998-2006, Free Software Foundation, Inc. --
12 -- GNARL is free software; you can redistribute it and/or modify it under --
13 -- terms of the GNU General Public License as published by the Free Soft- --
14 -- ware Foundation; either version 2, or (at your option) any later ver- --
15 -- sion. GNARL is distributed in the hope that it will be useful, but WITH- --
16 -- OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY --
17 -- or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License --
18 -- for more details. You should have received a copy of the GNU General --
19 -- Public License distributed with GNARL; see file COPYING. If not, write --
20 -- to the Free Software Foundation, 51 Franklin Street, Fifth Floor, --
21 -- Boston, MA 02110-1301, USA. --
23 -- As a special exception, if other files instantiate generics from this --
24 -- unit, or you link this unit with other files to produce an executable, --
25 -- this unit does not by itself cause the resulting executable to be --
26 -- covered by the GNU General Public License. This exception does not --
27 -- however invalidate any other reasons why the executable file might be --
28 -- covered by the GNU Public License. --
30 -- GNARL was developed by the GNARL team at Florida State University. --
31 -- Extensive contributions were provided by Ada Core Technologies, Inc. --
33 ------------------------------------------------------------------------------
35 -- This package contains all the simple primitives related to protected
36 -- objects with entries (i.e init, lock, unlock).
38 -- The handling of protected objects with no entries is done in
39 -- System.Tasking.Protected_Objects, the complex routines for protected
40 -- objects with entries in System.Tasking.Protected_Objects.Operations.
42 -- The split between Entries and Operations is needed to break circular
43 -- dependencies inside the run time.
45 -- Note: the compiler generates direct calls to this interface, via Rtsfind
48 -- Used for Exception_Occurrence_Access
51 with System
.Task_Primitives
.Operations
;
52 -- Used for Initialize_Lock
58 with System
.Tasking
.Initialization
;
59 -- Used for Defer_Abort,
61 -- Change_Base_Priority
63 pragma Elaborate_All
(System
.Tasking
.Initialization
);
64 -- This insures that tasking is initialized if any protected objects are
67 with System
.Parameters
;
68 -- Used for Single_Lock
70 package body System
.Tasking
.Protected_Objects
.Entries
is
72 package STPO
renames System
.Task_Primitives
.Operations
;
75 use Task_Primitives
.Operations
;
82 Locking_Policy
: Character;
83 pragma Import
(C
, Locking_Policy
, "__gl_locking_policy");
89 procedure Finalize
(Object
: in out Protection_Entries
) is
90 Entry_Call
: Entry_Call_Link
;
92 Ceiling_Violation
: Boolean;
93 Self_ID
: constant Task_Id
:= STPO
.Self
;
94 Old_Base_Priority
: System
.Any_Priority
;
97 if Object
.Finalized
then
101 STPO
.Write_Lock
(Object
.L
'Unrestricted_Access, Ceiling_Violation
);
107 if Ceiling_Violation
then
109 -- Dip our own priority down to ceiling of lock. See similar code in
110 -- Tasking.Entry_Calls.Lock_Server.
112 STPO
.Write_Lock
(Self_ID
);
113 Old_Base_Priority
:= Self_ID
.Common
.Base_Priority
;
114 Self_ID
.New_Base_Priority
:= Object
.Ceiling
;
115 Initialization
.Change_Base_Priority
(Self_ID
);
116 STPO
.Unlock
(Self_ID
);
122 STPO
.Write_Lock
(Object
.L
'Unrestricted_Access, Ceiling_Violation
);
124 if Ceiling_Violation
then
125 Raise_Exception
(Program_Error
'Identity, "Ceiling Violation");
132 Object
.Old_Base_Priority
:= Old_Base_Priority
;
133 Object
.Pending_Action
:= True;
136 -- Send program_error to all tasks still queued on this object
138 for E
in Object
.Entry_Queues
'Range loop
139 Entry_Call
:= Object
.Entry_Queues
(E
).Head
;
141 while Entry_Call
/= null loop
142 Caller
:= Entry_Call
.Self
;
143 Entry_Call
.Exception_To_Raise
:= Program_Error
'Identity;
145 STPO
.Write_Lock
(Caller
);
146 Initialization
.Wakeup_Entry_Caller
(Self_ID
, Entry_Call
, Done
);
147 STPO
.Unlock
(Caller
);
149 exit when Entry_Call
= Object
.Entry_Queues
(E
).Tail
;
150 Entry_Call
:= Entry_Call
.Next
;
154 Object
.Finalized
:= True;
160 STPO
.Unlock
(Object
.L
'Unrestricted_Access);
162 STPO
.Finalize_Lock
(Object
.L
'Unrestricted_Access);
170 (Object
: Protection_Entries_Access
) return System
.Any_Priority
is
172 return Object
.New_Ceiling
;
175 -------------------------------------
176 -- Has_Interrupt_Or_Attach_Handler --
177 -------------------------------------
179 function Has_Interrupt_Or_Attach_Handler
180 (Object
: Protection_Entries_Access
)
183 pragma Warnings
(Off
, Object
);
186 end Has_Interrupt_Or_Attach_Handler
;
188 -----------------------------------
189 -- Initialize_Protection_Entries --
190 -----------------------------------
192 procedure Initialize_Protection_Entries
193 (Object
: Protection_Entries_Access
;
194 Ceiling_Priority
: Integer;
195 Compiler_Info
: System
.Address
;
196 Entry_Bodies
: Protected_Entry_Body_Access
;
197 Find_Body_Index
: Find_Body_Index_Access
)
199 Init_Priority
: Integer := Ceiling_Priority
;
200 Self_ID
: constant Task_Id
:= STPO
.Self
;
203 if Init_Priority
= Unspecified_Priority
then
204 Init_Priority
:= System
.Priority
'Last;
207 if Locking_Policy
= 'C'
208 and then Has_Interrupt_Or_Attach_Handler
(Object
)
209 and then Init_Priority
not in System
.Interrupt_Priority
211 -- Required by C.3.1(11)
216 Initialization
.Defer_Abort
(Self_ID
);
217 Initialize_Lock
(Init_Priority
, Object
.L
'Access);
218 Initialization
.Undefer_Abort
(Self_ID
);
219 Object
.Ceiling
:= System
.Any_Priority
(Init_Priority
);
220 Object
.Owner
:= Null_Task
;
221 Object
.Compiler_Info
:= Compiler_Info
;
222 Object
.Pending_Action
:= False;
223 Object
.Call_In_Progress
:= null;
224 Object
.Entry_Bodies
:= Entry_Bodies
;
225 Object
.Find_Body_Index
:= Find_Body_Index
;
227 for E
in Object
.Entry_Queues
'Range loop
228 Object
.Entry_Queues
(E
).Head
:= null;
229 Object
.Entry_Queues
(E
).Tail
:= null;
231 end Initialize_Protection_Entries
;
237 procedure Lock_Entries
238 (Object
: Protection_Entries_Access
; Ceiling_Violation
: out Boolean)
241 if Object
.Finalized
then
243 (Program_Error
'Identity, "Protected Object is finalized");
246 -- If pragma Detect_Blocking is active then, as described in the ARM
247 -- 9.5.1, par. 15, we must check whether this is an external call on a
248 -- protected subprogram with the same target object as that of the
249 -- protected action that is currently in progress (i.e., if the caller
250 -- is already the protected object's owner). If this is the case hence
251 -- Program_Error must be raised.
253 if Detect_Blocking
and then Object
.Owner
= Self
then
257 -- The lock is made without defering abort
259 -- Therefore the abort has to be deferred before calling this routine.
260 -- This means that the compiler has to generate a Defer_Abort call
261 -- before the call to Lock.
263 -- The caller is responsible for undeferring abort, and compiler
264 -- generated calls must be protected with cleanup handlers to ensure
265 -- that abort is undeferred in all cases.
267 pragma Assert
(STPO
.Self
.Deferral_Level
> 0);
268 Write_Lock
(Object
.L
'Access, Ceiling_Violation
);
270 -- We are entering in a protected action, so that we increase the
271 -- protected object nesting level (if pragma Detect_Blocking is
272 -- active), and update the protected object's owner.
274 if Detect_Blocking
then
276 Self_Id
: constant Task_Id
:= Self
;
279 -- Update the protected object's owner
281 Object
.Owner
:= Self_Id
;
283 -- Increase protected object nesting level
285 Self_Id
.Common
.Protected_Action_Nesting
:=
286 Self_Id
.Common
.Protected_Action_Nesting
+ 1;
292 procedure Lock_Entries
(Object
: Protection_Entries_Access
) is
293 Ceiling_Violation
: Boolean;
296 Lock_Entries
(Object
, Ceiling_Violation
);
298 if Ceiling_Violation
then
299 Raise_Exception
(Program_Error
'Identity, "Ceiling Violation");
303 ----------------------------
304 -- Lock_Read_Only_Entries --
305 ----------------------------
307 procedure Lock_Read_Only_Entries
(Object
: Protection_Entries_Access
) is
308 Ceiling_Violation
: Boolean;
311 if Object
.Finalized
then
313 (Program_Error
'Identity, "Protected Object is finalized");
316 -- If pragma Detect_Blocking is active then, as described in the ARM
317 -- 9.5.1, par. 15, we must check whether this is an external call on a
318 -- protected subprogram with the same target object as that of the
319 -- protected action that is currently in progress (i.e., if the caller
320 -- is already the protected object's owner). If this is the case hence
321 -- Program_Error must be raised.
323 -- Note that in this case (getting read access), several tasks may
324 -- have read ownership of the protected object, so that this method of
325 -- storing the (single) protected object's owner does not work
326 -- reliably for read locks. However, this is the approach taken for two
327 -- major reasosn: first, this function is not currently being used (it
328 -- is provided for possible future use), and second, it largely
329 -- simplifies the implementation.
331 if Detect_Blocking
and then Object
.Owner
= Self
then
335 Read_Lock
(Object
.L
'Access, Ceiling_Violation
);
337 if Ceiling_Violation
then
338 Raise_Exception
(Program_Error
'Identity, "Ceiling Violation");
341 -- We are entering in a protected action, so that we increase the
342 -- protected object nesting level (if pragma Detect_Blocking is
343 -- active), and update the protected object's owner.
345 if Detect_Blocking
then
347 Self_Id
: constant Task_Id
:= Self
;
350 -- Update the protected object's owner
352 Object
.Owner
:= Self_Id
;
354 -- Increase protected object nesting level
356 Self_Id
.Common
.Protected_Action_Nesting
:=
357 Self_Id
.Common
.Protected_Action_Nesting
+ 1;
360 end Lock_Read_Only_Entries
;
366 procedure Set_Ceiling
367 (Object
: Protection_Entries_Access
;
368 Prio
: System
.Any_Priority
) is
370 Object
.New_Ceiling
:= Prio
;
377 procedure Unlock_Entries
(Object
: Protection_Entries_Access
) is
379 -- We are exiting from a protected action, so that we decrease the
380 -- protected object nesting level (if pragma Detect_Blocking is
381 -- active), and remove ownership of the protected object.
383 if Detect_Blocking
then
385 Self_Id
: constant Task_Id
:= Self
;
388 -- Calls to this procedure can only take place when being within
389 -- a protected action and when the caller is the protected
392 pragma Assert
(Self_Id
.Common
.Protected_Action_Nesting
> 0
393 and then Object
.Owner
= Self_Id
);
395 -- Remove ownership of the protected object
397 Object
.Owner
:= Null_Task
;
399 Self_Id
.Common
.Protected_Action_Nesting
:=
400 Self_Id
.Common
.Protected_Action_Nesting
- 1;
404 Unlock
(Object
.L
'Access);
407 end System
.Tasking
.Protected_Objects
.Entries
;