Kernel.txt 59 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107
  1. MODULE Kernel;
  2. (* THIS IS TEXT COPY OF Kernel.odc *)
  3. (* DO NOT EDIT *)
  4. (* A. V. Shiryaev, 2012.10
  5. OpenBSD Kernel
  6. Based on 1.6-rc6 Windows Kernel
  7. + 20120822 Marc changes
  8. Some parts taken from OpenBUGS linKernel
  9. Most Windows-specific code removed
  10. Some Windows-specific code commented and marked red
  11. Windows COM-specific code re-marked from green to gray
  12. OpenBSD(/Linux)-specific code marked green
  13. TODO:
  14. handle stack overflow exceptions
  15. Quit from TrapHandler
  16. *)
  17. IMPORT S := SYSTEM, Libc := LinLibc, Dl := LinDl;
  18. CONST
  19. strictStackSweep = TRUE;
  20. nameLen* = 256;
  21. littleEndian* = TRUE;
  22. timeResolution* = 1000; (* ticks per second *)
  23. processor* = 10; (* i386 *)
  24. objType* = "ocf"; (* file types *)
  25. symType* = "osf";
  26. docType* = "odc";
  27. (* loader constants *)
  28. done* = 0;
  29. fileNotFound* = 1;
  30. syntaxError* = 2;
  31. objNotFound* = 3;
  32. illegalFPrint* = 4;
  33. cyclicImport* = 5;
  34. noMem* = 6;
  35. commNotFound* = 7;
  36. commSyntaxError* = 8;
  37. moduleNotFound* = 9;
  38. any = 1000000;
  39. CX = 1;
  40. SP = 4; (* register number of stack pointer *)
  41. FP = 5; (* register number of frame pointer *)
  42. ML = 3; (* register which holds the module list at program start *)
  43. N = 128 DIV 16; (* free lists *)
  44. (* kernel flags in module desc *)
  45. init = 16; dyn = 17; dll = 24; iptrs = 30;
  46. (* meta interface consts *)
  47. mConst = 1; mTyp = 2; mVar = 3; mProc = 4; mField = 5;
  48. debug = FALSE;
  49. (*
  50. sigStackSize = MAX(Libc.SIGSTKSZ, 65536);
  51. *)
  52. trapReturn = 1; (* Return value for sigsetjmp given from siglongjmp *)
  53. (* constants for the message boxes *)
  54. mbClose* = -1; mbOk* = 0; mbCancel* =1; mbRetry* = 2; mbIgnore* = 3; mbYes* = 4; mbNo* = 5;
  55. TYPE
  56. Name* = ARRAY nameLen OF SHORTCHAR;
  57. Command* = PROCEDURE;
  58. Module* = POINTER TO RECORD [untagged]
  59. next-: Module;
  60. opts-: SET; (* 0..15: compiler opts, 16..31: kernel flags *)
  61. refcnt-: INTEGER; (* <0: module invalidated *)
  62. compTime-, loadTime-: ARRAY 6 OF SHORTINT;
  63. ext-: INTEGER; (* currently not used *)
  64. term-: Command; (* terminator *)
  65. nofimps-, nofptrs-: INTEGER;
  66. csize-, dsize-, rsize-: INTEGER;
  67. code-, data-, refs-: INTEGER;
  68. procBase-, varBase-: INTEGER; (* meta base addresses *)
  69. names-: POINTER TO ARRAY [untagged] OF SHORTCHAR; (* names[0] = 0X *)
  70. ptrs-: POINTER TO ARRAY [untagged] OF INTEGER;
  71. imports-: POINTER TO ARRAY [untagged] OF Module;
  72. export-: Directory; (* exported objects (name sorted) *)
  73. name-: Name
  74. END;
  75. Type* = POINTER TO RECORD [untagged]
  76. (* record: ptr to method n at offset - 4 * (n+1) *)
  77. size-: INTEGER; (* record: size, array: #elem, dyn array: 0, proc: sigfp *)
  78. mod-: Module;
  79. id-: INTEGER; (* name idx * 256 + lev * 16 + attr * 4 + form *)
  80. base-: ARRAY 16 OF Type; (* signature if form = ProcTyp *)
  81. fields-: Directory; (* new fields (declaration order) *)
  82. ptroffs-: ARRAY any OF INTEGER (* array of any length *)
  83. END;
  84. Object* = POINTER TO ObjDesc;
  85. ObjDesc* = RECORD [untagged]
  86. fprint-: INTEGER;
  87. offs-: INTEGER; (* pvfprint for record types *)
  88. id-: INTEGER; (* name idx * 256 + vis * 16 + mode *)
  89. struct-: Type (* id of basic type or pointer to typedesc/signature *)
  90. END;
  91. Directory* = POINTER TO RECORD [untagged]
  92. num-: INTEGER; (* number of entries *)
  93. obj-: ARRAY any OF ObjDesc (* array of any length *)
  94. END;
  95. Signature* = POINTER TO RECORD [untagged]
  96. retStruct-: Type; (* id of basic type or pointer to typedesc or 0 *)
  97. num-: INTEGER; (* number of parameters *)
  98. par-: ARRAY any OF RECORD [untagged] (* parameters *)
  99. id-: INTEGER; (* name idx * 256 + kind *)
  100. struct-: Type (* id of basic type or pointer to typedesc *)
  101. END
  102. END;
  103. Handler* = PROCEDURE;
  104. Reducer* = POINTER TO ABSTRACT RECORD
  105. next: Reducer
  106. END;
  107. Identifier* = ABSTRACT RECORD
  108. typ*: INTEGER;
  109. obj-: ANYPTR
  110. END;
  111. TrapCleaner* = POINTER TO ABSTRACT RECORD
  112. next: TrapCleaner
  113. END;
  114. TryHandler* = PROCEDURE (a, b, c: INTEGER);
  115. (* meta extension suport *)
  116. ItemExt* = POINTER TO ABSTRACT RECORD END;
  117. ItemAttr* = RECORD
  118. obj*, vis*, typ*, adr*: INTEGER;
  119. mod*: Module;
  120. desc*: Type;
  121. ptr*: S.PTR;
  122. ext*: ItemExt
  123. END;
  124. Hook* = POINTER TO ABSTRACT RECORD END;
  125. LoaderHook* = POINTER TO ABSTRACT RECORD (Hook)
  126. res*: INTEGER;
  127. importing*, imported*, object*: ARRAY 256 OF CHAR
  128. END;
  129. GuiHook* = POINTER TO ABSTRACT RECORD (Hook) END; (* Implemented by HostGnome *)
  130. Block = POINTER TO RECORD [untagged]
  131. tag: Type;
  132. last: INTEGER; (* arrays: last element *)
  133. actual: INTEGER; (* arrays: used during mark phase *)
  134. first: INTEGER (* arrays: first element *)
  135. END;
  136. FreeBlock = POINTER TO FreeDesc;
  137. FreeDesc = RECORD [untagged]
  138. tag: Type; (* f.tag = ADR(f.size) *)
  139. size: INTEGER;
  140. next: FreeBlock
  141. END;
  142. Cluster = POINTER TO RECORD [untagged]
  143. size: INTEGER; (* total size *)
  144. next: Cluster;
  145. max: INTEGER
  146. (* start of first block *)
  147. END;
  148. FList = POINTER TO RECORD
  149. next: FList;
  150. blk: Block;
  151. iptr, aiptr: BOOLEAN
  152. END;
  153. CList = POINTER TO RECORD
  154. next: CList;
  155. do: Command;
  156. trapped: BOOLEAN
  157. END;
  158. PtrType = RECORD v: S.PTR END; (* used for array of pointer *)
  159. Char8Type = RECORD v: SHORTCHAR END;
  160. Char16Type = RECORD v: CHAR END;
  161. Int8Type = RECORD v: BYTE END;
  162. Int16Type = RECORD v: SHORTINT END;
  163. Int32Type = RECORD v: INTEGER END;
  164. Int64Type = RECORD v: LONGINT END;
  165. BoolType = RECORD v: BOOLEAN END;
  166. SetType = RECORD v: SET END;
  167. Real32Type = RECORD v: SHORTREAL END;
  168. Real64Type = RECORD v: REAL END;
  169. ProcType = RECORD v: PROCEDURE END;
  170. UPtrType = RECORD v: INTEGER END;
  171. StrPtr = POINTER TO ARRAY [untagged] OF SHORTCHAR;
  172. (* Linux specific boot loader info. Record must be identical to struct in the loader. *)
  173. BootInfo* = POINTER TO RECORD [untagged]
  174. modList: Module;
  175. argc-: INTEGER;
  176. argv-: Libc.StrArray
  177. END;
  178. VAR
  179. baseStack: INTEGER; (* modList, root, and baseStack must be together for remote debugging *)
  180. root: Cluster; (* cluster list *)
  181. modList-: Module; (* root of module list *)
  182. trapCount-: INTEGER;
  183. err-, pc-, sp-, fp-, stack-, val-: INTEGER;
  184. free: ARRAY N OF FreeBlock; (* free list *)
  185. sentinelBlock: FreeDesc;
  186. sentinel: FreeBlock;
  187. candidates: ARRAY 1024 OF INTEGER;
  188. nofcand: INTEGER;
  189. allocated: INTEGER; (* bytes allocated on BlackBox heap *)
  190. total: INTEGER; (* current total size of BlackBox heap *)
  191. used: INTEGER; (* bytes allocated on system heap *)
  192. finalizers: FList;
  193. hotFinalizers: FList;
  194. cleaners: CList;
  195. reducers: Reducer;
  196. trapStack: TrapCleaner;
  197. actual: Module; (* valid during module initialization *)
  198. res: INTEGER; (* auxiliary global variables used for trap handling *)
  199. old: INTEGER;
  200. trapViewer, trapChecker: Handler;
  201. trapped, guarded, secondTrap: BOOLEAN;
  202. interrupted: BOOLEAN;
  203. static, inDll, terminating: BOOLEAN;
  204. restart: Command;
  205. told, shift: INTEGER; (* used in Time() *)
  206. loader: LoaderHook;
  207. loadres: INTEGER;
  208. wouldFinalize: BOOLEAN;
  209. watcher*: PROCEDURE (event: INTEGER); (* for debugging *)
  210. (*
  211. sigStack: Libc.PtrVoid;
  212. *)
  213. loopContext: Libc.sigjmp_buf; (* trap return context, if no Kernel.Try has been used. *)
  214. currentTryContext: POINTER TO Libc.sigjmp_buf; (* trap return context, if Kernel.Try has been used. *)
  215. isReadableContext: Libc.sigjmp_buf; (* for IsReadable *)
  216. isReadableCheck: BOOLEAN;
  217. guiHook: GuiHook;
  218. (* !!! This variable has to be the last variable in the list. !!! *)
  219. bootInfo-: BootInfo;
  220. (* code procedures for fpu *)
  221. PROCEDURE [1] FINIT 0DBH, 0E3H;
  222. PROCEDURE [1] FLDCW 0D9H, 06DH, 0FCH; (* -4, FP *)
  223. PROCEDURE [1] FSTCW 0D9H, 07DH, 0FCH; (* -4, FP *)
  224. (* code procedure for memory erase *)
  225. PROCEDURE [code] Erase (adr, words: INTEGER)
  226. 089H, 0C7H, (* MOV EDI, EAX *)
  227. 031H, 0C0H, (* XOR EAX, EAX *)
  228. 059H, (* POP ECX *)
  229. 0F2H, 0ABH; (* REP STOS *)
  230. (* code procedure for stack allocate *)
  231. PROCEDURE [code] ALLOC (* argument in CX *)
  232. (*
  233. PUSH EAX
  234. ADD ECX,-5
  235. JNS L0
  236. XOR ECX,ECX
  237. L0: AND ECX,-4 (n-8+3)/4*4
  238. MOV EAX,ECX
  239. AND EAX,4095
  240. SUB ESP,EAX
  241. MOV EAX,ECX
  242. SHR EAX,12
  243. JEQ L2
  244. L1: PUSH 0
  245. SUB ESP,4092
  246. DEC EAX
  247. JNE L1
  248. L2: ADD ECX,8
  249. MOV EAX,[ESP,ECX,-4]
  250. PUSH EAX
  251. MOV EAX,[ESP,ECX,-4]
  252. SHR ECX,2
  253. RET
  254. *);
  255. PROCEDURE (VAR id: Identifier) Identified* (): BOOLEAN, NEW, ABSTRACT;
  256. PROCEDURE (r: Reducer) Reduce* (full: BOOLEAN), NEW, ABSTRACT;
  257. PROCEDURE (c: TrapCleaner) Cleanup*, NEW, EMPTY;
  258. (* meta extension suport *)
  259. PROCEDURE (e: ItemExt) Lookup* (name: ARRAY OF CHAR; VAR i: ANYREC), NEW, ABSTRACT;
  260. PROCEDURE (e: ItemExt) Index* (index: INTEGER; VAR elem: ANYREC), NEW, ABSTRACT;
  261. PROCEDURE (e: ItemExt) Deref* (VAR ref: ANYREC), NEW, ABSTRACT;
  262. PROCEDURE (e: ItemExt) Valid* (): BOOLEAN, NEW, ABSTRACT;
  263. PROCEDURE (e: ItemExt) Size* (): INTEGER, NEW, ABSTRACT;
  264. PROCEDURE (e: ItemExt) BaseTyp* (): INTEGER, NEW, ABSTRACT;
  265. PROCEDURE (e: ItemExt) Len* (): INTEGER, NEW, ABSTRACT;
  266. PROCEDURE (e: ItemExt) Call* (OUT ok: BOOLEAN), NEW, ABSTRACT;
  267. PROCEDURE (e: ItemExt) BoolVal* (): BOOLEAN, NEW, ABSTRACT;
  268. PROCEDURE (e: ItemExt) PutBoolVal* (x: BOOLEAN), NEW, ABSTRACT;
  269. PROCEDURE (e: ItemExt) CharVal* (): CHAR, NEW, ABSTRACT;
  270. PROCEDURE (e: ItemExt) PutCharVal* (x: CHAR), NEW, ABSTRACT;
  271. PROCEDURE (e: ItemExt) IntVal* (): INTEGER, NEW, ABSTRACT;
  272. PROCEDURE (e: ItemExt) PutIntVal* (x: INTEGER), NEW, ABSTRACT;
  273. PROCEDURE (e: ItemExt) LongVal* (): LONGINT, NEW, ABSTRACT;
  274. PROCEDURE (e: ItemExt) PutLongVal* (x: LONGINT), NEW, ABSTRACT;
  275. PROCEDURE (e: ItemExt) RealVal* (): REAL, NEW, ABSTRACT;
  276. PROCEDURE (e: ItemExt) PutRealVal* (x: REAL), NEW, ABSTRACT;
  277. PROCEDURE (e: ItemExt) SetVal* (): SET, NEW, ABSTRACT;
  278. PROCEDURE (e: ItemExt) PutSetVal* (x: SET), NEW, ABSTRACT;
  279. PROCEDURE (e: ItemExt) PtrVal* (): ANYPTR, NEW, ABSTRACT;
  280. PROCEDURE (e: ItemExt) PutPtrVal* (x: ANYPTR), NEW, ABSTRACT;
  281. PROCEDURE (e: ItemExt) GetSStringVal* (OUT x: ARRAY OF SHORTCHAR;
  282. OUT ok: BOOLEAN), NEW, ABSTRACT;
  283. PROCEDURE (e: ItemExt) PutSStringVal* (IN x: ARRAY OF SHORTCHAR;
  284. OUT ok: BOOLEAN), NEW, ABSTRACT;
  285. PROCEDURE (e: ItemExt) GetStringVal* (OUT x: ARRAY OF CHAR; OUT ok: BOOLEAN), NEW, ABSTRACT;
  286. PROCEDURE (e: ItemExt) PutStringVal* (IN x: ARRAY OF CHAR; OUT ok: BOOLEAN), NEW, ABSTRACT;
  287. (* -------------------- miscellaneous tools -------------------- *)
  288. PROCEDURE Msg (IN str: ARRAY OF CHAR);
  289. VAR ss: ARRAY 1024 OF SHORTCHAR; res, l: INTEGER;
  290. BEGIN
  291. ss := SHORT(str);
  292. l := LEN(ss$);
  293. ss[l] := 0AX; ss[l + 1] := 0X;
  294. res := Libc.printf(ss)
  295. END Msg;
  296. PROCEDURE Int (x: LONGINT);
  297. VAR j, k: INTEGER; ch: CHAR; a, s: ARRAY 32 OF CHAR;
  298. BEGIN
  299. IF x # MIN(LONGINT) THEN
  300. IF x < 0 THEN s[0] := "-"; k := 1; x := -x ELSE k := 0 END;
  301. j := 0; REPEAT a[j] := CHR(x MOD 10 + ORD("0")); x := x DIV 10; INC(j) UNTIL x = 0
  302. ELSE
  303. a := "8085774586302733229"; s[0] := "-"; k := 1;
  304. j := 0; WHILE a[j] # 0X DO INC(j) END
  305. END;
  306. ASSERT(k + j < LEN(s), 20);
  307. REPEAT DEC(j); ch := a[j]; s[k] := ch; INC(k) UNTIL j = 0;
  308. s[k] := 0X;
  309. Msg(s);
  310. END Int;
  311. PROCEDURE (h: GuiHook) MessageBox* (
  312. title, msg: ARRAY OF CHAR; buttons: SET): INTEGER, NEW, ABSTRACT;
  313. PROCEDURE (h: GuiHook) Beep*, NEW, ABSTRACT;
  314. (* Is extended by HostGnome to show dialogs. If no dialog is present or
  315. if the dialog is not closed by using one button, then "mbClose" is returned *)
  316. PROCEDURE MessageBox* (title, msg: ARRAY OF CHAR; buttons: SET): INTEGER;
  317. VAR res: INTEGER;
  318. BEGIN
  319. IF guiHook # NIL THEN
  320. res := guiHook.MessageBox(title, msg, buttons)
  321. ELSE
  322. Msg(" ");
  323. Msg("****");
  324. Msg("* " + title);
  325. Msg("* " + msg);
  326. Msg("****");
  327. res := mbClose;
  328. END;
  329. RETURN res
  330. END MessageBox;
  331. PROCEDURE SetGuiHook* (hook: GuiHook);
  332. BEGIN
  333. guiHook := hook
  334. END SetGuiHook;
  335. PROCEDURE SplitName* (name: ARRAY OF CHAR; VAR head, tail: ARRAY OF CHAR);
  336. (* portable *)
  337. VAR i, j: INTEGER; ch, lch: CHAR;
  338. BEGIN
  339. i := 0; ch := name[0];
  340. IF ch # 0X THEN
  341. REPEAT
  342. head[i] := ch; lch := ch; INC(i); ch := name[i]
  343. UNTIL (ch = 0X)
  344. OR ((ch >= "A") & (ch <= "Z") OR (ch >= "À") & (ch # "×") & (ch <= "Þ"))
  345. & ((lch < "A") OR (lch > "Z") & (lch < "À") OR (lch = "×") OR (lch > "Þ"));
  346. head[i] := 0X; j := 0;
  347. WHILE ch # 0X DO tail[j] := ch; INC(i); INC(j); ch := name[i] END;
  348. tail[j] := 0X;
  349. IF tail = "" THEN tail := head$; head := "" END
  350. ELSE head := ""; tail := ""
  351. END
  352. END SplitName;
  353. PROCEDURE MakeFileName* (VAR name: ARRAY OF CHAR; type: ARRAY OF CHAR);
  354. VAR i, j: INTEGER; ext: ARRAY 8 OF CHAR; ch: CHAR;
  355. BEGIN
  356. i := 0;
  357. WHILE (name[i] # 0X) & (name[i] # ".") DO INC(i) END;
  358. IF name[i] = "." THEN
  359. IF name[i + 1] = 0X THEN name[i] := 0X END
  360. ELSIF i < LEN(name) - 4 THEN
  361. IF type = "" THEN ext := docType ELSE ext := type$ END;
  362. name[i] := "."; INC(i); j := 0; ch := ext[0];
  363. WHILE ch # 0X DO
  364. IF (ch >= "A") & (ch <= "Z") THEN
  365. ch := CHR(ORD(ch) + ORD("a") - ORD("A"))
  366. END;
  367. name[i] := ch; INC(i); INC(j); ch := ext[j]
  368. END;
  369. name[i] := 0X
  370. END
  371. END MakeFileName;
  372. PROCEDURE Time* (): LONGINT;
  373. VAR t: INTEGER;
  374. BEGIN
  375. (* t := WinApi.GetTickCount(); *)
  376. (* A. V. Shiryaev: OpenBSD *)
  377. ASSERT(Libc.CLOCKS_PER_SEC = 100);
  378. t := 10 * Libc.clock();
  379. IF t < told THEN INC(shift) END;
  380. told := t;
  381. RETURN shift * 100000000L + t
  382. END Time;
  383. PROCEDURE Beep* ();
  384. VAR ss: ARRAY 2 OF SHORTCHAR;
  385. BEGIN
  386. IF guiHook # NIL THEN
  387. guiHook.Beep
  388. ELSE
  389. ss[0] := 007X; ss[1] := 0X;
  390. res := Libc.printf(ss); res := Libc.fflush(Libc.NULL)
  391. END
  392. END Beep;
  393. PROCEDURE SearchProcVar* (var: INTEGER; VAR m: Module; VAR adr: INTEGER);
  394. BEGIN
  395. adr := var; m := NIL;
  396. IF var # 0 THEN
  397. m := modList;
  398. WHILE (m # NIL) & ((var < m.code) OR (var >= m.code + m.csize)) DO m := m.next END;
  399. IF m # NIL THEN DEC(adr, m.code) END
  400. END
  401. END SearchProcVar;
  402. (* -------------------- system memory management --------------------- *)
  403. (* A. V. Shiryaev, 2012.10: NOTE: it seems that GC works correctly with positive addesses only *)
  404. (*
  405. PROCEDURE HeapAlloc (adr: INTEGER; size: INTEGER; prot: SET): Libc.PtrVoid;
  406. VAR
  407. x: Libc.PtrVoid;
  408. res: INTEGER;
  409. BEGIN
  410. x := Libc.calloc(1, size); (* calloc initialize allocated space to zero *)
  411. IF x # Libc.NULL THEN
  412. res := Libc.mprotect(x, size, prot);
  413. IF res # 0 THEN
  414. Libc.free(x);
  415. x := Libc.NULL;
  416. Msg("Kernel.HeapAlloc: mprotect failed!");
  417. HALT(100)
  418. END
  419. END;
  420. RETURN x
  421. END HeapAlloc;
  422. *)
  423. PROCEDURE HeapAlloc (adr: Libc.PtrVoid; size: INTEGER; prot: SET): Libc.PtrVoid;
  424. VAR x: Libc.PtrVoid;
  425. BEGIN
  426. x := Libc.mmap(adr, size, prot, Libc.MAP_PRIVATE + Libc.MAP_ANON, -1, 0);
  427. IF x = Libc.MAP_FAILED THEN
  428. x := Libc.NULL
  429. ELSE
  430. ASSERT(size MOD 4 = 0, 100);
  431. Erase(x, size DIV 4)
  432. END;
  433. RETURN x
  434. END HeapAlloc;
  435. (*
  436. PROCEDURE HeapFree (adr: Libc.PtrVoid; size: INTEGER);
  437. VAR res: INTEGER;
  438. BEGIN
  439. (*
  440. ASSERT(size MOD 4 = 0, 100);
  441. Erase(adr, size DIV 4);
  442. res := Libc.mprotect(adr, size, Libc.PROT_NONE);
  443. ASSERT(res = 0, 101);
  444. *)
  445. Libc.free(adr)
  446. END HeapFree;
  447. *)
  448. PROCEDURE HeapFree (adr: Libc.PtrVoid; size: INTEGER);
  449. VAR res: INTEGER;
  450. BEGIN
  451. (*
  452. ASSERT(size MOD 4 = 0, 100);
  453. Erase(adr, size DIV 4);
  454. res := Libc.mprotect(adr, size, Libc.PROT_NONE);
  455. ASSERT(res = 0, 101);
  456. *)
  457. res := Libc.munmap(adr, size);
  458. ASSERT(res = 0, 102)
  459. END HeapFree;
  460. PROCEDURE AllocHeapMem (size: INTEGER; VAR c: Cluster);
  461. (* allocate at least size bytes, typically at least 256 kbytes are allocated *)
  462. CONST N = 65536; (* cluster size for dll *)
  463. prot = Libc.PROT_READ + Libc.PROT_WRITE (* + Libc.PROT_EXEC *);
  464. VAR adr: INTEGER;
  465. allocated: INTEGER;
  466. BEGIN
  467. INC(size, 16);
  468. ASSERT(size > 0, 100); adr := 0;
  469. IF size < N THEN adr := HeapAlloc(1, N, prot) END;
  470. IF adr = 0 THEN adr := HeapAlloc(1, size, prot); allocated := size ELSE allocated := N END;
  471. IF adr = 0 THEN c := NIL
  472. ELSE
  473. c := S.VAL(Cluster, (adr + 15) DIV 16 * 16); c.max := adr;
  474. c.size := allocated - (S.VAL(INTEGER, c) - adr);
  475. INC(used, c.size); INC(total, c.size)
  476. END
  477. (* post: (c = NIL) OR (c MOD 16 = 0) & (c.size >= size) *)
  478. END AllocHeapMem;
  479. PROCEDURE FreeHeapMem (c: Cluster);
  480. BEGIN
  481. DEC(used, c.size); DEC(total, c.size);
  482. HeapFree(c.max, (S.VAL(INTEGER, c) - c.max) + c.size)
  483. END FreeHeapMem;
  484. PROCEDURE AllocModMem* (descSize, modSize: INTEGER; VAR descAdr, modAdr: INTEGER);
  485. CONST
  486. prot = Libc.PROT_READ + Libc.PROT_WRITE (* + Libc.PROT_EXEC *);
  487. BEGIN
  488. descAdr := HeapAlloc(0, descSize, prot);
  489. IF descAdr # 0 THEN
  490. modAdr := HeapAlloc(0, modSize, prot);
  491. IF modAdr # 0 THEN INC(used, descSize + modSize)
  492. ELSE HeapFree(descAdr, descSize); descAdr := 0
  493. END
  494. ELSE modAdr := 0
  495. END
  496. END AllocModMem;
  497. PROCEDURE DeallocModMem* (descSize, modSize, descAdr, modAdr: INTEGER);
  498. BEGIN
  499. DEC(used, descSize + modSize);
  500. HeapFree(descAdr, descSize);
  501. HeapFree(modAdr, modSize)
  502. END DeallocModMem;
  503. PROCEDURE InvalModMem (modSize, modAdr: INTEGER);
  504. BEGIN
  505. DEC(used, modSize);
  506. HeapFree(modAdr, modSize)
  507. END InvalModMem;
  508. (*
  509. PROCEDURE IsReadable* (from, to: INTEGER): BOOLEAN;
  510. (* check wether memory between from (incl.) and to (excl.) may be read *)
  511. BEGIN
  512. RETURN WinApi.IsBadReadPtr(from, to - from) = 0
  513. END IsReadable;
  514. *)
  515. (* Alexander Shiryaev, 2012.10: I do not know other way that works in OpenBSD *)
  516. (* This procedure can be called from TrapHandler also *)
  517. PROCEDURE IsReadable* (from, to: INTEGER): BOOLEAN;
  518. (* check wether memory between from (incl.) and to (excl.) may be read *)
  519. VAR res: BOOLEAN; res1: INTEGER;
  520. x: SHORTCHAR;
  521. mask, omask: Libc.sigset_t;
  522. BEGIN
  523. (* save old sigmask and unblock SIGSEGV *)
  524. res1 := Libc.sigemptyset(S.ADR(mask));
  525. ASSERT(res1 = 0, 100);
  526. res1 := Libc.sigaddset(S.ADR(mask), Libc.SIGSEGV);
  527. ASSERT(res1 = 0, 101);
  528. res1 := Libc.sigprocmask(Libc.SIG_UNBLOCK, mask, omask);
  529. ASSERT(res1 = 0, 102);
  530. res := FALSE;
  531. res1 := Libc.sigsetjmp(isReadableContext, Libc.TRUE);
  532. IF res1 = 0 THEN
  533. isReadableCheck := TRUE;
  534. (* read memory *)
  535. REPEAT
  536. S.GET(from, x);
  537. INC(from)
  538. UNTIL from = to;
  539. res := TRUE
  540. ELSE
  541. ASSERT(res1 = 1, 103)
  542. END;
  543. isReadableCheck := FALSE;
  544. (* restore saved sigmask *)
  545. res1 := Libc.sigprocmask(Libc.SIG_SETMASK, omask, NIL);
  546. ASSERT(res1 = 0, 104);
  547. RETURN res
  548. END IsReadable;
  549. (* --------------------- NEW implementation (portable) -------------------- *)
  550. PROCEDURE^ NewBlock (size: INTEGER): Block;
  551. PROCEDURE NewRec* (typ: INTEGER): INTEGER; (* implementation of NEW(ptr) *)
  552. VAR size: INTEGER; b: Block; tag: Type; l: FList;
  553. BEGIN
  554. IF ODD(typ) THEN (* record contains interface pointers *)
  555. tag := S.VAL(Type, typ - 1);
  556. b := NewBlock(tag.size);
  557. IF b = NIL THEN RETURN 0 END;
  558. b.tag := tag;
  559. l := S.VAL(FList, S.ADR(b.last)); (* anchor new object! *)
  560. l := S.VAL(FList, NewRec(S.TYP(FList))); (* NEW(l) *)
  561. l.blk := b; l.iptr := TRUE; l.next := finalizers; finalizers := l;
  562. RETURN S.ADR(b.last)
  563. ELSE
  564. tag := S.VAL(Type, typ);
  565. b := NewBlock(tag.size);
  566. IF b = NIL THEN RETURN 0 END;
  567. b.tag := tag; S.GET(typ - 4, size);
  568. IF size # 0 THEN (* record uses a finalizer *)
  569. l := S.VAL(FList, S.ADR(b.last)); (* anchor new object! *)
  570. l := S.VAL(FList, NewRec(S.TYP(FList))); (* NEW(l) *)
  571. l.blk := b; l.next := finalizers; finalizers := l
  572. END;
  573. RETURN S.ADR(b.last)
  574. END
  575. END NewRec;
  576. PROCEDURE NewArr* (eltyp, nofelem, nofdim: INTEGER): INTEGER; (* impl. of NEW(ptr, dim0, dim1, ...) *)
  577. VAR b: Block; size, headSize: INTEGER; t: Type; fin: BOOLEAN; l: FList;
  578. BEGIN
  579. IF (nofdim < 0)OR(nofdim>1FFFFFFCH) THEN RETURN 0 END;(*20120822 Marc*)
  580. headSize := 4 * nofdim + 12; fin := FALSE;
  581. CASE eltyp OF
  582. (*
  583. | -1: eltyp := S.ADR(IntPtrType); fin := TRUE
  584. *)
  585. | -1: HALT(100)
  586. | 0: eltyp := S.ADR(PtrType)
  587. | 1: eltyp := S.ADR(Char8Type)
  588. | 2: eltyp := S.ADR(Int16Type)
  589. | 3: eltyp := S.ADR(Int8Type)
  590. | 4: eltyp := S.ADR(Int32Type)
  591. | 5: eltyp := S.ADR(BoolType)
  592. | 6: eltyp := S.ADR(SetType)
  593. | 7: eltyp := S.ADR(Real32Type)
  594. | 8: eltyp := S.ADR(Real64Type)
  595. | 9: eltyp := S.ADR(Char16Type)
  596. | 10: eltyp := S.ADR(Int64Type)
  597. | 11: eltyp := S.ADR(ProcType)
  598. | 12: eltyp := S.ADR(UPtrType)
  599. ELSE (* eltyp is desc *)
  600. IF ODD(eltyp) THEN DEC(eltyp); fin := TRUE END
  601. END;
  602. t := S.VAL(Type, eltyp);
  603. ASSERT(t .size> 0,100);
  604. IF (nofelem < 0) OR( (7FFFFFFFH-headSize) DIV t.size < nofelem) THEN (* 20120822 Marc*)
  605. RETURN 0
  606. END;
  607. size := headSize + nofelem * t.size;
  608. b := NewBlock(size);
  609. IF b = NIL THEN RETURN 0 END;
  610. b.tag := S.VAL(Type, eltyp + 2); (* tag + array mark *)
  611. b.last := S.ADR(b.last) + size - t.size; (* pointer to last elem *)
  612. b.first := S.ADR(b.last) + headSize; (* pointer to first elem *)
  613. IF fin THEN
  614. l := S.VAL(FList, S.ADR(b.last)); (* anchor new object! *)
  615. l := S.VAL(FList, NewRec(S.TYP(FList))); (* NEW(l) *)
  616. l.blk := b; l.aiptr := TRUE; l.next := finalizers; finalizers := l
  617. END;
  618. RETURN S.ADR(b.last)
  619. END NewArr;
  620. (* -------------------- handler installation (portable) --------------------- *)
  621. PROCEDURE ThisFinObj* (VAR id: Identifier): ANYPTR;
  622. VAR l: FList;
  623. BEGIN
  624. ASSERT(id.typ # 0, 100);
  625. l := finalizers;
  626. WHILE l # NIL DO
  627. IF S.VAL(INTEGER, l.blk.tag) = id.typ THEN
  628. id.obj := S.VAL(ANYPTR, S.ADR(l.blk.last));
  629. IF id.Identified() THEN RETURN id.obj END
  630. END;
  631. l := l.next
  632. END;
  633. RETURN NIL
  634. END ThisFinObj;
  635. PROCEDURE InstallReducer* (r: Reducer);
  636. BEGIN
  637. r.next := reducers; reducers := r
  638. END InstallReducer;
  639. PROCEDURE InstallTrapViewer* (h: Handler);
  640. BEGIN
  641. trapViewer := h
  642. END InstallTrapViewer;
  643. PROCEDURE InstallTrapChecker* (h: Handler);
  644. BEGIN
  645. trapChecker := h
  646. END InstallTrapChecker;
  647. PROCEDURE PushTrapCleaner* (c: TrapCleaner);
  648. VAR t: TrapCleaner;
  649. BEGIN
  650. t := trapStack; WHILE (t # NIL) & (t # c) DO t := t.next END;
  651. ASSERT(t = NIL, 20);
  652. c.next := trapStack; trapStack := c
  653. END PushTrapCleaner;
  654. PROCEDURE PopTrapCleaner* (c: TrapCleaner);
  655. VAR t: TrapCleaner;
  656. BEGIN
  657. t := NIL;
  658. WHILE (trapStack # NIL) & (t # c) DO
  659. t := trapStack; trapStack := trapStack.next
  660. END
  661. END PopTrapCleaner;
  662. PROCEDURE InstallCleaner* (p: Command);
  663. VAR c: CList;
  664. BEGIN
  665. c := S.VAL(CList, NewRec(S.TYP(CList))); (* NEW(c) *)
  666. c.do := p; c.trapped := FALSE; c.next := cleaners; cleaners := c
  667. END InstallCleaner;
  668. PROCEDURE RemoveCleaner* (p: Command);
  669. VAR c0, c: CList;
  670. BEGIN
  671. c := cleaners; c0 := NIL;
  672. WHILE (c # NIL) & (c.do # p) DO c0 := c; c := c.next END;
  673. IF c # NIL THEN
  674. IF c0 = NIL THEN cleaners := cleaners.next ELSE c0.next := c.next END
  675. END
  676. END RemoveCleaner;
  677. PROCEDURE Cleanup*;
  678. VAR c, c0: CList;
  679. BEGIN
  680. c := cleaners; c0 := NIL;
  681. WHILE c # NIL DO
  682. IF ~c.trapped THEN
  683. c.trapped := TRUE; c.do; c.trapped := FALSE; c0 := c
  684. ELSE
  685. IF c0 = NIL THEN cleaners := cleaners.next
  686. ELSE c0.next := c.next
  687. END
  688. END;
  689. c := c.next
  690. END
  691. END Cleanup;
  692. (* -------------------- meta information (portable) --------------------- *)
  693. PROCEDURE (h: LoaderHook) ThisMod* (IN name: ARRAY OF SHORTCHAR): Module, NEW, ABSTRACT;
  694. PROCEDURE SetLoaderHook*(h: LoaderHook);
  695. BEGIN
  696. loader := h
  697. END SetLoaderHook;
  698. PROCEDURE InitModule (mod: Module); (* initialize linked modules *)
  699. VAR body: Command;
  700. res: INTEGER; errno: INTEGER;
  701. BEGIN
  702. IF ~(dyn IN mod.opts) & (mod.next # NIL) & ~(init IN mod.next.opts) THEN InitModule(mod.next) END;
  703. IF ~(init IN mod.opts) THEN
  704. body := S.VAL(Command, mod.code);
  705. INCL(mod.opts, init);
  706. actual := mod;
  707. (* A. V. Shiryaev: Allow execution on code pages *)
  708. res := Libc.mprotect(mod.code, mod.csize,
  709. Libc.PROT_READ + Libc.PROT_WRITE + Libc.PROT_EXEC);
  710. IF res = -1 THEN
  711. S.GET( Libc.__errno_location(), errno );
  712. Msg("ERROR: Kernel.InitModule: mprotect failed!");
  713. Msg(mod.name$); Int(mod.code); Int(mod.csize); Int(errno);
  714. HALT(100)
  715. ELSE ASSERT(res = 0)
  716. END;
  717. body(); actual := NIL
  718. END
  719. END InitModule;
  720. PROCEDURE ThisLoadedMod* (IN name: ARRAY OF SHORTCHAR): Module; (* loaded modules only *)
  721. VAR m: Module;
  722. BEGIN
  723. loadres := done;
  724. m := modList;
  725. WHILE (m # NIL) & ((m.name # name) OR (m.refcnt < 0)) DO m := m.next END;
  726. IF (m # NIL) & ~(init IN m.opts) THEN InitModule(m) END;
  727. IF m = NIL THEN loadres := moduleNotFound END;
  728. RETURN m
  729. END ThisLoadedMod;
  730. PROCEDURE ThisMod* (IN name: ARRAY OF CHAR): Module;
  731. VAR n : Name;
  732. BEGIN
  733. n := SHORT(name$);
  734. IF loader # NIL THEN
  735. loader.res := done;
  736. RETURN loader.ThisMod(n)
  737. ELSE
  738. RETURN ThisLoadedMod(n)
  739. END
  740. END ThisMod;
  741. PROCEDURE LoadMod* (IN name: ARRAY OF CHAR);
  742. VAR m: Module;
  743. BEGIN
  744. m := ThisMod(name)
  745. END LoadMod;
  746. PROCEDURE GetLoaderResult* (OUT res: INTEGER; OUT importing, imported, object: ARRAY OF CHAR);
  747. BEGIN
  748. IF loader # NIL THEN
  749. res := loader.res;
  750. importing := loader.importing$;
  751. imported := loader.imported$;
  752. object := loader.object$
  753. ELSE
  754. res := loadres;
  755. importing := "";
  756. imported := "";
  757. object := ""
  758. END
  759. END GetLoaderResult;
  760. PROCEDURE ThisObject* (mod: Module; name: ARRAY OF SHORTCHAR): Object;
  761. VAR l, r, m: INTEGER; p: StrPtr;
  762. BEGIN
  763. l := 0; r := mod.export.num;
  764. WHILE l < r DO (* binary search *)
  765. m := (l + r) DIV 2;
  766. p := S.VAL(StrPtr, S.ADR(mod.names[mod.export.obj[m].id DIV 256]));
  767. IF p^ = name THEN RETURN S.VAL(Object, S.ADR(mod.export.obj[m])) END;
  768. IF p^ < name THEN l := m + 1 ELSE r := m END
  769. END;
  770. RETURN NIL
  771. END ThisObject;
  772. PROCEDURE ThisDesc* (mod: Module; fprint: INTEGER): Object;
  773. VAR i, n: INTEGER;
  774. BEGIN
  775. i := 0; n := mod.export.num;
  776. WHILE (i < n) & (mod.export.obj[i].id DIV 256 = 0) DO
  777. IF mod.export.obj[i].offs = fprint THEN RETURN S.VAL(Object, S.ADR(mod.export.obj[i])) END;
  778. INC(i)
  779. END;
  780. RETURN NIL
  781. END ThisDesc;
  782. PROCEDURE ThisField* (rec: Type; name: ARRAY OF SHORTCHAR): Object;
  783. VAR n: INTEGER; p: StrPtr; obj: Object; m: Module;
  784. BEGIN
  785. m := rec.mod;
  786. obj := S.VAL(Object, S.ADR(rec.fields.obj[0])); n := rec.fields.num;
  787. WHILE n > 0 DO
  788. p := S.VAL(StrPtr, S.ADR(m.names[obj.id DIV 256]));
  789. IF p^ = name THEN RETURN obj END;
  790. DEC(n); INC(S.VAL(INTEGER, obj), 16)
  791. END;
  792. RETURN NIL
  793. END ThisField;
  794. PROCEDURE ThisCommand* (mod: Module; name: ARRAY OF SHORTCHAR): Command;
  795. VAR x: Object; sig: Signature;
  796. BEGIN
  797. x := ThisObject(mod, name);
  798. IF (x # NIL) & (x.id MOD 16 = mProc) THEN
  799. sig := S.VAL(Signature, x.struct);
  800. IF (sig.retStruct = NIL) & (sig.num = 0) THEN RETURN S.VAL(Command, mod.procBase + x.offs) END
  801. END;
  802. RETURN NIL
  803. END ThisCommand;
  804. PROCEDURE ThisType* (mod: Module; name: ARRAY OF SHORTCHAR): Type;
  805. VAR x: Object;
  806. BEGIN
  807. x := ThisObject(mod, name);
  808. IF (x # NIL) & (x.id MOD 16 = mTyp) & (S.VAL(INTEGER, x.struct) DIV 256 # 0) THEN
  809. RETURN x.struct
  810. ELSE
  811. RETURN NIL
  812. END
  813. END ThisType;
  814. PROCEDURE TypeOf* (IN rec: ANYREC): Type;
  815. BEGIN
  816. RETURN S.VAL(Type, S.TYP(rec))
  817. END TypeOf;
  818. PROCEDURE LevelOf* (t: Type): SHORTINT;
  819. BEGIN
  820. RETURN SHORT(t.id DIV 16 MOD 16)
  821. END LevelOf;
  822. PROCEDURE NewObj* (VAR o: S.PTR; t: Type);
  823. VAR i: INTEGER;
  824. BEGIN
  825. IF t.size = -1 THEN o := NIL
  826. ELSE
  827. i := 0; WHILE t.ptroffs[i] >= 0 DO INC(i) END;
  828. IF t.ptroffs[i+1] >= 0 THEN INC(S.VAL(INTEGER, t)) END; (* with interface pointers *)
  829. o := S.VAL(S.PTR, NewRec(S.VAL(INTEGER, t))) (* generic NEW *)
  830. END
  831. END NewObj;
  832. PROCEDURE GetObjName* (mod: Module; obj: Object; VAR name: Name);
  833. VAR p: StrPtr;
  834. BEGIN
  835. p := S.VAL(StrPtr, S.ADR(mod.names[obj.id DIV 256]));
  836. name := p^$
  837. END GetObjName;
  838. PROCEDURE GetTypeName* (t: Type; VAR name: Name);
  839. VAR p: StrPtr;
  840. BEGIN
  841. p := S.VAL(StrPtr, S.ADR(t.mod.names[t.id DIV 256]));
  842. name := p^$
  843. END GetTypeName;
  844. PROCEDURE RegisterMod* (mod: Module);
  845. VAR i: INTEGER;
  846. t: Libc.time_t; tm: Libc.tm;
  847. BEGIN
  848. mod.next := modList; modList := mod; mod.refcnt := 0; INCL(mod.opts, dyn); i := 0;
  849. WHILE i < mod.nofimps DO
  850. IF mod.imports[i] # NIL THEN INC(mod.imports[i].refcnt) END;
  851. INC(i)
  852. END;
  853. t := Libc.time(NIL);
  854. tm := Libc.localtime(t);
  855. mod.loadTime[0] := SHORT(tm.tm_year + 1900); (* Linux counts years from 1900 but BlackBox from 0000 *)
  856. mod.loadTime[1] := SHORT(tm.tm_mon + 1) (* Linux month range 0-11 but BB month range 1-12 *);
  857. mod.loadTime[2] := SHORT(tm.tm_mday);
  858. mod.loadTime[3] := SHORT(tm.tm_hour);
  859. mod.loadTime[4] := SHORT(tm.tm_min);
  860. mod.loadTime[5] := SHORT(tm.tm_sec);
  861. tm := NIL;
  862. IF ~(init IN mod.opts) THEN InitModule(mod) END
  863. END RegisterMod;
  864. PROCEDURE^ Collect*;
  865. PROCEDURE UnloadMod* (mod: Module);
  866. VAR i: INTEGER; t: Command;
  867. BEGIN
  868. IF mod.refcnt = 0 THEN
  869. t := mod.term; mod.term := NIL;
  870. IF t # NIL THEN t() END; (* terminate module *)
  871. i := 0;
  872. WHILE i < mod.nofptrs DO (* release global pointers *)
  873. S.PUT(mod.varBase + mod.ptrs[i], 0); INC(i)
  874. END;
  875. (*
  876. ReleaseIPtrs(mod); (* release global interface pointers *)
  877. *)
  878. Collect; (* call finalizers *)
  879. i := 0;
  880. WHILE i < mod.nofimps DO (* release imported modules *)
  881. IF mod.imports[i] # NIL THEN DEC(mod.imports[i].refcnt) END;
  882. INC(i)
  883. END;
  884. mod.refcnt := -1;
  885. IF dyn IN mod.opts THEN (* release memory *)
  886. InvalModMem(mod.data + mod.dsize - mod.refs, mod.refs)
  887. END
  888. END
  889. END UnloadMod;
  890. (* -------------------- dynamic procedure call --------------------- *) (* COMPILER DEPENDENT *)
  891. PROCEDURE [1] PUSH (p: INTEGER) 050H; (* push AX *)
  892. PROCEDURE [1] CALL (a: INTEGER) 0FFH, 0D0H; (* call AX *)
  893. PROCEDURE [1] RETI (): LONGINT;
  894. PROCEDURE [1] RETR (): REAL;
  895. (*
  896. type par
  897. 32 bit scalar value
  898. 64 bit scalar low hi
  899. var scalar address
  900. record address tag
  901. array address size
  902. open array address length .. length
  903. *)
  904. PROCEDURE Call* (adr: INTEGER; sig: Signature; IN par: ARRAY OF INTEGER; n: INTEGER): LONGINT;
  905. VAR p, kind, sp, size: INTEGER; typ: Type; r: REAL;
  906. BEGIN
  907. p := sig.num;
  908. WHILE p > 0 DO (* push parameters from right to left *)
  909. DEC(p);
  910. typ := sig.par[p].struct;
  911. kind := sig.par[p].id MOD 16;
  912. IF (S.VAL(INTEGER, typ) DIV 256 = 0) OR (typ.id MOD 4 IN {0, 3}) THEN (* scalar *)
  913. IF (kind = 10) & ((S.VAL(INTEGER, typ) = 8) OR (S.VAL(INTEGER, typ) = 10)) THEN (* 64 bit *)
  914. DEC(n); PUSH(par[n]) (* push hi word *)
  915. END;
  916. DEC(n); PUSH(par[n]) (* push value/address *)
  917. ELSIF typ.id MOD 4 = 1 THEN (* record *)
  918. IF kind # 10 THEN (* var par *)
  919. DEC(n); PUSH(par[n]); (* push tag *)
  920. DEC(n); PUSH(par[n]) (* push address *)
  921. ELSE
  922. DEC(n, 2); (* skip tag *)
  923. S.GETREG(SP, sp); sp := (sp - typ.size) DIV 4 * 4; S.PUTREG(SP, sp); (* allocate space *)
  924. S.MOVE(par[n], sp, typ.size) (* copy to stack *)
  925. END
  926. ELSIF typ.size = 0 THEN (* open array *)
  927. size := typ.id DIV 16 MOD 16; (* number of open dimensions *)
  928. WHILE size > 0 DO
  929. DEC(size); DEC(n); PUSH(par[n]) (* push length *)
  930. END;
  931. DEC(n); PUSH(par[n]) (* push address *)
  932. ELSE (* fix array *)
  933. IF kind # 10 THEN (* var par *)
  934. DEC(n, 2); PUSH(par[n]) (* push address *)
  935. ELSE
  936. DEC(n); size := par[n]; DEC(n);
  937. S.GETREG(SP, sp); sp := (sp - size) DIV 4 * 4; S.PUTREG(SP, sp); (* allocate space *)
  938. S.MOVE(par[n], sp, size) (* copy to stack *)
  939. END
  940. END
  941. END;
  942. ASSERT(n = 0);
  943. IF S.VAL(INTEGER, sig.retStruct) = 7 THEN (* shortreal *)
  944. CALL(adr);
  945. RETURN S.VAL(INTEGER, SHORT(RETR())) (* return value in fpu register *)
  946. ELSIF S.VAL(INTEGER, sig.retStruct) = 8 THEN (* real *)
  947. CALL(adr); r := RETR();
  948. RETURN S.VAL(LONGINT, r) (* return value in fpu register *)
  949. ELSE
  950. CALL(adr);
  951. RETURN RETI() (* return value in integer registers *)
  952. END
  953. END Call;
  954. (* -------------------- reference information (portable) --------------------- *)
  955. PROCEDURE RefCh (VAR ref: INTEGER; VAR ch: SHORTCHAR);
  956. BEGIN
  957. S.GET(ref, ch); INC(ref)
  958. END RefCh;
  959. PROCEDURE RefNum (VAR ref: INTEGER; VAR x: INTEGER);
  960. VAR s, n: INTEGER; ch: SHORTCHAR;
  961. BEGIN
  962. s := 0; n := 0; RefCh(ref, ch);
  963. WHILE ORD(ch) >= 128 DO INC(n, ASH(ORD(ch) - 128, s) ); INC(s, 7); RefCh(ref, ch) END;
  964. x := n + ASH(ORD(ch) MOD 64 - ORD(ch) DIV 64 * 64, s)
  965. END RefNum;
  966. PROCEDURE RefName (VAR ref: INTEGER; VAR n: Name);
  967. VAR i: INTEGER; ch: SHORTCHAR;
  968. BEGIN
  969. i := 0; RefCh(ref, ch);
  970. WHILE ch # 0X DO n[i] := ch; INC(i); RefCh(ref, ch) END;
  971. n[i] := 0X
  972. END RefName;
  973. PROCEDURE GetRefProc* (VAR ref: INTEGER; VAR adr: INTEGER; VAR name: Name);
  974. VAR ch: SHORTCHAR;
  975. BEGIN
  976. S.GET(ref, ch);
  977. WHILE ch >= 0FDX DO (* skip variables *)
  978. INC(ref); RefCh(ref, ch);
  979. IF ch = 10X THEN INC(ref, 4) END;
  980. RefNum(ref, adr); RefName(ref, name); S.GET(ref, ch)
  981. END;
  982. WHILE (ch > 0X) & (ch < 0FCX) DO (* skip source refs *)
  983. INC(ref); RefNum(ref, adr); S.GET(ref, ch)
  984. END;
  985. IF ch = 0FCX THEN INC(ref); RefNum(ref, adr); RefName(ref, name)
  986. ELSE adr := 0
  987. END
  988. END GetRefProc;
  989. PROCEDURE GetRefVar* (VAR ref: INTEGER; VAR mode, form: SHORTCHAR; VAR desc: Type;
  990. VAR adr: INTEGER; VAR name: Name);
  991. BEGIN
  992. S.GET(ref, mode); desc := NIL;
  993. IF mode >= 0FDX THEN
  994. mode := SHORT(CHR(ORD(mode) - 0FCH));
  995. INC(ref); RefCh(ref, form);
  996. IF form = 10X THEN
  997. S.GET(ref, desc); INC(ref, 4); form := SHORT(CHR(16 + desc.id MOD 4))
  998. END;
  999. RefNum(ref, adr); RefName(ref, name)
  1000. ELSE
  1001. mode := 0X; form := 0X; adr := 0
  1002. END
  1003. END GetRefVar;
  1004. PROCEDURE SourcePos* (mod: Module; codePos: INTEGER): INTEGER;
  1005. VAR ref, pos, ad, d: INTEGER; ch: SHORTCHAR; name: Name;
  1006. BEGIN
  1007. ref := mod.refs; pos := 0; ad := 0; S.GET(ref, ch);
  1008. WHILE ch # 0X DO
  1009. WHILE (ch > 0X) & (ch < 0FCX) DO
  1010. INC(ad, ORD(ch)); INC(ref); RefNum(ref, d);
  1011. IF ad > codePos THEN RETURN pos END;
  1012. INC(pos, d); S.GET(ref, ch)
  1013. END;
  1014. IF ch = 0FCX THEN INC(ref); RefNum(ref, d); RefName(ref, name); S.GET(ref, ch) END;
  1015. WHILE ch >= 0FDX DO (* skip variables *)
  1016. INC(ref); RefCh(ref, ch);
  1017. IF ch = 10X THEN INC(ref, 4) END;
  1018. RefNum(ref, d); RefName(ref, name); S.GET(ref, ch)
  1019. END
  1020. END;
  1021. RETURN -1
  1022. END SourcePos;
  1023. (* -------------------- dynamic link libraries --------------------- *)
  1024. (*
  1025. PROCEDURE DlOpen (name: ARRAY OF SHORTCHAR): Dl.HANDLE;
  1026. CONST flags = Dl.RTLD_LAZY + Dl.RTLD_GLOBAL;
  1027. VAR h: Dl.HANDLE;
  1028. i: INTEGER;
  1029. BEGIN
  1030. h := Dl.NULL;
  1031. i := 0; WHILE (i < LEN(name)) & (name[i] # 0X) DO INC(i) END;
  1032. IF i < LEN(name) THEN
  1033. h := Dl.dlopen(name, flags);
  1034. WHILE (h = Dl.NULL) & (i > 0) DO
  1035. DEC(i);
  1036. WHILE (i > 0) & (name[i] # '.') DO DEC(i) END;
  1037. IF i > 0 THEN
  1038. name[i] := 0X;
  1039. h := Dl.dlopen(name, flags);
  1040. (* IF h # Dl.NULL THEN Msg(name$) END *)
  1041. END
  1042. END
  1043. END;
  1044. RETURN h
  1045. END DlOpen;
  1046. *)
  1047. PROCEDURE LoadDll* (IN name: ARRAY OF SHORTCHAR; VAR ok: BOOLEAN);
  1048. VAR h: Dl.HANDLE;
  1049. BEGIN
  1050. ok := FALSE;
  1051. h := Dl.dlopen(name, Dl.RTLD_LAZY + Dl.RTLD_GLOBAL);
  1052. IF h # Dl.NULL THEN ok := TRUE END
  1053. END LoadDll;
  1054. PROCEDURE ThisDllObj* (mode, fprint: INTEGER; IN dll, name: ARRAY OF SHORTCHAR): INTEGER;
  1055. VAR ad: INTEGER; h: Dl.HANDLE;
  1056. BEGIN
  1057. ad := 0;
  1058. IF mode IN {mVar, mProc} THEN
  1059. h := Dl.dlopen(dll, Dl.RTLD_LAZY+ Dl.RTLD_GLOBAL);
  1060. IF h # Dl.NULL THEN
  1061. ad := Dl.dlsym(h, name);
  1062. END
  1063. END;
  1064. RETURN ad
  1065. END ThisDllObj;
  1066. (* -------------------- garbage collector (portable) --------------------- *)
  1067. PROCEDURE Mark (this: Block);
  1068. VAR father, son: Block; tag: Type; flag, offset, actual: INTEGER;
  1069. BEGIN
  1070. IF ~ODD(S.VAL(INTEGER, this.tag)) THEN
  1071. father := NIL;
  1072. LOOP
  1073. INC(S.VAL(INTEGER, this.tag));
  1074. flag := S.VAL(INTEGER, this.tag) MOD 4;
  1075. tag := S.VAL(Type, S.VAL(INTEGER, this.tag) - flag);
  1076. IF flag >= 2 THEN actual := this.first; this.actual := actual
  1077. ELSE actual := S.ADR(this.last)
  1078. END;
  1079. LOOP
  1080. offset := tag.ptroffs[0];
  1081. IF offset < 0 THEN
  1082. INC(S.VAL(INTEGER, tag), offset + 4); (* restore tag *)
  1083. IF (flag >= 2) & (actual < this.last) & (offset < -4) THEN (* next array element *)
  1084. INC(actual, tag.size); this.actual := actual
  1085. ELSE (* up *)
  1086. this.tag := S.VAL(Type, S.VAL(INTEGER, tag) + flag);
  1087. IF father = NIL THEN RETURN END;
  1088. son := this; this := father;
  1089. flag := S.VAL(INTEGER, this.tag) MOD 4;
  1090. tag := S.VAL(Type, S.VAL(INTEGER, this.tag) - flag);
  1091. offset := tag.ptroffs[0];
  1092. IF flag >= 2 THEN actual := this.actual ELSE actual := S.ADR(this.last) END;
  1093. S.GET(actual + offset, father); S.PUT(actual + offset, S.ADR(son.last));
  1094. INC(S.VAL(INTEGER, tag), 4)
  1095. END
  1096. ELSE
  1097. S.GET(actual + offset, son);
  1098. IF son # NIL THEN
  1099. DEC(S.VAL(INTEGER, son), 4);
  1100. IF ~ODD(S.VAL(INTEGER, son.tag)) THEN (* down *)
  1101. this.tag := S.VAL(Type, S.VAL(INTEGER, tag) + flag);
  1102. S.PUT(actual + offset, father); father := this; this := son;
  1103. EXIT
  1104. END
  1105. END;
  1106. INC(S.VAL(INTEGER, tag), 4)
  1107. END
  1108. END
  1109. END
  1110. END
  1111. END Mark;
  1112. PROCEDURE MarkGlobals;
  1113. VAR m: Module; i, p: INTEGER;
  1114. BEGIN
  1115. m := modList;
  1116. WHILE m # NIL DO
  1117. IF m.refcnt >= 0 THEN
  1118. i := 0;
  1119. WHILE i < m.nofptrs DO
  1120. S.GET(m.varBase + m.ptrs[i], p); INC(i);
  1121. IF p # 0 THEN Mark(S.VAL(Block, p - 4)) END
  1122. END
  1123. END;
  1124. m := m.next
  1125. END
  1126. END MarkGlobals;
  1127. (* This is the specification for the code procedure following below:
  1128. PROCEDURE Next (b: Block): Block; (* next block in same cluster *)
  1129. VAR size: INTEGER;
  1130. BEGIN
  1131. S.GET(S.VAL(INTEGER, b.tag) DIV 4 * 4, size);
  1132. IF ODD(S.VAL(INTEGER, b.tag) DIV 2) THEN INC(size, b.last - S.ADR(b.last)) END;
  1133. RETURN S.VAL(Block, S.VAL(INTEGER, b) + (size + 19) DIV 16 * 16)
  1134. END Next;
  1135. *)
  1136. PROCEDURE [code] Next (b: Block): Block (* next block in same cluster *)
  1137. (*
  1138. MOV ECX,[EAX] b.tag
  1139. AND CL,0FCH b.tag DIV * 4
  1140. MOV ECX,[ECX] size
  1141. TESTB [EAX],02H ODD(b.tag DIV 2)
  1142. JE L1
  1143. ADD ECX,[EAX,4] size + b.last
  1144. SUB ECX,EAX
  1145. SUB ECX,4 size + b.last - ADR(b.last)
  1146. L1:
  1147. ADD ECX,19 size + 19
  1148. AND CL,0F0H (size + 19) DIV 16 * 16
  1149. ADD EAX,ECX b + size
  1150. *)
  1151. 08BH, 008H,
  1152. 080H, 0E1H, 0FCH,
  1153. 08BH, 009H,
  1154. 0F6H, 000H, 002H,
  1155. 074H, 008H,
  1156. 003H, 048H, 004H,
  1157. 029H, 0C1H,
  1158. 083H, 0E9H, 004H,
  1159. 083H, 0C1H, 013H,
  1160. 080H, 0E1H, 0F0H,
  1161. 001H, 0C8H;
  1162. PROCEDURE CheckCandidates;
  1163. (* pre: nofcand > 0 *)
  1164. VAR i, j, h, p, end: INTEGER; c: Cluster; blk, next: Block;
  1165. BEGIN
  1166. (* sort candidates (shellsort) *)
  1167. h := 1; REPEAT h := h*3 + 1 UNTIL h > nofcand;
  1168. REPEAT h := h DIV 3; i := h;
  1169. WHILE i < nofcand DO p := candidates[i]; j := i;
  1170. WHILE (j >= h) & (candidates[j-h] > p) DO
  1171. candidates[j] := candidates[j-h]; j := j-h
  1172. END;
  1173. candidates[j] := p; INC(i)
  1174. END
  1175. UNTIL h = 1;
  1176. (* sweep *)
  1177. c := root; i := 0;
  1178. WHILE c # NIL DO
  1179. blk := S.VAL(Block, S.VAL(INTEGER, c) + 12);
  1180. end := S.VAL(INTEGER, blk) + (c.size - 12) DIV 16 * 16;
  1181. WHILE candidates[i] < S.VAL(INTEGER, blk) DO
  1182. INC(i);
  1183. IF i = nofcand THEN RETURN END
  1184. END;
  1185. WHILE S.VAL(INTEGER, blk) < end DO
  1186. next := Next(blk);
  1187. IF candidates[i] < S.VAL(INTEGER, next) THEN
  1188. IF (S.VAL(INTEGER, blk.tag) # S.ADR(blk.last)) (* not a free block *)
  1189. & (~strictStackSweep OR (candidates[i] = S.ADR(blk.last))) THEN
  1190. Mark(blk)
  1191. END;
  1192. REPEAT
  1193. INC(i);
  1194. IF i = nofcand THEN RETURN END
  1195. UNTIL candidates[i] >= S.VAL(INTEGER, next)
  1196. END;
  1197. IF (S.VAL(INTEGER, blk.tag) MOD 4 = 0) & (S.VAL(INTEGER, blk.tag) # S.ADR(blk.last))
  1198. & (blk.tag.base[0] = NIL) & (blk.actual > 0) THEN (* referenced interface record *)
  1199. Mark(blk)
  1200. END;
  1201. blk := next
  1202. END;
  1203. c := c.next
  1204. END
  1205. END CheckCandidates;
  1206. PROCEDURE MarkLocals;
  1207. VAR sp, p, min, max: INTEGER; c: Cluster;
  1208. BEGIN
  1209. S.GETREG(FP, sp); nofcand := 0; c := root;
  1210. WHILE c.next # NIL DO c := c.next END;
  1211. min := S.VAL(INTEGER, root); max := S.VAL(INTEGER, c) + c.size;
  1212. WHILE sp < baseStack DO
  1213. S.GET(sp, p);
  1214. IF (p > min) & (p < max) & (~strictStackSweep OR (p MOD 16 = 0)) THEN
  1215. candidates[nofcand] := p; INC(nofcand);
  1216. IF nofcand = LEN(candidates) - 1 THEN CheckCandidates; nofcand := 0 END
  1217. END;
  1218. INC(sp, 4)
  1219. END;
  1220. candidates[nofcand] := max; INC(nofcand); (* ensure complete scan for interface mark*)
  1221. IF nofcand > 0 THEN CheckCandidates END
  1222. END MarkLocals;
  1223. PROCEDURE MarkFinObj;
  1224. VAR f: FList;
  1225. BEGIN
  1226. wouldFinalize := FALSE;
  1227. f := finalizers;
  1228. WHILE f # NIL DO
  1229. IF ~ODD(S.VAL(INTEGER, f.blk.tag)) THEN wouldFinalize := TRUE END;
  1230. Mark(f.blk);
  1231. f := f.next
  1232. END;
  1233. f := hotFinalizers;
  1234. WHILE f # NIL DO IF ~ODD(S.VAL(INTEGER, f.blk.tag)) THEN wouldFinalize := TRUE END;
  1235. Mark(f.blk);
  1236. f := f.next
  1237. END
  1238. END MarkFinObj;
  1239. PROCEDURE CheckFinalizers;
  1240. VAR f, g, h, k: FList;
  1241. BEGIN
  1242. f := finalizers; g := NIL;
  1243. IF hotFinalizers = NIL THEN k := NIL
  1244. ELSE
  1245. k := hotFinalizers;
  1246. WHILE k.next # NIL DO k := k.next END
  1247. END;
  1248. WHILE f # NIL DO
  1249. h := f; f := f.next;
  1250. IF ~ODD(S.VAL(INTEGER, h.blk.tag)) THEN
  1251. IF g = NIL THEN finalizers := f ELSE g.next := f END;
  1252. IF k = NIL THEN hotFinalizers := h ELSE k.next := h END;
  1253. k := h; h.next := NIL
  1254. ELSE g := h
  1255. END
  1256. END;
  1257. h := hotFinalizers;
  1258. WHILE h # NIL DO Mark(h.blk); h := h.next END
  1259. END CheckFinalizers;
  1260. PROCEDURE ExecFinalizer (a, b, c: INTEGER);
  1261. VAR f: FList; fin: PROCEDURE(this: ANYPTR);
  1262. BEGIN
  1263. f := S.VAL(FList, a);
  1264. IF f.aiptr THEN (* ArrFinalizer(S.VAL(ANYPTR, S.ADR(f.blk.last))) *)
  1265. ELSE
  1266. S.GET(S.VAL(INTEGER, f.blk.tag) - 4, fin); (* method 0 *)
  1267. IF (fin # NIL) & (f.blk.tag.mod.refcnt >= 0) THEN fin(S.VAL(ANYPTR, S.ADR(f.blk.last))) END;
  1268. (*
  1269. IF f.iptr THEN RecFinalizer(S.VAL(ANYPTR, S.ADR(f.blk.last))) END
  1270. *)
  1271. END
  1272. END ExecFinalizer;
  1273. PROCEDURE^ Try* (h: TryHandler; a, b, c: INTEGER); (* COMPILER DEPENDENT *)
  1274. PROCEDURE CallFinalizers;
  1275. VAR f: FList;
  1276. BEGIN
  1277. WHILE hotFinalizers # NIL DO
  1278. f := hotFinalizers; hotFinalizers := hotFinalizers.next;
  1279. Try(ExecFinalizer, S.VAL(INTEGER, f), 0, 0)
  1280. END;
  1281. wouldFinalize := FALSE
  1282. END CallFinalizers;
  1283. PROCEDURE Insert (blk: FreeBlock; size: INTEGER); (* insert block in free list *)
  1284. VAR i: INTEGER;
  1285. BEGIN
  1286. blk.size := size - 4; blk.tag := S.VAL(Type, S.ADR(blk.size));
  1287. i := MIN(N - 1, (blk.size DIV 16));
  1288. blk.next := free[i]; free[i] := blk
  1289. END Insert;
  1290. PROCEDURE Sweep (dealloc: BOOLEAN);
  1291. VAR cluster, last, c: Cluster; blk, next: Block; fblk, b, t: FreeBlock; end, i: INTEGER;
  1292. BEGIN
  1293. cluster := root; last := NIL; allocated := 0;
  1294. i := N;
  1295. REPEAT DEC(i); free[i] := sentinel UNTIL i = 0;
  1296. WHILE cluster # NIL DO
  1297. blk := S.VAL(Block, S.VAL(INTEGER, cluster) + 12);
  1298. end := S.VAL(INTEGER, blk) + (cluster.size - 12) DIV 16 * 16;
  1299. fblk := NIL;
  1300. WHILE S.VAL(INTEGER, blk) < end DO
  1301. next := Next(blk);
  1302. IF ODD(S.VAL(INTEGER, blk.tag)) THEN
  1303. IF fblk # NIL THEN
  1304. Insert(fblk, S.VAL(INTEGER, blk) - S.VAL(INTEGER, fblk));
  1305. fblk := NIL
  1306. END;
  1307. DEC(S.VAL(INTEGER, blk.tag)); (* unmark *)
  1308. INC(allocated, S.VAL(INTEGER, next) - S.VAL(INTEGER, blk))
  1309. ELSIF fblk = NIL THEN
  1310. fblk := S.VAL(FreeBlock, blk)
  1311. END;
  1312. blk := next
  1313. END;
  1314. IF dealloc & (S.VAL(INTEGER, fblk) = S.VAL(INTEGER, cluster) + 12) THEN (* deallocate cluster *)
  1315. c := cluster; cluster := cluster.next;
  1316. IF last = NIL THEN root := cluster ELSE last.next := cluster END;
  1317. FreeHeapMem(c)
  1318. ELSE
  1319. IF fblk # NIL THEN Insert(fblk, end - S.VAL(INTEGER, fblk)) END;
  1320. last := cluster; cluster := cluster.next
  1321. END
  1322. END;
  1323. (* reverse free list *)
  1324. i := N;
  1325. REPEAT
  1326. DEC(i);
  1327. b := free[i]; fblk := sentinel;
  1328. WHILE b # sentinel DO t := b; b := t.next; t.next := fblk; fblk := t END;
  1329. free[i] := fblk
  1330. UNTIL i = 0
  1331. END Sweep;
  1332. PROCEDURE Collect*;
  1333. BEGIN
  1334. IF root # NIL THEN
  1335. CallFinalizers; (* trap cleanup *)
  1336. IF debug & (watcher # NIL) THEN watcher(1) END;
  1337. MarkGlobals;
  1338. MarkLocals;
  1339. CheckFinalizers;
  1340. Sweep(TRUE);
  1341. CallFinalizers
  1342. END
  1343. END Collect;
  1344. PROCEDURE FastCollect*;
  1345. BEGIN
  1346. IF root # NIL THEN
  1347. IF debug & (watcher # NIL) THEN watcher(2) END;
  1348. MarkGlobals;
  1349. MarkLocals;
  1350. MarkFinObj;
  1351. Sweep(FALSE)
  1352. END
  1353. END FastCollect;
  1354. PROCEDURE WouldFinalize* (): BOOLEAN;
  1355. BEGIN
  1356. RETURN wouldFinalize
  1357. END WouldFinalize;
  1358. (* --------------------- memory allocation (portable) -------------------- *)
  1359. PROCEDURE OldBlock (size: INTEGER): FreeBlock; (* size MOD 16 = 0 *)
  1360. VAR b, l: FreeBlock; s, i: INTEGER;
  1361. BEGIN
  1362. IF debug & (watcher # NIL) THEN watcher(3) END;
  1363. s := size - 4;
  1364. i := MIN(N - 1, s DIV 16);
  1365. WHILE (i # N - 1) & (free[i] = sentinel) DO INC(i) END;
  1366. b := free[i]; l := NIL;
  1367. WHILE b.size < s DO l := b; b := b.next END;
  1368. IF b # sentinel THEN
  1369. IF l = NIL THEN free[i] := b.next ELSE l.next := b.next END
  1370. ELSE b := NIL
  1371. END;
  1372. RETURN b
  1373. END OldBlock;
  1374. PROCEDURE LastBlock (limit: INTEGER): FreeBlock; (* size MOD 16 = 0 *)
  1375. VAR b, l: FreeBlock; s, i: INTEGER;
  1376. BEGIN
  1377. s := limit - 4;
  1378. i := 0;
  1379. REPEAT
  1380. b := free[i]; l := NIL;
  1381. WHILE (b # sentinel) & (S.VAL(INTEGER, b) + b.size # s) DO l := b; b := b.next END;
  1382. IF b # sentinel THEN
  1383. IF l = NIL THEN free[i] := b.next ELSE l.next := b.next END
  1384. ELSE b := NIL
  1385. END;
  1386. INC(i)
  1387. UNTIL (b # NIL) OR (i = N);
  1388. RETURN b
  1389. END LastBlock;
  1390. PROCEDURE NewBlock (size: INTEGER): Block;
  1391. VAR tsize, a, s: INTEGER; b: FreeBlock; new, c: Cluster; r: Reducer;
  1392. BEGIN
  1393. ASSERT(size>=0,20);
  1394. IF size >7FFFFFECH THEN RETURN NIL END; (*20120822 Marc*)
  1395. tsize := (size + 19) DIV 16 * 16;
  1396. b := OldBlock(tsize); (* 1) search for free block *)
  1397. IF b = NIL THEN
  1398. FastCollect; b := OldBlock(tsize); (* 2) collect *)
  1399. IF b = NIL THEN
  1400. Collect; b := OldBlock(tsize); (* 2a) fully collect *)
  1401. END;
  1402. IF b = NIL THEN
  1403. AllocHeapMem(tsize + 12, new); (* 3) allocate new cluster *)
  1404. IF new # NIL THEN
  1405. IF (root = NIL) OR (S.VAL(INTEGER, new) < S.VAL(INTEGER, root)) THEN
  1406. new.next := root; root := new
  1407. ELSE
  1408. c := root;
  1409. WHILE (c.next # NIL) & (S.VAL(INTEGER, new) > S.VAL(INTEGER, c.next)) DO c := c.next END;
  1410. new.next := c.next; c.next := new
  1411. END;
  1412. b := S.VAL(FreeBlock, S.VAL(INTEGER, new) + 12);
  1413. b.size := (new.size - 12) DIV 16 * 16 - 4
  1414. ELSE
  1415. RETURN NIL (* 4) give up *)
  1416. END
  1417. END
  1418. END;
  1419. (* b # NIL *)
  1420. a := b.size + 4 - tsize;
  1421. IF a > 0 THEN Insert(S.VAL(FreeBlock, S.VAL(INTEGER, b) + tsize), a) END;
  1422. IF size > 0 THEN Erase(S.ADR(b.size), (size + 3) DIV 4) END;
  1423. INC(allocated, tsize);
  1424. RETURN S.VAL(Block, b)
  1425. END NewBlock;
  1426. PROCEDURE Allocated* (): INTEGER;
  1427. BEGIN
  1428. RETURN allocated
  1429. END Allocated;
  1430. PROCEDURE Used* (): INTEGER;
  1431. BEGIN
  1432. RETURN used
  1433. END Used;
  1434. PROCEDURE Root* (): INTEGER;
  1435. BEGIN
  1436. RETURN S.VAL(INTEGER, root)
  1437. END Root;
  1438. (* -------------------- Trap Handling --------------------- *)
  1439. PROCEDURE^ InitFpu;
  1440. PROCEDURE Start* (code: Command);
  1441. BEGIN
  1442. restart := code;
  1443. (*
  1444. S.GETREG(SP, baseStack); (* save base stack *)
  1445. *)
  1446. res := Libc.sigsetjmp(loopContext, Libc.TRUE);
  1447. code()
  1448. END Start;
  1449. PROCEDURE Quit* (exitCode: INTEGER);
  1450. VAR m: Module; term: Command; t: BOOLEAN;
  1451. res: INTEGER;
  1452. BEGIN
  1453. trapViewer := NIL; trapChecker := NIL; restart := NIL;
  1454. t := terminating; terminating := TRUE; m := modList;
  1455. WHILE m # NIL DO (* call terminators *)
  1456. IF ~static OR ~t THEN
  1457. term := m.term; m.term := NIL;
  1458. IF term # NIL THEN term() END
  1459. END;
  1460. (*
  1461. ReleaseIPtrs(m);
  1462. *)
  1463. m := m.next
  1464. END;
  1465. CallFinalizers;
  1466. hotFinalizers := finalizers; finalizers := NIL;
  1467. CallFinalizers;
  1468. (*
  1469. IF ~inDll THEN
  1470. RemoveExcp(excpPtr^);
  1471. WinApi.ExitProcess(exitCode) (* never returns *)
  1472. END
  1473. *)
  1474. res := Libc.fflush(0);
  1475. Libc.exit(exitCode)
  1476. END Quit;
  1477. PROCEDURE FatalError* (id: INTEGER; str: ARRAY OF CHAR);
  1478. VAR res: INTEGER; title: ARRAY 16 OF CHAR; text: ARRAY 256 OF SHORTCHAR;
  1479. BEGIN
  1480. title := "Error xy";
  1481. title[6] := CHR(id DIV 10 + ORD("0"));
  1482. title[7] := CHR(id MOD 10 + ORD("0"));
  1483. (*
  1484. res := WinApi.MessageBoxW(0, str, title, {});
  1485. *)
  1486. text := SHORT(str$);
  1487. res := MessageBox(title$, SHORT(str), {mbOk});
  1488. (*
  1489. IF ~inDll THEN RemoveExcp(excpPtr^) END;
  1490. *)
  1491. (*
  1492. WinApi.ExitProcess(1)
  1493. *)
  1494. Libc.exit(1)
  1495. (* never returns *)
  1496. END FatalError;
  1497. PROCEDURE DefaultTrapViewer;
  1498. VAR len, ref, end, x, a, b, c: INTEGER; mod: Module;
  1499. name: Name; out: ARRAY 1024 OF SHORTCHAR;
  1500. PROCEDURE WriteString (s: ARRAY OF SHORTCHAR);
  1501. VAR i: INTEGER;
  1502. BEGIN
  1503. i := 0;
  1504. WHILE (len < LEN(out) - 1) & (s[i] # 0X) DO out[len] := s[i]; INC(i); INC(len) END
  1505. END WriteString;
  1506. PROCEDURE WriteHex (x, n: INTEGER);
  1507. VAR i, y: INTEGER;
  1508. BEGIN
  1509. IF len + n < LEN(out) THEN
  1510. i := len + n - 1;
  1511. WHILE i >= len DO
  1512. y := x MOD 16; x := x DIV 16;
  1513. IF y > 9 THEN y := y + (ORD("A") - ORD("0") - 10) END;
  1514. out[i] := SHORT(CHR(y + ORD("0"))); DEC(i)
  1515. END;
  1516. INC(len, n)
  1517. END
  1518. END WriteHex;
  1519. PROCEDURE WriteLn;
  1520. BEGIN
  1521. IF len < LEN(out) - 1 THEN out[len] := 0AX (* 0DX on Windows *); INC(len) END
  1522. END WriteLn;
  1523. BEGIN
  1524. len := 0;
  1525. IF err = 129 THEN WriteString("invalid with")
  1526. ELSIF err = 130 THEN WriteString("invalid case")
  1527. ELSIF err = 131 THEN WriteString("function without return")
  1528. ELSIF err = 132 THEN WriteString("type guard")
  1529. ELSIF err = 133 THEN WriteString("implied type guard")
  1530. ELSIF err = 134 THEN WriteString("value out of range")
  1531. ELSIF err = 135 THEN WriteString("index out of range")
  1532. ELSIF err = 136 THEN WriteString("string too long")
  1533. ELSIF err = 137 THEN WriteString("stack overflow")
  1534. ELSIF err = 138 THEN WriteString("integer overflow")
  1535. ELSIF err = 139 THEN WriteString("division by zero")
  1536. ELSIF err = 140 THEN WriteString("infinite real result")
  1537. ELSIF err = 141 THEN WriteString("real underflow")
  1538. ELSIF err = 142 THEN WriteString("real overflow")
  1539. ELSIF err = 143 THEN WriteString("undefined real result")
  1540. ELSIF err = 200 THEN WriteString("keyboard interrupt")
  1541. ELSIF err = 202 THEN WriteString("illegal instruction: ");
  1542. WriteHex(val, 4)
  1543. ELSIF err = 203 THEN WriteString("illegal memory read [ad = ");
  1544. WriteHex(val, 8); WriteString("]")
  1545. ELSIF err = 204 THEN WriteString("illegal memory write [ad = ");
  1546. WriteHex(val, 8); WriteString("]")
  1547. ELSIF err = 205 THEN WriteString("illegal execution [ad = ");
  1548. WriteHex(val, 8); WriteString("]")
  1549. ELSIF err < 0 THEN WriteString("exception #"); WriteHex(-err, 2)
  1550. ELSE err := err DIV 100 * 256 + err DIV 10 MOD 10 * 16 + err MOD 10;
  1551. WriteString("trap #"); WriteHex(err, 3)
  1552. END;
  1553. a := pc; b := fp; c := 12;
  1554. REPEAT
  1555. WriteLn; WriteString("- ");
  1556. mod := modList;
  1557. WHILE (mod # NIL) & ((a < mod.code) OR (a >= mod.code + mod.csize)) DO mod := mod.next END;
  1558. IF mod # NIL THEN
  1559. DEC(a, mod.code);
  1560. IF mod.refcnt >= 0 THEN
  1561. WriteString(mod.name); ref := mod.refs;
  1562. REPEAT GetRefProc(ref, end, name) UNTIL (end = 0) OR (a < end);
  1563. IF a < end THEN
  1564. WriteString("."); WriteString(name)
  1565. END
  1566. ELSE
  1567. WriteString("("); WriteString(mod.name); WriteString(")")
  1568. END;
  1569. WriteString(" ")
  1570. END;
  1571. WriteString("(pc="); WriteHex(a, 8);
  1572. WriteString(", fp="); WriteHex(b, 8); WriteString(")");
  1573. IF (b >= sp) & (b < stack) THEN
  1574. S.GET(b+4, a); (* stacked pc *)
  1575. S.GET(b, b); (* dynamic link *)
  1576. DEC(c)
  1577. ELSE c := 0
  1578. END
  1579. UNTIL c = 0;
  1580. out[len] := 0X;
  1581. x := MessageBox("BlackBox", out$, {mbOk})
  1582. END DefaultTrapViewer;
  1583. PROCEDURE TrapCleanup;
  1584. VAR t: TrapCleaner;
  1585. BEGIN
  1586. WHILE trapStack # NIL DO
  1587. t := trapStack; trapStack := trapStack.next; t.Cleanup
  1588. END;
  1589. IF (trapChecker # NIL) & (err # 128) THEN trapChecker END
  1590. END TrapCleanup;
  1591. PROCEDURE SetTrapGuard* (on: BOOLEAN);
  1592. BEGIN
  1593. guarded := on
  1594. END SetTrapGuard;
  1595. PROCEDURE Try* (h: TryHandler; a, b, c: INTEGER);
  1596. VAR res: INTEGER; context: Libc.sigjmp_buf; oldContext: POINTER TO Libc.sigjmp_buf;
  1597. BEGIN
  1598. oldContext := currentTryContext;
  1599. res := Libc.sigsetjmp(context, Libc.TRUE);
  1600. currentTryContext := S.ADR(context);
  1601. IF res = 0 THEN (* first time around *)
  1602. h(a, b, c);
  1603. ELSIF res = trapReturn THEN (* after a trap *)
  1604. ELSE
  1605. HALT(100)
  1606. END;
  1607. currentTryContext := oldContext;
  1608. END Try;
  1609. (* -------------------- Initialization --------------------- *)
  1610. PROCEDURE InitFpu; (* COMPILER DEPENDENT *)
  1611. (* could be eliminated, delayed for backward compatibility *)
  1612. VAR cw: SET;
  1613. BEGIN
  1614. FINIT;
  1615. FSTCW;
  1616. (* denorm, underflow, precision, zero div, overflow masked *)
  1617. (* invalid trapped *)
  1618. (* round to nearest, temp precision *)
  1619. cw := cw - {0..5, 8..11} + {1, 2, 3, 4, 5, 8, 9};
  1620. FLDCW
  1621. END InitFpu;
  1622. (* A. V. Shiryaev: Show extended trap information (OpenBSD) *)
  1623. PROCEDURE ShowTrap (sig: INTEGER; siginfo: Libc.Ptrsiginfo_t; context: Libc.Ptrucontext_t);
  1624. PROCEDURE WriteChar (c: SHORTCHAR);
  1625. VAR s: ARRAY [untagged] 2 OF SHORTCHAR;
  1626. BEGIN
  1627. s[0] := c; s[1] := 0X;
  1628. res := Libc.printf(s)
  1629. END WriteChar;
  1630. PROCEDURE WriteString (s: ARRAY OF SHORTCHAR);
  1631. VAR res: INTEGER;
  1632. BEGIN
  1633. res := Libc.printf(s)
  1634. END WriteString;
  1635. PROCEDURE WriteHex (x, n: INTEGER);
  1636. VAR i, y: INTEGER;
  1637. s: ARRAY 9 OF SHORTCHAR;
  1638. BEGIN
  1639. s[n] := 0X;
  1640. i := 0 + n - 1;
  1641. WriteChar("$");
  1642. WHILE i >= 0 DO
  1643. y := x MOD 16; x := x DIV 16;
  1644. IF y > 9 THEN y := y + (ORD("A") - ORD("0") - 10) END;
  1645. s[i] := SHORT(CHR(y + ORD("0")));
  1646. DEC(i)
  1647. END;
  1648. WriteString(s)
  1649. END WriteHex;
  1650. PROCEDURE WriteLn;
  1651. BEGIN
  1652. WriteChar(0AX)
  1653. END WriteLn;
  1654. PROCEDURE KV (name: ARRAY OF SHORTCHAR; x: INTEGER);
  1655. BEGIN
  1656. WriteString(name); WriteString(" = "); WriteHex(x, 8)
  1657. END KV;
  1658. BEGIN
  1659. WriteString("================================"); WriteLn;
  1660. WriteString("TRAP:"); WriteLn;
  1661. WriteString("--------------------------------"); WriteLn;
  1662. KV("sig", sig); WriteString(", ");
  1663. KV("baseStack", baseStack); WriteLn;
  1664. KV("GS ", context.sc_gs); WriteString(", ");
  1665. KV("FS ", context.sc_fs); WriteString(", ");
  1666. KV("ES ", context.sc_es); WriteString(", ");
  1667. KV("DS ", context.sc_ds); WriteLn;
  1668. KV("EDI", context.sc_edi); WriteString(", ");
  1669. KV("ESI", context.sc_esi); WriteString(", ");
  1670. KV("EBP", context.sc_ebp); WriteString(", ");
  1671. KV("EBX", context.sc_ebx); WriteLn;
  1672. KV("EDX", context.sc_edx); WriteString(", ");
  1673. KV("ECX", context.sc_ecx); WriteString(", ");
  1674. KV("EAX", context.sc_eax); WriteString(", ");
  1675. KV("EIP", context.sc_eip); WriteLn;
  1676. KV("CS", context.sc_cs); WriteString(", ");
  1677. KV("EFLAGS", context.sc_eflags); WriteString(", ");
  1678. KV("ESP", context.sc_esp); WriteString(", ");
  1679. KV("SS", context.sc_ss); WriteLn;
  1680. KV("ONSTACK", context.sc_onstack); WriteString(", ");
  1681. KV("MASK", context.sc_mask); WriteString(", ");
  1682. KV("TRAPNO", context.sc_trapno); WriteString(", ");
  1683. KV("ERR", context.sc_err); WriteLn;
  1684. (* WriteString("--------------------------------"); WriteLn; *)
  1685. (* TODO: show siginfo *)
  1686. WriteString("================================"); WriteLn
  1687. END ShowTrap;
  1688. PROCEDURE [ccall] TrapHandler (sig: INTEGER; siginfo: Libc.Ptrsiginfo_t; context: Libc.Ptrucontext_t);
  1689. BEGIN
  1690. IF isReadableCheck THEN
  1691. isReadableCheck := FALSE;
  1692. Msg("~IsReadable");
  1693. Libc.siglongjmp(isReadableContext, 1)
  1694. END;
  1695. (*
  1696. S.GETREG(SP, sp);
  1697. S.GETREG(FP, fp);
  1698. *)
  1699. stack := baseStack;
  1700. (* A. V. Shiryaev *)
  1701. (*
  1702. sp := context.uc_mcontext.gregs[7]; (* TODO: is the stack pointer really stored in register 7? *)
  1703. fp := context.uc_mcontext.gregs[6]; (* TODO: is the frame pointer really stored in register 6? *)
  1704. pc := context.uc_mcontext.gregs[14]; (* TODO: is the pc really stored in register 14? *)
  1705. *)
  1706. (* val := siginfo.si_addr; *)
  1707. (* OpenBSD *)
  1708. ShowTrap(sig, siginfo, context);
  1709. sp := context.sc_esp; fp := context.sc_ebp; pc := context.sc_eip;
  1710. val := siginfo.si_pid; (* XXX *)
  1711. (*
  1712. Int(sig); Int(siginfo.si_signo); Int(siginfo.si_code); Int(siginfo.si_errno);
  1713. Int(siginfo.si_status); Int(siginfo.si_value); Int(siginfo.si_int);
  1714. *)
  1715. err := sig;
  1716. IF trapped THEN DefaultTrapViewer END;
  1717. CASE sig OF
  1718. Libc.SIGINT:
  1719. err := 200 (* Interrupt (ANSI). *)
  1720. | Libc.SIGILL: (* Illegal instruction (ANSI). *)
  1721. err := 202; val := 0;
  1722. IF IsReadable(pc, pc + 4) THEN
  1723. S.GET(pc, val);
  1724. IF val MOD 100H = 8DH THEN (* lea reg,reg *)
  1725. IF val DIV 100H MOD 100H = 0F0H THEN
  1726. err := val DIV 10000H MOD 100H (* trap *)
  1727. ELSIF val DIV 1000H MOD 10H = 0EH THEN
  1728. err := 128 + val DIV 100H MOD 10H (* run time error *)
  1729. END
  1730. END
  1731. END
  1732. | Libc.SIGFPE:
  1733. CASE siginfo.si_code OF
  1734. 0: (* TODO: ?????? *)
  1735. (* A. V. Shiryaev: OpenBSD *)
  1736. (*
  1737. IF siginfo.si_int = 8 THEN
  1738. err := 139
  1739. ELSIF siginfo.si_int = 0 THEN
  1740. err := 143
  1741. END
  1742. *)
  1743. err := 143;
  1744. | Libc.FPE_INTDIV: err := 139 (* Integer divide by zero. *)
  1745. | Libc.FPE_INTOVF: err := 138 (* Integer overflow. *)
  1746. | Libc.FPE_FLTDIV: err := 140 (* Floating point divide by zero. *)
  1747. | Libc.FPE_FLTOVF: err := 142 (* Floating point overflow. *)
  1748. | Libc.FPE_FLTUND: err := 141 (* Floating point underflow. *)
  1749. | Libc.FPE_FLTRES: err := 143 (* Floating point inexact result. *)
  1750. | Libc.FPE_FLTINV: err := 143 (* Floating point invalid operation. *)
  1751. | Libc.FPE_FLTSUB: err := 134 (* Subscript out of range. *)
  1752. ELSE
  1753. END
  1754. | Libc.SIGSEGV: (* Segmentation violation (ANSI). *)
  1755. err := 203
  1756. ELSE
  1757. END;
  1758. INC(trapCount);
  1759. InitFpu;
  1760. TrapCleanup;
  1761. IF err # 128 THEN
  1762. IF (trapViewer = NIL) OR trapped THEN
  1763. DefaultTrapViewer
  1764. ELSE
  1765. trapped := TRUE;
  1766. trapViewer();
  1767. trapped := FALSE
  1768. END
  1769. END;
  1770. IF currentTryContext # NIL THEN (* Try failed *)
  1771. Libc.siglongjmp(currentTryContext, trapReturn)
  1772. ELSE
  1773. IF restart # NIL THEN (* Start failed *)
  1774. Libc.siglongjmp(loopContext, trapReturn)
  1775. END;
  1776. Quit(1); (* FIXME *)
  1777. END;
  1778. trapped := FALSE
  1779. END TrapHandler;
  1780. PROCEDURE InstallSignals*;
  1781. VAR sa, old: Libc.sigaction_t; res, i: INTEGER;
  1782. (*
  1783. sigstk: Libc.sigaltstack_t;
  1784. errno: INTEGER;
  1785. *)
  1786. BEGIN
  1787. (*
  1788. (* A. V. Shiryaev: Set alternative stack on which signals are to be processed *)
  1789. sigstk.ss_sp := sigStack;
  1790. sigstk.ss_size := sigStackSize;
  1791. sigstk.ss_flags := 0;
  1792. res := Libc.sigaltstack(sigstk, NIL);
  1793. IF res # 0 THEN Msg("ERROR: Kernel.InstallSignals: sigaltstack failed!");
  1794. S.GET( Libc.__errno_location(), errno );
  1795. Int(errno);
  1796. Libc.exit(1)
  1797. END;
  1798. *)
  1799. sa.sa_sigaction := TrapHandler;
  1800. (*
  1801. res := LinLibc.sigemptyset(S.ADR(sa.sa_mask));
  1802. *)
  1803. res := Libc.sigfillset(S.ADR(sa.sa_mask));
  1804. sa.sa_flags := (* Libc.SA_ONSTACK + *) Libc.SA_SIGINFO; (* TrapHandler takes three arguments *)
  1805. (*
  1806. IF LinLibc.sigaction(LinLibc.SIGINT, sa, old) # 0 THEN Msg("failed to install SIGINT") END;
  1807. IF LinLibc.sigaction(LinLibc.SIGILL, sa, old) # 0 THEN Msg("failed to install SIGILL") END;
  1808. IF LinLibc.sigaction(LinLibc.SIGFPE, sa, old) # 0 THEN Msg("failed to install SIGFPE") END;
  1809. IF LinLibc.sigaction(LinLibc.SIGSEGV, sa, old) # 0 THEN Msg("failed to install SIGSEGV") END;
  1810. IF LinLibc.sigaction(LinLibc.SIGPIPE, sa, old) # 0 THEN Msg("failed to install SIGPIPE") END;
  1811. IF LinLibc.sigaction(LinLibc.SIGTERM, sa, old) # 0 THEN Msg("failed to install SIGTERM") END;
  1812. *)
  1813. (* respond to all possible signals *)
  1814. FOR i := 1 TO Libc._NSIG - 1 DO
  1815. IF (i # Libc.SIGKILL)
  1816. & (i # Libc.SIGSTOP)
  1817. & (i # Libc.SIGWINCH)
  1818. & (i # Libc.SIGTHR) (* A. V. Shiryaev: OpenBSD -pthread *)
  1819. THEN
  1820. IF Libc.sigaction(i, sa, old) # 0 THEN Msg("failed to install signal"); Int(i) END;
  1821. END
  1822. END
  1823. END InstallSignals;
  1824. PROCEDURE Init;
  1825. VAR i: INTEGER;
  1826. BEGIN
  1827. (*
  1828. (* for sigaltstack *)
  1829. sigStack := Libc.calloc(1, sigStackSize);
  1830. IF sigStack = Libc.NULL THEN
  1831. Msg("ERROR: Kernel.Init: calloc(1, sigStackSize) failed!");
  1832. Libc.exit(1)
  1833. END;
  1834. *)
  1835. isReadableCheck := FALSE;
  1836. InstallSignals; (* init exception handling *)
  1837. currentTryContext := NIL;
  1838. allocated := 0; total := 0; used := 0;
  1839. sentinelBlock.size := MAX(INTEGER);
  1840. sentinel := S.ADR(sentinelBlock);
  1841. (*
  1842. S.PUTREG(ML, S.ADR(modList));
  1843. *)
  1844. i := N;
  1845. REPEAT DEC(i); free[i] := sentinel UNTIL i = 0;
  1846. IF inDll THEN
  1847. (*
  1848. baseStack := FPageWord(4); (* begin of stack segment *)
  1849. *)
  1850. END;
  1851. InitFpu;
  1852. IF ~static THEN
  1853. InitModule(modList);
  1854. IF ~inDll THEN Quit(1) END
  1855. END;
  1856. told := 0; shift := 0
  1857. END Init;
  1858. BEGIN
  1859. IF modList = NIL THEN (* only once *)
  1860. S.GETREG(SP, baseStack); (* TODO: Check that this is ok. *)
  1861. IF bootInfo # NIL THEN
  1862. modList := bootInfo.modList (* boot loader initializes the bootInfo struct *)
  1863. ELSE
  1864. S.GETREG(ML, modList) (* linker loads module list to BX *)
  1865. END;
  1866. static := init IN modList.opts;
  1867. inDll := dll IN modList.opts;
  1868. Init
  1869. END
  1870. CLOSE
  1871. IF ~terminating THEN
  1872. terminating := TRUE;
  1873. Quit(0)
  1874. END
  1875. END Kernel.