Generic.Unix.Objects.Mod 25 KB

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