2
0

Unix.Objects.Mod 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806
  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. AddrSize = SIZEOF( ADDRESS )
  25. VAR
  26. (* timer *)
  27. timerActivity : TimerActivity;
  28. timers : Timer;
  29. timerListMutex : Unix.Mutex_t;
  30. (* processes *)
  31. mainProcess : Process; (* runs the GC *)
  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. childrunning : Unix.Condition_t;
  38. newProcess: Process;
  39. nextPID: LONGINT;
  40. (* garbage colletion *)
  41. gcFinished: Unix.Condition_t; igc: Unix.Mutex_t;
  42. collect: BOOLEAN;
  43. finalizerCaller : FinalizerCaller;
  44. finCaller : Process;
  45. (* the dummy parameters assure proper stack alignment when compiled with
  46. option "\A" or "--darwinHost" *)
  47. mtxInit: PROCEDURE {REALTIME, C} ( dummy: LONGINT ): Unix.Mutex_t;
  48. mtxDestroy: PROCEDURE {REALTIME, C} ( mtx: Unix.Mutex_t );
  49. mtxLock: PROCEDURE {REALTIME, C} ( mtx: Unix.Mutex_t );
  50. mtxUnlock: PROCEDURE {REALTIME, C} ( mtx: Unix.Mutex_t );
  51. conInit: PROCEDURE {REALTIME, C} ( dummy: LONGINT ): Unix.Condition_t;
  52. conDestroy: PROCEDURE {REALTIME, C} ( cond: Unix.Condition_t );
  53. conWait: PROCEDURE {REALTIME, C} ( cond: Unix.Condition_t; mtx: Unix.Mutex_t );
  54. conSignal: PROCEDURE {REALTIME, C} ( cond: Unix.Condition_t );
  55. thrStart: PROCEDURE {REALTIME, C} ( p: PROCEDURE; stackLen: LONGINT ): Unix.Thread_t;
  56. thrThis: PROCEDURE {REALTIME, C} ( dummy: LONGINT ): Unix.Thread_t;
  57. thrSleep: PROCEDURE {REALTIME, C} ( ms: LONGINT );
  58. thrYield: PROCEDURE {REALTIME, C} ( dummy: LONGINT );
  59. thrExit: PROCEDURE {REALTIME, C} ( dummy: LONGINT );
  60. thrSuspend: PROCEDURE {REALTIME, C} ( t: Unix.Thread_t );
  61. thrResume: PROCEDURE {REALTIME, C} ( t: Unix.Thread_t );
  62. thrSetPriority: PROCEDURE {REALTIME, C} ( t: Unix.Thread_t; prio: LONGINT );
  63. thrGetPriority: PROCEDURE {REALTIME, C} ( t: Unix.Thread_t ): LONGINT;
  64. thrKill: PROCEDURE {REALTIME, C} ( t: Unix.Thread_t );
  65. TYPE
  66. CpuCyclesArray* = ARRAY Machine.MaxCPU OF HUGEINT;
  67. ProtectedObject = POINTER TO RECORD END;
  68. ObjectHeader = Heaps.ProtRecBlock;
  69. ProcessQueue = Heaps.ProcessQueue;
  70. EventHandler* = PROCEDURE {DELEGATE};
  71. Timer* = OBJECT
  72. VAR
  73. next: Timer;
  74. trigger: LONGINT;
  75. handler: EventHandler
  76. END Timer;
  77. TimerActivity = OBJECT
  78. VAR
  79. t, r: Timer; h: EventHandler; restart: BOOLEAN;
  80. PROCEDURE UpdateTicks;
  81. BEGIN {EXCLUSIVE}
  82. Machine.UpdateTicks
  83. END UpdateTicks;
  84. PROCEDURE Restart;
  85. BEGIN {EXCLUSIVE}
  86. restart := TRUE
  87. END Restart;
  88. BEGIN {ACTIVE, SAFE, PRIORITY(High)}
  89. restart := FALSE;
  90. LOOP
  91. t := timers;
  92. IF t # NIL THEN
  93. h := NIL; r := NIL;
  94. BEGIN {EXCLUSIVE}
  95. AWAIT( (Machine.ticks >= t.trigger) OR restart ); restart := FALSE;
  96. IF Machine.ticks >= t.trigger THEN
  97. h := t.handler; r := t
  98. END
  99. END;
  100. IF r # NIL THEN Remove( r ) END;
  101. IF h # NIL THEN (* not canceled *) h END
  102. ELSE
  103. BEGIN{EXCLUSIVE}
  104. AWAIT( restart ); restart := FALSE;
  105. END
  106. END
  107. END
  108. END TimerActivity;
  109. FinalizedCollection* = OBJECT (* base type for collection, extended in Kernel.Mod *)
  110. PROCEDURE RemoveAll*(obj: ANY); (** abstract *)
  111. BEGIN HALT(301) END RemoveAll;
  112. END FinalizedCollection;
  113. FinalizerNode* = POINTER TO RECORD (Heaps.FinalizerNode)
  114. c*: FinalizedCollection (* base type for collection containing object *)
  115. END;
  116. FinalizerCaller = OBJECT (* separate active object that calls finalizers *)
  117. VAR
  118. n: Heaps.FinalizerNode; start: BOOLEAN;
  119. PROCEDURE Start;
  120. BEGIN {EXCLUSIVE}
  121. start := TRUE
  122. END Start;
  123. BEGIN {ACTIVE, SAFE, PRIORITY(High)}
  124. finCaller := CurrentProcess( ); start := FALSE;
  125. LOOP
  126. BEGIN {EXCLUSIVE} AWAIT( start ) END;
  127. start := FALSE;
  128. LOOP
  129. n := Heaps.GetFinalizer();
  130. IF n = NIL THEN EXIT END;
  131. IF n IS FinalizerNode THEN
  132. n(FinalizerNode).c.RemoveAll(n.objStrong) (* remove it if it is not removed yet *)
  133. END;
  134. IF n.finalizer # NIL THEN
  135. n.finalizer(n.objStrong) (* may acquire locks *)
  136. END;
  137. END;
  138. Machine.ReleaseGC
  139. END
  140. END FinalizerCaller;
  141. Body = PROCEDURE ( self: ProtectedObject );
  142. Condition = PROCEDURE ( slink: ADDRESS ): BOOLEAN;
  143. Process* = OBJECT (Heaps.RootObject)
  144. VAR
  145. threadId : Unix.Thread_t;
  146. nextProcess- : Process; (* next in list of all processes *)
  147. stackBottom - : ADDRESS;
  148. SP- : ADDRESS; (* SP value at last NEW *)
  149. id- : LONGINT;
  150. body : Body;
  151. mode- : LONGINT;
  152. flags- : SET;
  153. priority- : LONGINT; (* only effective if Aos is running SUID root *)
  154. succ : Process; (* in ProcessQueue *)
  155. obj- : ProtectedObject; (* associated active object *)
  156. condition- : Condition; (* awaited process' condition *)
  157. condFP- : ADDRESS; (* awaited process' condition's context *)
  158. continue : Unix.Condition_t; (* gets signaled when condition yields true *)
  159. waitingOn- : ProtectedObject;
  160. procID- : LONGINT; (* processor ID where running, not used in UnixAos *)
  161. state- : Machine.State; (*! not used in UnixAos! *)
  162. state0 : ARRAY 2048 OF CHAR; (* thread state at body start, used for restart after trap *)
  163. PROCEDURE FindRoots*;
  164. VAR sp, ptr: ADDRESS;
  165. BEGIN
  166. IF mode # Terminated THEN
  167. sp := SP;
  168. WHILE sp < stackBottom DO
  169. S.GET( sp, ptr );
  170. IF (ptr # 0) & (ptr MOD 8 = 0) THEN Heaps.AddCandidate( ptr ) END;
  171. INC( sp, AddrSize )
  172. END;
  173. END;
  174. Heaps.Mark( nextProcess )
  175. END FindRoots;
  176. PROCEDURE Cancel;
  177. VAR pt, t: Process; kt: Unix.Thread_t;
  178. BEGIN
  179. IF SELF = CurrentProcess() THEN Exit
  180. ELSE
  181. Machine.Acquire( Machine.X11 ); (* let the thread to be killed first finish its last I/O, if any *)
  182. mtxLock( processList );
  183. pt := NIL; t := root; kt := 0;
  184. WHILE (t # NIL ) & (t # SELF) DO pt := t; t := t.nextProcess END;
  185. IF t = SELF THEN
  186. kt := threadId;
  187. IF pt = NIL THEN root := t.nextProcess ELSE pt.nextProcess := t.nextProcess END;
  188. END;
  189. mtxUnlock( processList );
  190. IF kt # 0 THEN thrKill( kt ) END;
  191. Machine.Release( Machine.X11 );
  192. END
  193. END Cancel;
  194. PROCEDURE GetPriority( ): LONGINT;
  195. BEGIN
  196. RETURN thrGetPriority( threadId )
  197. END GetPriority;
  198. PROCEDURE SetPriority( prio: LONGINT );
  199. VAR pr: LONGINT;
  200. BEGIN
  201. pr := max( Machine.prioLow, min( prio, Machine.prioHigh ) );
  202. thrSetPriority( threadId, pr ); (* works only if SUID root *)
  203. priority := GetPriority( )
  204. END SetPriority;
  205. PROCEDURE & Initialize( obj: ProtectedObject; bodyProc: Body; prio: LONGINT; fl: SET; stacksize: LONGINT);
  206. VAR thr: Unix.Thread_t;
  207. BEGIN
  208. SELF.obj := obj; condition := NIL; continue := conInit(0);
  209. flags := fl;
  210. priority := prio;
  211. nextProcess := NIL;
  212. IF root # NIL THEN
  213. newProcess := SELF;
  214. ASSERT( bodyProc # NIL );
  215. body := bodyProc;
  216. mtxLock( startProcess );
  217. thr := thrStart( BodyStarter, stacksize );
  218. conWait( childrunning, startProcess );
  219. mtxUnlock( startProcess );
  220. RegisterFinalizer( SELF, FinalizeProcess );
  221. ELSE
  222. (* first process *)
  223. stackBottom := Glue.stackBottom;
  224. SP := Machine.CurrentSP( );
  225. threadId := thrThis(0);
  226. id := 0; nextPID := 1;
  227. root := SELF;
  228. mainProcess := SELF;
  229. mode := Running;
  230. END;
  231. END Initialize;
  232. END Process;
  233. PROCEDURE BodyStarter;
  234. VAR p{UNTRACED}: Process; res: LONGINT; prevBP: ADDRESS;
  235. BEGIN
  236. mtxLock( startProcess );
  237. p := newProcess; newProcess := NIL;
  238. p.threadId := thrThis(0);
  239. p.id := nextPID; INC( nextPID );
  240. p.SP := Machine.CurrentSP( );
  241. p.stackBottom := Machine.CurrentBP( );
  242. S.GET( p.stackBottom, prevBP );
  243. S.PUT( prevBP, S.VAL( ADDRESS, 0 ) ); (* for terminating Reflection.StackTraceBack *)
  244. mtxLock( processList );
  245. p.nextProcess := root; root := p;
  246. mtxUnlock( processList );
  247. conSignal( childrunning );
  248. mtxUnlock( startProcess );
  249. p.SetPriority( p.priority );
  250. IF Restart IN p.flags THEN
  251. res := Unix.sigsetjmp( ADDRESSOF( p.state0[0] ), 1 );
  252. END;
  253. p.mode := Running;
  254. p.body( p.obj );
  255. p.mode := Terminated;
  256. Exit
  257. END BodyStarter;
  258. (*--------------------- create, lock, await, unlock -------------------------*)
  259. PROCEDURE InitProtHeader( hdr: ObjectHeader );
  260. BEGIN
  261. hdr.mtx := mtxInit( 0 ); hdr.enter := conInit( 0 ); hdr.lockedBy := NIL;
  262. END InitProtHeader;
  263. PROCEDURE CreateProcess*( body: Body; priority: LONGINT; flags: SET; obj: ProtectedObject );
  264. VAR p: Process; hdr: ObjectHeader;
  265. BEGIN
  266. mtxLock( createProcess );
  267. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr ); InitProtHeader( hdr );
  268. IF priority = 0 THEN priority := Normal END;
  269. NEW( p, obj, body, priority, flags, stacksize ) ; (* execute BodyStarter as new (posix or solaris) thread *)
  270. mtxUnlock( createProcess );
  271. RegisterFinalizer( obj, FinalizeActiveObj )
  272. END CreateProcess;
  273. PROCEDURE Lock*( obj: ProtectedObject; exclusive: BOOLEAN );
  274. VAR hdr: ObjectHeader; p: Process;
  275. BEGIN
  276. ASSERT( exclusive ); (* shared not implemented yet *)
  277. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr );
  278. p := CurrentProcess();
  279. p.mode := AwaitingLock;
  280. IF hdr.mtx = 0 THEN InitProtHeader( hdr ) END;
  281. mtxLock( hdr.mtx );
  282. WHILE hdr.lockedBy # NIL DO
  283. (* wait until threads with complied AWAIT conditions have left the monitor *)
  284. conWait( hdr.enter, hdr.mtx );
  285. END;
  286. p.mode := Running; hdr.lockedBy := p; p.waitingOn := NIL
  287. END Lock;
  288. PROCEDURE Await*( cond: Condition; slink: ADDRESS; obj: ProtectedObject; flags: SET );
  289. VAR hdr: ObjectHeader; p, c: Process;
  290. BEGIN
  291. IF 1 IN flags THEN (* compiler did not generate IF *)
  292. IF cond( slink ) THEN (* condition already true *) RETURN END
  293. END;
  294. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr ); c := NIL;
  295. IF hdr.awaitingCond.head # NIL THEN c := FindCondition( hdr.awaitingCond ) END;
  296. p := CurrentProcess(); p.succ := NIL; p.condition := cond; p.condFP := slink;
  297. p.waitingOn := obj; p.mode := AwaitingCond;
  298. Put( hdr.awaitingCond, p );
  299. hdr.lockedBy := c;
  300. IF c # NIL THEN conSignal( c.continue ) ELSE conSignal( hdr.enter ) END;
  301. conWait( p.continue, hdr.mtx );
  302. p.mode := Running; hdr.lockedBy := p; p.waitingOn := NIL
  303. END Await;
  304. PROCEDURE Unlock*( obj: ProtectedObject; dummy: BOOLEAN );
  305. VAR hdr: ObjectHeader; c: Process;
  306. BEGIN
  307. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr ); c := NIL;
  308. IF hdr.awaitingCond.head # NIL THEN c := FindCondition( hdr.awaitingCond ) END;
  309. hdr.lockedBy := c;
  310. IF c # NIL THEN conSignal( c.continue ) ELSE conSignal( hdr.enter ) END;
  311. mtxUnlock( hdr.mtx );
  312. END Unlock;
  313. PROCEDURE FindCondition( VAR q: ProcessQueue ): Process;
  314. VAR first, cand: Process;
  315. BEGIN
  316. Get( q, first );
  317. IF first.condition( first.condFP ) THEN RETURN first ELSE Put( q, first ) END;
  318. WHILE q.head # first DO
  319. Get( q, cand );
  320. IF cand.condition( cand.condFP ) THEN RETURN cand ELSE Put( q, cand ) END;
  321. END;
  322. RETURN NIL
  323. END FindCondition;
  324. PROCEDURE Get( VAR queue: ProcessQueue; VAR new: Process );
  325. VAR t: Process;
  326. BEGIN
  327. t := queue.head(Process);
  328. IF t # NIL THEN
  329. IF t = queue.tail THEN queue.head := NIL; queue.tail := NIL
  330. ELSE queue.head := t.succ; t.succ := NIL
  331. END
  332. END;
  333. new := t
  334. END Get;
  335. PROCEDURE Put( VAR queue: ProcessQueue; t: Process );
  336. BEGIN
  337. IF queue.head = NIL THEN queue.head := t ELSE queue.tail(Process).succ := t END;
  338. queue.tail := t
  339. END Put;
  340. (*-------------------------------------------------------------------------*)
  341. PROCEDURE Terminate*;
  342. BEGIN
  343. Exit
  344. END Terminate;
  345. PROCEDURE TerminateThis*( p: Process; unbreakable: BOOLEAN );
  346. BEGIN
  347. p.mode := Terminated;
  348. p.Cancel
  349. END TerminateThis;
  350. PROCEDURE SetPriority*( pri: LONGINT ); (* Set the current process' priority. *)
  351. VAR me: Process;
  352. BEGIN
  353. me := CurrentProcess();
  354. me.SetPriority( pri )
  355. END SetPriority;
  356. PROCEDURE Sleep*( ms: LONGINT );
  357. BEGIN
  358. thrSleep( ms )
  359. END Sleep;
  360. PROCEDURE Yield*; (* Relinquish control. *)
  361. BEGIN
  362. thrYield(0);
  363. END Yield;
  364. (* Return current process. (DEPRECATED, use ActiveObject) *)
  365. PROCEDURE CurrentProcess*( ): Process;
  366. VAR me: Unix.Thread_t; p: Process;
  367. BEGIN
  368. me := thrThis(0);
  369. mtxLock( processList );
  370. p := root;
  371. WHILE (p # NIL) & (p.threadId # me) DO p := p.nextProcess END;
  372. mtxUnlock( processList );
  373. RETURN p
  374. END CurrentProcess;
  375. (* Return the active object currently executing. *)
  376. PROCEDURE ActiveObject*( ): ANY;
  377. VAR p: Process;
  378. BEGIN
  379. p := CurrentProcess();
  380. RETURN p.obj
  381. END ActiveObject;
  382. (* Return stack bottom of process. For compatibility WinAos/UnixAos/NativeAos *)
  383. PROCEDURE GetStackBottom*(p: Process): ADDRESS;
  384. BEGIN
  385. RETURN p.stackBottom
  386. END GetStackBottom;
  387. PROCEDURE GetProcessID*( ): LONGINT;
  388. VAR p: Process;
  389. BEGIN
  390. p := CurrentProcess();
  391. RETURN p.id;
  392. END GetProcessID;
  393. PROCEDURE GetCpuCycles*( process : Process; VAR cpuCycles: CpuCyclesArray; all: BOOLEAN );
  394. VAR i: LONGINT;
  395. BEGIN
  396. ASSERT( process # NIL );
  397. FOR i := 0 TO Machine.MaxCPU-1 DO cpuCycles[i] := 0 END;
  398. END GetCpuCycles;
  399. (*-----------------------------------------------------------------------*)
  400. PROCEDURE min( a, b: LONGINT ): LONGINT;
  401. BEGIN
  402. IF a <= b THEN RETURN a ELSE RETURN b END
  403. END min;
  404. PROCEDURE max( a, b: LONGINT ): LONGINT;
  405. BEGIN
  406. IF a >= b THEN RETURN a ELSE RETURN b END
  407. END max;
  408. PROCEDURE RegisterFinalizer( obj: ANY; fin: Heaps.Finalizer );
  409. VAR n: Heaps.FinalizerNode;
  410. BEGIN
  411. NEW( n ); n.finalizer := fin; Heaps.AddFinalizer( obj, n );
  412. END RegisterFinalizer;
  413. PROCEDURE FinalizeActiveObj( obj: ANY );
  414. VAR p: Process;
  415. BEGIN
  416. mtxLock( processList );
  417. p := root;
  418. WHILE (p # NIL) & (p.obj # obj) DO p := p.nextProcess END;
  419. mtxUnlock( processList );
  420. IF (p # NIL) & (p.obj = obj) THEN
  421. p.mode := Terminated;
  422. conDestroy( p.continue ); p.continue := 0;
  423. FinalizeProtObject( obj );
  424. p.Cancel
  425. END;
  426. END FinalizeActiveObj;
  427. PROCEDURE FinalizeProtObject( obj: ANY );
  428. VAR hdr: ObjectHeader;
  429. BEGIN
  430. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr );
  431. IF hdr.mtx # 0 THEN
  432. mtxDestroy( hdr.mtx ); hdr.mtx := 0
  433. END
  434. END FinalizeProtObject;
  435. PROCEDURE FinalizeProcess( obj: ANY );
  436. VAR p: Process;
  437. BEGIN
  438. p := obj(Process);
  439. IF p.continue # 0 THEN
  440. conDestroy( p.continue ); p.continue := 0
  441. END
  442. END FinalizeProcess;
  443. (* Terminate calling thread. *)
  444. PROCEDURE Exit;
  445. VAR prev, p, me: Process;
  446. BEGIN
  447. me := CurrentProcess();
  448. me.mode := Terminated;
  449. mtxLock( processList );
  450. prev := NIL; p := root;
  451. WHILE (p # NIL ) & (p # me) DO prev := p; p := p.nextProcess END;
  452. IF p = me THEN
  453. IF prev = NIL THEN root := p.nextProcess ELSE prev.nextProcess := p.nextProcess END;
  454. END;
  455. mtxUnlock( processList );
  456. thrExit(0)
  457. END Exit;
  458. PROCEDURE ExitTrap*;
  459. VAR p: Process;
  460. BEGIN
  461. p := CurrentProcess();
  462. (* restart the object body if it was given the SAFE flag *)
  463. IF Restart IN p.flags THEN
  464. Unix.siglongjmp( ADDRESSOF( p.state0[0] ), 1 )
  465. END;
  466. Exit
  467. END ExitTrap;
  468. (*---------------------------- Timer --------------------------------*)
  469. PROCEDURE Remove( t: Timer ); (* remove timer from list of active timers *)
  470. VAR p, x: Timer;
  471. BEGIN
  472. mtxLock( timerListMutex );
  473. t.trigger := 0; t.handler := NIL;
  474. IF timers # NIL THEN
  475. IF t = timers THEN
  476. timers := t.next
  477. ELSE
  478. p := timers; x := p.next;
  479. WHILE (x # NIL) & (x # t) DO p := x; x := p.next END;
  480. IF x = t THEN p.next := t.next END
  481. END;
  482. t.next := NIL
  483. END;
  484. mtxUnlock( timerListMutex )
  485. END Remove;
  486. PROCEDURE Insert( t: Timer );
  487. VAR p, x: Timer;
  488. BEGIN
  489. mtxLock( timerListMutex );
  490. p := NIL; x := timers;
  491. WHILE (x # NIL) & (x.trigger < t.trigger) DO p := x; x := p.next END;
  492. t.next := x;
  493. IF p = NIL THEN timers := t ELSE p.next := t END;
  494. mtxUnlock( timerListMutex )
  495. END Insert;
  496. PROCEDURE SetTimeout*( t: Timer; h: EventHandler; ms: LONGINT );
  497. BEGIN
  498. ASSERT( ( t # NIL) & ( h # NIL) );
  499. Remove( t );
  500. IF ms < 1 THEN ms := 1 END;
  501. t.trigger := Machine.ticks + ms; t.handler := h;
  502. Insert( t );
  503. timerActivity.Restart
  504. END SetTimeout;
  505. PROCEDURE SetTimeoutAt*( t: Timer; h: EventHandler; ms: LONGINT );
  506. BEGIN
  507. ASSERT( (t # NIL) & (h # NIL) );
  508. Remove( t );
  509. t.trigger := ms; t.handler := h;
  510. Insert( t );
  511. timerActivity.Restart
  512. END SetTimeoutAt;
  513. PROCEDURE CancelTimeout*( t: Timer );
  514. BEGIN
  515. Remove( t )
  516. END CancelTimeout;
  517. (*-------------------- Garbage Collection ------------------------------------*)
  518. (* called by WMProcessInfo to obtain the current state of a running process *)
  519. PROCEDURE UpdateProcessState*( p: Process );
  520. BEGIN
  521. (* update p.stat.{PC,BP,SP} *)
  522. END UpdateProcessState;
  523. PROCEDURE SuspendActivities;
  524. VAR t: Process;
  525. BEGIN
  526. t := root;
  527. WHILE t # NIL DO
  528. IF (t # mainProcess) & (t # finCaller) THEN thrSuspend( t.threadId ) END;
  529. t := t.nextProcess
  530. END;
  531. END SuspendActivities;
  532. PROCEDURE ResumeActivities;
  533. VAR t: Process;
  534. BEGIN
  535. t := root;
  536. WHILE t # NIL DO
  537. IF (t # mainProcess) & (t # finCaller) THEN thrResume( t.threadId ) END;
  538. t := t.nextProcess
  539. END;
  540. END ResumeActivities;
  541. PROCEDURE SaveSP; (* save current SP for usage by the GC *)
  542. VAR me: Unix.Thread_t; t: Process;
  543. BEGIN
  544. me := thrThis(0); t := root;
  545. WHILE (t # NIL ) & (t.threadId # me) DO t := t.nextProcess END;
  546. IF t # NIL THEN t.SP := Machine.CurrentSP( ) END
  547. END SaveSP;
  548. PROCEDURE InvokeGC;
  549. BEGIN
  550. IF Machine.AcquireGC() THEN (* gets released by FinalizerCaller *)
  551. collect := TRUE;
  552. conWait( gcFinished, igc )
  553. END;
  554. END InvokeGC;
  555. (*! GCLoop gets called as last procedure in BootConsole (main thread).
  556. The stack of the main thread is not limited by the boot parameter 'StackSize' !!
  557. *)
  558. PROCEDURE GCLoop*; (* Timer and GC activity *)
  559. BEGIN
  560. SetPriority( GCPriority );
  561. LOOP
  562. IF collect THEN
  563. collect := FALSE;
  564. Machine.Acquire( Machine.Heaps );
  565. SuspendActivities;
  566. Heaps.CollectGarbage( Modules.root );
  567. Machine.Release( Machine.Heaps );
  568. ResumeActivities;
  569. finalizerCaller.Start;
  570. conSignal( gcFinished );
  571. ELSE
  572. thrSleep( 10 );
  573. END;
  574. timerActivity.UpdateTicks
  575. END
  576. END GCLoop;
  577. PROCEDURE CurrentProcessTime*(): HUGEINT;
  578. BEGIN
  579. RETURN Machine.GetTimer()
  580. END CurrentProcessTime;
  581. PROCEDURE TimerFrequency*(): HUGEINT;
  582. BEGIN
  583. RETURN Machine.mhz * 1000000
  584. END TimerFrequency;
  585. (*----------------------------- initialization ----------------------------------*)
  586. PROCEDURE StartTimerActivity;
  587. BEGIN
  588. timerListMutex := mtxInit(0); timers := NIL;
  589. NEW( timerActivity );
  590. END StartTimerActivity;
  591. PROCEDURE GetStacksize;
  592. VAR str: ARRAY 32 OF CHAR; i: LONGINT;
  593. BEGIN
  594. Machine.GetConfig( "StackSize", str );
  595. IF str = "" THEN stacksize := DefaultStacksize
  596. ELSE
  597. i := 0; stacksize := Machine.StrToInt( i, str );
  598. stacksize := stacksize * 1024;
  599. END;
  600. IF Glue.debug # {} THEN
  601. Trace.String( "Stacksize of active objects = " );
  602. Trace.Int( stacksize DIV 1024, 0 ); Trace.StringLn( "K" )
  603. END;
  604. END GetStacksize;
  605. PROCEDURE Convert;
  606. VAR p: Process;
  607. BEGIN
  608. (* make current thread the first active object *)
  609. NEW( p, NIL, NIL, 0, {}, 0 );
  610. END Convert;
  611. PROCEDURE Init;
  612. BEGIN
  613. Unix.Dlsym( 0, "mtxInit", ADDRESSOF( mtxInit ) );
  614. Unix.Dlsym( 0, "mtxDestroy", ADDRESSOF( mtxDestroy ) );
  615. Unix.Dlsym( 0, "mtxLock", ADDRESSOF( mtxLock ) );
  616. Unix.Dlsym( 0, "mtxUnlock", ADDRESSOF( mtxUnlock ) );
  617. Unix.Dlsym( 0, "conInit", ADDRESSOF( conInit ) );
  618. Unix.Dlsym( 0, "conDestroy", ADDRESSOF( conDestroy ) );
  619. Unix.Dlsym( 0, "conWait", ADDRESSOF( conWait ) );
  620. Unix.Dlsym( 0, "conSignal", ADDRESSOF( conSignal ) );
  621. Unix.Dlsym( 0, "thrStart", ADDRESSOF( thrStart ) );
  622. Unix.Dlsym( 0, "thrThis", ADDRESSOF( thrThis ) );
  623. Unix.Dlsym( 0, "thrSleep", ADDRESSOF( thrSleep ) );
  624. Unix.Dlsym( 0, "thrYield", ADDRESSOF( thrYield ) );
  625. Unix.Dlsym( 0, "thrExit", ADDRESSOF( thrExit ) );
  626. Unix.Dlsym( 0, "thrSuspend", ADDRESSOF( thrSuspend ) );
  627. Unix.Dlsym( 0, "thrResume", ADDRESSOF( thrResume ) );
  628. Unix.Dlsym( 0, "thrGetPriority", ADDRESSOF( thrGetPriority ) );
  629. Unix.Dlsym( 0, "thrSetPriority", ADDRESSOF( thrSetPriority ) );
  630. Unix.Dlsym( 0, "thrKill", ADDRESSOF( thrKill ) );
  631. createProcess := mtxInit( 0 ); processList := mtxInit( 0 );
  632. startProcess := mtxInit(0); childrunning := conInit(0);
  633. collect := FALSE;
  634. igc := mtxInit( 0 ); gcFinished := conInit( 0 );
  635. GetStacksize;
  636. Convert;
  637. StartTimerActivity;
  638. NEW( finalizerCaller );
  639. Heaps.saveSP := SaveSP;
  640. Heaps.GC := InvokeGC;
  641. Heaps.InvokeGC := InvokeGC;
  642. END Init;
  643. VAR
  644. (* for compatibility and later extension *)
  645. TraceProcessHook*: PROCEDURE (prcoess: Process; pc, bp: ADDRESS; stacklow, stackhigh: ADDRESS);
  646. BEGIN
  647. TraceProcessHook := NIL;
  648. Init;
  649. END Objects.