Generic.Solaris.Objects.Mod 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818
  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; Locked = 7;
  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. clock : Clock;
  29. timers : Timer;
  30. timerListMutex : Unix.Mutex_t;
  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. TYPE
  43. LockT= POINTER TO RECORD
  44. mtx, enter: ADDRESS;
  45. END;
  46. CpuCyclesArray* = ARRAY Machine.MaxCPU OF HUGEINT;
  47. ProtectedObject = POINTER TO RECORD END;
  48. ObjectHeader = Heaps.ProtRecBlock;
  49. ProcessQueue = Heaps.ProcessQueue;
  50. EventHandler* = PROCEDURE {DELEGATE};
  51. Timer* = OBJECT
  52. VAR
  53. next: Timer;
  54. trigger: LONGINT;
  55. handler: EventHandler
  56. END Timer;
  57. TimerActivity = OBJECT
  58. VAR
  59. t, r: Timer; h: EventHandler; restart: BOOLEAN;
  60. PROCEDURE UpdateTicks;
  61. BEGIN {EXCLUSIVE}
  62. Machine.UpdateTicks
  63. END UpdateTicks;
  64. PROCEDURE Restart;
  65. BEGIN {EXCLUSIVE}
  66. restart := TRUE
  67. END Restart;
  68. BEGIN {ACTIVE, SAFE, PRIORITY(High)}
  69. restart := FALSE;
  70. LOOP
  71. t := timers;
  72. IF t # NIL THEN
  73. h := NIL; r := NIL;
  74. BEGIN {EXCLUSIVE}
  75. AWAIT( (Machine.ticks >= t.trigger) OR restart ); restart := FALSE;
  76. IF Machine.ticks >= t.trigger THEN
  77. h := t.handler; r := t
  78. END
  79. END;
  80. IF r # NIL THEN Remove( r ) END;
  81. IF h # NIL THEN (* not canceled *) h END
  82. ELSE
  83. BEGIN{EXCLUSIVE}
  84. AWAIT( restart ); restart := FALSE;
  85. END
  86. END
  87. END
  88. END TimerActivity;
  89. Clock* = OBJECT
  90. VAR i: LONGINT;
  91. BEGIN{ACTIVE}
  92. LOOP
  93. Unix.ThrSleep( 10 );
  94. timerActivity.UpdateTicks
  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; (*! not used in UnixAos! *)
  149. context : Unix.McontextDesc;
  150. contsp : ADDRESS;
  151. state0 : ARRAY 2048 OF CHAR; (* thread state at body start, used for restart after trap *)
  152. PROCEDURE FindRoots*;
  153. VAR sp, ptr, bp, n, a0, a1, adr: ADDRESS; desc: Modules.ProcedureDescPointer; i: LONGINT; p {UNTRACED}: ANY;
  154. me: Process;
  155. BEGIN
  156. IF mode # Terminated THEN
  157. IF SELF = CurrentProcess() THEN
  158. context.r_sp := Machine.CurrentSP();
  159. context.r_bp := Machine.CurrentBP();
  160. END;
  161. sp := context.r_sp; bp := context.r_bp;
  162. Heaps.Candidate( context.r_di ); Heaps.Candidate( context.r_si );
  163. Heaps.Candidate( context.r_bx ); Heaps.Candidate( context.r_dx );
  164. Heaps.Candidate( context.r_cx ); Heaps.Candidate( context.r_ax) ;
  165. IF (stackBottom # 0) & (sp # 0) & (sp <= stackBottom) THEN
  166. Heaps.RegisterCandidates( sp, stackBottom - sp );
  167. END;
  168. END;
  169. Heaps.Mark( nextProcess )
  170. END FindRoots;
  171. PROCEDURE Cancel;
  172. VAR pt, t: Process; kt: Unix.Thread_t;
  173. BEGIN
  174. IF SELF = CurrentProcess() THEN Exit
  175. ELSE
  176. Machine.Acquire( Machine.X11 ); (* let the thread to be killed first finish its last I/O, if any *)
  177. Unix.MtxLock( processList );
  178. pt := NIL; t := root; kt := 0;
  179. WHILE (t # NIL ) & (t # SELF) DO pt := t; t := t.nextProcess END;
  180. IF t = SELF THEN
  181. kt := threadId;
  182. IF pt = NIL THEN root := t.nextProcess ELSE pt.nextProcess := t.nextProcess END;
  183. END;
  184. Unix.MtxUnlock( processList );
  185. IF kt # 0 THEN Unix.ThrKill( kt ) END;
  186. Machine.Release( Machine.X11 );
  187. END
  188. END Cancel;
  189. PROCEDURE GetPriority( ): LONGINT;
  190. BEGIN
  191. RETURN Unix.ThrGetPriority( threadId )
  192. END GetPriority;
  193. PROCEDURE SetPriority( prio: LONGINT );
  194. VAR pr: LONGINT;
  195. BEGIN
  196. pr := max( Machine.prioLow, min( prio, Machine.prioHigh ) );
  197. Unix.ThrSetPriority( threadId, pr ); (* works only if SUID root *)
  198. priority := GetPriority( )
  199. END SetPriority;
  200. PROCEDURE & Initialize( obj: ProtectedObject; bodyProc: Body; prio: LONGINT; fl: SET; stacksize: LONGINT);
  201. VAR thr: Unix.Thread_t;
  202. BEGIN
  203. SELF.obj := obj; condition := NIL; continue := Unix.ConInit(0);
  204. flags := fl;
  205. priority := prio;
  206. nextProcess := NIL;
  207. IF root # NIL THEN
  208. newProcess := SELF;
  209. ASSERT( bodyProc # NIL );
  210. body := bodyProc;
  211. Unix.MtxLock( startProcess );
  212. thr := Unix.ThrStart( BodyStarter, stacksize );
  213. Unix.ConWait( childrunning, startProcess );
  214. Unix.MtxUnlock( startProcess );
  215. RegisterFinalizer( SELF, FinalizeProcess );
  216. ELSE
  217. (* first process *)
  218. stackBottom := Glue.stackBottom;
  219. threadId := Unix.ThrThis(0);
  220. id := 0; nextPID := 1;
  221. root := SELF;
  222. mode := Running;
  223. END;
  224. END Initialize;
  225. END Process;
  226. GCStatusExt = OBJECT (Heaps.GCStatus)
  227. PROCEDURE SetgcOngoing( value: BOOLEAN );
  228. VAR p: Heaps.ProcessLink; cur, r: Process; res: LONGINT; num: LONGINT; time: LONGINT;
  229. BEGIN
  230. IF value THEN
  231. IF Machine.AcquireGC( ) THEN
  232. (* Trace.StringLn( "[GC" ); *)
  233. Machine.Acquire( Machine.X11 );
  234. Machine.Acquire( Machine.Heaps );
  235. SuspendActivities;
  236. Heaps.CollectGarbage( Modules.root );
  237. Machine.Release( Machine.Heaps );
  238. Machine.Release( Machine.X11 );
  239. ResumeActivities;
  240. finalizerCaller.Activate;
  241. (* Trace.StringLn( "GC]" ) *)
  242. (* ELSE
  243. Trace.StringLn( "-GC-" ) *)
  244. END
  245. END;
  246. END SetgcOngoing;
  247. END GCStatusExt;
  248. PROCEDURE BodyStarter;
  249. VAR p{UNTRACED}: Process; res: LONGINT; prevBP: ADDRESS; i: LONGINT;
  250. BEGIN
  251. Unix.MtxLock( startProcess );
  252. p := newProcess; newProcess := NIL;
  253. p.threadId := Unix.ThrThis(0);
  254. p.id := nextPID; INC( nextPID );
  255. p.stackBottom := Machine.CurrentBP( );
  256. S.GET( p.stackBottom, prevBP );
  257. S.PUT( prevBP, S.VAL( ADDRESS, 0 ) ); (* for terminating Reflection.StackTraceBack *)
  258. Unix.MtxLock( processList );
  259. p.nextProcess := root; root := p;
  260. Unix.MtxUnlock( processList );
  261. Unix.ConSignal( childrunning );
  262. Unix.MtxUnlock( startProcess );
  263. p.SetPriority( p.priority );
  264. IF Restart IN p.flags THEN
  265. res := Unix.sigsetjmp( ADDRESSOF( p.state0[0] ), 1 );
  266. END;
  267. p.mode := Running;
  268. p.body( p.obj );
  269. p.mode := Terminated;
  270. Exit
  271. END BodyStarter;
  272. (*--------------------- create, lock, await, unlock -------------------------*)
  273. (* initialize the ObjectHeader, requires lockMutex temporarily *)
  274. PROCEDURE InitProtHeader( hdr: ObjectHeader);
  275. VAR lock: LockT;
  276. BEGIN
  277. (* we cannot hold the lockMute here because allocation can trigger the GC that requires the lock when activating the finalizers *)
  278. NEW(lock);
  279. Unix.MtxLock(lockMutex);
  280. IF hdr.lock = NIL THEN
  281. hdr.lock := lock;
  282. lock.mtx := Unix.MtxInit( 0 ); lock.enter := Unix.ConInit( 0 ); hdr.lockedBy := NIL;
  283. END;
  284. Unix.MtxUnlock(lockMutex);
  285. END InitProtHeader;
  286. PROCEDURE CreateProcess*( body: Body; priority: LONGINT; flags: SET; obj: ProtectedObject );
  287. VAR p: Process; hdr: ObjectHeader;
  288. BEGIN
  289. Unix.MtxLock( createProcess );
  290. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr );
  291. InitProtHeader( hdr );
  292. IF priority = 0 THEN priority := Normal END;
  293. NEW( p, obj, body, priority, flags, stacksize ) ; (* execute BodyStarter as new (posix or solaris) thread *)
  294. Unix.MtxUnlock( createProcess );
  295. RegisterFinalizer( obj, FinalizeActiveObj )
  296. END CreateProcess;
  297. PROCEDURE Lock*( obj: ProtectedObject; exclusive: BOOLEAN );
  298. VAR hdr: ObjectHeader; p: Process; lock: LockT;
  299. BEGIN
  300. ASSERT( exclusive ); (* shared not implemented yet *)
  301. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr );
  302. p := CurrentProcess();
  303. p.mode := AwaitingLock;
  304. (*! we might want to replace the lock mutex by a lock free construct *)
  305. IF hdr.lock = NIL THEN InitProtHeader( hdr ) END;
  306. lock := S.VAL(LockT, hdr.lock);
  307. p.mode := Locked;
  308. Unix.MtxLock( lock.mtx );
  309. WHILE hdr.lockedBy # NIL DO
  310. (* wait until threads with complied AWAIT conditions have left the monitor *)
  311. p.mode := AwaitingLock;
  312. Unix.ConWait( lock.enter, lock.mtx );
  313. END;
  314. p.mode := Running; hdr.lockedBy := p; p.waitingOn := NIL
  315. END Lock;
  316. PROCEDURE Await*( cond: Condition; slink: ADDRESS; obj: ProtectedObject; flags: SET );
  317. VAR hdr: ObjectHeader; p, c: Process; lock: LockT; sp: ADDRESS;
  318. BEGIN
  319. IF 1 IN flags THEN (* compiler did not generate IF *)
  320. IF cond( slink ) THEN (* condition already true *) RETURN END
  321. END;
  322. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr ); c := NIL;
  323. lock := S.VAL(LockT, hdr.lock);
  324. IF hdr.awaitingCond.head # NIL THEN c := FindCondition( hdr.awaitingCond ) END;
  325. p := CurrentProcess(); p.succ := NIL; p.condition := cond; p.condFP := slink;
  326. p.waitingOn := obj; p.mode := AwaitingCond;
  327. Put( hdr.awaitingCond, p );
  328. hdr.lockedBy := c;
  329. IF c # NIL THEN Unix.ConSignal( c.continue ) ELSE Unix.ConSignal( lock.enter ) END;
  330. sp := Machine.CurrentSP( );
  331. IF p.contsp # sp THEN
  332. p.contsp := sp;
  333. Unix.GetThreadContext( p.threadId )
  334. END;
  335. Unix.ConWait( p.continue, lock.mtx );
  336. p.mode := Running; hdr.lockedBy := p; p.waitingOn := NIL
  337. END Await;
  338. PROCEDURE Unlock*( obj: ProtectedObject; dummy: BOOLEAN );
  339. VAR hdr: ObjectHeader; c: Process; lock: LockT;
  340. BEGIN
  341. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr ); c := NIL;
  342. lock := S.VAL(LockT,hdr.lock);
  343. IF hdr.awaitingCond.head # NIL THEN c := FindCondition( hdr.awaitingCond ) END;
  344. hdr.lockedBy := c;
  345. IF c # NIL THEN Unix.ConSignal( c.continue ) ELSE Unix.ConSignal( lock.enter ) END;
  346. Unix.MtxUnlock( lock.mtx );
  347. END Unlock;
  348. PROCEDURE FindCondition( VAR q: ProcessQueue ): Process;
  349. VAR first, cand: Process;
  350. BEGIN
  351. Get( q, first );
  352. IF first.condition( first.condFP ) THEN RETURN first ELSE Put( q, first ) END;
  353. WHILE q.head # first DO
  354. Get( q, cand );
  355. IF cand.condition( cand.condFP ) THEN RETURN cand ELSE Put( q, cand ) END;
  356. END;
  357. RETURN NIL
  358. END FindCondition;
  359. PROCEDURE Get( VAR queue: ProcessQueue; VAR new: Process );
  360. VAR t: Process;
  361. BEGIN
  362. t := queue.head(Process);
  363. IF t # NIL THEN
  364. IF t = queue.tail THEN queue.head := NIL; queue.tail := NIL
  365. ELSE queue.head := t.succ; t.succ := NIL
  366. END
  367. END;
  368. new := t
  369. END Get;
  370. PROCEDURE Put( VAR queue: ProcessQueue; t: Process );
  371. BEGIN
  372. IF queue.head = NIL THEN queue.head := t ELSE queue.tail(Process).succ := t END;
  373. queue.tail := t
  374. END Put;
  375. (*-------------------------------------------------------------------------*)
  376. PROCEDURE Terminate*;
  377. BEGIN
  378. Exit
  379. END Terminate;
  380. PROCEDURE TerminateThis*( p: Process; unbreakable: BOOLEAN );
  381. BEGIN
  382. p.mode := Terminated;
  383. p.Cancel
  384. END TerminateThis;
  385. PROCEDURE SetPriority*( pri: LONGINT ); (* Set the current process' priority. *)
  386. VAR me: Process;
  387. BEGIN
  388. me := CurrentProcess();
  389. me.SetPriority( pri )
  390. END SetPriority;
  391. PROCEDURE Sleep*( ms: LONGINT );
  392. BEGIN
  393. Unix.ThrSleep( ms )
  394. END Sleep;
  395. PROCEDURE Yield*; (* Relinquish control. *)
  396. BEGIN
  397. Unix.ThrYield(0);
  398. END Yield;
  399. (* Return current process. (DEPRECATED, use ActiveObject) *)
  400. PROCEDURE CurrentProcess*( ): Process;
  401. VAR me: Unix.Thread_t; p: Process;
  402. BEGIN
  403. me := Unix.ThrThis(0);
  404. Unix.MtxLock( processList );
  405. p := root;
  406. WHILE (p # NIL) & (p.threadId # me) DO p := p.nextProcess END;
  407. Unix.MtxUnlock( processList );
  408. RETURN p
  409. END CurrentProcess;
  410. (* Return the active object currently executing. *)
  411. PROCEDURE ActiveObject*( ): ANY;
  412. VAR p: Process;
  413. BEGIN
  414. p := CurrentProcess();
  415. RETURN p.obj
  416. END ActiveObject;
  417. (* Return stack bottom of process. For compatibility WinAos/UnixAos/NativeAos *)
  418. PROCEDURE GetStackBottom*(p: Process): ADDRESS;
  419. BEGIN
  420. RETURN p.stackBottom
  421. END GetStackBottom;
  422. PROCEDURE GetProcessID*( ): LONGINT;
  423. VAR p: Process;
  424. BEGIN
  425. p := CurrentProcess();
  426. RETURN p.id;
  427. END GetProcessID;
  428. PROCEDURE GetCpuCycles*( process : Process; VAR cpuCycles: CpuCyclesArray; all: BOOLEAN );
  429. VAR i: LONGINT;
  430. BEGIN
  431. ASSERT( process # NIL );
  432. FOR i := 0 TO Machine.MaxCPU-1 DO cpuCycles[i] := 0 END;
  433. END GetCpuCycles;
  434. (*-----------------------------------------------------------------------*)
  435. PROCEDURE min( a, b: LONGINT ): LONGINT;
  436. BEGIN
  437. IF a <= b THEN RETURN a ELSE RETURN b END
  438. END min;
  439. PROCEDURE max( a, b: LONGINT ): LONGINT;
  440. BEGIN
  441. IF a >= b THEN RETURN a ELSE RETURN b END
  442. END max;
  443. PROCEDURE RegisterFinalizer( obj: ANY; fin: Heaps.Finalizer );
  444. VAR n: Heaps.FinalizerNode;
  445. BEGIN
  446. NEW( n ); n.finalizer := fin; Heaps.AddFinalizer( obj, n );
  447. END RegisterFinalizer;
  448. PROCEDURE FinalizeActiveObj( obj: ANY );
  449. VAR p: Process;
  450. BEGIN
  451. Unix.MtxLock( processList );
  452. p := root;
  453. WHILE (p # NIL) & (p.obj # obj) DO p := p.nextProcess END;
  454. Unix.MtxUnlock( processList );
  455. IF (p # NIL) & (p.obj = obj) THEN
  456. p.mode := Terminated;
  457. Unix.ConDestroy( p.continue ); p.continue := 0;
  458. FinalizeProtObject( obj );
  459. p.Cancel
  460. END;
  461. END FinalizeActiveObj;
  462. PROCEDURE FinalizeProtObject( obj: ANY );
  463. VAR hdr: ObjectHeader; lock: LockT;
  464. BEGIN
  465. S.GET( S.VAL( ADDRESS, obj ) + Heaps.HeapBlockOffset, hdr );
  466. IF hdr.lock # NIL THEN
  467. lock := S.VAL(LockT, hdr.lock);
  468. Unix.MtxDestroy( lock.mtx ); lock.mtx := 0
  469. END
  470. END FinalizeProtObject;
  471. PROCEDURE FinalizeProcess( obj: ANY );
  472. VAR p: Process;
  473. BEGIN
  474. p := obj(Process);
  475. IF p.continue # 0 THEN
  476. Unix.ConDestroy( p.continue ); p.continue := 0
  477. END
  478. END FinalizeProcess;
  479. (* Terminate calling thread. *)
  480. PROCEDURE Exit;
  481. VAR prev, p, me: Process;
  482. BEGIN
  483. me := CurrentProcess();
  484. me.mode := Terminated;
  485. Unix.MtxLock( processList );
  486. prev := NIL; p := root;
  487. WHILE (p # NIL ) & (p # me) DO prev := p; p := p.nextProcess END;
  488. IF p = me THEN
  489. IF prev = NIL THEN root := p.nextProcess ELSE prev.nextProcess := p.nextProcess END;
  490. END;
  491. Unix.MtxUnlock( processList );
  492. Unix.ThrExit(0);
  493. END Exit;
  494. PROCEDURE ExitTrap*;
  495. VAR p: Process;
  496. BEGIN
  497. p := CurrentProcess();
  498. (* restart the object body if it was given the SAFE flag *)
  499. IF Restart IN p.flags THEN
  500. Unix.siglongjmp( ADDRESSOF( p.state0[0] ), 1 )
  501. END;
  502. Exit
  503. END ExitTrap;
  504. (*---------------------------- Timer --------------------------------*)
  505. PROCEDURE Remove( t: Timer ); (* remove timer from list of active timers *)
  506. VAR p, x: Timer;
  507. BEGIN
  508. Unix.MtxLock( timerListMutex );
  509. t.trigger := 0; t.handler := NIL;
  510. IF timers # NIL THEN
  511. IF t = timers THEN
  512. timers := t.next
  513. ELSE
  514. p := timers; x := p.next;
  515. WHILE (x # NIL) & (x # t) DO p := x; x := p.next END;
  516. IF x = t THEN p.next := t.next END
  517. END;
  518. t.next := NIL
  519. END;
  520. Unix.MtxUnlock( timerListMutex )
  521. END Remove;
  522. PROCEDURE Insert( t: Timer );
  523. VAR p, x: Timer;
  524. BEGIN
  525. Unix.MtxLock( timerListMutex );
  526. p := NIL; x := timers;
  527. WHILE (x # NIL) & (x.trigger < t.trigger) DO p := x; x := p.next END;
  528. t.next := x;
  529. IF p = NIL THEN timers := t ELSE p.next := t END;
  530. Unix.MtxUnlock( timerListMutex )
  531. END Insert;
  532. PROCEDURE SetTimeout*( t: Timer; h: EventHandler; ms: LONGINT );
  533. BEGIN
  534. ASSERT( ( t # NIL) & ( h # NIL) );
  535. Remove( t );
  536. IF ms < 1 THEN ms := 1 END;
  537. t.trigger := Machine.ticks + ms; t.handler := h;
  538. Insert( t );
  539. timerActivity.Restart
  540. END SetTimeout;
  541. PROCEDURE SetTimeoutAt*( t: Timer; h: EventHandler; ms: LONGINT );
  542. BEGIN
  543. ASSERT( (t # NIL) & (h # NIL) );
  544. Remove( t );
  545. t.trigger := ms; t.handler := h;
  546. Insert( t );
  547. timerActivity.Restart
  548. END SetTimeoutAt;
  549. PROCEDURE CancelTimeout*( t: Timer );
  550. BEGIN
  551. Remove( t )
  552. END CancelTimeout;
  553. (*-------------------- Garbage Collection ------------------------------------*)
  554. PROCEDURE GetContext( ctxt: Unix.Ucontext );
  555. VAR t: Process; bp: ADDRESS;
  556. BEGIN
  557. t := CurrentProcess();
  558. Unix.CopyContext( ctxt.mc, t.context );
  559. END GetContext;
  560. PROCEDURE SuspendActivities;
  561. VAR t,me: Process; res: LONGINT;
  562. BEGIN
  563. me := CurrentProcess();
  564. t := root;
  565. WHILE t # NIL DO
  566. IF t # me THEN
  567. (* Trace.String( "suspend " ); Trace.Hex( t.id, 2 ); Trace.Ln; *)
  568. IF t.mode = Running THEN
  569. t.contsp := 0;
  570. Unix.ThrSuspend( t.threadId, TRUE );
  571. ELSE
  572. Unix.ThrSuspend( t.threadId, FALSE )
  573. END
  574. END;
  575. t := t.nextProcess
  576. END;
  577. END SuspendActivities;
  578. PROCEDURE ResumeActivities;
  579. VAR t, me: Process;
  580. BEGIN
  581. me := CurrentProcess();
  582. t := root;
  583. WHILE t # NIL DO
  584. IF (t # me) THEN
  585. (* Trace.String( "resume " ); Trace.Hex( t.id, 2 ); Trace.Ln; *)
  586. Unix.ThrResume( t.threadId );
  587. END;
  588. t := t.nextProcess
  589. END;
  590. END ResumeActivities;
  591. (*! GCLoop gets called as last procedure in BootConsole (main thread).
  592. The stack of the main thread is not limited by the boot parameter 'StackSize' !!
  593. *)
  594. PROCEDURE GCLoop*; (* Timer and GC activity *)
  595. BEGIN
  596. LOOP Unix.ThrSleep( 10 ) END; (* in Solaris the main thread must not terminate !! *)
  597. (* RETURN; *)
  598. END GCLoop;
  599. PROCEDURE CurrentProcessTime*(): HUGEINT;
  600. BEGIN
  601. RETURN Machine.GetTimer()
  602. END CurrentProcessTime;
  603. PROCEDURE TimerFrequency*(): HUGEINT;
  604. BEGIN
  605. RETURN Machine.mhz * 1000000
  606. END TimerFrequency;
  607. (*----------------------------- initialization ----------------------------------*)
  608. PROCEDURE StartTimerActivity;
  609. BEGIN
  610. timerListMutex := Unix.MtxInit(0); timers := NIL;
  611. NEW( timerActivity );
  612. END StartTimerActivity;
  613. PROCEDURE GetStacksize;
  614. VAR str: ARRAY 32 OF CHAR; i: LONGINT;
  615. BEGIN
  616. Machine.GetConfig( "StackSize", str );
  617. IF str = "" THEN stacksize := DefaultStacksize
  618. ELSE
  619. i := 0; stacksize := Machine.StrToInt( i, str );
  620. stacksize := stacksize * 1024;
  621. END;
  622. IF Glue.debug # {} THEN
  623. Trace.String( "Stacksize of active objects = " );
  624. Trace.Int( stacksize DIV 1024, 0 ); Trace.StringLn( "K" )
  625. END;
  626. END GetStacksize;
  627. PROCEDURE Convert;
  628. VAR p: Process;
  629. BEGIN
  630. (* make current thread the first active object *)
  631. NEW( p, NIL, NIL, 0, {}, 0 );
  632. END Convert;
  633. PROCEDURE Init;
  634. BEGIN
  635. Unix.suspendHandler := GetContext;
  636. createProcess := Unix.MtxInit( 0 ); processList := Unix.MtxInit( 0 );
  637. startProcess := Unix.MtxInit(0); childrunning := Unix.ConInit(0);
  638. lockMutex := Unix.MtxInit(0);
  639. GetStacksize;
  640. Convert;
  641. NEW( clock ); StartTimerActivity;
  642. NEW( finalizerCaller );
  643. Heaps.gcStatus := GCStatusFactory()
  644. END Init;
  645. PROCEDURE GCStatusFactory(): Heaps.GCStatus;
  646. VAR gcStatusExt : GCStatusExt;
  647. BEGIN
  648. ASSERT( Heaps.gcStatus = NIL );
  649. NEW( gcStatusExt );
  650. RETURN gcStatusExt
  651. END GCStatusFactory;
  652. BEGIN
  653. Init;
  654. END Objects.