Generic.Unix.Objects.Mod 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935
  1. (* Aos, Copyright 2001, Pieter Muller, ETH Zurich *)
  2. MODULE Objects; (** AUTHOR "pjm, G.F."; PURPOSE "Active object runtime support"; *)
  3. IMPORT S := SYSTEM, Trace, Glue, Unix, Machine, Heaps, Modules;
  4. CONST
  5. (*! Process flags, meaningless in Unix ports !!! *)
  6. PleaseHalt* = 10; (* Process requested to Halt itself soon *)
  7. Unbreakable*= 11; (* FINALLY shall not catch HALT exception (PleaseHalt is also set) *)
  8. SelfTermination*=12; (* Indicates the process has requested to terminate ifself (PleaseHalt is also set) *)
  9. Preempted* = 27; (* Has been preempted. *)
  10. Resistant* = 28; (* Can only be destroyed by itself *)
  11. MinPriority* = Unix.ThreadLow;
  12. Low* = Unix.ThreadLow + 1;
  13. Normal* = Unix.ThreadNormal;
  14. High* = Unix.ThreadHigh - 2;
  15. GCPriority* = Unix.ThreadHigh - 1;
  16. Realtime* = Unix.ThreadHigh;
  17. (* Process flag defined by compiler in OPC.CallRecBody *)
  18. Restart* = 0; (* Restart/Destroy process on exception *)
  19. (* Process modes (in UnixAos Running means Running or Ready!) *)
  20. Unknown* = 0; Ready* = 1; Running* = 2; AwaitingLock* = 3;
  21. AwaitingCond* = 4; AwaitingEvent* = 5; Terminated* = 6;
  22. Second* = 1000; (* frequency of ticks increments in Hz *)
  23. DefaultStacksize = 128*1024;
  24. VAR
  25. (* timer *)
  26. timerActivity : TimerActivity;
  27. clock : Clock;
  28. timers : Timer;
  29. timerListMutex : Unix.Mutex_t;
  30. timerStopped: BOOLEAN;
  31. (* processes *)
  32. root- : Process; (*! Anchor of all instantiated threads in system *)
  33. stacksize: LONGINT; (* stack size of active objects, adjustable via boot parameter *)
  34. processList : Unix.Mutex_t;
  35. createProcess : Unix.Mutex_t;
  36. startProcess : Unix.Mutex_t;
  37. lockMutex : Unix.Mutex_t;
  38. childrunning : Unix.Condition_t;
  39. newProcess: Process;
  40. nextPID: LONGINT;
  41. finalizerCaller : FinalizerCaller;
  42. mainthread: Unix.Thread_t;
  43. TYPE
  44. LockT= POINTER TO RECORD
  45. mtx, enter: ADDRESS;
  46. END;
  47. CpuCyclesArray* = ARRAY Machine.MaxCPU OF HUGEINT;
  48. ProtectedObject = POINTER TO RECORD END;
  49. ObjectHeader = Heaps.ProtRecBlock;
  50. ProcessQueue = Heaps.ProcessQueue;
  51. EventHandler* = PROCEDURE {DELEGATE};
  52. Timer* = OBJECT
  53. VAR
  54. next: Timer;
  55. trigger: LONGINT;
  56. handler: EventHandler
  57. END Timer;
  58. TimerActivity = OBJECT
  59. VAR
  60. t, r: Timer; h: EventHandler; restart: BOOLEAN;
  61. PROCEDURE UpdateTicks;
  62. BEGIN {EXCLUSIVE}
  63. Machine.UpdateTicks
  64. END UpdateTicks;
  65. PROCEDURE Restart;
  66. BEGIN {EXCLUSIVE}
  67. restart := TRUE
  68. END Restart;
  69. BEGIN {ACTIVE, SAFE, PRIORITY(High)}
  70. restart := FALSE;
  71. LOOP
  72. t := timers;
  73. IF t # NIL THEN
  74. h := NIL; r := NIL;
  75. BEGIN {EXCLUSIVE}
  76. AWAIT( (Machine.ticks >= t.trigger) OR restart ); restart := FALSE;
  77. IF Machine.ticks >= t.trigger THEN
  78. h := t.handler; r := t
  79. END
  80. END;
  81. IF r # NIL THEN Remove( r ) END;
  82. IF h # NIL THEN (* not canceled *) h END
  83. ELSE
  84. BEGIN{EXCLUSIVE}
  85. AWAIT( restart ); restart := FALSE;
  86. END
  87. END
  88. END
  89. END TimerActivity;
  90. Clock* = OBJECT
  91. BEGIN {ACTIVE}
  92. LOOP
  93. Unix.ThrSleep( 10 );
  94. IF ~timerStopped THEN timerActivity.UpdateTicks END
  95. END;
  96. END Clock;
  97. FinalizedCollection* = OBJECT (* base type for collection, extended in Kernel.Mod *)
  98. PROCEDURE RemoveAll*(obj: ANY); (** abstract *)
  99. BEGIN HALT(301) END RemoveAll;
  100. END FinalizedCollection;
  101. FinalizerNode* = POINTER TO RECORD (Heaps.FinalizerNode)
  102. c*: FinalizedCollection (* base type for collection containing object *)
  103. END;
  104. FinalizerCaller = OBJECT (* separate active object that calls finalizers *)
  105. VAR
  106. n: Heaps.FinalizerNode; start: BOOLEAN;
  107. PROCEDURE Activate;
  108. BEGIN {EXCLUSIVE}
  109. start := TRUE
  110. END Activate;
  111. BEGIN {ACTIVE, SAFE, PRIORITY(High)}
  112. start := FALSE;
  113. LOOP
  114. BEGIN {EXCLUSIVE} AWAIT( start ) END;
  115. start := FALSE;
  116. LOOP
  117. n := Heaps.GetFinalizer();
  118. IF n = NIL THEN EXIT END;
  119. IF n IS FinalizerNode THEN
  120. n(FinalizerNode).c.RemoveAll(n.objStrong) (* remove it if it is not removed yet *)
  121. END;
  122. IF n.finalizer # NIL THEN
  123. n.finalizer(n.objStrong) (* may acquire locks *)
  124. END;
  125. END;
  126. (* Machine.ReleaseGC( ) *)
  127. END
  128. END FinalizerCaller;
  129. Body = PROCEDURE ( self: ProtectedObject );
  130. Condition = PROCEDURE ( slink: ADDRESS ): BOOLEAN;
  131. Process* = OBJECT (Heaps.ProcessLink)
  132. VAR
  133. threadId- : Unix.Thread_t;
  134. nextProcess- : Process; (* next in list of all processes *)
  135. stackBottom - : ADDRESS;
  136. id- : LONGINT;
  137. body : Body;
  138. mode- : LONGINT;
  139. flags- : SET;
  140. priority- : LONGINT; (* only effective if Aos is running SUID root *)
  141. succ : Process; (* in ProcessQueue *)
  142. obj- : ProtectedObject; (* associated active object *)
  143. condition- : Condition; (* awaited process' condition *)
  144. condFP- : ADDRESS; (* awaited process' condition's context *)
  145. continue : Unix.Condition_t; (* gets signaled when condition yields true *)
  146. waitingOn- : ProtectedObject;
  147. procID- : LONGINT; (*! processor ID where running, not used in UnixAos *)
  148. state- : Machine.State; (*! only PC, SP and BP are updated in UnixAos *)
  149. context- : Unix.McontextDesc;
  150. state0 : ARRAY 2048 OF CHAR; (* thread state at body start, used for restart after trap *)
  151. PROCEDURE FindRoots*;
  152. VAR sp, bp, n, a0, a1, adr: ADDRESS; desc: Modules.ProcedureDescPointer; i: LONGINT; p {UNTRACED}: ANY;
  153. BEGIN
  154. IF mode # Terminated THEN
  155. IF SELF = CurrentProcess() THEN
  156. context.r_sp := Machine.CurrentSP( );
  157. context.r_bp := Machine.CurrentBP( );
  158. END;
  159. sp := context.r_sp; bp := context.r_bp;
  160. IF Heaps.GCType = Heaps.HeuristicStackInspectionGC THEN
  161. Heaps.Candidate( context.r_di ); Heaps.Candidate( context.r_si );
  162. Heaps.Candidate( context.r_bx ); Heaps.Candidate( context.r_dx );
  163. Heaps.Candidate( context.r_cx ); Heaps.Candidate( context.r_ax);
  164. IF (stackBottom # 0) & (sp # 0) & (sp <= stackBottom) THEN
  165. Heaps.RegisterCandidates( sp, stackBottom - sp );
  166. END;
  167. ELSIF Heaps.GCType = Heaps.MetaDataForStackGC THEN
  168. IF bp < stackBottom THEN
  169. WHILE (bp # Heaps.NilVal) & (bp > 1024) & (bp < stackBottom) DO (* do not test for bp >= sp: could be wrong temporarily! *)
  170. TRACE( bp );
  171. S.GET( bp, n );
  172. TRACE( n );
  173. IF ODD( n ) THEN (* procedure descriptor at bp *)
  174. IF n > 1024 THEN
  175. desc := S.VAL( Modules.ProcedureDescPointer, n-1 );
  176. IF desc # NIL THEN
  177. a0 := ADDRESSOF( desc.offsets );
  178. a1 := S.VAL( ADDRESS, desc.offsets );
  179. ASSERT( a0+SIZEOF( ADDRESS ) = a1, 54321 );
  180. FOR i := 0 TO LEN( desc.offsets ) - 1 DO
  181. adr := bp + desc.offsets[i]; (* pointer at offset *)
  182. S.GET( adr, p ); (* load pointer *)
  183. IF p # NIL THEN
  184. Heaps.Mark( p );
  185. END
  186. END
  187. END
  188. END;
  189. S.GET( bp + SIZEOF( ADDRESS ), bp );
  190. ELSE (* classical stack frame *)
  191. bp := n;
  192. END;
  193. TRACE( bp );
  194. END;
  195. ASSERT( (bp = stackBottom) OR (bp < 1024) , 12345 );
  196. END;
  197. END
  198. END;
  199. Heaps.Mark( nextProcess )
  200. END FindRoots;
  201. PROCEDURE Cancel;
  202. VAR pt, t: Process; kt: Unix.Thread_t;
  203. BEGIN
  204. IF SELF = CurrentProcess() THEN Exit
  205. ELSE
  206. Machine.Acquire( Machine.X11 ); (* let the thread to be killed first finish its last I/O, if any *)
  207. Unix.MtxLock( processList );
  208. pt := NIL; t := root; kt := 0;
  209. WHILE (t # NIL ) & (t # SELF) DO pt := t; t := t.nextProcess END;
  210. IF t = SELF THEN
  211. kt := threadId;
  212. IF pt = NIL THEN root := t.nextProcess ELSE pt.nextProcess := t.nextProcess END;
  213. END;
  214. Unix.MtxUnlock( processList );
  215. IF kt # 0 THEN Unix.ThrKill( kt ) END;
  216. Machine.Release( Machine.X11 );
  217. END
  218. END Cancel;
  219. PROCEDURE GetPriority( ): LONGINT;
  220. BEGIN
  221. RETURN Unix.ThrGetPriority( threadId )
  222. END GetPriority;
  223. PROCEDURE SetPriority( prio: LONGINT );
  224. VAR pr: LONGINT;
  225. BEGIN
  226. pr := max( Machine.prioLow, min( prio, Machine.prioHigh ) );
  227. Unix.ThrSetPriority( threadId, pr ); (* works only if SUID root *)
  228. priority := GetPriority( )
  229. END SetPriority;
  230. PROCEDURE UpdateState;
  231. BEGIN
  232. state.PC := context.r_pc;
  233. state.BP := context.r_bp;
  234. state.SP := context.r_sp
  235. END UpdateState;
  236. PROCEDURE & Initialize( obj: ProtectedObject; bodyProc: Body; prio: LONGINT; fl: SET; stacksize: LONGINT);
  237. BEGIN
  238. SELF.obj := obj; condition := NIL; continue := Unix.NewCond( );
  239. flags := fl;
  240. priority := prio;
  241. nextProcess := NIL;
  242. IF root # NIL THEN
  243. newProcess := SELF;
  244. ASSERT( bodyProc # NIL );
  245. body := bodyProc;
  246. Unix.MtxLock( startProcess );
  247. threadId := Unix.ThrStart( BodyStarter, stacksize );
  248. Unix.CondWait( childrunning, startProcess );
  249. Unix.MtxUnlock( startProcess );
  250. RegisterFinalizer( SELF, FinalizeProcess );
  251. ELSE
  252. (* first process *)
  253. stackBottom := Glue.stackBottom;
  254. threadId := Unix.ThrThis( );
  255. id := 0; nextPID := 1;
  256. root := SELF;
  257. mode := Running;
  258. END;
  259. END Initialize;
  260. END Process;
  261. GCStatusExt = OBJECT (Heaps.GCStatus)
  262. (* called from Heaps.InvokeGC, i.e. this is a hidden upcall. However, it is necessary to take the Machine.Objects
  263. lock here since writing the set of variables here must not be interrupted, i.e. atomic writing of the set of variables
  264. is absolutely necessary. They system may hang if the lock is not taken. *)
  265. PROCEDURE SetgcOngoing( value: BOOLEAN );
  266. VAR cur: Process;
  267. BEGIN (* serialize writers *)
  268. cur := CurrentProcess();
  269. IF value THEN
  270. Machine.Acquire( Machine.Objects );
  271. Machine.Acquire( Machine.Heaps );
  272. cur.context.r_sp := Machine.CurrentSP();
  273. cur.context.r_bp := Machine.CurrentBP();
  274. cur.context.r_pc := ADDRESSOF( GCLoop );
  275. cur.UpdateState;
  276. SuspendActivities;
  277. Heaps.CollectGarbage( Modules.root );
  278. Machine.Release( Machine.Heaps );
  279. Machine.Release( Machine.Objects );
  280. ResumeActivities;
  281. finalizerCaller.Activate;
  282. END;
  283. END SetgcOngoing;
  284. END GCStatusExt;
  285. PROCEDURE BodyStarter;
  286. VAR p{UNTRACED}: Process; res: LONGINT; prevBP: ADDRESS;
  287. BEGIN
  288. Unix.MtxLock( startProcess );
  289. p := newProcess; newProcess := NIL;
  290. p.id := nextPID; INC( nextPID );
  291. p.stackBottom := Machine.CurrentBP( );
  292. S.GET( p.stackBottom, prevBP );
  293. S.PUT( prevBP, S.VAL( ADDRESS, 0 ) ); (* for terminating Reflection.StackTraceBack *)
  294. Unix.MtxLock( processList );
  295. p.nextProcess := root; root := p;
  296. Unix.MtxUnlock( processList );
  297. Unix.CondSignal( childrunning );
  298. Unix.MtxUnlock( startProcess );
  299. p.SetPriority( p.priority );
  300. IF Restart IN p.flags THEN
  301. res := Unix.sigsetjmp( ADDRESSOF( p.state0[0] ), 1 );
  302. END;
  303. p.mode := Running;
  304. p.body( p.obj );
  305. p.mode := Terminated;
  306. Exit
  307. END BodyStarter;
  308. (*--------------------- create, lock, await, unlock -------------------------*)
  309. (* initialize the ObjectHeader, requires lockMutex temporarily *)
  310. PROCEDURE InitProtHeader( hdr: ObjectHeader);
  311. VAR lock: LockT;
  312. BEGIN
  313. (* we cannot hold the lockMute here because allocation can trigger the GC that requires the lock when activating the finalizers *)
  314. NEW(lock);
  315. Unix.MtxLock(lockMutex);
  316. IF hdr.lock = NIL THEN
  317. hdr.lock := lock;
  318. lock.mtx := Unix.NewMtx( ); lock.enter := Unix.NewCond( ); hdr.lockedBy := NIL;
  319. END;
  320. Unix.MtxUnlock(lockMutex);
  321. END InitProtHeader;
  322. PROCEDURE CreateProcess*( body: Body; priority: LONGINT; flags: SET; obj: ProtectedObject );
  323. VAR p: Process; hdr: ObjectHeader;
  324. BEGIN
  325. Unix.MtxLock( createProcess );
  326. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr );
  327. InitProtHeader( hdr );
  328. IF priority = 0 THEN priority := Normal END;
  329. NEW( p, obj, body, priority, flags, stacksize ) ; (* execute BodyStarter as new (posix or solaris) thread *)
  330. Unix.MtxUnlock( createProcess );
  331. RegisterFinalizer( obj, FinalizeActiveObj )
  332. END CreateProcess;
  333. PROCEDURE Lock*( obj: ProtectedObject; exclusive: BOOLEAN );
  334. VAR hdr: ObjectHeader; p: Process; lock: LockT;
  335. BEGIN
  336. ASSERT( exclusive ); (* shared not implemented yet *)
  337. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr );
  338. p := CurrentProcess();
  339. p.context.r_sp := Machine.CurrentSP( );
  340. p.context.r_bp := Machine.CurrentBP( );
  341. p.context.r_pc := Machine.CurrentPC( );
  342. p.UpdateState;
  343. p.mode := AwaitingLock;
  344. (*! we might want to replace the lock mutex by a lock free construct *)
  345. IF hdr.lock = NIL THEN InitProtHeader( hdr ) END;
  346. lock := S.VAL(LockT, hdr.lock);
  347. p.mode := AwaitingLock;
  348. Unix.MtxLock( lock.mtx );
  349. WHILE hdr.lockedBy # NIL DO
  350. (* wait until threads with complied AWAIT conditions have left the monitor *)
  351. Unix.CondWait( lock.enter, lock.mtx );
  352. END;
  353. p.mode := Running; hdr.lockedBy := p; p.waitingOn := NIL
  354. END Lock;
  355. PROCEDURE Await*( cond: Condition; slink: ADDRESS; obj: ProtectedObject; flags: SET );
  356. VAR hdr: ObjectHeader; p, c: Process; lock: LockT;
  357. BEGIN
  358. IF 1 IN flags THEN (* compiler did not generate IF *)
  359. IF cond( slink ) THEN (* condition already true *) RETURN END
  360. END;
  361. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr ); c := NIL;
  362. lock := S.VAL(LockT, hdr.lock);
  363. IF hdr.awaitingCond.head # NIL THEN c := FindCondition( hdr.awaitingCond ) END;
  364. p := CurrentProcess(); p.succ := NIL; p.condition := cond; p.condFP := slink;
  365. p.waitingOn := obj; p.mode := AwaitingCond;
  366. Put( hdr.awaitingCond, p );
  367. hdr.lockedBy := c;
  368. IF c # NIL THEN Unix.CondSignal( c.continue ) ELSE Unix.CondSignal( lock.enter ) END;
  369. p.context.r_sp := Machine.CurrentSP( );
  370. p.context.r_bp := Machine.CurrentBP( );
  371. p.context.r_pc := Machine.CurrentPC( );
  372. p.UpdateState;
  373. Unix.CondWait( p.continue, lock.mtx );
  374. p.mode := Running; hdr.lockedBy := p; p.waitingOn := NIL
  375. END Await;
  376. PROCEDURE Unlock*( obj: ProtectedObject; dummy: BOOLEAN );
  377. VAR hdr: ObjectHeader; c: Process; lock: LockT;
  378. BEGIN
  379. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr ); c := NIL;
  380. lock := S.VAL(LockT,hdr.lock);
  381. IF hdr.awaitingCond.head # NIL THEN c := FindCondition( hdr.awaitingCond ) END;
  382. hdr.lockedBy := c;
  383. IF c # NIL THEN Unix.CondSignal( c.continue ) ELSE Unix.CondSignal( lock.enter ) END;
  384. Unix.MtxUnlock( lock.mtx );
  385. END Unlock;
  386. PROCEDURE FindCondition( VAR q: ProcessQueue ): Process;
  387. VAR first, cand: Process;
  388. BEGIN
  389. Get( q, first );
  390. IF first.condition( first.condFP ) THEN RETURN first ELSE Put( q, first ) END;
  391. WHILE q.head # first DO
  392. Get( q, cand );
  393. IF cand.condition( cand.condFP ) THEN RETURN cand ELSE Put( q, cand ) END;
  394. END;
  395. RETURN NIL
  396. END FindCondition;
  397. PROCEDURE Get( VAR queue: ProcessQueue; VAR new: Process );
  398. VAR t: Process;
  399. BEGIN
  400. t := queue.head(Process);
  401. IF t # NIL THEN
  402. IF t = queue.tail THEN queue.head := NIL; queue.tail := NIL
  403. ELSE queue.head := t.succ; t.succ := NIL
  404. END
  405. END;
  406. new := t
  407. END Get;
  408. PROCEDURE Put( VAR queue: ProcessQueue; t: Process );
  409. BEGIN
  410. IF queue.head = NIL THEN queue.head := t ELSE queue.tail(Process).succ := t END;
  411. queue.tail := t
  412. END Put;
  413. (*-------------------------------------------------------------------------*)
  414. PROCEDURE Terminate*;
  415. BEGIN
  416. Exit
  417. END Terminate;
  418. PROCEDURE TerminateThis*( p: Process; unbreakable: BOOLEAN );
  419. BEGIN
  420. p.mode := Terminated;
  421. p.Cancel
  422. END TerminateThis;
  423. PROCEDURE SetPriority*( pri: LONGINT ); (* Set the current process' priority. *)
  424. VAR me: Process;
  425. BEGIN
  426. me := CurrentProcess();
  427. me.SetPriority( pri )
  428. END SetPriority;
  429. PROCEDURE Sleep*( ms: LONGINT );
  430. BEGIN
  431. Unix.ThrSleep( ms );
  432. END Sleep;
  433. PROCEDURE Yield*; (* Relinquish control. *)
  434. BEGIN
  435. Unix.ThrYield( );
  436. END Yield;
  437. (* Return current process. (DEPRECATED, use ActiveObject) *)
  438. PROCEDURE CurrentProcess*( ): Process;
  439. VAR me: Unix.Thread_t; p: Process;
  440. BEGIN
  441. me := Unix.ThrThis( );
  442. Unix.MtxLock( processList );
  443. p := root;
  444. WHILE (p # NIL) & (p.threadId # me) DO p := p.nextProcess END;
  445. Unix.MtxUnlock( processList );
  446. RETURN p
  447. END CurrentProcess;
  448. PROCEDURE CurrentProcess0( ): Process;
  449. VAR me: Unix.Thread_t; p: Process;
  450. BEGIN
  451. me := Unix.ThrThis( );
  452. p := root;
  453. WHILE (p # NIL) & (p.threadId # me) DO p := p.nextProcess END;
  454. RETURN p
  455. END CurrentProcess0;
  456. (* Return the active object currently executing. *)
  457. PROCEDURE ActiveObject*( ): ANY;
  458. VAR p: Process;
  459. BEGIN
  460. p := CurrentProcess();
  461. RETURN p.obj
  462. END ActiveObject;
  463. (* Return stack bottom of process. For compatibility WinAos/UnixAos/NativeAos *)
  464. PROCEDURE GetStackBottom*(p: Process): ADDRESS;
  465. BEGIN
  466. RETURN p.stackBottom
  467. END GetStackBottom;
  468. PROCEDURE GetProcessID*( ): LONGINT;
  469. VAR p: Process;
  470. BEGIN
  471. p := CurrentProcess();
  472. RETURN p.id;
  473. END GetProcessID;
  474. PROCEDURE GetCpuCycles*( process : Process; VAR cpuCycles: CpuCyclesArray; all: BOOLEAN );
  475. VAR i: LONGINT;
  476. BEGIN
  477. ASSERT( process # NIL );
  478. FOR i := 0 TO Machine.MaxCPU-1 DO cpuCycles[i] := 0 END;
  479. END GetCpuCycles;
  480. (*-----------------------------------------------------------------------*)
  481. PROCEDURE min( a, b: LONGINT ): LONGINT;
  482. BEGIN
  483. IF a <= b THEN RETURN a ELSE RETURN b END
  484. END min;
  485. PROCEDURE max( a, b: LONGINT ): LONGINT;
  486. BEGIN
  487. IF a >= b THEN RETURN a ELSE RETURN b END
  488. END max;
  489. PROCEDURE RegisterFinalizer( obj: ANY; fin: Heaps.Finalizer );
  490. VAR n: Heaps.FinalizerNode;
  491. BEGIN
  492. NEW( n ); n.finalizer := fin; Heaps.AddFinalizer( obj, n );
  493. END RegisterFinalizer;
  494. PROCEDURE FinalizeActiveObj( obj: ANY );
  495. VAR p: Process;
  496. BEGIN
  497. Unix.MtxLock( processList );
  498. p := root;
  499. WHILE (p # NIL) & (p.obj # obj) DO p := p.nextProcess END;
  500. Unix.MtxUnlock( processList );
  501. IF (p # NIL) & (p.obj = obj) THEN
  502. p.mode := Terminated;
  503. Unix.CondDestroy( p.continue ); p.continue := 0;
  504. FinalizeProtObject( obj );
  505. p.Cancel
  506. END;
  507. END FinalizeActiveObj;
  508. PROCEDURE FinalizeProtObject( obj: ANY );
  509. VAR hdr: ObjectHeader; lock: LockT;
  510. BEGIN
  511. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr );
  512. IF hdr.lock # NIL THEN
  513. lock := S.VAL(LockT, hdr.lock);
  514. Unix.MtxDestroy( lock.mtx ); lock.mtx := 0
  515. END
  516. END FinalizeProtObject;
  517. PROCEDURE FinalizeProcess( obj: ANY );
  518. VAR p: Process;
  519. BEGIN
  520. p := obj(Process);
  521. IF p.continue # 0 THEN
  522. Unix.CondDestroy( p.continue ); p.continue := 0
  523. END
  524. END FinalizeProcess;
  525. (* Terminate calling thread. *)
  526. PROCEDURE Exit;
  527. VAR prev, p, me: Process;
  528. BEGIN
  529. me := CurrentProcess();
  530. me.mode := Terminated;
  531. Unix.MtxLock( processList );
  532. prev := NIL; p := root;
  533. WHILE (p # NIL ) & (p # me) DO prev := p; p := p.nextProcess END;
  534. IF p = me THEN
  535. IF prev = NIL THEN root := p.nextProcess ELSE prev.nextProcess := p.nextProcess END;
  536. END;
  537. Unix.MtxUnlock( processList );
  538. Unix.ThrExit( );
  539. END Exit;
  540. PROCEDURE ExitTrap*;
  541. VAR p: Process;
  542. BEGIN
  543. p := CurrentProcess();
  544. (* restart the object body if it was given the SAFE flag *)
  545. IF Restart IN p.flags THEN
  546. Unix.siglongjmp( ADDRESSOF( p.state0[0] ), 1 )
  547. END;
  548. Exit
  549. END ExitTrap;
  550. (*---------------------------- Timer --------------------------------*)
  551. PROCEDURE Remove( t: Timer ); (* remove timer from list of active timers *)
  552. VAR p, x: Timer;
  553. BEGIN
  554. Unix.MtxLock( timerListMutex );
  555. t.trigger := 0; t.handler := NIL;
  556. IF timers # NIL THEN
  557. IF t = timers THEN
  558. timers := t.next
  559. ELSE
  560. p := timers; x := p.next;
  561. WHILE (x # NIL) & (x # t) DO p := x; x := p.next END;
  562. IF x = t THEN p.next := t.next END
  563. END;
  564. t.next := NIL
  565. END;
  566. Unix.MtxUnlock( timerListMutex )
  567. END Remove;
  568. PROCEDURE Insert( t: Timer );
  569. VAR p, x: Timer;
  570. BEGIN
  571. Unix.MtxLock( timerListMutex );
  572. p := NIL; x := timers;
  573. WHILE (x # NIL) & (x.trigger < t.trigger) DO p := x; x := p.next END;
  574. t.next := x;
  575. IF p = NIL THEN timers := t ELSE p.next := t END;
  576. Unix.MtxUnlock( timerListMutex )
  577. END Insert;
  578. PROCEDURE SetTimeout*( t: Timer; h: EventHandler; ms: LONGINT );
  579. BEGIN
  580. ASSERT( ( t # NIL) & ( h # NIL) );
  581. Remove( t );
  582. IF ms < 1 THEN ms := 1 END;
  583. t.trigger := Machine.ticks + ms; t.handler := h;
  584. Insert( t );
  585. timerActivity.Restart
  586. END SetTimeout;
  587. PROCEDURE SetTimeoutAt*( t: Timer; h: EventHandler; ms: LONGINT );
  588. BEGIN
  589. ASSERT( (t # NIL) & (h # NIL) );
  590. Remove( t );
  591. t.trigger := ms; t.handler := h;
  592. Insert( t );
  593. timerActivity.Restart
  594. END SetTimeoutAt;
  595. PROCEDURE CancelTimeout*( t: Timer );
  596. BEGIN
  597. Remove( t )
  598. END CancelTimeout;
  599. (*-------------------- Garbage Collection ------------------------------------*)
  600. PROCEDURE GetContext( ctxt: Unix.Ucontext );
  601. VAR t: Process;
  602. BEGIN
  603. (* use CurrentProcess0 here instead of CurrentProcess in order to
  604. avoid a possible deadlock *)
  605. t := CurrentProcess0();
  606. Unix.CopyContext( ctxt.mc, t.context );
  607. t.UpdateState
  608. END GetContext;
  609. (* called by WMProcessInfo to obtain the current state of a running process *)
  610. PROCEDURE UpdateProcessState*( p: Process );
  611. BEGIN
  612. (* update p.stat.{PC,BP,SP} *)
  613. IF p.threadId # Unix.ThrThis() THEN
  614. timerStopped := TRUE;
  615. Unix.ThrSleep( 2 );
  616. IF p.mode = Running THEN (* still running *)
  617. Unix.ThrSuspend( p.threadId, TRUE );
  618. Unix.ThrResume( p.threadId );
  619. END;
  620. timerStopped := FALSE
  621. END
  622. END UpdateProcessState;
  623. PROCEDURE SuspendActivities;
  624. VAR t, me: Process;
  625. BEGIN
  626. me := CurrentProcess();
  627. t := root;
  628. WHILE t # NIL DO
  629. IF t # me THEN
  630. Unix.ThrSuspend( t.threadId, t.mode = Running );
  631. END;
  632. t := t.nextProcess
  633. END;
  634. END SuspendActivities;
  635. PROCEDURE ResumeActivities;
  636. VAR t, me: Process;
  637. BEGIN
  638. me := CurrentProcess();
  639. t := root;
  640. WHILE t # NIL DO
  641. IF (t # me) THEN
  642. Unix.ThrResume( t.threadId );
  643. END;
  644. t := t.nextProcess
  645. END;
  646. END ResumeActivities;
  647. PROCEDURE GCLoop*; (* Timer and GC activity *)
  648. BEGIN
  649. (*
  650. If using BootConsole keep main thread running.
  651. If the thread is blocked by a mutex (MainThreadSleep)
  652. the signals SIGHUP, SIGINT and SIGQUIT don't branch into SignalHandler
  653. but terminate A2 (at least in Solaris).
  654. *)
  655. LOOP Sleep( 100 ) END
  656. END GCLoop;
  657. PROCEDURE CurrentProcessTime*(): HUGEINT;
  658. BEGIN
  659. RETURN Machine.GetTimer()
  660. END CurrentProcessTime;
  661. PROCEDURE TimerFrequency*(): HUGEINT;
  662. BEGIN
  663. RETURN Machine.mhz * 1000000
  664. END TimerFrequency;
  665. PROCEDURE LeaveA2;
  666. VAR cur: Process; bp, n: ADDRESS;
  667. BEGIN
  668. IF clock = NIL THEN RETURN END;
  669. cur := CurrentProcess();
  670. IF cur # NIL THEN
  671. bp := Machine.CurrentBP();
  672. S.GET( bp, n );
  673. IF ODD( n ) THEN S.GET( bp + SIZEOF( ADDRESS ), bp ) ELSE bp := n END;
  674. cur.context.r_bp := bp;
  675. END;
  676. END LeaveA2;
  677. PROCEDURE ReenterA2;
  678. VAR cur: Process;
  679. BEGIN
  680. IF clock = NIL THEN RETURN END;
  681. cur := CurrentProcess();
  682. IF cur # NIL THEN
  683. cur.context.r_bp := 0;
  684. END;
  685. END ReenterA2;
  686. (*----------------------------- initialization ----------------------------------*)
  687. PROCEDURE StartTimerActivity;
  688. BEGIN
  689. timerListMutex := Unix.NewMtx( ); timers := NIL;
  690. NEW( timerActivity );
  691. END StartTimerActivity;
  692. PROCEDURE GetStacksize;
  693. VAR str: ARRAY 32 OF CHAR; i: LONGINT;
  694. BEGIN
  695. Machine.GetConfig( "StackSize", str );
  696. IF str = "" THEN stacksize := DefaultStacksize
  697. ELSE
  698. i := 0; stacksize := Machine.StrToInt( i, str );
  699. stacksize := stacksize * 1024;
  700. END;
  701. IF Glue.debug # {} THEN
  702. Trace.String( "Stacksize of active objects = " );
  703. Trace.Int( stacksize DIV 1024, 0 ); Trace.StringLn( "K" )
  704. END;
  705. END GetStacksize;
  706. PROCEDURE Convert;
  707. VAR p: Process;
  708. BEGIN
  709. (* make current thread the first active object *)
  710. mainthread := Unix.ThrThis();
  711. NEW( p, NIL, NIL, 0, {}, 0 );
  712. END Convert;
  713. PROCEDURE Init;
  714. BEGIN
  715. Unix.suspendHandler := GetContext;
  716. createProcess := Unix.NewMtx( ); processList := Unix.NewMtx( );
  717. startProcess := Unix.NewMtx( ); childrunning := Unix.NewCond( );
  718. lockMutex := Unix.NewMtx( );
  719. GetStacksize;
  720. Convert;
  721. NEW( clock ); StartTimerActivity; timerStopped := FALSE;
  722. NEW( finalizerCaller );
  723. Heaps.gcStatus := GCStatusFactory()
  724. END Init;
  725. TYPE
  726. MainThread = OBJECT
  727. VAR exit: BOOLEAN;
  728. PROCEDURE & Init;
  729. BEGIN
  730. exit := FALSE;
  731. END Init;
  732. PROCEDURE Await();
  733. BEGIN {EXCLUSIVE}
  734. AWAIT( exit );
  735. END Await;
  736. END MainThread;
  737. VAR main: MainThread;
  738. PROCEDURE MainThreadSleep;
  739. BEGIN
  740. NEW( main );
  741. main.Await( );
  742. Unix.exit( 0 );
  743. END MainThreadSleep;
  744. PROCEDURE {FINAL} Final;
  745. BEGIN
  746. MainThreadSleep;
  747. END Final;
  748. PROCEDURE GCStatusFactory(): Heaps.GCStatus;
  749. VAR gcStatusExt : GCStatusExt;
  750. BEGIN
  751. ASSERT( Heaps.gcStatus = NIL );
  752. NEW( gcStatusExt );
  753. RETURN gcStatusExt
  754. END GCStatusFactory;
  755. VAR
  756. (* for compatibility and later extension *)
  757. TraceProcessHook*: PROCEDURE (prcoess: Process; pc, bp: ADDRESS; stacklow, stackhigh: ADDRESS);
  758. BEGIN
  759. TraceProcessHook := NIL;
  760. Init;
  761. END Objects.