Obsd.linKernel.txt 74 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573
  1. MODULE Kernel;
  2. (* THIS IS TEXT COPY OF Obsd.linKernel.odc *)
  3. (* DO NOT EDIT *)
  4. (* TODO: Stack overflow is not cought *)
  5. IMPORT SYSTEM, LinDl, LinLibc;
  6. CONST
  7. dllMem = TRUE; (* should be a variable, but for easier memory managment it is always true. *)
  8. strictStackSweep = TRUE;
  9. nameLen* = 256;
  10. littleEndian* = TRUE;
  11. timeResolution* = 1000; (* ticks per second *)
  12. processor* = 10; (* i386 *)
  13. objType* = "ocf"; (* file types *)
  14. symType* = "osf";
  15. docType* = "odc";
  16. (* loader constants *)
  17. done* = 0;
  18. fileNotFound* = 1;
  19. syntaxError* = 2;
  20. objNotFound* = 3;
  21. illegalFPrint* = 4;
  22. cyclicImport* = 5;
  23. noMem* = 6;
  24. commNotFound* = 7;
  25. commSyntaxError* = 8;
  26. moduleNotFound* = 9;
  27. any = 1000000;
  28. CX = 1;
  29. SP = 4; (* register number of stack pointer *)
  30. FP = 5; (* register number of frame pointer *)
  31. ML = 3; (* register which holds the module list at program start *)
  32. N = 128 DIV 16; (* free lists *)
  33. (* kernel flags in module desc *)
  34. init = 16; dyn = 17; dll = 24; iptrs = 30;
  35. (* meta interface consts *)
  36. mConst = 1; mTyp = 2; mVar = 3; mProc = 4; mField = 5;
  37. debug = FALSE;
  38. trapReturn = 1; (* Return value for sigsetjmp given from siglongjmp *)
  39. (* constants for the message boxes *)
  40. mbClose* = -1; mbOk* = 0; mbCancel* =1; mbRetry* = 2; mbIgnore* = 3; mbYes* = 4; mbNo* = 5;
  41. TYPE
  42. Name* = ARRAY nameLen OF SHORTCHAR;
  43. Command* = PROCEDURE;
  44. Module* = POINTER TO RECORD [untagged]
  45. next-: Module;
  46. opts-: SET; (* 0..15: compiler opts, 16..31: kernel flags *)
  47. refcnt-: INTEGER; (* <0: module invalidated *)
  48. compTime-, loadTime-: ARRAY 6 OF SHORTINT;
  49. ext-: INTEGER; (* currently not used *)
  50. term-: Command; (* terminator *)
  51. nofimps-, nofptrs-: INTEGER;
  52. csize-, dsize-, rsize-: INTEGER;
  53. code-, data-, refs-: INTEGER;
  54. procBase-, varBase-: INTEGER; (* meta base addresses *)
  55. names-: POINTER TO ARRAY [untagged] OF SHORTCHAR; (* names[0] = 0X *)
  56. ptrs-: POINTER TO ARRAY [untagged] OF INTEGER;
  57. imports-: POINTER TO ARRAY [untagged] OF Module;
  58. export-: Directory; (* exported objects (name sorted) *)
  59. name-: Name
  60. END;
  61. Type* = POINTER TO RECORD [untagged]
  62. (* record: ptr to method n at offset - 4 * (n+1) *)
  63. size-: INTEGER; (* record: size, array: #elem, dyn array: 0, proc: sigfp *)
  64. mod-: Module;
  65. id-: INTEGER; (* name idx * 256 + lev * 16 + attr * 4 + form *)
  66. base-: ARRAY 16 OF Type;
  67. fields-: Directory; (* new fields (declaration order) *)
  68. ptroffs-: ARRAY any OF INTEGER (* array of any length *)
  69. END;
  70. Object* = POINTER TO ObjDesc;
  71. ObjDesc* = RECORD [untagged]
  72. fprint-: INTEGER;
  73. offs-: INTEGER; (* pvfprint for record types *)
  74. id-: INTEGER; (* name idx * 256 + vis * 16 + mode *)
  75. struct-: Type (* id of basic type or pointer to typedesc *)
  76. END;
  77. Directory* = POINTER TO RECORD [untagged]
  78. num-: INTEGER; (* number of entries *)
  79. obj-: ARRAY any OF ObjDesc (* array of any length *)
  80. END;
  81. Signature* = POINTER TO RECORD [untagged]
  82. retStruct-: Type; (* id of basic type or pointer to typedesc or 0 *)
  83. num-: INTEGER; (* number of parameters *)
  84. par-: ARRAY any OF RECORD [untagged] (* parameters *)
  85. id-: INTEGER; (* name idx * 256 + kind *)
  86. struct-: Type (* id of basic type or pointer to typedesc *)
  87. END
  88. END;
  89. Handler* = PROCEDURE;
  90. Reducer* = POINTER TO ABSTRACT RECORD
  91. next: Reducer
  92. END;
  93. Identifier* = ABSTRACT RECORD
  94. typ*: INTEGER;
  95. obj-: ANYPTR
  96. END;
  97. TrapCleaner* = POINTER TO ABSTRACT RECORD
  98. next: TrapCleaner
  99. END;
  100. TryHandler* = PROCEDURE (a, b, c: INTEGER);
  101. (* meta extension suport *)
  102. ItemExt* = POINTER TO ABSTRACT RECORD END;
  103. ItemAttr* = RECORD
  104. obj*, vis*, typ*, adr*: INTEGER;
  105. mod*: Module;
  106. desc*: Type;
  107. ptr*: SYSTEM.PTR;
  108. ext*: ItemExt
  109. END;
  110. Hook* = POINTER TO ABSTRACT RECORD END;
  111. LoaderHook* = POINTER TO ABSTRACT RECORD (Hook)
  112. res*: INTEGER;
  113. importing*, imported*, object*: ARRAY 256 OF CHAR
  114. END;
  115. GuiHook* = POINTER TO ABSTRACT RECORD (Hook) END; (* Implemented by HostGnome *)
  116. Block = POINTER TO RECORD [untagged]
  117. tag: Type;
  118. last: INTEGER; (* arrays: last element *)
  119. actual: INTEGER; (* arrays: used during mark phase *)
  120. first: INTEGER (* arrays: first element *)
  121. END;
  122. FreeBlock = POINTER TO FreeDesc;
  123. FreeDesc = RECORD [untagged]
  124. tag: Type; (* f.tag = ADR(f.size) *)
  125. size: INTEGER;
  126. next: FreeBlock
  127. END;
  128. Cluster = POINTER TO RECORD [untagged]
  129. size: INTEGER; (* total size *)
  130. next: Cluster;
  131. max: INTEGER
  132. (* start of first block *)
  133. END;
  134. FList = POINTER TO RECORD
  135. next: FList;
  136. blk: Block;
  137. iptr, aiptr: BOOLEAN
  138. END;
  139. CList = POINTER TO RECORD
  140. next: CList;
  141. do: Command;
  142. trapped: BOOLEAN
  143. END;
  144. PtrType = RECORD v: SYSTEM.PTR END; (* used for array of pointer *)
  145. Char8Type = RECORD v: SHORTCHAR END;
  146. Char16Type = RECORD v: CHAR END;
  147. Int8Type = RECORD v: BYTE END;
  148. Int16Type = RECORD v: SHORTINT END;
  149. Int32Type = RECORD v: INTEGER END;
  150. Int64Type = RECORD v: LONGINT END;
  151. BoolType = RECORD v: BOOLEAN END;
  152. SetType = RECORD v: SET END;
  153. Real32Type = RECORD v: SHORTREAL END;
  154. Real64Type = RECORD v: REAL END;
  155. ProcType = RECORD v: PROCEDURE END;
  156. UPtrType = RECORD v: INTEGER END;
  157. StrPtr = POINTER TO ARRAY [untagged] OF SHORTCHAR;
  158. (*
  159. IntPtrType = RECORD p: COM.IUnknown END; (* used for array of interface pointer *)
  160. IntPtr = POINTER TO RECORD [untagged] p: COM.IUnknown END;
  161. ExcpFramePtr = POINTER TO RECORD (KERNEL32.ExcpFrm)
  162. par: INTEGER
  163. END;
  164. Interface = POINTER TO RECORD (* COMPILER DEPENDENT *)
  165. vtab: INTEGER;
  166. ref: INTEGER; (* must correspond to Block.actual *)
  167. unk: COM.IUnknown
  168. END;
  169. *)
  170. (* Linux specific boot loader info. Record must be identical to struct in the loader. *)
  171. BootInfo* = POINTER TO RECORD [untagged]
  172. modList: Module;
  173. argc-: INTEGER;
  174. argv-: LinLibc.StrArray
  175. END;
  176. VAR
  177. baseStack: INTEGER; (* modList, root, and baseStack must be together for remote debugging *)
  178. root: Cluster; (* cluster list *)
  179. modList-: Module; (* root of module list *)
  180. trapCount-: INTEGER;
  181. err-, pc-, sp-, fp-, stack-, val-: INTEGER;
  182. comSig-: INTEGER; (* command signature *)
  183. free: ARRAY N OF FreeBlock; (* free list *)
  184. sentinelBlock: FreeDesc;
  185. sentinel: FreeBlock;
  186. candidates: ARRAY 1024 OF INTEGER;
  187. nofcand: INTEGER;
  188. allocated: INTEGER; (* bytes allocated on BlackBox heap *)
  189. total: INTEGER; (* current total size of BlackBox heap *)
  190. used: INTEGER; (* bytes allocated on system heap *)
  191. finalizers: FList;
  192. hotFinalizers: FList;
  193. cleaners: CList;
  194. reducers: Reducer;
  195. trapStack: TrapCleaner;
  196. actual: Module; (* valid during module initialization *)
  197. res: INTEGER; (* auxiliary global variables used for trap handling *)
  198. old: SET;
  199. trapViewer, trapChecker: Handler;
  200. trapped, guarded, secondTrap: BOOLEAN;
  201. interrupted: BOOLEAN;
  202. static, inDll, terminating: BOOLEAN;
  203. retAd: INTEGER;
  204. restart: Command;
  205. (*
  206. heap: LinLibc.PtrVoid; (*heap: KERNEL32.Handle;*)
  207. excpPtr: KERNEL32.ExcpFrmPtr;
  208. mainThread: KERNEL32.Handle;
  209. *)
  210. told, shift: INTEGER; (* used in Time() *)
  211. loader: LoaderHook;
  212. loadres: INTEGER;
  213. wouldFinalize: BOOLEAN;
  214. watcher*: PROCEDURE (event: INTEGER); (* for debug *)
  215. loopContext: LinLibc.sigjmp_buf; (* trap return context, if no Kernel.Try has been used. *)
  216. currentTryContext: POINTER TO LinLibc.sigjmp_buf; (* trap return context, if Kernel.Try has been used. *)
  217. guiHook: GuiHook;
  218. cmdLine-: ARRAY 1024 OF CHAR;
  219. (* !!! This variable has to be the last variable in the list. !!! *)
  220. bootInfo-: BootInfo;
  221. (* code procedures for exception handling *)
  222. PROCEDURE [1] PushFP 055H;
  223. PROCEDURE [1] PopFP 05DH;
  224. PROCEDURE [1] PushBX 053H;
  225. PROCEDURE [1] PopBX 05BH;
  226. PROCEDURE [1] PushSI 056H;
  227. PROCEDURE [1] PopSI 05EH;
  228. PROCEDURE [1] PushDI 057H;
  229. PROCEDURE [1] PopDI 05FH;
  230. PROCEDURE [1] LdSP8 08DH, 065H, 0F8H;
  231. PROCEDURE [1] Return0 (ret: INTEGER) 0C3H;
  232. PROCEDURE [1] ReturnCX (ret: INTEGER) 05AH, 001H, 0CCH, 0FFH, 0E2H; (* POP DX; ADD SP,CX; JP DX *)
  233. PROCEDURE [1] FPageWord (offs: INTEGER): INTEGER 64H, 8BH, 0H; (* MOV EAX,FS:[EAX] *)
  234. (* code procedures for fpu *)
  235. PROCEDURE [1] FINIT 0DBH, 0E3H;
  236. PROCEDURE [1] FLDCW 0D9H, 06DH, 0FCH; (* -4, FP *)
  237. PROCEDURE [1] FSTCW 0D9H, 07DH, 0FCH; (* -4, FP *)
  238. (* code procedure for memory erase *)
  239. PROCEDURE [code] Erase (adr, words: INTEGER)
  240. 089H, 0C7H, (* MOV EDI, EAX *)
  241. 031H, 0C0H, (* XOR EAX, EAX *)
  242. 059H, (* POP ECX *)
  243. 0F2H, 0ABH; (* REP STOS *)
  244. (* code procedure for stack allocate *)
  245. PROCEDURE [code] ALLOC (* argument in CX *)
  246. (*
  247. PUSH EAX
  248. ADD ECX,-5
  249. JNS L0
  250. XOR ECX,ECX
  251. L0: AND ECX,-4 (n-8+3)/4*4
  252. MOV EAX,ECX
  253. AND EAX,4095
  254. SUB ESP,EAX
  255. MOV EAX,ECX
  256. SHR EAX,12
  257. JEQ L2
  258. L1: PUSH 0
  259. SUB ESP,4092
  260. DEC EAX
  261. JNE L1
  262. L2: ADD ECX,8
  263. MOV EAX,[ESP,ECX,-4]
  264. PUSH EAX
  265. MOV EAX,[ESP,ECX,-4]
  266. SHR ECX,2
  267. RET
  268. *);
  269. (* code procedures for COM support *)
  270. PROCEDURE [code] ADDREF
  271. (*
  272. MOV ECX,[ESP,4]
  273. INC [ECX,4]
  274. MOV EAX,[ECX,8]
  275. OR EAX,EAX
  276. JE L1
  277. PUSH EAX
  278. MOV EAX,[EAX]
  279. CALL [EAX,4]
  280. MOV ECX,[ESP,4]
  281. L1: MOV EAX,[ECX,4]
  282. RET 4
  283. *)
  284. 08BH, 04CH, 024H, 004H,
  285. 0FFH, 041H, 004H,
  286. 08BH, 041H, 008H,
  287. 009H, 0C0H,
  288. 074H, 00AH,
  289. 050H,
  290. 08BH, 000H,
  291. 0FFH, 050H, 004H,
  292. 08BH, 04CH, 024H, 004H,
  293. 08BH, 041H, 004H,
  294. 0C2H, 004H, 000H;
  295. PROCEDURE [code] RELEASE
  296. (*
  297. MOV ECX,[ESP,4]
  298. MOV EAX,[ECX,8]
  299. OR EAX,EAX
  300. JE L1
  301. PUSH EAX
  302. MOV EAX,[EAX]
  303. CALL [EAX,8]
  304. MOV ECX,[ESP,4]
  305. L1: DEC [ECX,4]
  306. MOV EAX,[ECX,4]
  307. RET 4
  308. *)
  309. 08BH, 04CH, 024H, 004H,
  310. 08BH, 041H, 008H,
  311. 009H, 0C0H,
  312. 074H, 00AH,
  313. 050H,
  314. 08BH, 000H,
  315. 0FFH, 050H, 008H,
  316. 08BH, 04CH, 024H, 004H,
  317. 0FFH, 049H, 004H,
  318. 08BH, 041H, 004H,
  319. 0C2H, 004H, 000H;
  320. PROCEDURE [code] CALLREL
  321. (*
  322. MOV EAX,[ESP,4]
  323. CMP [EAX,4],1
  324. JNE L1
  325. PUSH ESI
  326. PUSH EDI
  327. PUSH EAX
  328. MOV EAX,[EAX,-4]
  329. CALL [EAX,-8]
  330. POP EDI
  331. POP ESI
  332. L1:
  333. *)
  334. 08BH, 044H, 024H, 004H,
  335. 083H, 078H, 004H, 001H,
  336. 075H, 00BH,
  337. 056H,
  338. 057H,
  339. 050H,
  340. 08BH, 040H, 0FCH,
  341. 0FFH, 050H, 0F8H,
  342. 05FH,
  343. 05EH;
  344. PROCEDURE (VAR id: Identifier) Identified* (): BOOLEAN, NEW, ABSTRACT;
  345. PROCEDURE (r: Reducer) Reduce* (full: BOOLEAN), NEW, ABSTRACT;
  346. PROCEDURE (c: TrapCleaner) Cleanup*, NEW, EMPTY;
  347. (* meta extension suport *)
  348. PROCEDURE (e: ItemExt) Lookup* (name: ARRAY OF CHAR; VAR i: ANYREC), NEW, ABSTRACT;
  349. PROCEDURE (e: ItemExt) Index* (index: INTEGER; VAR elem: ANYREC), NEW, ABSTRACT;
  350. PROCEDURE (e: ItemExt) Deref* (VAR ref: ANYREC), NEW, ABSTRACT;
  351. PROCEDURE (e: ItemExt) Valid* (): BOOLEAN, NEW, ABSTRACT;
  352. PROCEDURE (e: ItemExt) Size* (): INTEGER, NEW, ABSTRACT;
  353. PROCEDURE (e: ItemExt) BaseTyp* (): INTEGER, NEW, ABSTRACT;
  354. PROCEDURE (e: ItemExt) Len* (): INTEGER, NEW, ABSTRACT;
  355. PROCEDURE (e: ItemExt) Call* (OUT ok: BOOLEAN), NEW, ABSTRACT;
  356. PROCEDURE (e: ItemExt) BoolVal* (): BOOLEAN, NEW, ABSTRACT;
  357. PROCEDURE (e: ItemExt) PutBoolVal* (x: BOOLEAN), NEW, ABSTRACT;
  358. PROCEDURE (e: ItemExt) CharVal* (): CHAR, NEW, ABSTRACT;
  359. PROCEDURE (e: ItemExt) PutCharVal* (x: CHAR), NEW, ABSTRACT;
  360. PROCEDURE (e: ItemExt) IntVal* (): INTEGER, NEW, ABSTRACT;
  361. PROCEDURE (e: ItemExt) PutIntVal* (x: INTEGER), NEW, ABSTRACT;
  362. PROCEDURE (e: ItemExt) LongVal* (): LONGINT, NEW, ABSTRACT;
  363. PROCEDURE (e: ItemExt) PutLongVal* (x: LONGINT), NEW, ABSTRACT;
  364. PROCEDURE (e: ItemExt) RealVal* (): REAL, NEW, ABSTRACT;
  365. PROCEDURE (e: ItemExt) PutRealVal* (x: REAL), NEW, ABSTRACT;
  366. PROCEDURE (e: ItemExt) SetVal* (): SET, NEW, ABSTRACT;
  367. PROCEDURE (e: ItemExt) PutSetVal* (x: SET), NEW, ABSTRACT;
  368. PROCEDURE (e: ItemExt) PtrVal* (): ANYPTR, NEW, ABSTRACT;
  369. PROCEDURE (e: ItemExt) PutPtrVal* (x: ANYPTR), NEW, ABSTRACT;
  370. PROCEDURE (e: ItemExt) GetSStringVal* (
  371. OUT x: ARRAY OF SHORTCHAR; OUT ok: BOOLEAN), NEW, ABSTRACT;
  372. PROCEDURE (e: ItemExt) PutSStringVal* (
  373. IN x: ARRAY OF SHORTCHAR; OUT ok: BOOLEAN), NEW, ABSTRACT;
  374. PROCEDURE (e: ItemExt) GetStringVal* (OUT x: ARRAY OF CHAR; OUT ok: BOOLEAN), NEW, ABSTRACT;
  375. PROCEDURE (e: ItemExt) PutStringVal* (IN x: ARRAY OF CHAR; OUT ok: BOOLEAN), NEW, ABSTRACT;
  376. (* -------------------- miscellaneous tools -------------------- *)
  377. PROCEDURE Msg (IN str: ARRAY OF CHAR);
  378. VAR ss: ARRAY 1024 OF SHORTCHAR; res, l: INTEGER;
  379. BEGIN
  380. ss := SHORT(str);
  381. l := LEN(ss$);
  382. ss[l] := 0AX; ss[l + 1] := 0X;
  383. res := LinLibc.printf(ss);
  384. END Msg;
  385. PROCEDURE Int (x: LONGINT);
  386. VAR j, k: INTEGER; ch: CHAR; a, s: ARRAY 32 OF CHAR;
  387. BEGIN
  388. IF x # MIN(LONGINT) THEN
  389. IF x < 0 THEN s[0] := "-"; k := 1; x := -x ELSE k := 0 END;
  390. j := 0; REPEAT a[j] := CHR(x MOD 10 + ORD("0")); x := x DIV 10; INC(j) UNTIL x = 0
  391. ELSE
  392. a := "8085774586302733229"; s[0] := "-"; k := 1;
  393. j := 0; WHILE a[j] # 0X DO INC(j) END
  394. END;
  395. ASSERT(k + j < LEN(s), 20);
  396. REPEAT DEC(j); ch := a[j]; s[k] := ch; INC(k) UNTIL j = 0;
  397. s[k] := 0X;
  398. Msg(s);
  399. END Int;
  400. PROCEDURE (h: GuiHook) MessageBox* (
  401. title, msg: ARRAY OF CHAR; buttons: SET): INTEGER, NEW, ABSTRACT;
  402. PROCEDURE (h: GuiHook) Beep*, NEW, ABSTRACT;
  403. (* Is extended by HostGnome to show dialogs. If no dialog is present or
  404. if the dialog is not closed by using one button, then "mbClose" is returned *)
  405. PROCEDURE MessageBox* (title, msg: ARRAY OF CHAR; buttons: SET): INTEGER;
  406. VAR res: INTEGER;
  407. BEGIN
  408. IF guiHook # NIL THEN
  409. res := guiHook.MessageBox(title, msg, buttons)
  410. ELSE
  411. Msg(" ");
  412. Msg("****");
  413. Msg("* " + title);
  414. Msg("* " + msg);
  415. Msg("****");
  416. res := mbClose;
  417. END;
  418. RETURN res
  419. END MessageBox;
  420. PROCEDURE SetGuiHook* (hook: GuiHook);
  421. BEGIN
  422. guiHook := hook
  423. END SetGuiHook;
  424. PROCEDURE SplitName* (name: ARRAY OF CHAR; VAR head, tail: ARRAY OF CHAR);
  425. (* portable *)
  426. VAR i, j: INTEGER; ch, lch: CHAR;
  427. BEGIN
  428. i := 0; ch := name[0];
  429. REPEAT
  430. head[i] := ch; lch := ch; INC(i); ch := name[i]
  431. UNTIL (ch = 0X)
  432. OR ((ch >= "A") & (ch <= "Z") OR (ch >= "À") & (ch # "×") & (ch <= "Þ"))
  433. & ((lch < "A") OR (lch > "Z") & (lch < "À") OR (lch = "×") OR (lch > "Þ"));
  434. head[i] := 0X; j := 0;
  435. WHILE ch # 0X DO tail[j] := ch; INC(i); INC(j); ch := name[i] END;
  436. tail[j] := 0X;
  437. IF tail = "" THEN tail := head$; head := "" END
  438. END SplitName;
  439. PROCEDURE MakeFileName* (VAR name: ARRAY OF CHAR; type: ARRAY OF CHAR);
  440. VAR i, j: INTEGER; ext: ARRAY 8 OF CHAR; ch: CHAR;
  441. BEGIN
  442. i := 0;
  443. WHILE (name[i] # 0X) & (name[i] # ".") DO INC(i) END;
  444. IF name[i] = "." THEN
  445. IF name[i + 1] = 0X THEN name[i] := 0X END
  446. ELSIF i < LEN(name) - 4 THEN
  447. IF type = "" THEN ext := docType ELSE ext := type$ END;
  448. name[i] := "."; INC(i); j := 0; ch := ext[0];
  449. WHILE ch # 0X DO
  450. IF (ch >= "A") & (ch <= "Z") THEN
  451. ch := CHR(ORD(ch) + ORD("a") - ORD("A"))
  452. END;
  453. name[i] := ch; INC(i); INC(j); ch := ext[j]
  454. END;
  455. name[i] := 0X
  456. END
  457. END MakeFileName;
  458. PROCEDURE Time* (): LONGINT;
  459. VAR t: INTEGER;
  460. BEGIN
  461. (* A. V. Shiryaev *)
  462. (* processor time to milliseconds *)
  463. t := (1000 * LinLibc.clock()) DIV LinLibc.CLOCKS_PER_SEC;
  464. IF t < told THEN INC(shift) END;
  465. told := t;
  466. RETURN shift * 100000000L + t
  467. END Time;
  468. PROCEDURE Beep* ();
  469. VAR ss: ARRAY 2 OF SHORTCHAR;
  470. BEGIN
  471. IF guiHook # NIL THEN
  472. guiHook.Beep
  473. ELSE
  474. ss[0] := 007X; ss[1] := 0X;
  475. res := LinLibc.printf(ss); res := LinLibc.fflush(LinLibc.NULL)
  476. END
  477. END Beep;
  478. PROCEDURE SearchProcVar* (var: INTEGER; VAR m: Module; VAR adr: INTEGER);
  479. BEGIN
  480. adr := var; m := NIL;
  481. IF var # 0 THEN
  482. m := modList;
  483. WHILE (m # NIL) & ((var < m.code) OR (var >= m.code + m.csize)) DO m := m.next END;
  484. IF m # NIL THEN DEC(adr, m.code) END
  485. END
  486. END SearchProcVar;
  487. (* -------------------- system memory management --------------------- *)
  488. PROCEDURE GrowHeapMem (size: INTEGER; VAR c: Cluster);
  489. (* grow to at least size bytes, typically at least 256 kbytes are allocated *)
  490. CONST N = 262144;
  491. VAR adr, s: INTEGER;
  492. BEGIN
  493. ASSERT(size >= c.size, 100);
  494. IF size <= c.max THEN
  495. s := (size + (N - 1)) DIV N * N;
  496. (*
  497. adr := KERNEL32.VirtualAlloc(SYSTEM.VAL(INTEGER, c), s, {12}, {6}); (* commit; exec, read, write *)
  498. *)
  499. adr := LinLibc.calloc(1, s);
  500. IF adr # 0 THEN
  501. INC(used, s - c.size); INC(total, s - c.size); c.size := s
  502. END
  503. END
  504. (* post: (c.size unchanged) OR (c.size >= size) *)
  505. END GrowHeapMem;
  506. PROCEDURE AllocHeapMem (size: INTEGER; VAR c: Cluster);
  507. (* allocate at least size bytes, typically at least 256 kbytes are allocated *)
  508. CONST M = 1536 * 100000H; (* 1.5 GByte *)
  509. CONST N = 65536; (* cluster size for dll *)
  510. VAR adr, s: INTEGER;
  511. BEGIN
  512. IF dllMem THEN
  513. INC(size, 16);
  514. ASSERT(size > 0, 100); adr := 0;
  515. (*
  516. IF size < N THEN adr := KERNEL32.HeapAlloc(heap, {0}, N) END;
  517. IF adr = 0 THEN adr := KERNEL32.HeapAlloc(heap, {0}, size) END;
  518. *)
  519. IF size < N THEN adr := LinLibc.calloc(1, N) END;
  520. IF adr = 0 THEN adr := LinLibc.calloc(1, size)
  521. ELSE size := N
  522. END;
  523. IF adr = 0 THEN c := NIL
  524. ELSE
  525. c := SYSTEM.VAL(Cluster, ((adr + 15) DIV 16) * 16); c.max := adr;
  526. (*
  527. c.size := KERNEL32.HeapSize(heap, {0}, adr) - (SYSTEM.VAL(INTEGER, c) - adr);
  528. *)
  529. c.size := size - (SYSTEM.VAL(INTEGER, c) - adr);
  530. INC(used, c.size); INC(total, c.size)
  531. END;
  532. ELSE
  533. adr := 0; s := M;
  534. REPEAT
  535. (*
  536. adr := KERNEL32.VirtualAlloc(01000000H, s, {13}, {6}); (* reserve; exec, read, write *)
  537. *)
  538. IF adr = 0 THEN
  539. (*
  540. adr := KERNEL32.VirtualAlloc(0, s, {13}, {6}) (* reserve; exec, read, write *)
  541. *)
  542. END;
  543. s := s DIV 2
  544. UNTIL adr # 0;
  545. IF adr = 0 THEN c := NIL
  546. ELSE
  547. (*
  548. adr := KERNEL32.VirtualAlloc(adr, 1024, {12}, {6}); (* commit; exec, read, write *)
  549. *)
  550. c := SYSTEM.VAL(Cluster, adr);
  551. c.max := s * 2; c.size := 0; c.next := NIL;
  552. GrowHeapMem(size, c);
  553. IF c.size < size THEN c := NIL END
  554. END
  555. END
  556. (* post: (c = NIL) OR (c MOD 16 = 0) & (c.size >= size) *)
  557. END AllocHeapMem;
  558. PROCEDURE FreeHeapMem (c: Cluster);
  559. VAR res: INTEGER;
  560. BEGIN
  561. DEC(used, c.size); DEC(total, c.size);
  562. IF dllMem THEN
  563. (*
  564. res := KERNEL32.HeapFree(heap, {0}, c.max)
  565. *)
  566. LinLibc.free(c.max)
  567. END
  568. END FreeHeapMem;
  569. PROCEDURE HeapFull (size: INTEGER): BOOLEAN;
  570. (*
  571. VAR ms: KERNEL32.MemStatus;
  572. *)
  573. BEGIN
  574. RETURN used + size > 4000000 (* TODO: Do this right!!! Well, maybe not, since it isn't used for dllMem *)
  575. (*
  576. ms.size := SIZE(KERNEL32.MemStatus);
  577. ms.memLoad := -1;
  578. KERNEL32.GlobalMemoryStatus(ms);
  579. IF ms.memLoad >= 0 THEN
  580. RETURN used + size > ms.totPhys
  581. ELSE (* old win32s *)
  582. RETURN used + size > 4000000
  583. END
  584. *)
  585. END HeapFull;
  586. PROCEDURE AllocModMem* (descSize, modSize: INTEGER; VAR descAdr, modAdr: INTEGER);
  587. VAR res: INTEGER;
  588. BEGIN
  589. (*
  590. descAdr := KERNEL32.VirtualAlloc(0, descSize, {12, 13}, {6}); (* reserve & commit; exec, read, write *)
  591. IF descAdr # 0 THEN
  592. modAdr := KERNEL32.VirtualAlloc(0, modSize, {12, 13}, {6}); (* reserve & commit; exec, read, write *)
  593. IF modAdr # 0 THEN INC(used, descSize + modSize)
  594. ELSE res := KERNEL32.VirtualFree(descAdr, 0, {15}); descAdr := 0
  595. END
  596. ELSE modAdr := 0
  597. END
  598. *)
  599. descAdr := LinLibc.calloc(1, descSize);
  600. IF descAdr # LinLibc.NULL THEN
  601. modAdr := LinLibc.calloc(1, modSize);
  602. IF modAdr # LinLibc.NULL THEN INC(used, descSize + modSize)
  603. ELSE LinLibc.free(descAdr); descAdr := 0
  604. END
  605. ELSE modAdr := 0
  606. END
  607. END AllocModMem;
  608. PROCEDURE DeallocModMem* (descSize, modSize, descAdr, modAdr: INTEGER);
  609. VAR res: INTEGER;
  610. BEGIN
  611. DEC(used, descSize + modSize);
  612. (*
  613. res := KERNEL32.VirtualFree(descAdr, 0, {15}); (* release *)
  614. res := KERNEL32.VirtualFree(modAdr, 0, {15}) (* release *)
  615. *)
  616. LinLibc.free(descAdr);
  617. LinLibc.free(modAdr)
  618. END DeallocModMem;
  619. PROCEDURE InvalModMem (modSize, modAdr: INTEGER);
  620. VAR res: INTEGER;
  621. BEGIN
  622. DEC(used, modSize);
  623. (*
  624. res := KERNEL32.VirtualFree(modAdr, modSize, {14}) (* decommit *)
  625. *)
  626. LinLibc.free(modAdr)
  627. END InvalModMem;
  628. PROCEDURE IsReadable* (from, to: INTEGER): BOOLEAN;
  629. (* check wether memory between from (incl.) and to (excl.) may be read *)
  630. BEGIN
  631. (*
  632. RETURN KERNEL32.IsBadReadPtr(from, to - from) = 0
  633. *)
  634. RETURN TRUE (* TODO: Do this correct!!! *)
  635. END IsReadable;
  636. (* --------------------- COM reference counting -------------------- *)
  637. PROCEDURE [noframe] AddRef* (p: INTEGER): INTEGER; (* COMPILER DEPENDENT *)
  638. BEGIN
  639. ADDREF
  640. (*
  641. INC(p.ref);
  642. IF p.unk # NIL THEN p.unk.AddRef() END;
  643. RETURN p.ref
  644. *)
  645. END AddRef;
  646. PROCEDURE [noframe] Release* (p: INTEGER): INTEGER; (* COMPILER DEPENDENT *)
  647. BEGIN
  648. RELEASE
  649. (*
  650. IF p.unk # NIL THEN p.unk.Release() END;
  651. DEC(p.ref);
  652. RETURN p.ref
  653. *)
  654. END Release;
  655. PROCEDURE [noframe] Release2* (p: INTEGER): INTEGER; (* COMPILER DEPENDENT *)
  656. BEGIN
  657. CALLREL;
  658. RELEASE
  659. (*
  660. IF p.ref = 1 THEN p.RELEASE END;
  661. IF p.unk # NIL THEN p.unk.Release() END;
  662. DEC(p.ref);
  663. RETURN p.ref
  664. *)
  665. END Release2;
  666. (*
  667. PROCEDURE RecFinalizer (obj: ANYPTR);
  668. VAR i: INTEGER; type: Type; p: IntPtr;
  669. BEGIN
  670. SYSTEM.GET(SYSTEM.VAL(INTEGER, obj) - 4, type);
  671. i := 0;
  672. WHILE type.ptroffs[i] >= 0 DO INC(i) END;
  673. INC(i);
  674. WHILE type.ptroffs[i] >= 0 DO
  675. p := SYSTEM.VAL(IntPtr, SYSTEM.VAL(INTEGER, obj) + type.ptroffs[i]); INC(i);
  676. p.p := NIL (* calls p.p.Release *)
  677. END
  678. END RecFinalizer;
  679. *)
  680. (*
  681. PROCEDURE ArrFinalizer (obj: SYSTEM.PTR);
  682. VAR last, adr, i, j: INTEGER; type: Type; p: IntPtr;
  683. BEGIN
  684. SYSTEM.GET(SYSTEM.VAL(INTEGER, obj) - 4, type);
  685. type := SYSTEM.VAL(Type, SYSTEM.VAL(INTEGER, type) - 2); (* remove array flag *)
  686. SYSTEM.GET(SYSTEM.VAL(INTEGER, obj), last);
  687. SYSTEM.GET(SYSTEM.VAL(INTEGER, obj) + 8, adr);
  688. j := 0;
  689. WHILE type.ptroffs[j] >= 0 DO INC(j) END;
  690. INC(j);
  691. WHILE adr <= last DO
  692. i := j;
  693. WHILE type.ptroffs[i] >= 0 DO
  694. p := SYSTEM.VAL(IntPtr, adr + type.ptroffs[i]); INC(i);
  695. p.p := NIL (* calls p.p.Release *)
  696. END;
  697. INC(adr, type.size)
  698. END
  699. END ArrFinalizer;
  700. *)
  701. (*
  702. PROCEDURE ReleaseIPtrs (mod: Module);
  703. VAR i: INTEGER; p: IntPtr;
  704. BEGIN
  705. IF iptrs IN mod.opts THEN
  706. EXCL(mod.opts, iptrs);
  707. i := mod.nofptrs;
  708. WHILE mod.ptrs[i] # -1 DO
  709. p := SYSTEM.VAL(IntPtr, mod.varBase + mod.ptrs[i]); INC(i);
  710. p.p := NIL (* calls p.p.Release *)
  711. END
  712. END
  713. END ReleaseIPtrs;
  714. *)
  715. (* --------------------- NEW implementation (portable) -------------------- *)
  716. PROCEDURE^ NewBlock (size: INTEGER): Block;
  717. PROCEDURE NewRec* (typ: INTEGER): INTEGER; (* implementation of NEW(ptr) *)
  718. VAR size: INTEGER; b: Block; tag: Type; l: FList;
  719. BEGIN
  720. IF ODD(typ) THEN (* record contains interface pointers *)
  721. tag := SYSTEM.VAL(Type, typ - 1);
  722. b := NewBlock(tag.size);
  723. IF b = NIL THEN RETURN 0 END;
  724. b.tag := tag;
  725. l := SYSTEM.VAL(FList, NewRec(SYSTEM.TYP(FList))); (* NEW(l) *)
  726. l.blk := b; l.iptr := TRUE; l.next := finalizers; finalizers := l;
  727. RETURN SYSTEM.ADR(b.last)
  728. ELSE
  729. tag := SYSTEM.VAL(Type, typ);
  730. b := NewBlock(tag.size);
  731. IF b = NIL THEN RETURN 0 END;
  732. b.tag := tag; SYSTEM.GET(typ - 4, size);
  733. IF size # 0 THEN (* record uses a finalizer *)
  734. l := SYSTEM.VAL(FList, NewRec(SYSTEM.TYP(FList))); (* NEW(l) *)
  735. l.blk := b; l.next := finalizers; finalizers := l
  736. END;
  737. RETURN SYSTEM.ADR(b.last)
  738. END
  739. END NewRec;
  740. PROCEDURE NewArr* (eltyp, nofelem, nofdim: INTEGER): INTEGER; (* impl. of NEW(ptr, dim0, dim1, ...) *)
  741. VAR b: Block; size, headSize: INTEGER; t: Type; fin: BOOLEAN; l: FList;
  742. BEGIN
  743. headSize := 4 * nofdim + 12; fin := FALSE;
  744. CASE eltyp OF
  745. (*
  746. | -1: eltyp := SYSTEM.ADR(IntPtrType); fin := TRUE
  747. *)
  748. | 0: eltyp := SYSTEM.ADR(PtrType)
  749. | 1: eltyp := SYSTEM.ADR(Char8Type)
  750. | 2: eltyp := SYSTEM.ADR(Int16Type)
  751. | 3: eltyp := SYSTEM.ADR(Int8Type)
  752. | 4: eltyp := SYSTEM.ADR(Int32Type)
  753. | 5: eltyp := SYSTEM.ADR(BoolType)
  754. | 6: eltyp := SYSTEM.ADR(SetType)
  755. | 7: eltyp := SYSTEM.ADR(Real32Type)
  756. | 8: eltyp := SYSTEM.ADR(Real64Type)
  757. | 9: eltyp := SYSTEM.ADR(Char16Type)
  758. | 10: eltyp := SYSTEM.ADR(Int64Type)
  759. | 11: eltyp := SYSTEM.ADR(ProcType)
  760. | 12: eltyp := SYSTEM.ADR(UPtrType)
  761. ELSE (* eltyp is desc *)
  762. IF ODD(eltyp) THEN DEC(eltyp); fin := TRUE END
  763. END;
  764. t := SYSTEM.VAL(Type, eltyp);
  765. size := headSize + nofelem * t.size;
  766. b := NewBlock(size);
  767. IF b = NIL THEN RETURN 0 END;
  768. b.tag := SYSTEM.VAL(Type, eltyp + 2); (* tag + array mark *)
  769. b.last := SYSTEM.ADR(b.last) + size - t.size; (* pointer to last elem *)
  770. b.first := SYSTEM.ADR(b.last) + headSize; (* pointer to first elem *)
  771. IF fin THEN
  772. l := SYSTEM.VAL(FList, NewRec(SYSTEM.TYP(FList))); (* NEW(l) *)
  773. l.blk := b; l.aiptr := TRUE; l.next := finalizers; finalizers := l
  774. END;
  775. RETURN SYSTEM.ADR(b.last)
  776. END NewArr;
  777. (* -------------------- handler installation (portable) --------------------- *)
  778. PROCEDURE ThisFinObj* (VAR id: Identifier): ANYPTR;
  779. VAR l: FList;
  780. BEGIN
  781. ASSERT(id.typ # 0, 100); ASSERT(hotFinalizers = NIL, 101);
  782. l := finalizers;
  783. WHILE l # NIL DO
  784. IF SYSTEM.VAL(INTEGER, l.blk.tag) = id.typ THEN
  785. id.obj := SYSTEM.VAL(ANYPTR, SYSTEM.ADR(l.blk.last));
  786. IF id.Identified() THEN RETURN id.obj END
  787. END;
  788. l := l.next
  789. END;
  790. RETURN NIL
  791. END ThisFinObj;
  792. PROCEDURE InstallReducer* (r: Reducer);
  793. BEGIN
  794. r.next := reducers; reducers := r
  795. END InstallReducer;
  796. PROCEDURE InstallTrapViewer* (h: Handler);
  797. BEGIN
  798. trapViewer := h
  799. END InstallTrapViewer;
  800. PROCEDURE InstallTrapChecker* (h: Handler);
  801. BEGIN
  802. trapChecker := h
  803. END InstallTrapChecker;
  804. PROCEDURE PushTrapCleaner* (c: TrapCleaner);
  805. VAR t: TrapCleaner;
  806. BEGIN
  807. t := trapStack; WHILE (t # NIL) & (t # c) DO t := t.next END;
  808. ASSERT(t = NIL, 20);
  809. c.next := trapStack; trapStack := c
  810. END PushTrapCleaner;
  811. PROCEDURE PopTrapCleaner* (c: TrapCleaner);
  812. VAR t: TrapCleaner;
  813. BEGIN
  814. t := NIL;
  815. WHILE (trapStack # NIL) & (t # c) DO
  816. t := trapStack; trapStack := trapStack.next
  817. END
  818. END PopTrapCleaner;
  819. PROCEDURE InstallCleaner* (p: Command);
  820. VAR c: CList;
  821. BEGIN
  822. c := SYSTEM.VAL(CList, NewRec(SYSTEM.TYP(CList))); (* NEW(c) *)
  823. c.do := p; c.trapped := FALSE; c.next := cleaners; cleaners := c
  824. END InstallCleaner;
  825. PROCEDURE RemoveCleaner* (p: Command);
  826. VAR c0, c: CList;
  827. BEGIN
  828. c := cleaners; c0 := NIL;
  829. WHILE (c # NIL) & (c.do # p) DO c0 := c; c := c.next END;
  830. IF c # NIL THEN
  831. IF c0 = NIL THEN cleaners := cleaners.next ELSE c0.next := c.next END
  832. END
  833. END RemoveCleaner;
  834. PROCEDURE Cleanup*;
  835. VAR c, c0: CList;
  836. BEGIN
  837. c := cleaners; c0 := NIL;
  838. WHILE c # NIL DO
  839. IF ~c.trapped THEN
  840. c.trapped := TRUE; c.do; c.trapped := FALSE; c0 := c
  841. ELSE
  842. IF c0 = NIL THEN cleaners := cleaners.next
  843. ELSE c0.next := c.next
  844. END
  845. END;
  846. c := c.next
  847. END
  848. END Cleanup;
  849. (* -------------------- meta information (portable) --------------------- *)
  850. PROCEDURE (h: LoaderHook) ThisMod* (IN name: ARRAY OF SHORTCHAR): Module, NEW, ABSTRACT;
  851. PROCEDURE SetLoaderHook*(h: LoaderHook);
  852. BEGIN
  853. loader := h
  854. END SetLoaderHook;
  855. PROCEDURE InitModule (mod: Module); (* initialize linked modules *)
  856. VAR body: Command;
  857. res, errno: INTEGER;
  858. BEGIN
  859. IF ~(dyn IN mod.opts) & (mod.next # NIL) & ~(init IN mod.next.opts) THEN InitModule(mod.next) END;
  860. IF ~(init IN mod.opts) THEN
  861. body := SYSTEM.VAL(Command, mod.code);
  862. INCL(mod.opts, init);
  863. actual := mod;
  864. (* A. V. Shiryaev: OpenBSD-specific *)
  865. (*
  866. res := LinLibc.mprotect(
  867. (mod.code DIV LinLibc.PAGE_SIZE) * LinLibc.PAGE_SIZE,
  868. ((mod.csize + mod.code MOD LinLibc.PAGE_SIZE - 1) DIV LinLibc.PAGE_SIZE) * LinLibc.PAGE_SIZE + LinLibc.PAGE_SIZE,
  869. LinLibc.PROT_READ + LinLibc.PROT_WRITE + LinLibc.PROT_EXEC);
  870. *)
  871. res := LinLibc.mprotect(mod.code, mod.csize,
  872. LinLibc.PROT_READ + LinLibc.PROT_WRITE + LinLibc.PROT_EXEC);
  873. IF res = -1 THEN
  874. SYSTEM.GET( LinLibc.__errno_location(), errno );
  875. res := LinLibc.printf("Kernel.InitModule('"); res := LinLibc.printf(mod.name);
  876. res := LinLibc.printf("'): mprotect("); Int(mod.code);
  877. res := LinLibc.printf(", "); Int(mod.csize);
  878. res := LinLibc.printf(", R|W|E) failed: errno = "); Int(errno);
  879. res := LinLibc.printf(0AX);
  880. (* HALT(100) *)
  881. ELSE ASSERT(res = 0)
  882. END;
  883. body();
  884. actual := NIL
  885. END
  886. END InitModule;
  887. PROCEDURE ThisLoadedMod* (IN name: ARRAY OF SHORTCHAR): Module; (* loaded modules only *)
  888. VAR m: Module;
  889. BEGIN
  890. loadres := done;
  891. m := modList;
  892. WHILE (m # NIL) & ((m.name # name) OR (m.refcnt < 0)) DO m := m.next END;
  893. IF (m # NIL) & ~(init IN m.opts) THEN InitModule(m) END;
  894. IF m = NIL THEN loadres := moduleNotFound END;
  895. RETURN m
  896. END ThisLoadedMod;
  897. PROCEDURE ThisMod* (IN name: ARRAY OF CHAR): Module;
  898. VAR n : Name;
  899. BEGIN
  900. n := SHORT(name$);
  901. IF loader # NIL THEN
  902. loader.res := done;
  903. RETURN loader.ThisMod(n)
  904. ELSE
  905. RETURN ThisLoadedMod(n)
  906. END
  907. END ThisMod;
  908. PROCEDURE LoadMod* (IN name: ARRAY OF CHAR);
  909. VAR m: Module;
  910. BEGIN
  911. m := ThisMod(name)
  912. END LoadMod;
  913. PROCEDURE GetLoaderResult* (OUT res: INTEGER; OUT importing, imported, object: ARRAY OF CHAR);
  914. BEGIN
  915. IF loader # NIL THEN
  916. res := loader.res;
  917. importing := loader.importing$;
  918. imported := loader.imported$;
  919. object := loader.object$
  920. ELSE
  921. res := loadres;
  922. importing := "";
  923. imported := "";
  924. object := ""
  925. END
  926. END GetLoaderResult;
  927. PROCEDURE ThisObject* (mod: Module; name: ARRAY OF SHORTCHAR): Object;
  928. VAR l, r, m: INTEGER; p: StrPtr;
  929. BEGIN
  930. l := 0; r := mod.export.num;
  931. WHILE l < r DO (* binary search *)
  932. m := (l + r) DIV 2;
  933. p := SYSTEM.VAL(StrPtr, SYSTEM.ADR(mod.names[mod.export.obj[m].id DIV 256]));
  934. IF p^ = name THEN RETURN SYSTEM.VAL(Object, SYSTEM.ADR(mod.export.obj[m])) END;
  935. IF p^ < name THEN l := m + 1 ELSE r := m END
  936. END;
  937. RETURN NIL
  938. END ThisObject;
  939. PROCEDURE ThisDesc* (mod: Module; fprint: INTEGER): Object;
  940. VAR i, n: INTEGER;
  941. BEGIN
  942. i := 0; n := mod.export.num;
  943. WHILE (i < n) & (mod.export.obj[i].id DIV 256 = 0) DO
  944. IF mod.export.obj[i].offs = fprint THEN
  945. RETURN SYSTEM.VAL(Object, SYSTEM.ADR(mod.export.obj[i]))
  946. END;
  947. INC(i)
  948. END;
  949. RETURN NIL
  950. END ThisDesc;
  951. PROCEDURE ThisField* (rec: Type; name: ARRAY OF SHORTCHAR): Object;
  952. VAR n: INTEGER; p: StrPtr; obj: Object; m: Module;
  953. BEGIN
  954. m := rec.mod;
  955. obj := SYSTEM.VAL(Object, SYSTEM.ADR(rec.fields.obj[0])); n := rec.fields.num;
  956. WHILE n > 0 DO
  957. p := SYSTEM.VAL(StrPtr, SYSTEM.ADR(m.names[obj.id DIV 256]));
  958. IF p^ = name THEN RETURN obj END;
  959. DEC(n); INC(SYSTEM.VAL(INTEGER, obj), 16)
  960. END;
  961. RETURN NIL
  962. END ThisField;
  963. (*PROCEDURE ThisCommand* (mod: Module; name: ARRAY OF SHORTCHAR): Command;
  964. VAR x: Object;
  965. BEGIN
  966. x := ThisObject(mod, name);
  967. IF (x # NIL) & (x.id MOD 16 = mProc) & (x.fprint = comSig) THEN
  968. RETURN SYSTEM.VAL(Command, mod.procBase + x.offs)
  969. ELSE
  970. RETURN NIL
  971. END
  972. END ThisCommand;*)
  973. PROCEDURE ThisCommand* (mod: Module; name: ARRAY OF SHORTCHAR): Command;
  974. VAR x: Object; sig: Signature;
  975. BEGIN
  976. x := ThisObject(mod, name);
  977. IF (x # NIL) & (x.id MOD 16 = mProc) THEN
  978. sig := SYSTEM.VAL(Signature, x.struct);
  979. IF (sig.retStruct = NIL) & (sig.num = 0) THEN RETURN SYSTEM.VAL(Command, mod.procBase + x.offs) END
  980. END;
  981. RETURN NIL
  982. END ThisCommand;
  983. PROCEDURE ThisType* (mod: Module; name: ARRAY OF SHORTCHAR): Type;
  984. VAR x: Object;
  985. BEGIN
  986. x := ThisObject(mod, name);
  987. IF (x # NIL) & (x.id MOD 16 = mTyp) & (SYSTEM.VAL(INTEGER, x.struct) DIV 256 # 0) THEN
  988. RETURN x.struct
  989. ELSE
  990. RETURN NIL
  991. END
  992. END ThisType;
  993. PROCEDURE TypeOf* (IN rec: ANYREC): Type;
  994. BEGIN
  995. RETURN SYSTEM.VAL(Type, SYSTEM.TYP(rec))
  996. END TypeOf;
  997. PROCEDURE LevelOf* (t: Type): SHORTINT;
  998. BEGIN
  999. RETURN SHORT(t.id DIV 16 MOD 16)
  1000. END LevelOf;
  1001. PROCEDURE NewObj* (VAR o: SYSTEM.PTR; t: Type);
  1002. VAR i: INTEGER;
  1003. BEGIN
  1004. IF t.size = -1 THEN o := NIL
  1005. ELSE
  1006. i := 0; WHILE t.ptroffs[i] >= 0 DO INC(i) END;
  1007. IF t.ptroffs[i+1] >= 0 THEN INC(SYSTEM.VAL(INTEGER, t)) END; (* with interface pointers *)
  1008. o := SYSTEM.VAL(SYSTEM.PTR, NewRec(SYSTEM.VAL(INTEGER, t))) (* generic NEW *)
  1009. END
  1010. END NewObj;
  1011. PROCEDURE GetObjName* (mod: Module; obj: Object; VAR name: Name);
  1012. VAR p: StrPtr;
  1013. BEGIN
  1014. p := SYSTEM.VAL(StrPtr, SYSTEM.ADR(mod.names[obj.id DIV 256]));
  1015. name := p^$
  1016. END GetObjName;
  1017. PROCEDURE GetTypeName* (t: Type; VAR name: Name);
  1018. VAR p: StrPtr;
  1019. BEGIN
  1020. p := SYSTEM.VAL(StrPtr, SYSTEM.ADR(t.mod.names[t.id DIV 256]));
  1021. name := p^$
  1022. END GetTypeName;
  1023. PROCEDURE RegisterMod* (mod: Module);
  1024. VAR i: INTEGER;(* t: KERNEL32.SystemTime;*) obj: Object; s: SET; c: Command; str: Name;
  1025. t: LinLibc.time_t; tm: LinLibc.tm;
  1026. BEGIN
  1027. mod.next := modList; modList := mod; mod.refcnt := 0; INCL(mod.opts, dyn); i := 0;
  1028. WHILE i < mod.nofimps DO
  1029. IF mod.imports[i] # NIL THEN INC(mod.imports[i].refcnt) END;
  1030. INC(i)
  1031. END;
  1032. t := LinLibc.time(NIL);
  1033. tm := LinLibc.localtime(t);
  1034. mod.loadTime[0] := SHORT(tm.tm_year + 1900); (* Linux counts years from 1900 but BlackBox from 0000 *)
  1035. mod.loadTime[1] := SHORT(tm.tm_mon + 1) (* Linux month range 0-11 but BB month range 1-12 *);
  1036. mod.loadTime[2] := SHORT(tm.tm_mday);
  1037. mod.loadTime[3] := SHORT(tm.tm_hour);
  1038. mod.loadTime[4] := SHORT(tm.tm_min);
  1039. mod.loadTime[5] := SHORT(tm.tm_sec);
  1040. tm := NIL;
  1041. IF ~(init IN mod.opts) THEN InitModule(mod) END
  1042. END RegisterMod;
  1043. PROCEDURE^ Collect*;
  1044. PROCEDURE UnloadMod* (mod: Module);
  1045. VAR i: INTEGER; t: Command;
  1046. BEGIN
  1047. IF mod.refcnt = 0 THEN
  1048. t := mod.term; mod.term := NIL;
  1049. IF t # NIL THEN t() END; (* terminate module *)
  1050. i := 0;
  1051. WHILE i < mod.nofptrs DO (* release global pointers *)
  1052. SYSTEM.PUT(mod.varBase + mod.ptrs[i], 0); INC(i)
  1053. END;
  1054. (*
  1055. ReleaseIPtrs(mod); (* release global interface pointers *)
  1056. *)
  1057. Collect; (* call finalizers *)
  1058. i := 0;
  1059. WHILE i < mod.nofimps DO (* release imported modules *)
  1060. IF mod.imports[i] # NIL THEN DEC(mod.imports[i].refcnt) END;
  1061. INC(i)
  1062. END;
  1063. mod.refcnt := -1;
  1064. IF dyn IN mod.opts THEN (* release memory *)
  1065. InvalModMem(mod.data + mod.dsize - mod.refs, mod.refs)
  1066. END
  1067. END
  1068. END UnloadMod;
  1069. (* -------------------- dynamic procedure call --------------------- *) (* COMPILER DEPENDENT *)
  1070. PROCEDURE [1] PUSH (p: INTEGER) 050H; (* push AX *)
  1071. PROCEDURE [1] CALL (a: INTEGER) 0FFH, 0D0H; (* call AX *)
  1072. PROCEDURE [1] RETI (): LONGINT;
  1073. PROCEDURE [1] RETR (): REAL;
  1074. (*
  1075. type par
  1076. 32 bit scalar value
  1077. 64 bit scalar low hi
  1078. var scalar address
  1079. record address tag
  1080. array address size
  1081. open array address length .. length
  1082. *)
  1083. PROCEDURE Call* (adr: INTEGER; sig: Signature; IN par: ARRAY OF INTEGER; n: INTEGER): LONGINT;
  1084. VAR p, kind, sp, size: INTEGER; typ: Type; r: REAL;
  1085. BEGIN
  1086. p := sig.num;
  1087. WHILE p > 0 DO (* push parameters from right to left *)
  1088. DEC(p);
  1089. typ := sig.par[p].struct;
  1090. kind := sig.par[p].id MOD 16;
  1091. IF (SYSTEM.VAL(INTEGER, typ) DIV 256 = 0) OR (typ.id MOD 4 IN {0, 3}) THEN (* scalar *)
  1092. IF (kind = 10) & ((SYSTEM.VAL(INTEGER, typ) = 8) OR (SYSTEM.VAL(INTEGER, typ) = 10)) THEN (* 64 bit *)
  1093. DEC(n); PUSH(par[n]) (* push hi word *)
  1094. END;
  1095. DEC(n); PUSH(par[n]) (* push value/address *)
  1096. ELSIF typ.id MOD 4 = 1 THEN (* record *)
  1097. IF kind # 10 THEN (* var par *)
  1098. DEC(n); PUSH(par[n]); (* push tag *)
  1099. DEC(n); PUSH(par[n]) (* push address *)
  1100. ELSE
  1101. DEC(n, 2); (* skip tag *)
  1102. SYSTEM.GETREG(SP, sp); sp := (sp - typ.size) DIV 4 * 4; SYSTEM.PUTREG(SP, sp); (* allocate space *)
  1103. SYSTEM.MOVE(par[n], sp, typ.size) (* copy to stack *)
  1104. END
  1105. ELSIF typ.size = 0 THEN (* open array *)
  1106. size := typ.id DIV 16 MOD 16; (* number of open dimensions *)
  1107. WHILE size > 0 DO
  1108. DEC(size); DEC(n); PUSH(par[n]) (* push length *)
  1109. END;
  1110. DEC(n); PUSH(par[n]) (* push address *)
  1111. ELSE (* fix array *)
  1112. IF kind # 10 THEN (* var par *)
  1113. DEC(n, 2); PUSH(par[n]) (* push address *)
  1114. ELSE
  1115. DEC(n); size := par[n]; DEC(n);
  1116. SYSTEM.GETREG(SP, sp); sp := (sp - size) DIV 4 * 4; SYSTEM.PUTREG(SP, sp); (* allocate space *)
  1117. SYSTEM.MOVE(par[n], sp, size) (* copy to stack *)
  1118. END
  1119. END
  1120. END;
  1121. ASSERT(n = 0);
  1122. IF SYSTEM.VAL(INTEGER, sig.retStruct) = 7 THEN (* shortreal *)
  1123. CALL(adr);
  1124. RETURN SYSTEM.VAL(INTEGER, SHORT(RETR())) (* return value in fpu register *)
  1125. ELSIF SYSTEM.VAL(INTEGER, sig.retStruct) = 8 THEN (* real *)
  1126. CALL(adr); r := RETR();
  1127. RETURN SYSTEM.VAL(LONGINT, r) (* return value in fpu register *)
  1128. ELSE
  1129. CALL(adr);
  1130. RETURN RETI() (* return value in integer registers *)
  1131. END
  1132. END Call;
  1133. (* -------------------- reference information (portable) --------------------- *)
  1134. PROCEDURE RefCh (VAR ref: INTEGER; VAR ch: SHORTCHAR);
  1135. BEGIN
  1136. SYSTEM.GET(ref, ch); INC(ref)
  1137. END RefCh;
  1138. PROCEDURE RefNum (VAR ref: INTEGER; VAR x: INTEGER);
  1139. VAR s, n: INTEGER; ch: SHORTCHAR;
  1140. BEGIN
  1141. s := 0; n := 0; RefCh(ref, ch);
  1142. WHILE ORD(ch) >= 128 DO INC(n, ASH(ORD(ch) - 128, s) ); INC(s, 7); RefCh(ref, ch) END;
  1143. x := n + ASH(ORD(ch) MOD 64 - ORD(ch) DIV 64 * 64, s)
  1144. END RefNum;
  1145. PROCEDURE RefName (VAR ref: INTEGER; VAR n: Name);
  1146. VAR i: INTEGER; ch: SHORTCHAR;
  1147. BEGIN
  1148. i := 0; RefCh(ref, ch);
  1149. WHILE ch # 0X DO n[i] := ch; INC(i); RefCh(ref, ch) END;
  1150. n[i] := 0X
  1151. END RefName;
  1152. PROCEDURE GetRefProc* (VAR ref: INTEGER; VAR adr: INTEGER; VAR name: Name);
  1153. VAR ch: SHORTCHAR;
  1154. BEGIN
  1155. SYSTEM.GET(ref, ch);
  1156. WHILE ch >= 0FDX DO (* skip variables *)
  1157. INC(ref); RefCh(ref, ch);
  1158. IF ch = 10X THEN INC(ref, 4) END;
  1159. RefNum(ref, adr); RefName(ref, name); SYSTEM.GET(ref, ch)
  1160. END;
  1161. WHILE (ch > 0X) & (ch < 0FCX) DO (* skip source refs *)
  1162. INC(ref); RefNum(ref, adr); SYSTEM.GET(ref, ch)
  1163. END;
  1164. IF ch = 0FCX THEN INC(ref); RefNum(ref, adr); RefName(ref, name)
  1165. ELSE adr := 0
  1166. END
  1167. END GetRefProc;
  1168. PROCEDURE GetRefVar* (VAR ref: INTEGER; VAR mode, form: SHORTCHAR; VAR desc: Type;
  1169. VAR adr: INTEGER; VAR name: Name
  1170. );
  1171. BEGIN
  1172. SYSTEM.GET(ref, mode); desc := NIL;
  1173. IF mode >= 0FDX THEN
  1174. mode := SHORT(CHR(ORD(mode) - 0FCH));
  1175. INC(ref); RefCh(ref, form);
  1176. IF form = 10X THEN
  1177. SYSTEM.GET(ref, desc); INC(ref, 4); form := SHORT(CHR(16 + desc.id MOD 4))
  1178. END;
  1179. RefNum(ref, adr); RefName(ref, name)
  1180. ELSE
  1181. mode := 0X; form := 0X; adr := 0
  1182. END
  1183. END GetRefVar;
  1184. PROCEDURE SourcePos* (mod: Module; codePos: INTEGER): INTEGER;
  1185. VAR ref, pos, ad, d: INTEGER; ch: SHORTCHAR; name: Name;
  1186. BEGIN
  1187. ref := mod.refs; pos := 0; ad := 0; SYSTEM.GET(ref, ch);
  1188. WHILE ch # 0X DO
  1189. WHILE (ch > 0X) & (ch < 0FCX) DO
  1190. INC(ad, ORD(ch)); INC(ref); RefNum(ref, d);
  1191. IF ad > codePos THEN RETURN pos END;
  1192. INC(pos, d); SYSTEM.GET(ref, ch)
  1193. END;
  1194. IF ch = 0FCX THEN INC(ref); RefNum(ref, d); RefName(ref, name); SYSTEM.GET(ref, ch) END;
  1195. WHILE ch >= 0FDX DO (* skip variables *)
  1196. INC(ref); RefCh(ref, ch);
  1197. IF ch = 10X THEN INC(ref, 4) END;
  1198. RefNum(ref, d); RefName(ref, name); SYSTEM.GET(ref, ch)
  1199. END
  1200. END;
  1201. RETURN -1
  1202. END SourcePos;
  1203. (* -------------------- dynamic link libraries --------------------- *)
  1204. (*
  1205. PROCEDURE LoadDll* (IN name: ARRAY OF SHORTCHAR; VAR ok: BOOLEAN);
  1206. VAR h: KERNEL32.Handle;
  1207. BEGIN
  1208. ok := FALSE;
  1209. h := KERNEL32.LoadLibraryA(name);
  1210. IF h # 0 THEN ok := TRUE END
  1211. END LoadDll;
  1212. PROCEDURE ThisDllObj* (mode, fprint: INTEGER; IN dll, name: ARRAY OF SHORTCHAR): INTEGER;
  1213. VAR ad: INTEGER; h: KERNEL32.Handle;
  1214. BEGIN
  1215. ad := 0;
  1216. IF mode = mProc THEN
  1217. h := KERNEL32.GetModuleHandleA(dll);
  1218. IF h # 0 THEN ad := KERNEL32.GetProcAddress(h, name) END
  1219. END;
  1220. RETURN ad
  1221. END ThisDllObj;
  1222. *)
  1223. PROCEDURE LoadDll* (IN name: ARRAY OF SHORTCHAR; VAR ok: BOOLEAN);
  1224. VAR h: LinDl.HANDLE;
  1225. BEGIN
  1226. ok := FALSE;
  1227. h := LinDl.dlopen(name, LinDl.RTLD_LAZY + LinDl.RTLD_GLOBAL);
  1228. IF h # LinDl.NULL THEN ok := TRUE END
  1229. END LoadDll;
  1230. PROCEDURE ThisDllObj* (mode, fprint: INTEGER; IN dll, name: ARRAY OF SHORTCHAR): INTEGER;
  1231. VAR ad: INTEGER; h: LinDl.HANDLE;
  1232. BEGIN
  1233. ad := 0;
  1234. IF mode IN {mVar, mProc} THEN
  1235. h := LinDl.dlopen(dll, LinDl.RTLD_LAZY+ LinDl.RTLD_GLOBAL);
  1236. IF h # LinDl.NULL THEN
  1237. ad := LinDl.dlsym(h, name);
  1238. END
  1239. END;
  1240. RETURN ad
  1241. END ThisDllObj;
  1242. (* -------------------- garbage collector (portable) --------------------- *)
  1243. PROCEDURE Mark (this: Block);
  1244. VAR father, son: Block; tag: Type; flag, offset, actual: INTEGER;
  1245. BEGIN
  1246. IF ~ODD(SYSTEM.VAL(INTEGER, this.tag)) THEN
  1247. father := NIL;
  1248. LOOP
  1249. INC(SYSTEM.VAL(INTEGER, this.tag));
  1250. flag := SYSTEM.VAL(INTEGER, this.tag) MOD 4;
  1251. tag := SYSTEM.VAL(Type, SYSTEM.VAL(INTEGER, this.tag) - flag);
  1252. IF flag >= 2 THEN actual := this.first; this.actual := actual
  1253. ELSE actual := SYSTEM.ADR(this.last)
  1254. END;
  1255. LOOP
  1256. offset := tag.ptroffs[0];
  1257. IF offset < 0 THEN
  1258. INC(SYSTEM.VAL(INTEGER, tag), offset + 4); (* restore tag *)
  1259. IF (flag >= 2) & (actual < this.last) & (offset < -4) THEN (* next array element *)
  1260. INC(actual, tag.size); this.actual := actual
  1261. ELSE (* up *)
  1262. this.tag := SYSTEM.VAL(Type, SYSTEM.VAL(INTEGER, tag) + flag);
  1263. IF father = NIL THEN RETURN END;
  1264. son := this; this := father;
  1265. flag := SYSTEM.VAL(INTEGER, this.tag) MOD 4;
  1266. tag := SYSTEM.VAL(Type, SYSTEM.VAL(INTEGER, this.tag) - flag);
  1267. offset := tag.ptroffs[0];
  1268. IF flag >= 2 THEN actual := this.actual ELSE actual := SYSTEM.ADR(this.last) END;
  1269. SYSTEM.GET(actual + offset, father); SYSTEM.PUT(actual + offset, SYSTEM.ADR(son.last));
  1270. INC(SYSTEM.VAL(INTEGER, tag), 4)
  1271. END
  1272. ELSE
  1273. SYSTEM.GET(actual + offset, son);
  1274. IF son # NIL THEN
  1275. DEC(SYSTEM.VAL(INTEGER, son), 4);
  1276. IF ~ODD(SYSTEM.VAL(INTEGER, son.tag)) THEN (* down *)
  1277. this.tag := SYSTEM.VAL(Type, SYSTEM.VAL(INTEGER, tag) + flag);
  1278. SYSTEM.PUT(actual + offset, father); father := this; this := son;
  1279. EXIT
  1280. END
  1281. END;
  1282. INC(SYSTEM.VAL(INTEGER, tag), 4)
  1283. END
  1284. END
  1285. END
  1286. END
  1287. END Mark;
  1288. PROCEDURE MarkGlobals;
  1289. VAR m: Module; i, p: INTEGER;
  1290. BEGIN
  1291. m := modList;
  1292. WHILE m # NIL DO
  1293. IF m.refcnt >= 0 THEN
  1294. i := 0;
  1295. WHILE i < m.nofptrs DO
  1296. SYSTEM.GET(m.varBase + m.ptrs[i], p); INC(i);
  1297. IF p # 0 THEN
  1298. Mark(SYSTEM.VAL(Block, p - 4))
  1299. END
  1300. END
  1301. END;
  1302. m := m.next
  1303. END
  1304. END MarkGlobals;
  1305. (*
  1306. PROCEDURE Next (b: Block): Block; (* next block in same cluster *)
  1307. VAR size: INTEGER;
  1308. BEGIN
  1309. SYSTEM.GET(SYSTEM.VAL(INTEGER, b.tag) DIV 4 * 4, size);
  1310. IF ODD(SYSTEM.VAL(INTEGER, b.tag) DIV 2) THEN INC(size, b.last - SYSTEM.ADR(b.last)) END;
  1311. RETURN SYSTEM.VAL(Block, SYSTEM.VAL(INTEGER, b) + (size + 19) DIV 16 * 16)
  1312. END Next;
  1313. *)
  1314. PROCEDURE [code] Next (b: Block): Block (* next block in same cluster *)
  1315. (*
  1316. MOV ECX,[EAX] b.tag
  1317. AND CL,0FCH b.tag DIV * 4
  1318. MOV ECX,[ECX] size
  1319. TESTB [EAX],02H ODD(b.tag DIV 2)
  1320. JE L1
  1321. ADD ECX,[EAX,4] size + b.last
  1322. SUB ECX,EAX
  1323. SUB ECX,4 size + b.last - ADR(b.last)
  1324. L1:
  1325. ADD ECX,19 size + 19
  1326. AND CL,0F0H (size + 19) DIV 16 * 16
  1327. ADD EAX,ECX b + size
  1328. *)
  1329. 08BH, 008H,
  1330. 080H, 0E1H, 0FCH,
  1331. 08BH, 009H,
  1332. 0F6H, 000H, 002H,
  1333. 074H, 008H,
  1334. 003H, 048H, 004H,
  1335. 029H, 0C1H,
  1336. 083H, 0E9H, 004H,
  1337. 083H, 0C1H, 013H,
  1338. 080H, 0E1H, 0F0H,
  1339. 001H, 0C8H;
  1340. PROCEDURE CheckCandidates;
  1341. (* pre: nofcand > 0 *)
  1342. VAR i, j, h, p, end: INTEGER; c: Cluster; blk, next: Block;
  1343. BEGIN
  1344. (* sort candidates (shellsort) *)
  1345. h := 1; REPEAT h := h*3 + 1 UNTIL h > nofcand;
  1346. REPEAT h := h DIV 3; i := h;
  1347. WHILE i < nofcand DO p := candidates[i]; j := i;
  1348. WHILE (j >= h) & (candidates[j-h] > p) DO
  1349. candidates[j] := candidates[j-h]; j := j-h
  1350. END;
  1351. candidates[j] := p; INC(i)
  1352. END
  1353. UNTIL h = 1;
  1354. (* sweep *)
  1355. c := root; i := 0;
  1356. WHILE c # NIL DO
  1357. blk := SYSTEM.VAL(Block, SYSTEM.VAL(INTEGER, c) + 12);
  1358. end := SYSTEM.VAL(INTEGER, blk) + (c.size - 12) DIV 16 * 16;
  1359. WHILE candidates[i] < SYSTEM.VAL(INTEGER, blk) DO
  1360. INC(i);
  1361. IF i = nofcand THEN RETURN END
  1362. END;
  1363. WHILE SYSTEM.VAL(INTEGER, blk) < end DO
  1364. next := Next(blk);
  1365. IF candidates[i] < SYSTEM.VAL(INTEGER, next) THEN
  1366. IF (SYSTEM.VAL(INTEGER, blk.tag) # SYSTEM.ADR(blk.last)) (* not a free block *)
  1367. & (~strictStackSweep OR (candidates[i] = SYSTEM.ADR(blk.last))) THEN
  1368. Mark(blk)
  1369. END;
  1370. REPEAT
  1371. INC(i);
  1372. IF i = nofcand THEN RETURN END
  1373. UNTIL candidates[i] >= SYSTEM.VAL(INTEGER, next)
  1374. END;
  1375. IF (SYSTEM.VAL(INTEGER, blk.tag) MOD 4 = 0)
  1376. & (SYSTEM.VAL(INTEGER, blk.tag) # SYSTEM.ADR(blk.last))
  1377. & (blk.tag.base[0] = NIL) & (blk.actual > 0)
  1378. THEN (* referenced interface record *)
  1379. Mark(blk)
  1380. END;
  1381. blk := next
  1382. END;
  1383. c := c.next
  1384. END
  1385. END CheckCandidates;
  1386. PROCEDURE MarkLocals;
  1387. VAR sp, p, min, max: INTEGER; c: Cluster;
  1388. BEGIN
  1389. SYSTEM.GETREG(FP, sp); nofcand := 0; c := root;
  1390. WHILE c.next # NIL DO c := c.next END;
  1391. min := SYSTEM.VAL(INTEGER, root); max := SYSTEM.VAL(INTEGER, c) + c.size;
  1392. WHILE sp < baseStack DO
  1393. SYSTEM.GET(sp, p);
  1394. IF (p > min) & (p < max) & (~strictStackSweep OR (p MOD 16 = 0)) THEN
  1395. candidates[nofcand] := p; INC(nofcand);
  1396. IF nofcand = LEN(candidates) - 1 THEN CheckCandidates; nofcand := 0 END
  1397. END;
  1398. INC(sp, 4)
  1399. END;
  1400. candidates[nofcand] := max; INC(nofcand); (* ensure complete scan for interface mark*)
  1401. IF nofcand > 0 THEN CheckCandidates END
  1402. END MarkLocals;
  1403. PROCEDURE MarkFinObj;
  1404. VAR f: FList;
  1405. BEGIN
  1406. wouldFinalize := FALSE;
  1407. f := finalizers;
  1408. WHILE f # NIL DO
  1409. IF ~ODD(SYSTEM.VAL(INTEGER, f.blk.tag)) THEN wouldFinalize := TRUE END;
  1410. Mark(f.blk);
  1411. f := f.next
  1412. END;
  1413. f := hotFinalizers;
  1414. WHILE f # NIL DO IF ~ODD(SYSTEM.VAL(INTEGER, f.blk.tag)) THEN wouldFinalize := TRUE END;
  1415. Mark(f.blk);
  1416. f := f.next
  1417. END
  1418. END MarkFinObj;
  1419. PROCEDURE CheckFinalizers;
  1420. VAR f, g, h, k: FList;
  1421. BEGIN
  1422. f := finalizers; g := NIL;
  1423. (* hotFinalizers := NIL; k := NIL; *)
  1424. IF hotFinalizers = NIL THEN k := NIL
  1425. ELSE
  1426. k := hotFinalizers;
  1427. WHILE k.next # NIL DO k := k.next END
  1428. END;
  1429. WHILE f # NIL DO
  1430. h := f; f := f.next;
  1431. IF ~ODD(SYSTEM.VAL(INTEGER, h.blk.tag)) THEN
  1432. IF g = NIL THEN finalizers := f ELSE g.next := f END;
  1433. IF k = NIL THEN hotFinalizers := h ELSE k.next := h END;
  1434. k := h; h.next := NIL
  1435. ELSE g := h
  1436. END
  1437. END;
  1438. h := hotFinalizers;
  1439. WHILE h # NIL DO Mark(h.blk); h := h.next END
  1440. END CheckFinalizers;
  1441. PROCEDURE ExecFinalizer (a, b, c: INTEGER);
  1442. VAR f: FList; fin: PROCEDURE(this: ANYPTR);
  1443. BEGIN
  1444. f := hotFinalizers; hotFinalizers := hotFinalizers.next;
  1445. IF f.aiptr THEN (*ArrFinalizer(SYSTEM.VAL(ANYPTR, S.ADR(f.blk.last)))*)
  1446. ELSE
  1447. SYSTEM.GET(SYSTEM.VAL(INTEGER, f.blk.tag) - 4, fin); (* method 0 *)
  1448. IF fin # NIL THEN fin(SYSTEM.VAL(ANYPTR, SYSTEM.ADR(f.blk.last))) END;
  1449. (*
  1450. IF f.iptr THEN RecFinalizer(SYSTEM.VAL(ANYPTR, SYSTEM.ADR(f.blk.last))) END
  1451. *)
  1452. END
  1453. END ExecFinalizer;
  1454. PROCEDURE^ Try* (h: TryHandler; a, b, c: INTEGER); (* COMPILER DEPENDENT *)
  1455. PROCEDURE CallFinalizers;
  1456. VAR f: FList;
  1457. BEGIN
  1458. WHILE hotFinalizers # NIL DO
  1459. f := hotFinalizers.next; hotFinalizers.next := NIL;
  1460. Try(ExecFinalizer, 0, 0, 0);
  1461. hotFinalizers := f
  1462. END;
  1463. wouldFinalize := FALSE
  1464. END CallFinalizers;
  1465. PROCEDURE Insert (blk: FreeBlock; size: INTEGER); (* insert block in free list *)
  1466. VAR i: INTEGER;
  1467. BEGIN
  1468. blk.size := size - 4; blk.tag := SYSTEM.VAL(Type, SYSTEM.ADR(blk.size));
  1469. i := MIN(N - 1, (blk.size DIV 16));
  1470. blk.next := free[i]; free[i] := blk
  1471. END Insert;
  1472. PROCEDURE Sweep (dealloc: BOOLEAN);
  1473. VAR cluster, last, c: Cluster; blk, next: Block; fblk, b, t: FreeBlock; end, i: INTEGER;
  1474. BEGIN
  1475. cluster := root; last := NIL; allocated := 0;
  1476. i := N;
  1477. REPEAT DEC(i); free[i] := sentinel UNTIL i = 0;
  1478. WHILE cluster # NIL DO
  1479. blk := SYSTEM.VAL(Block, SYSTEM.VAL(INTEGER, cluster) + 12);
  1480. end := SYSTEM.VAL(INTEGER, blk) + (cluster.size - 12) DIV 16 * 16;
  1481. fblk := NIL;
  1482. WHILE SYSTEM.VAL(INTEGER, blk) < end DO
  1483. next := Next(blk);
  1484. IF ODD(SYSTEM.VAL(INTEGER, blk.tag)) THEN
  1485. IF fblk # NIL THEN
  1486. Insert(fblk, SYSTEM.VAL(INTEGER, blk) - SYSTEM.VAL(INTEGER, fblk));
  1487. fblk := NIL
  1488. END;
  1489. DEC(SYSTEM.VAL(INTEGER, blk.tag)); (* unmark *)
  1490. INC(allocated, SYSTEM.VAL(INTEGER, next) - SYSTEM.VAL(INTEGER, blk))
  1491. ELSIF fblk = NIL THEN
  1492. fblk := SYSTEM.VAL(FreeBlock, blk)
  1493. END;
  1494. blk := next
  1495. END;
  1496. IF dealloc & (SYSTEM.VAL(INTEGER, fblk) = SYSTEM.VAL(INTEGER, cluster) + 12) THEN
  1497. (* deallocate cluster *)
  1498. c := cluster; cluster := cluster.next;
  1499. IF last = NIL THEN root := cluster ELSE last.next := cluster END;
  1500. FreeHeapMem(c)
  1501. ELSE
  1502. IF fblk # NIL THEN Insert(fblk, end - SYSTEM.VAL(INTEGER, fblk)) END;
  1503. last := cluster; cluster := cluster.next
  1504. END
  1505. END;
  1506. (* reverse free list *)
  1507. i := N;
  1508. REPEAT
  1509. DEC(i);
  1510. b := free[i]; fblk := sentinel;
  1511. WHILE b # sentinel DO t := b; b := t.next; t.next := fblk; fblk := t END;
  1512. free[i] := fblk
  1513. UNTIL i = 0
  1514. END Sweep;
  1515. PROCEDURE Collect*;
  1516. BEGIN
  1517. IF root # NIL THEN
  1518. CallFinalizers; (* trap cleanup *)
  1519. IF debug & (watcher # NIL) THEN watcher(1) END;
  1520. MarkGlobals;
  1521. MarkLocals;
  1522. CheckFinalizers;
  1523. Sweep(TRUE);
  1524. CallFinalizers
  1525. END
  1526. END Collect;
  1527. PROCEDURE FastCollect*;
  1528. BEGIN
  1529. IF root # NIL THEN
  1530. (*
  1531. CallFinalizers; (* trap cleanup *)
  1532. *)
  1533. IF debug & (watcher # NIL) THEN watcher(2) END;
  1534. MarkGlobals;
  1535. MarkLocals;
  1536. (* CheckFinalizers; *)
  1537. MarkFinObj;
  1538. Sweep(FALSE);
  1539. (*
  1540. CallFinalizers
  1541. *)
  1542. END
  1543. END FastCollect;
  1544. (*
  1545. PROCEDURE GlobalCollect*;
  1546. BEGIN
  1547. IF root # NIL THEN
  1548. MarkGlobals;
  1549. (* MarkLocals; *)
  1550. CheckFinalizers;
  1551. Sweep(FALSE);
  1552. END
  1553. END GlobalCollect;
  1554. *)
  1555. PROCEDURE WouldFinalize* (): BOOLEAN;
  1556. BEGIN
  1557. RETURN wouldFinalize
  1558. END WouldFinalize;
  1559. (* --------------------- memory allocation (portable) -------------------- *)
  1560. PROCEDURE OldBlock (size: INTEGER): FreeBlock; (* size MOD 16 = 0 *)
  1561. VAR b, l: FreeBlock; s, i: INTEGER;
  1562. BEGIN
  1563. IF debug & (watcher # NIL) THEN watcher(3) END;
  1564. s := size - 4;
  1565. i := MIN(N - 1, s DIV 16);
  1566. WHILE (i # N - 1) & (free[i] = sentinel) DO INC(i) END;
  1567. b := free[i]; l := NIL;
  1568. WHILE b.size < s DO l := b; b := b.next END;
  1569. IF b # sentinel THEN
  1570. IF l = NIL THEN free[i] := b.next ELSE l.next := b.next END
  1571. ELSE b := NIL
  1572. END;
  1573. RETURN b
  1574. END OldBlock;
  1575. PROCEDURE LastBlock (limit: INTEGER): FreeBlock; (* size MOD 16 = 0 *)
  1576. VAR b, l: FreeBlock; s, i: INTEGER;
  1577. BEGIN
  1578. s := limit - 4;
  1579. i := 0;
  1580. REPEAT
  1581. b := free[i]; l := NIL;
  1582. WHILE (b # sentinel) & (SYSTEM.VAL(INTEGER, b) + b.size # s) DO l := b; b := b.next END;
  1583. IF b # sentinel THEN
  1584. IF l = NIL THEN free[i] := b.next ELSE l.next := b.next END
  1585. ELSE b := NIL
  1586. END;
  1587. INC(i)
  1588. UNTIL (b # NIL) OR (i = N);
  1589. RETURN b
  1590. END LastBlock;
  1591. PROCEDURE NewBlock (size: INTEGER): Block;
  1592. VAR tsize, a, s: INTEGER; b: FreeBlock; new, c: Cluster; r: Reducer;
  1593. BEGIN
  1594. tsize := (size + 19) DIV 16 * 16;
  1595. b := OldBlock(tsize); (* 1) search for free block *)
  1596. IF b = NIL THEN
  1597. IF dllMem THEN
  1598. FastCollect; b := OldBlock(tsize); (* 2) collect *)
  1599. IF b = NIL THEN
  1600. AllocHeapMem(tsize + 12, new); (* 3) allocate new cluster *)
  1601. IF new # NIL THEN
  1602. IF (root = NIL) OR (SYSTEM.VAL(INTEGER, new) < SYSTEM.VAL(INTEGER, root)) THEN
  1603. new.next := root; root := new
  1604. ELSE
  1605. c := root;
  1606. WHILE (c.next # NIL) & (SYSTEM.VAL(INTEGER, new) > SYSTEM.VAL(INTEGER, c.next)) DO
  1607. c := c.next
  1608. END;
  1609. new.next := c.next; c.next := new
  1610. END;
  1611. b := SYSTEM.VAL(FreeBlock, SYSTEM.VAL(INTEGER, new) + 12);
  1612. b.size := (new.size - 12) DIV 16 * 16 - 4
  1613. ELSE
  1614. RETURN NIL (* 4) give up *)
  1615. END
  1616. END
  1617. ELSE
  1618. FastCollect; (* 2) collect *)
  1619. IF (b = NIL) & (HeapFull(tsize)) & (reducers # NIL) THEN (* 3) little space => reduce once *)
  1620. r := reducers; reducers := NIL;
  1621. WHILE r # NIL DO r.Reduce(FALSE); r := r.next END;
  1622. Collect
  1623. END;
  1624. s := 3 * (allocated + tsize) DIV 2;
  1625. a := 12 + (root.size - 12) DIV 16 * 16;
  1626. IF s <= total THEN
  1627. b := OldBlock(tsize);
  1628. IF b = NIL THEN s := a + tsize END
  1629. ELSIF s < a + tsize THEN
  1630. s := a + tsize
  1631. END;
  1632. IF total < s THEN (* 4) enlarge heap *)
  1633. GrowHeapMem(s, root);
  1634. IF root.size >= s THEN
  1635. b := LastBlock(SYSTEM.VAL(INTEGER, root) + a);
  1636. IF b # NIL THEN
  1637. b.size := (root.size - a + b.size + 4) DIV 16 * 16 - 4
  1638. ELSE
  1639. b := SYSTEM.VAL(FreeBlock, SYSTEM.VAL(INTEGER, root) + a);
  1640. b.size := (root.size - a) DIV 16 * 16 - 4
  1641. END
  1642. ELSIF reducers # NIL THEN (* 5) no space => fully reduce *)
  1643. r := reducers; reducers := NIL;
  1644. WHILE r # NIL DO r.Reduce(TRUE); r := r.next END;
  1645. Collect
  1646. END
  1647. END;
  1648. IF b = NIL THEN
  1649. b := OldBlock(tsize);
  1650. IF b = NIL THEN RETURN NIL END (* 6) give up *)
  1651. END
  1652. END
  1653. END;
  1654. (* b # NIL *)
  1655. a := b.size + 4 - tsize;
  1656. IF a > 0 THEN Insert(SYSTEM.VAL(FreeBlock, SYSTEM.VAL(INTEGER, b) + tsize), a) END;
  1657. IF size > 0 THEN Erase(SYSTEM.ADR(b.size), (size + 3) DIV 4) END;
  1658. INC(allocated, tsize);
  1659. RETURN SYSTEM.VAL(Block, b)
  1660. END NewBlock;
  1661. (*
  1662. PROCEDURE NewBlock (size: INTEGER): Block;
  1663. VAR tsize, a, s: INTEGER; b: FreeBlock; new, c: Cluster; r: Reducer;
  1664. BEGIN
  1665. tsize := (size + 19) DIV 16 * 16;
  1666. b := OldBlock(tsize); (* 1) search for free block *)
  1667. IF b = NIL THEN
  1668. (*FastCollect;*) b := OldBlock(tsize); (* 2) collect *)
  1669. IF b = NIL THEN
  1670. AllocHeapMem(tsize + 12, new); (* 3) allocate new cluster *)
  1671. IF new # NIL THEN
  1672. IF (root = NIL) OR (SYSTEM.VAL(INTEGER, new) < SYSTEM.VAL(INTEGER, root)) THEN
  1673. new.next := root; root := new
  1674. ELSE
  1675. c := root;
  1676. WHILE (c.next # NIL) & (SYSTEM.VAL(INTEGER, new) > SYSTEM.VAL(INTEGER, c.next)) DO
  1677. c := c.next
  1678. END;
  1679. new.next := c.next; c.next := new
  1680. END;
  1681. b := SYSTEM.VAL(FreeBlock, SYSTEM.VAL(INTEGER, new) + 12);
  1682. b.size := (new.size - 12) DIV 16 * 16 - 4
  1683. ELSE
  1684. RETURN NIL (* 4) give up *)
  1685. END
  1686. END
  1687. END;
  1688. (* b # NIL *)
  1689. a := b.size + 4 - tsize;
  1690. IF a > 0 THEN Insert(SYSTEM.VAL(FreeBlock, SYSTEM.VAL(INTEGER, b) + tsize), a) END;
  1691. IF size > 0 THEN Erase(SYSTEM.ADR(b.size), (size + 3) DIV 4) END;
  1692. INC(allocated, tsize);
  1693. RETURN SYSTEM.VAL(Block, b)
  1694. END NewBlock;
  1695. *)
  1696. PROCEDURE Allocated* (): INTEGER;
  1697. BEGIN
  1698. RETURN allocated
  1699. END Allocated;
  1700. PROCEDURE Used* (): INTEGER;
  1701. BEGIN
  1702. RETURN used
  1703. END Used;
  1704. PROCEDURE Root* (): INTEGER;
  1705. BEGIN
  1706. RETURN SYSTEM.VAL(INTEGER, root)
  1707. END Root;
  1708. (* -------------------- Trap Handling --------------------- *)
  1709. PROCEDURE^ InitFpu;
  1710. PROCEDURE Start* (code: Command);
  1711. BEGIN
  1712. restart := code;
  1713. res := LinLibc.sigsetjmp(loopContext, LinLibc.TRUE);
  1714. restart()
  1715. END Start;
  1716. PROCEDURE Quit* (exitCode: INTEGER);
  1717. VAR m: Module; term: Command; t: BOOLEAN; res: INTEGER;
  1718. BEGIN
  1719. trapViewer := NIL; trapChecker := NIL; restart := NIL;
  1720. t := terminating; terminating := TRUE; m := modList;
  1721. WHILE m # NIL DO (* call terminators *)
  1722. IF ~static OR ~t THEN
  1723. term := m.term; m.term := NIL;
  1724. IF term # NIL THEN term() END
  1725. END;
  1726. (*
  1727. ReleaseIPtrs(m);
  1728. *)
  1729. m := m.next
  1730. END;
  1731. CallFinalizers;
  1732. hotFinalizers := finalizers; finalizers := NIL;
  1733. CallFinalizers;
  1734. (*
  1735. WinOle.OleUninitialize();
  1736. *)
  1737. (*
  1738. IF ~inDll THEN
  1739. KERNEL32.RemoveExcp(excpPtr^);
  1740. KERNEL32.ExitProcess(exitCode) (* never returns *)
  1741. END
  1742. *)
  1743. res := LinLibc.fflush(0);
  1744. LinLibc.exit(exitCode)
  1745. END Quit;
  1746. PROCEDURE FatalError* (id: INTEGER; str: ARRAY OF CHAR);
  1747. VAR res: INTEGER; title: ARRAY 16 OF SHORTCHAR; text: ARRAY 256 OF SHORTCHAR;
  1748. BEGIN
  1749. title := "Error xy";
  1750. title[6] := SHORT(CHR(id DIV 10 + ORD("0")));
  1751. title[7] := SHORT(CHR(id MOD 10 + ORD("0")));
  1752. text := SHORT(str$);
  1753. res := MessageBox(title$, text$, {mbOk});
  1754. (*
  1755. WinOle.OleUninitialize();
  1756. *)
  1757. (*
  1758. IF ~inDll THEN KERNEL32.RemoveExcp(excpPtr^) END;
  1759. KERNEL32.ExitProcess(1)
  1760. *)
  1761. LinLibc.exit(1);
  1762. (* never returns *)
  1763. END FatalError;
  1764. PROCEDURE DefaultTrapViewer;
  1765. VAR len, ref, end, x, a, b, c: INTEGER; mod: Module;
  1766. name: Name; out: ARRAY 1024 OF SHORTCHAR;
  1767. PROCEDURE WriteString (s: ARRAY OF SHORTCHAR);
  1768. VAR i: INTEGER;
  1769. BEGIN
  1770. i := 0;
  1771. WHILE (len < LEN(out) - 1) & (s[i] # 0X) DO out[len] := s[i]; INC(i); INC(len) END
  1772. END WriteString;
  1773. PROCEDURE WriteHex (x, n: INTEGER);
  1774. VAR i, y: INTEGER;
  1775. BEGIN
  1776. IF len + n < LEN(out) THEN
  1777. i := len + n - 1;
  1778. WHILE i >= len DO
  1779. y := x MOD 16; x := x DIV 16;
  1780. IF y > 9 THEN y := y + (ORD("A") - ORD("0") - 10) END;
  1781. out[i] := SHORT(CHR(y + ORD("0"))); DEC(i)
  1782. END;
  1783. INC(len, n)
  1784. END
  1785. END WriteHex;
  1786. PROCEDURE WriteLn;
  1787. BEGIN
  1788. IF len < LEN(out) - 1 THEN out[len] := 0AX (* 0DX on Windows *); INC(len) END
  1789. END WriteLn;
  1790. BEGIN
  1791. len := 0;
  1792. IF err = 129 THEN WriteString("invalid with")
  1793. ELSIF err = 130 THEN WriteString("invalid case")
  1794. ELSIF err = 131 THEN WriteString("function without return")
  1795. ELSIF err = 132 THEN WriteString("type guard")
  1796. ELSIF err = 133 THEN WriteString("implied type guard")
  1797. ELSIF err = 134 THEN WriteString("value out of range")
  1798. ELSIF err = 135 THEN WriteString("index out of range")
  1799. ELSIF err = 136 THEN WriteString("string too long")
  1800. ELSIF err = 137 THEN WriteString("stack overflow")
  1801. ELSIF err = 138 THEN WriteString("integer overflow")
  1802. ELSIF err = 139 THEN WriteString("division by zero")
  1803. ELSIF err = 140 THEN WriteString("infinite real result")
  1804. ELSIF err = 141 THEN WriteString("real underflow")
  1805. ELSIF err = 142 THEN WriteString("real overflow")
  1806. ELSIF err = 143 THEN WriteString("undefined real result")
  1807. ELSIF err = 200 THEN WriteString("keyboard interrupt")
  1808. ELSIF err = 202 THEN WriteString("illegal instruction: ");
  1809. WriteHex(val, 4)
  1810. ELSIF err = 203 THEN WriteString("illegal memory read [ad = ");
  1811. WriteHex(val, 8); WriteString("]")
  1812. ELSIF err = 204 THEN WriteString("illegal memory write [ad = ");
  1813. WriteHex(val, 8); WriteString("]")
  1814. ELSIF err = 205 THEN WriteString("illegal execution [ad = ");
  1815. WriteHex(val, 8); WriteString("]")
  1816. ELSIF err < 0 THEN WriteString("exception #"); WriteHex(-err, 2)
  1817. ELSE err := err DIV 100 * 256 + err DIV 10 MOD 10 * 16 + err MOD 10;
  1818. WriteString("trap #"); WriteHex(err, 3)
  1819. END;
  1820. a := pc; b := fp; c := 12;
  1821. REPEAT
  1822. WriteLn; WriteString("- ");
  1823. mod := modList;
  1824. WHILE (mod # NIL) & ((a < mod.code) OR (a >= mod.code + mod.csize)) DO mod := mod.next END;
  1825. IF mod # NIL THEN
  1826. DEC(a, mod.code);
  1827. IF mod.refcnt >= 0 THEN
  1828. WriteString(mod.name); ref := mod.refs;
  1829. REPEAT GetRefProc(ref, end, name) UNTIL (end = 0) OR (a < end);
  1830. IF a < end THEN
  1831. WriteString("."); WriteString(name)
  1832. END
  1833. ELSE
  1834. WriteString("("); WriteString(mod.name); WriteString(")")
  1835. END;
  1836. WriteString(" ")
  1837. END;
  1838. WriteString("(pc="); WriteHex(a, 8);
  1839. WriteString(", fp="); WriteHex(b, 8); WriteString(")");
  1840. IF (b >= sp) & (b < stack) THEN
  1841. SYSTEM.GET(b+4, a); (* stacked pc *)
  1842. SYSTEM.GET(b, b); (* dynamic link *)
  1843. DEC(c)
  1844. ELSE c := 0
  1845. END
  1846. UNTIL c = 0;
  1847. out[len] := 0X;
  1848. x := MessageBox("BlackBox", out$, {mbOk})
  1849. END DefaultTrapViewer;
  1850. PROCEDURE TrapCleanup;
  1851. VAR t: TrapCleaner;
  1852. BEGIN
  1853. WHILE trapStack # NIL DO
  1854. t := trapStack; trapStack := trapStack.next; t.Cleanup
  1855. END;
  1856. IF (trapChecker # NIL) & (err # 128) THEN trapChecker END
  1857. END TrapCleanup;
  1858. (*
  1859. PROCEDURE Unwind(f: KERNEL32.ExcpFrmPtr); (* COMPILER DEPENDENT *)
  1860. CONST Label = 27; (* offset of Label: from proc start *)
  1861. BEGIN
  1862. PushFP;
  1863. KERNEL32.RtlUnwind(f, SYSTEM.ADR(Unwind) + Label, NIL, 0);
  1864. (* Label: *)
  1865. PopFP
  1866. END Unwind;
  1867. *)
  1868. (*
  1869. PROCEDURE TrapHandler (excpRec: KERNEL32.ExcpRecPtr; estFrame: KERNEL32.ExcpFrmPtr;
  1870. context: KERNEL32.ContextPtr; dispCont: INTEGER): INTEGER;
  1871. (* same parameter size as Try *)
  1872. BEGIN
  1873. IF excpRec^.flags * {1, 2} = {} THEN
  1874. IF (excpRec.code MOD 256 = 4) & ~interrupted THEN (* wrong trace trap *)
  1875. context.debug[5] := 0; (* disable all debug traps *)
  1876. LdSP8; PopSI; PopDI; PopFP; (* COMPILER DEPENDENT *)
  1877. Return0(0) (* return continueExecution without parameter remove *)
  1878. END;
  1879. Unwind(estFrame);
  1880. IF trapped & (excpRec.code MOD 256 # 1) & (excpRec.code MOD 256 # 253) THEN
  1881. DefaultTrapViewer;
  1882. IF ~secondTrap THEN trapped := FALSE; secondTrap := TRUE END
  1883. END;
  1884. err := -(excpRec.code MOD 256);
  1885. pc := context.ip; sp := context.sp; fp := context.bp; stack := baseStack;
  1886. IF err = -4 THEN err := 200 (* keyboard interrupt *)
  1887. ELSIF err = -5 THEN
  1888. val := excpRec.info[1];
  1889. IF val = pc THEN (* call to undef adr *)
  1890. err := 205; SYSTEM.GET(sp, pc); INC(sp, 4); DEC(pc)
  1891. ELSIF excpRec.info[0] = 0 THEN (* illegal read *)
  1892. err := 203
  1893. ELSE (* illegal write *)
  1894. err := 204
  1895. END
  1896. ELSIF (err = -29) OR (err = -30) THEN (* illegal instruction *)
  1897. err := 202; val := 0;
  1898. IF IsReadable(excpRec.adr, excpRec.adr + 4) THEN
  1899. SYSTEM.GET(excpRec.adr, val);
  1900. IF val MOD 100H = 8DH THEN (* lea reg,reg *)
  1901. IF val DIV 100H MOD 100H = 0F0H THEN
  1902. err := val DIV 10000H MOD 100H (* trap *)
  1903. ELSIF val DIV 1000H MOD 10H = 0EH THEN
  1904. err := 128 + val DIV 100H MOD 10H (* run time error *)
  1905. END
  1906. END
  1907. END
  1908. ELSIF err = -142 THEN DEC(pc); err := 140 (* fpu: div by zero *)
  1909. ELSIF (err = -144) OR (err = -146) THEN DEC(pc); err := 143 ; (* fpu: invalid op *)
  1910. val := context.float[0] MOD 4096 * 65536 + context.float[1] MOD 65536
  1911. ELSIF err = -145 THEN DEC(pc); err := 142 (* fpu: overflow *)
  1912. ELSIF err = -147 THEN DEC(pc); err := 141 (* fpu: underflow *)
  1913. ELSIF err = -148 THEN err := 139 (* division by zero *)
  1914. ELSIF err = -149 THEN err := 138 (* integer overflow *)
  1915. ELSIF (err = -1) OR (err = -253) THEN err := 137 (* stack overflow *)
  1916. END;
  1917. INC(trapCount);
  1918. InitFpu;
  1919. IF err # 137 THEN (* stack overflow handling is delayed *)
  1920. TrapCleanup;
  1921. IF err = 128 THEN (* do nothing *)
  1922. ELSIF(trapViewer # NIL) & (restart # NIL) & ~trapped & ~guarded THEN
  1923. trapped := TRUE; trapViewer()
  1924. ELSE DefaultTrapViewer
  1925. END
  1926. END;
  1927. trapped := FALSE; secondTrap := FALSE;
  1928. IF dispCont = 0 THEN (* InterfaceTrapHandler *) (* COMPILER DEPENDENT *)
  1929. KERNEL32.RemoveExcp(estFrame^);
  1930. SYSTEM.PUTREG(CX, estFrame(ExcpFramePtr).par);
  1931. SYSTEM.PUTREG(SP, SYSTEM.VAL(INTEGER, estFrame) + 12);
  1932. IF err = 137 THEN (* retrigger stack overflow *)
  1933. TrapCleanup; DefaultTrapViewer;
  1934. res := KERNEL32.VirtualProtect(FPageWord(8), 1024, {2, 8}, old);
  1935. IF res = 0 THEN res := KERNEL32.VirtualProtect(FPageWord(8), 1024, {0}, old) END
  1936. END;
  1937. PopSI; PopDI; PopBX; PopFP;
  1938. ReturnCX(WinApi.E_UNEXPECTED)
  1939. ELSIF estFrame # excpPtr THEN (* Try failed *) (* COMPILER DEPENDENT *)
  1940. KERNEL32.RemoveExcp(estFrame^);
  1941. res := SYSTEM.VAL(INTEGER, estFrame);
  1942. SYSTEM.PUTREG(FP, res + (SIZE(KERNEL32.ExcpFrm) + 8)); (* restore fp *)
  1943. SYSTEM.PUTREG(SP, res - 4); (* restore stack *)
  1944. IF err = 137 THEN (* retrigger stack overflow *)
  1945. TrapCleanup; DefaultTrapViewer;
  1946. res := KERNEL32.VirtualProtect(FPageWord(8), 1024, {2, 8}, old);
  1947. IF res = 0 THEN res := KERNEL32.VirtualProtect(FPageWord(8), 1024, {0}, old) END
  1948. END;
  1949. PopBX;
  1950. RETURN 0 (* return from Try *)
  1951. ELSIF restart # NIL THEN (* Start failed *)
  1952. SYSTEM.PUTREG(FP, baseStack); (* restore fp *)
  1953. SYSTEM.PUTREG(SP, baseStack); (* restore stack *)
  1954. IF err = 137 THEN (* retrigger stack overflow *)
  1955. TrapCleanup; DefaultTrapViewer;
  1956. res := KERNEL32.VirtualProtect(FPageWord(8), 1024, {2, 8}, old);
  1957. IF res = 0 THEN res := KERNEL32.VirtualProtect(FPageWord(8), 1024, {0}, old) END
  1958. END;
  1959. restart();
  1960. Quit(1)
  1961. ELSE (* boot process failed *)
  1962. Quit(1)
  1963. END
  1964. (* never returns *)
  1965. ELSE
  1966. LdSP8; PopSI; PopDI; PopFP; (* COMPILER DEPENDENT *)
  1967. Return0(1) (* return continueSearch without parameter remove *)
  1968. END
  1969. END TrapHandler;
  1970. *)
  1971. PROCEDURE SetTrapGuard* (on: BOOLEAN);
  1972. BEGIN
  1973. guarded := on
  1974. END SetTrapGuard;
  1975. (*
  1976. PROCEDURE Try* (h: TryHandler; a, b, c: INTEGER); (* COMPILER DEPENDENT *)
  1977. (* same parameter size as TrapHandler *)
  1978. VAR excp: KERNEL32.ExcpFrm; (* no other local variables! *)
  1979. BEGIN
  1980. PushBX;
  1981. excp.handler := TrapHandler;
  1982. KERNEL32.InstallExcp(excp);
  1983. h(a, b, c);
  1984. KERNEL32.RemoveExcp(excp);
  1985. PopBX
  1986. END Try;
  1987. *)
  1988. PROCEDURE Try* (h: TryHandler; a, b, c: INTEGER);
  1989. VAR res: INTEGER; context: LinLibc.sigjmp_buf; oldContext: POINTER TO LinLibc.sigjmp_buf;
  1990. BEGIN
  1991. oldContext := currentTryContext;
  1992. res := LinLibc.sigsetjmp(context, LinLibc.TRUE);
  1993. currentTryContext := SYSTEM.ADR(context);
  1994. IF res = 0 THEN (* first time around *)
  1995. h(a, b, c);
  1996. ELSIF res = trapReturn THEN (* after a trap *)
  1997. ELSE
  1998. HALT(100)
  1999. END;
  2000. currentTryContext := oldContext;
  2001. END Try;
  2002. PROCEDURE InterfaceTrapHandler* (excpRec, estFrame, context, dispCont: INTEGER): INTEGER;
  2003. (* known to compiler *)
  2004. VAR res: INTEGER;
  2005. BEGIN
  2006. (*
  2007. res := TrapHandler(SYSTEM.VAL(KERNEL32.ExcpRecPtr, excpRec),
  2008. SYSTEM.VAL(KERNEL32.ExcpFrmPtr, estFrame),
  2009. SYSTEM.VAL(KERNEL32.ContextPtr, context),
  2010. 0);
  2011. (* LdSP8 removes parameters of TrapHandler *)
  2012. LdSP8; PopSI; PopDI; PopFP; (* COMPILER DEPENDENT *)
  2013. Return0(1); (* return continueSearch without parameter remove *)
  2014. IF FALSE THEN RETURN 0 END
  2015. *)
  2016. RETURN 0
  2017. END InterfaceTrapHandler;
  2018. (* -------------------- keyboard interrupt handling --------------------- *)
  2019. (*
  2020. PROCEDURE KeyboardWatcher (main: INTEGER): INTEGER; (* runs in a thread *)
  2021. VAR res, id, a, to: INTEGER; msg: USER32.Message; wnd: USER32.Handle;
  2022. context: KERNEL32.Context; mod: Module;
  2023. BEGIN
  2024. wnd := USER32.CreateWindowExA({}, "Edit", "", {}, 0, 0, 0, 0, 0, 0, KERNEL32.GetModuleHandleA(NIL), 0);
  2025. res := USER32.RegisterHotKey(wnd, 13, {1}, 3); (* ctrl break *)
  2026. IF res = 0 THEN
  2027. res := USER32.RegisterHotKey(wnd, 14, {1, 2}, 3) (* shift ctrl break *)
  2028. END;
  2029. LOOP
  2030. res := USER32.GetMessageA(msg, 0, 0, 0);
  2031. IF msg.message = USER32.WMHotKey THEN
  2032. wnd := USER32.GetForegroundWindow();
  2033. res := USER32.GetWindowThreadProcessId(wnd, id);
  2034. IF (msg.wParam = 14) OR (id = KERNEL32.GetCurrentProcessId()) THEN
  2035. to := KERNEL32.GetTickCount() + 1000; (* 1 sec timeout *)
  2036. REPEAT
  2037. res := KERNEL32.SuspendThread(main);
  2038. context.flags := {0, 16};
  2039. res := KERNEL32.GetThreadContext(main, context);
  2040. mod := modList; a := context.ip;
  2041. WHILE (mod # NIL) & ((a < mod.code) OR (a >= mod.code + mod.csize)) DO
  2042. mod := mod.next
  2043. END;
  2044. IF (mod # NIL) & (mod.name = "Kernel") THEN mod := NIL END;
  2045. IF mod # NIL THEN
  2046. interrupted := TRUE;
  2047. INCL(SYSTEM.VAL(SET, context.pf), 8); (* set trap flag *)
  2048. res := KERNEL32.SetThreadContext(main, context)
  2049. END;
  2050. res := KERNEL32.ResumeThread(main);
  2051. KERNEL32.Sleep(0);
  2052. interrupted := FALSE
  2053. UNTIL (mod # NIL) OR (KERNEL32.GetTickCount() > to)
  2054. END
  2055. END
  2056. END;
  2057. RETURN 0
  2058. END KeyboardWatcher;
  2059. *)
  2060. (*
  2061. PROCEDURE InstallKeyboardInt;
  2062. VAR res, id: INTEGER; t, main: KERNEL32.Handle;
  2063. BEGIN
  2064. res := KERNEL32.DuplicateHandle(KERNEL32.GetCurrentProcess(), KERNEL32.GetCurrentThread(),
  2065. KERNEL32.GetCurrentProcess(), main, {1, 3, 4, 16..19}, 0, {});
  2066. t := KERNEL32.CreateThread(NIL, 4096, KeyboardWatcher, main, {}, id)
  2067. END InstallKeyboardInt;
  2068. *)
  2069. (* -------------------- Initialization --------------------- *)
  2070. PROCEDURE InitFpu; (* COMPILER DEPENDENT *)
  2071. (* could be eliminated, delayed for backward compatibility *)
  2072. VAR cw: SET;
  2073. BEGIN
  2074. FINIT;
  2075. FSTCW;
  2076. (* denorm, underflow, precision, zero div, overflow masked *)
  2077. (* invalid trapped *)
  2078. (* round to nearest, temp precision *)
  2079. cw := cw - {0..5, 8..11} + {1, 2, 3, 4, 5, 8, 9};
  2080. FLDCW
  2081. END InitFpu;
  2082. (* A. V. Shiryaev *)
  2083. (* show extended trap information *)
  2084. PROCEDURE ShowTrap (sig: INTEGER; siginfo: LinLibc.Ptrsiginfo_t; context: LinLibc.Ptrucontext_t);
  2085. PROCEDURE WriteChar (c: SHORTCHAR);
  2086. VAR s: ARRAY [untagged] 2 OF SHORTCHAR;
  2087. BEGIN
  2088. s[0] := c; s[1] := 0X;
  2089. res := LinLibc.printf(s)
  2090. END WriteChar;
  2091. PROCEDURE WriteString (s: ARRAY OF SHORTCHAR);
  2092. VAR res: INTEGER;
  2093. BEGIN
  2094. res := LinLibc.printf(s)
  2095. END WriteString;
  2096. PROCEDURE WriteHex (x, n: INTEGER);
  2097. VAR i, y: INTEGER;
  2098. s: ARRAY 9 OF SHORTCHAR;
  2099. BEGIN
  2100. s[n] := 0X;
  2101. i := 0 + n - 1;
  2102. WriteChar("$");
  2103. WHILE i >= 0 DO
  2104. y := x MOD 16; x := x DIV 16;
  2105. IF y > 9 THEN y := y + (ORD("A") - ORD("0") - 10) END;
  2106. s[i] := SHORT(CHR(y + ORD("0")));
  2107. DEC(i)
  2108. END;
  2109. WriteString(s)
  2110. END WriteHex;
  2111. PROCEDURE WriteLn;
  2112. BEGIN
  2113. WriteChar(0AX)
  2114. END WriteLn;
  2115. PROCEDURE KV (name: ARRAY OF SHORTCHAR; x: INTEGER);
  2116. BEGIN
  2117. WriteString(name); WriteString(" = "); WriteHex(x, 8)
  2118. END KV;
  2119. BEGIN
  2120. WriteString("================================"); WriteLn;
  2121. WriteString("TRAP:"); WriteLn;
  2122. WriteString("--------------------------------"); WriteLn;
  2123. KV("sig", sig); WriteString(", ");
  2124. KV("baseStack", baseStack); WriteLn;
  2125. KV("GS ", context.sc_gs); WriteString(", ");
  2126. KV("FS ", context.sc_fs); WriteString(", ");
  2127. KV("ES ", context.sc_es); WriteString(", ");
  2128. KV("DS ", context.sc_ds); WriteLn;
  2129. KV("EDI", context.sc_edi); WriteString(", ");
  2130. KV("ESI", context.sc_esi); WriteString(", ");
  2131. KV("EBP", context.sc_ebp); WriteString(", ");
  2132. KV("EBX", context.sc_ebx); WriteLn;
  2133. KV("EDX", context.sc_edx); WriteString(", ");
  2134. KV("ECX", context.sc_ecx); WriteString(", ");
  2135. KV("EAX", context.sc_eax); WriteString(", ");
  2136. KV("EIP", context.sc_eip); WriteLn;
  2137. KV("CS", context.sc_cs); WriteString(", ");
  2138. KV("EFLAGS", context.sc_eflags); WriteString(", ");
  2139. KV("ESP", context.sc_esp); WriteString(", ");
  2140. KV("SS", context.sc_ss); WriteLn;
  2141. KV("ONSTACK", context.sc_onstack); WriteString(", ");
  2142. KV("MASK", context.sc_mask); WriteString(", ");
  2143. KV("TRAPNO", context.sc_trapno); WriteString(", ");
  2144. KV("ERR", context.sc_err); WriteLn;
  2145. (* WriteString("--------------------------------"); WriteLn; *)
  2146. (* TODO: show siginfo *)
  2147. WriteString("================================"); WriteLn
  2148. END ShowTrap;
  2149. PROCEDURE TrapHandler (sig: INTEGER; siginfo: LinLibc.Ptrsiginfo_t; context: LinLibc.Ptrucontext_t);
  2150. BEGIN
  2151. (*
  2152. SYSTEM.GETREG(SP, sp);
  2153. SYSTEM.GETREG(FP, fp);
  2154. *)
  2155. stack := baseStack;
  2156. (* A. V. Shiryaev *)
  2157. ShowTrap(sig, siginfo, context);
  2158. (*
  2159. sp := context.uc_mcontext.gregs[7]; (* TODO: is the stack pointer really stored in register 7? *)
  2160. fp := context.uc_mcontext.gregs[6]; (* TODO: is the frame pointer really stored in register 6? *)
  2161. pc := context.uc_mcontext.gregs[14]; (* TODO: is the pc really stored in register 14? *)
  2162. *)
  2163. sp := context.sc_esp; fp := context.sc_ebp; pc := context.sc_eip;
  2164. (* val := siginfo.si_addr; *)
  2165. val := siginfo.si_pid; (* XXX *)
  2166. (*
  2167. Int(sig); Int(siginfo.si_signo); Int(siginfo.si_code); Int(siginfo.si_errno);
  2168. Int(siginfo.si_status); Int(siginfo.si_value); Int(siginfo.si_int);
  2169. *)
  2170. err := sig;
  2171. IF trapped THEN DefaultTrapViewer END;
  2172. CASE sig OF
  2173. LinLibc.SIGINT:
  2174. err := 200 (* Interrupt (ANSI). *)
  2175. | LinLibc.SIGILL: (* Illegal instruction (ANSI). *)
  2176. err := 202; val := 0;
  2177. IF IsReadable(pc, pc + 4) THEN
  2178. SYSTEM.GET(pc, val);
  2179. IF val MOD 100H = 8DH THEN (* lea reg,reg *)
  2180. IF val DIV 100H MOD 100H = 0F0H THEN
  2181. err := val DIV 10000H MOD 100H (* trap *)
  2182. ELSIF val DIV 1000H MOD 10H = 0EH THEN
  2183. err := 128 + val DIV 100H MOD 10H (* run time error *)
  2184. END
  2185. END
  2186. END
  2187. | LinLibc.SIGFPE:
  2188. CASE siginfo.si_code OF
  2189. 0: (* TODO: ?????? *)
  2190. (* A. V. Shiryaev *)
  2191. (*
  2192. IF siginfo.si_int = 8 THEN
  2193. err := 139
  2194. ELSIF siginfo.si_int = 0 THEN
  2195. err := 143
  2196. END
  2197. *)
  2198. err := 143;
  2199. | LinLibc.FPE_INTDIV: err := 139 (* Integer divide by zero. *)
  2200. | LinLibc.FPE_INTOVF: err := 138 (* Integer overflow. *)
  2201. | LinLibc.FPE_FLTDIV: err := 140 (* Floating point divide by zero. *)
  2202. | LinLibc.FPE_FLTOVF: err := 142 (* Floating point overflow. *)
  2203. | LinLibc.FPE_FLTUND: err := 141 (* Floating point underflow. *)
  2204. | LinLibc.FPE_FLTRES: err := 143 (* Floating point inexact result. *)
  2205. | LinLibc.FPE_FLTINV: err := 143 (* Floating point invalid operation. *)
  2206. | LinLibc.FPE_FLTSUB: err := 134 (* Subscript out of range. *)
  2207. ELSE
  2208. END
  2209. | LinLibc.SIGSEGV: (* Segmentation violation (ANSI). *)
  2210. err := 203
  2211. ELSE
  2212. END;
  2213. INC(trapCount);
  2214. InitFpu;
  2215. TrapCleanup;
  2216. IF err # 128 THEN
  2217. IF (trapViewer = NIL) OR trapped THEN
  2218. DefaultTrapViewer
  2219. ELSE
  2220. trapped := TRUE;
  2221. trapViewer();
  2222. trapped := FALSE
  2223. END
  2224. END;
  2225. IF currentTryContext # NIL THEN (* Try failed *)
  2226. LinLibc.siglongjmp(currentTryContext, trapReturn)
  2227. ELSE
  2228. IF restart # NIL THEN (* Start failed *)
  2229. LinLibc.siglongjmp(loopContext, trapReturn)
  2230. END;
  2231. Quit(1);
  2232. END;
  2233. trapped := FALSE
  2234. END TrapHandler;
  2235. PROCEDURE InstallSignals*;
  2236. VAR sa, old: LinLibc.sigaction_t; res, i: INTEGER;
  2237. BEGIN
  2238. sa.sa_sigaction := TrapHandler;
  2239. (*
  2240. res := LinLibc.sigemptyset(SYSTEM.ADR(sa.sa_mask));
  2241. *)
  2242. res := LinLibc.sigfillset(SYSTEM.ADR(sa.sa_mask));
  2243. sa.sa_flags := LinLibc.SA_SIGINFO; (* TrapHandler takes three arguments *)
  2244. (*
  2245. IF LinLibc.sigaction(LinLibc.SIGINT, sa, old) # 0 THEN Msg("failed to install SIGINT") END;
  2246. IF LinLibc.sigaction(LinLibc.SIGILL, sa, old) # 0 THEN Msg("failed to install SIGILL") END;
  2247. IF LinLibc.sigaction(LinLibc.SIGFPE, sa, old) # 0 THEN Msg("failed to install SIGFPE") END;
  2248. IF LinLibc.sigaction(LinLibc.SIGSEGV, sa, old) # 0 THEN Msg("failed to install SIGSEGV") END;
  2249. IF LinLibc.sigaction(LinLibc.SIGPIPE, sa, old) # 0 THEN Msg("failed to install SIGPIPE") END;
  2250. IF LinLibc.sigaction(LinLibc.SIGTERM, sa, old) # 0 THEN Msg("failed to install SIGTERM") END;
  2251. *)
  2252. (* respond to all possible signals *)
  2253. FOR i := 1 TO LinLibc._NSIG - 1 DO
  2254. IF (i # LinLibc.SIGKILL)
  2255. & (i # LinLibc.SIGSTOP)
  2256. & (i # LinLibc.SIGWINCH)
  2257. THEN
  2258. IF LinLibc.sigaction(i, sa, old) # 0 THEN Msg("failed to install signal"); Int(i) END;
  2259. END
  2260. END
  2261. END InstallSignals;
  2262. PROCEDURE SetOpts;
  2263. VAR k: Module;
  2264. BEGIN
  2265. k := modList;
  2266. WHILE (k # NIL) & (k.name # "Kernel") DO k := k.next END;
  2267. ASSERT(k # NIL);
  2268. static := init IN k.opts;
  2269. inDll := dll IN k.opts
  2270. END SetOpts;
  2271. PROCEDURE SetCmdLine;
  2272. VAR i, l: INTEGER;
  2273. BEGIN
  2274. l := LEN(cmdLine);
  2275. cmdLine := bootInfo.argv[0]$;
  2276. FOR i := 1 TO bootInfo.argc - 1 DO cmdLine := cmdLine + " " + bootInfo.argv[i]END
  2277. END SetCmdLine;
  2278. PROCEDURE Init;
  2279. VAR (*excp: KERNEL32.ExcpFrm; *) t: Type; (*res: COM.RESULT; *) i: INTEGER;
  2280. env: LinLibc.jmp_buf; res: LONGINT;
  2281. BEGIN
  2282. InstallSignals; (* init exception handling *)
  2283. currentTryContext := NIL;
  2284. t := SYSTEM.VAL(Type, SYSTEM.ADR(Command)); (* type desc of Command *)
  2285. comSig := t.size; (* size = signature fprint for proc types *)
  2286. allocated := 0; total := 0; used := 0;
  2287. sentinelBlock.size := MAX(INTEGER);
  2288. sentinel := SYSTEM.ADR(sentinelBlock);
  2289. (* cdg/mf, 4.2.2004, dll support
  2290. SYSTEM.PUTREG(ML, SYSTEM.ADR(modList));
  2291. *)
  2292. IF dllMem THEN
  2293. i := N;
  2294. REPEAT DEC(i); free[i] := sentinel UNTIL i = 0;
  2295. root := NIL;
  2296. (*
  2297. heap := KERNEL32.GetProcessHeap()
  2298. *)
  2299. ELSE
  2300. i := N;
  2301. REPEAT DEC(i); free[i] := sentinel UNTIL i = 0;
  2302. AllocHeapMem(1, root); ASSERT(root # NIL, 100);
  2303. i := MIN(N - 1, (root.size - 12) DIV 16 - 1);
  2304. free[i] := SYSTEM.VAL(FreeBlock, SYSTEM.VAL(INTEGER, root) + 12);
  2305. free[i].next := sentinel;
  2306. free[i].size := (root.size - 12) DIV 16 * 16 - 4
  2307. END;
  2308. (*
  2309. res := WinOle.OleInitialize(0);
  2310. IF inDll THEN
  2311. baseStack := FPageWord(4) (* begin of stack segment *)
  2312. ELSE
  2313. InstallKeyboardInt;
  2314. InitFpu
  2315. END;
  2316. *)
  2317. InitFpu;
  2318. IF ~static THEN
  2319. InitModule(modList);
  2320. IF ~inDll THEN Quit(1) END
  2321. END;
  2322. told := 0; shift := 0;
  2323. END Init;
  2324. BEGIN
  2325. IF modList = NIL THEN (* only once *)
  2326. IF bootInfo # NIL THEN
  2327. modList := bootInfo.modList; (* boot loader initializes the bootInfo struct *)
  2328. SYSTEM.GETREG(SP, baseStack); (* TODO: Check that this is ok. *)
  2329. SetOpts;
  2330. SetCmdLine
  2331. ELSE
  2332. SYSTEM.GETREG(ML, modList); (* linker loads module list to BX *)
  2333. SYSTEM.GETREG(SP, baseStack);
  2334. static := init IN modList.opts;
  2335. inDll := dll IN modList.opts;
  2336. END;
  2337. (*
  2338. dllMem := inDll;
  2339. *)
  2340. Init
  2341. END
  2342. CLOSE
  2343. IF ~terminating THEN
  2344. terminating := TRUE;
  2345. Quit(0)
  2346. END
  2347. END Kernel.
  2348. (!)DevDecoder.Decode Kernel