RTL.ob07 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072
  1. (*
  2. BSD 2-Clause License
  3. Copyright (c) 2018-2021, 2023, Anton Krotov
  4. All rights reserved.
  5. *)
  6. (*---------------------x86_64---------------------*)
  7. $IF (CPU_X8664)
  8. MODULE RTL;
  9. IMPORT SYSTEM, API;
  10. CONST
  11. minint = ROR(1, 1);
  12. WORD = API.BIT_DEPTH DIV 8;
  13. VAR
  14. name: INTEGER;
  15. types: INTEGER;
  16. PROCEDURE [oberon] _move* (bytes, dest, source: INTEGER);
  17. BEGIN
  18. SYSTEM.CODE(
  19. 048H, 08BH, 045H, 010H, (* mov rax, qword [rbp + 16] *)
  20. 048H, 085H, 0C0H, (* test rax, rax *)
  21. 07EH, 020H, (* jle L *)
  22. 0FCH, (* cld *)
  23. 057H, (* push rdi *)
  24. 056H, (* push rsi *)
  25. 048H, 08BH, 075H, 020H, (* mov rsi, qword [rbp + 32] *)
  26. 048H, 08BH, 07DH, 018H, (* mov rdi, qword [rbp + 24] *)
  27. 048H, 089H, 0C1H, (* mov rcx, rax *)
  28. 048H, 0C1H, 0E9H, 003H, (* shr rcx, 3 *)
  29. 0F3H, 048H, 0A5H, (* rep movsd *)
  30. 048H, 089H, 0C1H, (* mov rcx, rax *)
  31. 048H, 083H, 0E1H, 007H, (* and rcx, 7 *)
  32. 0F3H, 0A4H, (* rep movsb *)
  33. 05EH, (* pop rsi *)
  34. 05FH (* pop rdi *)
  35. (* L: *)
  36. )
  37. END _move;
  38. PROCEDURE [oberon] _arrcpy* (base_size, len_dst, dst, len_src, src: INTEGER): BOOLEAN;
  39. VAR
  40. res: BOOLEAN;
  41. BEGIN
  42. IF len_src > len_dst THEN
  43. res := FALSE
  44. ELSE
  45. _move(len_src * base_size, dst, src);
  46. res := TRUE
  47. END
  48. RETURN res
  49. END _arrcpy;
  50. PROCEDURE [oberon] _strcpy* (chr_size, len_src, src, len_dst, dst: INTEGER);
  51. BEGIN
  52. _move(MIN(len_dst, len_src) * chr_size, dst, src)
  53. END _strcpy;
  54. PROCEDURE [oberon] _rot* (Len, Ptr: INTEGER);
  55. BEGIN
  56. SYSTEM.CODE(
  57. 048H, 08BH, 04DH, 010H, (* mov rcx, qword [rbp + 16] *) (* rcx <- Len *)
  58. 048H, 08BH, 045H, 018H, (* mov rax, qword [rbp + 24] *) (* rax <- Ptr *)
  59. 048H, 0FFH, 0C9H, (* dec rcx *)
  60. 04CH, 08BH, 010H, (* mov r10, qword [rax] *)
  61. (* L: *)
  62. 048H, 08BH, 050H, 008H, (* mov rdx, qword [rax + 8] *)
  63. 048H, 089H, 010H, (* mov qword [rax], rdx *)
  64. 048H, 083H, 0C0H, 008H, (* add rax, 8 *)
  65. 048H, 0FFH, 0C9H, (* dec rcx *)
  66. 075H, 0F0H, (* jnz L *)
  67. 04CH, 089H, 010H, (* mov qword [rax], r10 *)
  68. 05DH, (* pop rbp *)
  69. 0C2H, 010H, 000H (* ret 16 *)
  70. )
  71. END _rot;
  72. PROCEDURE [oberon] _set* (b, a: INTEGER); (* {a..b} -> rax *)
  73. BEGIN
  74. SYSTEM.CODE(
  75. 048H, 08BH, 04DH, 010H, (* mov rcx, qword ptr [rbp + 16] *) (* rcx <- b *)
  76. 048H, 08BH, 045H, 018H, (* mov rax, qword ptr [rbp + 24] *) (* rax <- a *)
  77. 048H, 039H, 0C8H, (* cmp rax, rcx *)
  78. 07FH, 047H, (* jg L1 *)
  79. 048H, 083H, 0F8H, 03FH, (* cmp rax, 63 *)
  80. 07FH, 041H, (* jg L1 *)
  81. 048H, 085H, 0C9H, (* test rcx, rcx *)
  82. 07CH, 03CH, (* jl L1 *)
  83. 048H, 083H, 0F9H, 03FH, (* cmp rcx, 63 *)
  84. 07EH, 007H, (* jle L3 *)
  85. 048H, 0C7H, 0C1H, 03FH, (* mov rcx, 63 *)
  86. 000H, 000H, 000H,
  87. (* L3: *)
  88. 048H, 085H, 0C0H, (* test rax, rax *)
  89. 07DH, 003H, (* jge L2 *)
  90. 048H, 031H, 0C0H, (* xor rax, rax *)
  91. (* L2: *)
  92. 048H, 089H, 0CAH, (* mov rdx, rcx *)
  93. 048H, 029H, 0C2H, (* sub rdx, rax *)
  94. 048H, 0B8H, 000H, 000H, (* movabs rax, minint *)
  95. 000H, 000H, 000H, 000H,
  96. 000H, 080H,
  97. 048H, 087H, 0CAH, (* xchg rdx, rcx *)
  98. 048H, 0D3H, 0F8H, (* sar rax, cl *)
  99. 048H, 087H, 0CAH, (* xchg rdx, rcx *)
  100. 048H, 083H, 0E9H, 03FH, (* sub rcx, 63 *)
  101. 048H, 0F7H, 0D9H, (* neg rcx *)
  102. 048H, 0D3H, 0E8H, (* shr rax, cl *)
  103. 05DH, (* pop rbp *)
  104. 0C2H, 010H, 000H, (* ret 16 *)
  105. (* L1: *)
  106. 048H, 031H, 0C0H, (* xor rax, rax *)
  107. 05DH, (* pop rbp *)
  108. 0C2H, 010H, 000H (* ret 16 *)
  109. )
  110. END _set;
  111. PROCEDURE [oberon] _set1* (a: INTEGER); (* {a} -> rax *)
  112. BEGIN
  113. SYSTEM.CODE(
  114. 048H, 031H, 0C0H, (* xor rax, rax *)
  115. 048H, 08BH, 04DH, 010H, (* mov rcx, qword [rbp + 16] *) (* rcx <- a *)
  116. 048H, 083H, 0F9H, 03FH, (* cmp rcx, 63 *)
  117. 077H, 004H, (* ja L *)
  118. 048H, 00FH, 0ABH, 0C8H (* bts rax, rcx *)
  119. (* L: *)
  120. )
  121. END _set1;
  122. PROCEDURE [oberon] _divmod* (y, x: INTEGER); (* (x div y) -> rax; (x mod y) -> rdx *)
  123. BEGIN
  124. SYSTEM.CODE(
  125. 048H, 08BH, 045H, 018H, (* mov rax, qword [rbp + 24] *) (* rax <- x *)
  126. 048H, 031H, 0D2H, (* xor rdx, rdx *)
  127. 048H, 085H, 0C0H, (* test rax, rax *)
  128. 074H, 022H, (* je L2 *)
  129. 07FH, 003H, (* jg L1 *)
  130. 048H, 0F7H, 0D2H, (* not rdx *)
  131. (* L1: *)
  132. 049H, 089H, 0C0H, (* mov r8, rax *)
  133. 048H, 08BH, 04DH, 010H, (* mov rcx, qword [rbp + 16] *) (* rcx <- y *)
  134. 048H, 0F7H, 0F9H, (* idiv rcx *)
  135. 048H, 085H, 0D2H, (* test rdx, rdx *)
  136. 074H, 00EH, (* je L2 *)
  137. 049H, 031H, 0C8H, (* xor r8, rcx *)
  138. 04DH, 085H, 0C0H, (* test r8, r8 *)
  139. 07DH, 006H, (* jge L2 *)
  140. 048H, 0FFH, 0C8H, (* dec rax *)
  141. 048H, 001H, 0CAH (* add rdx, rcx *)
  142. (* L2: *)
  143. )
  144. END _divmod;
  145. PROCEDURE [oberon] _new* (t, size: INTEGER; VAR ptr: INTEGER);
  146. BEGIN
  147. ptr := API._NEW(size);
  148. IF ptr # 0 THEN
  149. SYSTEM.PUT(ptr + 8, t);
  150. INC(ptr, 16)
  151. END
  152. END _new;
  153. PROCEDURE [oberon] _dispose* (VAR ptr: INTEGER);
  154. BEGIN
  155. IF ptr # 0 THEN
  156. ptr := API._DISPOSE(ptr - 16)
  157. END
  158. END _dispose;
  159. PROCEDURE [oberon] _length* (len, str: INTEGER);
  160. BEGIN
  161. SYSTEM.CODE(
  162. 048H, 08BH, 045H, 018H, (* mov rax, qword [rbp + 24] *)
  163. 048H, 08BH, 04DH, 010H, (* mov rcx, qword [rbp + 16] *)
  164. 048H, 0FFH, 0C8H, (* dec rax *)
  165. (* L1: *)
  166. 048H, 0FFH, 0C0H, (* inc rax *)
  167. 080H, 038H, 000H, (* cmp byte [rax], 0 *)
  168. 074H, 005H, (* jz L2 *)
  169. 0E2H, 0F6H, (* loop L1 *)
  170. 048H, 0FFH, 0C0H, (* inc rax *)
  171. (* L2: *)
  172. 048H, 02BH, 045H, 018H (* sub rax, qword [rbp + 24] *)
  173. )
  174. END _length;
  175. PROCEDURE [oberon] _lengthw* (len, str: INTEGER);
  176. BEGIN
  177. SYSTEM.CODE(
  178. 048H, 08BH, 045H, 018H, (* mov rax, qword [rbp + 24] *)
  179. 048H, 08BH, 04DH, 010H, (* mov rcx, qword [rbp + 16] *)
  180. 048H, 083H, 0E8H, 002H, (* sub rax, 2 *)
  181. (* L1: *)
  182. 048H, 083H, 0C0H, 002H, (* add rax, 2 *)
  183. 066H, 083H, 038H, 000H, (* cmp word [rax], 0 *)
  184. 074H, 006H, (* jz L2 *)
  185. 0E2H, 0F4H, (* loop L1 *)
  186. 048H, 083H, 0C0H, 002H, (* add rax, 2 *)
  187. (* L2: *)
  188. 048H, 02BH, 045H, 018H, (* sub rax, qword [rbp + 24] *)
  189. 048H, 0D1H, 0E8H (* shr rax, 1 *)
  190. )
  191. END _lengthw;
  192. PROCEDURE [oberon] strncmp (a, b, n: INTEGER): INTEGER;
  193. BEGIN
  194. SYSTEM.CODE(
  195. 048H, 08BH, 04DH, 010H, (* mov rcx, qword[rbp + 16]; rcx <- a *)
  196. 048H, 08BH, 055H, 018H, (* mov rdx, qword[rbp + 24]; rdx <- b *)
  197. 04CH, 08BH, 045H, 020H, (* mov r8, qword[rbp + 32]; r8 <- n *)
  198. 04DH, 031H, 0C9H, (* xor r9, r9 *)
  199. 04DH, 031H, 0D2H, (* xor r10, r10 *)
  200. 048H, 0B8H, 000H, 000H, (* movabs rax, minint *)
  201. 000H, 000H, 000H, 000H,
  202. 000H, 080H,
  203. (* L1: *)
  204. 04DH, 085H, 0C0H, (* test r8, r8 *)
  205. 07EH, 024H, (* jle L3 *)
  206. 044H, 08AH, 009H, (* mov r9b, byte[rcx] *)
  207. 044H, 08AH, 012H, (* mov r10b, byte[rdx] *)
  208. 048H, 0FFH, 0C1H, (* inc rcx *)
  209. 048H, 0FFH, 0C2H, (* inc rdx *)
  210. 049H, 0FFH, 0C8H, (* dec r8 *)
  211. 04DH, 039H, 0D1H, (* cmp r9, r10 *)
  212. 074H, 008H, (* je L2 *)
  213. 04CH, 089H, 0C8H, (* mov rax, r9 *)
  214. 04CH, 029H, 0D0H, (* sub rax, r10 *)
  215. 0EBH, 008H, (* jmp L3 *)
  216. (* L2: *)
  217. 04DH, 085H, 0C9H, (* test r9, r9 *)
  218. 075H, 0DAH, (* jne L1 *)
  219. 048H, 031H, 0C0H, (* xor rax, rax *)
  220. (* L3: *)
  221. 05DH, (* pop rbp *)
  222. 0C2H, 018H, 000H (* ret 24 *)
  223. )
  224. RETURN 0
  225. END strncmp;
  226. PROCEDURE [oberon] strncmpw (a, b, n: INTEGER): INTEGER;
  227. BEGIN
  228. SYSTEM.CODE(
  229. 048H, 08BH, 04DH, 010H, (* mov rcx, qword[rbp + 16]; rcx <- a *)
  230. 048H, 08BH, 055H, 018H, (* mov rdx, qword[rbp + 24]; rdx <- b *)
  231. 04CH, 08BH, 045H, 020H, (* mov r8, qword[rbp + 32]; r8 <- n *)
  232. 04DH, 031H, 0C9H, (* xor r9, r9 *)
  233. 04DH, 031H, 0D2H, (* xor r10, r10 *)
  234. 048H, 0B8H, 000H, 000H, (* movabs rax, minint *)
  235. 000H, 000H, 000H, 000H,
  236. 000H, 080H,
  237. (* L1: *)
  238. 04DH, 085H, 0C0H, (* test r8, r8 *)
  239. 07EH, 028H, (* jle L3 *)
  240. 066H, 044H, 08BH, 009H, (* mov r9w, word[rcx] *)
  241. 066H, 044H, 08BH, 012H, (* mov r10w, word[rdx] *)
  242. 048H, 083H, 0C1H, 002H, (* add rcx, 2 *)
  243. 048H, 083H, 0C2H, 002H, (* add rdx, 2 *)
  244. 049H, 0FFH, 0C8H, (* dec r8 *)
  245. 04DH, 039H, 0D1H, (* cmp r9, r10 *)
  246. 074H, 008H, (* je L2 *)
  247. 04CH, 089H, 0C8H, (* mov rax, r9 *)
  248. 04CH, 029H, 0D0H, (* sub rax, r10 *)
  249. 0EBH, 008H, (* jmp L3 *)
  250. (* L2: *)
  251. 04DH, 085H, 0C9H, (* test r9, r9 *)
  252. 075H, 0D6H, (* jne L1 *)
  253. 048H, 031H, 0C0H, (* xor rax, rax *)
  254. (* L3: *)
  255. 05DH, (* pop rbp *)
  256. 0C2H, 018H, 000H (* ret 24 *)
  257. )
  258. RETURN 0
  259. END strncmpw;
  260. PROCEDURE [oberon] _strcmp* (op, len2, str2, len1, str1: INTEGER): BOOLEAN;
  261. VAR
  262. res: INTEGER;
  263. bRes: BOOLEAN;
  264. c: CHAR;
  265. BEGIN
  266. res := strncmp(str1, str2, MIN(len1, len2));
  267. IF res = minint THEN
  268. IF len1 > len2 THEN
  269. SYSTEM.GET(str1 + len2, c);
  270. res := ORD(c)
  271. ELSIF len1 < len2 THEN
  272. SYSTEM.GET(str2 + len1, c);
  273. res := -ORD(c)
  274. ELSE
  275. res := 0
  276. END
  277. END;
  278. CASE op OF
  279. |0: bRes := res = 0
  280. |1: bRes := res # 0
  281. |2: bRes := res < 0
  282. |3: bRes := res <= 0
  283. |4: bRes := res > 0
  284. |5: bRes := res >= 0
  285. END
  286. RETURN bRes
  287. END _strcmp;
  288. PROCEDURE [oberon] _strcmpw* (op, len2, str2, len1, str1: INTEGER): BOOLEAN;
  289. VAR
  290. res: INTEGER;
  291. bRes: BOOLEAN;
  292. c: WCHAR;
  293. BEGIN
  294. res := strncmpw(str1, str2, MIN(len1, len2));
  295. IF res = minint THEN
  296. IF len1 > len2 THEN
  297. SYSTEM.GET(str1 + len2 * 2, c);
  298. res := ORD(c)
  299. ELSIF len1 < len2 THEN
  300. SYSTEM.GET(str2 + len1 * 2, c);
  301. res := -ORD(c)
  302. ELSE
  303. res := 0
  304. END
  305. END;
  306. CASE op OF
  307. |0: bRes := res = 0
  308. |1: bRes := res # 0
  309. |2: bRes := res < 0
  310. |3: bRes := res <= 0
  311. |4: bRes := res > 0
  312. |5: bRes := res >= 0
  313. END
  314. RETURN bRes
  315. END _strcmpw;
  316. PROCEDURE PCharToStr (pchar: INTEGER; VAR s: ARRAY OF CHAR);
  317. VAR
  318. c: CHAR;
  319. i: INTEGER;
  320. BEGIN
  321. i := 0;
  322. REPEAT
  323. SYSTEM.GET(pchar, c);
  324. s[i] := c;
  325. INC(pchar);
  326. INC(i)
  327. UNTIL c = 0X
  328. END PCharToStr;
  329. PROCEDURE IntToStr (x: INTEGER; VAR str: ARRAY OF CHAR);
  330. VAR
  331. i, a: INTEGER;
  332. BEGIN
  333. i := 0;
  334. a := x;
  335. REPEAT
  336. INC(i);
  337. a := a DIV 10
  338. UNTIL a = 0;
  339. str[i] := 0X;
  340. REPEAT
  341. DEC(i);
  342. str[i] := CHR(x MOD 10 + ORD("0"));
  343. x := x DIV 10
  344. UNTIL x = 0
  345. END IntToStr;
  346. PROCEDURE append (VAR s1: ARRAY OF CHAR; s2: ARRAY OF CHAR);
  347. VAR
  348. n1, n2: INTEGER;
  349. BEGIN
  350. n1 := LENGTH(s1);
  351. n2 := LENGTH(s2);
  352. ASSERT(n1 + n2 < LEN(s1));
  353. SYSTEM.MOVE(SYSTEM.ADR(s2[0]), SYSTEM.ADR(s1[n1]), n2);
  354. s1[n1 + n2] := 0X
  355. END append;
  356. PROCEDURE [oberon] _error* (modnum, _module, err, line: INTEGER);
  357. VAR
  358. s, temp: ARRAY 1024 OF CHAR;
  359. BEGIN
  360. CASE err OF
  361. | 1: s := "assertion failure"
  362. | 2: s := "NIL dereference"
  363. | 3: s := "bad divisor"
  364. | 4: s := "NIL procedure call"
  365. | 5: s := "type guard error"
  366. | 6: s := "index out of range"
  367. | 7: s := "invalid CASE"
  368. | 8: s := "array assignment error"
  369. | 9: s := "CHR out of range"
  370. |10: s := "WCHR out of range"
  371. |11: s := "BYTE out of range"
  372. END;
  373. append(s, API.eol + "module: "); PCharToStr(_module, temp); append(s, temp);
  374. append(s, API.eol + "line: "); IntToStr(line, temp); append(s, temp);
  375. API.DebugMsg(SYSTEM.ADR(s[0]), name);
  376. API.exit_thread(0)
  377. END _error;
  378. PROCEDURE [oberon] _isrec* (t0, t1, r: INTEGER): INTEGER;
  379. BEGIN
  380. SYSTEM.GET(t0 + t1 + types, t0)
  381. RETURN t0 MOD 2
  382. END _isrec;
  383. PROCEDURE [oberon] _is* (t0, p: INTEGER): INTEGER;
  384. BEGIN
  385. IF p # 0 THEN
  386. SYSTEM.GET(p - WORD, p);
  387. SYSTEM.GET(t0 + p + types, p)
  388. END
  389. RETURN p MOD 2
  390. END _is;
  391. PROCEDURE [oberon] _guardrec* (t0, t1: INTEGER): INTEGER;
  392. BEGIN
  393. SYSTEM.GET(t0 + t1 + types, t0)
  394. RETURN t0 MOD 2
  395. END _guardrec;
  396. PROCEDURE [oberon] _guard* (t0, p: INTEGER): INTEGER;
  397. BEGIN
  398. SYSTEM.GET(p, p);
  399. IF p # 0 THEN
  400. SYSTEM.GET(p - WORD, p);
  401. SYSTEM.GET(t0 + p + types, p)
  402. ELSE
  403. p := 1
  404. END
  405. RETURN p MOD 2
  406. END _guard;
  407. PROCEDURE [oberon] _dllentry* (hinstDLL, fdwReason, lpvReserved: INTEGER): INTEGER;
  408. RETURN API.dllentry(hinstDLL, fdwReason, lpvReserved)
  409. END _dllentry;
  410. PROCEDURE [oberon] _sofinit*;
  411. BEGIN
  412. API.sofinit
  413. END _sofinit;
  414. PROCEDURE [oberon] _exit* (code: INTEGER);
  415. BEGIN
  416. API.exit(code)
  417. END _exit;
  418. PROCEDURE [oberon] _init* (modname: INTEGER; tcount, _types: INTEGER; code, param: INTEGER);
  419. VAR
  420. t0, t1, i, j: INTEGER;
  421. BEGIN
  422. API.init(param, code);
  423. types := API._NEW(tcount * tcount + SYSTEM.SIZE(INTEGER));
  424. ASSERT(types # 0);
  425. FOR i := 0 TO tcount - 1 DO
  426. FOR j := 0 TO tcount - 1 DO
  427. t0 := i; t1 := j;
  428. WHILE (t1 # 0) & (t1 # t0) DO
  429. SYSTEM.GET(_types + t1 * WORD, t1)
  430. END;
  431. SYSTEM.PUT8(i * tcount + j + types, ORD(t0 = t1))
  432. END
  433. END;
  434. name := modname
  435. END _init;
  436. END RTL.
  437. $END
  438. (*---------------------x86------------------------*)
  439. $IF (CPU_X86)
  440. MODULE RTL;
  441. IMPORT SYSTEM, API;
  442. CONST
  443. minint = ROR(1, 1);
  444. WORD = API.BIT_DEPTH DIV 8;
  445. VAR
  446. name: INTEGER;
  447. types: INTEGER;
  448. PROCEDURE [stdcall] _move* (bytes, dest, source: INTEGER);
  449. BEGIN
  450. SYSTEM.CODE(
  451. 08BH, 045H, 008H, (* mov eax, dword [ebp + 8] *)
  452. 085H, 0C0H, (* test eax, eax *)
  453. 07EH, 019H, (* jle L *)
  454. 0FCH, (* cld *)
  455. 057H, (* push edi *)
  456. 056H, (* push esi *)
  457. 08BH, 075H, 010H, (* mov esi, dword [ebp + 16] *)
  458. 08BH, 07DH, 00CH, (* mov edi, dword [ebp + 12] *)
  459. 089H, 0C1H, (* mov ecx, eax *)
  460. 0C1H, 0E9H, 002H, (* shr ecx, 2 *)
  461. 0F3H, 0A5H, (* rep movsd *)
  462. 089H, 0C1H, (* mov ecx, eax *)
  463. 083H, 0E1H, 003H, (* and ecx, 3 *)
  464. 0F3H, 0A4H, (* rep movsb *)
  465. 05EH, (* pop esi *)
  466. 05FH (* pop edi *)
  467. (* L: *)
  468. )
  469. END _move;
  470. PROCEDURE [stdcall] _arrcpy* (base_size, len_dst, dst, len_src, src: INTEGER): BOOLEAN;
  471. VAR
  472. res: BOOLEAN;
  473. BEGIN
  474. IF len_src > len_dst THEN
  475. res := FALSE
  476. ELSE
  477. _move(len_src * base_size, dst, src);
  478. res := TRUE
  479. END
  480. RETURN res
  481. END _arrcpy;
  482. PROCEDURE [stdcall] _strcpy* (chr_size, len_src, src, len_dst, dst: INTEGER);
  483. BEGIN
  484. _move(MIN(len_dst, len_src) * chr_size, dst, src)
  485. END _strcpy;
  486. PROCEDURE [stdcall] _rot* (Len, Ptr: INTEGER);
  487. BEGIN
  488. SYSTEM.CODE(
  489. 08BH, 04DH, 008H, (* mov ecx, dword [ebp + 8] *) (* ecx <- Len *)
  490. 08BH, 045H, 00CH, (* mov eax, dword [ebp + 12] *) (* eax <- Ptr *)
  491. 049H, (* dec ecx *)
  492. 053H, (* push ebx *)
  493. 08BH, 018H, (* mov ebx, dword [eax] *)
  494. (* L: *)
  495. 08BH, 050H, 004H, (* mov edx, dword [eax + 4] *)
  496. 089H, 010H, (* mov dword [eax], edx *)
  497. 083H, 0C0H, 004H, (* add eax, 4 *)
  498. 049H, (* dec ecx *)
  499. 075H, 0F5H, (* jnz L *)
  500. 089H, 018H, (* mov dword [eax], ebx *)
  501. 05BH, (* pop ebx *)
  502. 05DH, (* pop ebp *)
  503. 0C2H, 008H, 000H (* ret 8 *)
  504. )
  505. END _rot;
  506. PROCEDURE [stdcall] _set* (b, a: INTEGER); (* {a..b} -> eax *)
  507. BEGIN
  508. SYSTEM.CODE(
  509. 08BH, 04DH, 008H, (* mov ecx, dword [ebp + 8] *) (* ecx <- b *)
  510. 08BH, 045H, 00CH, (* mov eax, dword [ebp + 12] *) (* eax <- a *)
  511. 039H, 0C8H, (* cmp eax, ecx *)
  512. 07FH, 033H, (* jg L1 *)
  513. 083H, 0F8H, 01FH, (* cmp eax, 31 *)
  514. 07FH, 02EH, (* jg L1 *)
  515. 085H, 0C9H, (* test ecx, ecx *)
  516. 07CH, 02AH, (* jl L1 *)
  517. 083H, 0F9H, 01FH, (* cmp ecx, 31 *)
  518. 07EH, 005H, (* jle L3 *)
  519. 0B9H, 01FH, 000H, 000H, 000H, (* mov ecx, 31 *)
  520. (* L3: *)
  521. 085H, 0C0H, (* test eax, eax *)
  522. 07DH, 002H, (* jge L2 *)
  523. 031H, 0C0H, (* xor eax, eax *)
  524. (* L2: *)
  525. 089H, 0CAH, (* mov edx, ecx *)
  526. 029H, 0C2H, (* sub edx, eax *)
  527. 0B8H, 000H, 000H, 000H, 080H, (* mov eax, 0x80000000 *)
  528. 087H, 0CAH, (* xchg edx, ecx *)
  529. 0D3H, 0F8H, (* sar eax, cl *)
  530. 087H, 0CAH, (* xchg edx, ecx *)
  531. 083H, 0E9H, 01FH, (* sub ecx, 31 *)
  532. 0F7H, 0D9H, (* neg ecx *)
  533. 0D3H, 0E8H, (* shr eax, cl *)
  534. 05DH, (* pop ebp *)
  535. 0C2H, 008H, 000H, (* ret 8 *)
  536. (* L1: *)
  537. 031H, 0C0H, (* xor eax, eax *)
  538. 05DH, (* pop ebp *)
  539. 0C2H, 008H, 000H (* ret 8 *)
  540. )
  541. END _set;
  542. PROCEDURE [stdcall] _set1* (a: INTEGER); (* {a} -> eax *)
  543. BEGIN
  544. SYSTEM.CODE(
  545. 031H, 0C0H, (* xor eax, eax *)
  546. 08BH, 04DH, 008H, (* mov ecx, dword [ebp + 8] *) (* ecx <- a *)
  547. 083H, 0F9H, 01FH, (* cmp ecx, 31 *)
  548. 077H, 003H, (* ja L *)
  549. 00FH, 0ABH, 0C8H (* bts eax, ecx *)
  550. (* L: *)
  551. )
  552. END _set1;
  553. PROCEDURE [stdcall] _divmod* (y, x: INTEGER); (* (x div y) -> eax; (x mod y) -> edx *)
  554. BEGIN
  555. SYSTEM.CODE(
  556. 053H, (* push ebx *)
  557. 08BH, 045H, 00CH, (* mov eax, dword [ebp + 12] *) (* eax <- x *)
  558. 031H, 0D2H, (* xor edx, edx *)
  559. 085H, 0C0H, (* test eax, eax *)
  560. 074H, 018H, (* je L2 *)
  561. 07FH, 002H, (* jg L1 *)
  562. 0F7H, 0D2H, (* not edx *)
  563. (* L1: *)
  564. 089H, 0C3H, (* mov ebx, eax *)
  565. 08BH, 04DH, 008H, (* mov ecx, dword [ebp + 8] *) (* ecx <- y *)
  566. 0F7H, 0F9H, (* idiv ecx *)
  567. 085H, 0D2H, (* test edx, edx *)
  568. 074H, 009H, (* je L2 *)
  569. 031H, 0CBH, (* xor ebx, ecx *)
  570. 085H, 0DBH, (* test ebx, ebx *)
  571. 07DH, 003H, (* jge L2 *)
  572. 048H, (* dec eax *)
  573. 001H, 0CAH, (* add edx, ecx *)
  574. (* L2: *)
  575. 05BH (* pop ebx *)
  576. )
  577. END _divmod;
  578. PROCEDURE [stdcall] _new* (t, size: INTEGER; VAR ptr: INTEGER);
  579. BEGIN
  580. ptr := API._NEW(size);
  581. IF ptr # 0 THEN
  582. SYSTEM.PUT(ptr + ORD(API.OS = "LINUX")*12, t);
  583. INC(ptr, 4 + ORD(API.OS = "LINUX")*12)
  584. END
  585. END _new;
  586. PROCEDURE [stdcall] _dispose* (VAR ptr: INTEGER);
  587. BEGIN
  588. IF ptr # 0 THEN
  589. ptr := API._DISPOSE(ptr - (4 + ORD(API.OS = "LINUX")*12))
  590. END
  591. END _dispose;
  592. PROCEDURE [stdcall] _length* (len, str: INTEGER);
  593. BEGIN
  594. SYSTEM.CODE(
  595. 08BH, 045H, 00CH, (* mov eax, dword [ebp + 0Ch] *)
  596. 08BH, 04DH, 008H, (* mov ecx, dword [ebp + 08h] *)
  597. 048H, (* dec eax *)
  598. (* L1: *)
  599. 040H, (* inc eax *)
  600. 080H, 038H, 000H, (* cmp byte [eax], 0 *)
  601. 074H, 003H, (* jz L2 *)
  602. 0E2H, 0F8H, (* loop L1 *)
  603. 040H, (* inc eax *)
  604. (* L2: *)
  605. 02BH, 045H, 00CH (* sub eax, dword [ebp + 0Ch] *)
  606. )
  607. END _length;
  608. PROCEDURE [stdcall] _lengthw* (len, str: INTEGER);
  609. BEGIN
  610. SYSTEM.CODE(
  611. 08BH, 045H, 00CH, (* mov eax, dword [ebp + 0Ch] *)
  612. 08BH, 04DH, 008H, (* mov ecx, dword [ebp + 08h] *)
  613. 048H, (* dec eax *)
  614. 048H, (* dec eax *)
  615. (* L1: *)
  616. 040H, (* inc eax *)
  617. 040H, (* inc eax *)
  618. 066H, 083H, 038H, 000H, (* cmp word [eax], 0 *)
  619. 074H, 004H, (* jz L2 *)
  620. 0E2H, 0F6H, (* loop L1 *)
  621. 040H, (* inc eax *)
  622. 040H, (* inc eax *)
  623. (* L2: *)
  624. 02BH, 045H, 00CH, (* sub eax, dword [ebp + 0Ch] *)
  625. 0D1H, 0E8H (* shr eax, 1 *)
  626. )
  627. END _lengthw;
  628. PROCEDURE [stdcall] strncmp (a, b, n: INTEGER): INTEGER;
  629. BEGIN
  630. SYSTEM.CODE(
  631. 056H, (* push esi *)
  632. 057H, (* push edi *)
  633. 053H, (* push ebx *)
  634. 08BH, 075H, 008H, (* mov esi, dword[ebp + 8]; esi <- a *)
  635. 08BH, 07DH, 00CH, (* mov edi, dword[ebp + 12]; edi <- b *)
  636. 08BH, 05DH, 010H, (* mov ebx, dword[ebp + 16]; ebx <- n *)
  637. 031H, 0C9H, (* xor ecx, ecx *)
  638. 031H, 0D2H, (* xor edx, edx *)
  639. 0B8H,
  640. 000H, 000H, 000H, 080H, (* mov eax, minint *)
  641. (* L1: *)
  642. 085H, 0DBH, (* test ebx, ebx *)
  643. 07EH, 017H, (* jle L3 *)
  644. 08AH, 00EH, (* mov cl, byte[esi] *)
  645. 08AH, 017H, (* mov dl, byte[edi] *)
  646. 046H, (* inc esi *)
  647. 047H, (* inc edi *)
  648. 04BH, (* dec ebx *)
  649. 039H, 0D1H, (* cmp ecx, edx *)
  650. 074H, 006H, (* je L2 *)
  651. 089H, 0C8H, (* mov eax, ecx *)
  652. 029H, 0D0H, (* sub eax, edx *)
  653. 0EBH, 006H, (* jmp L3 *)
  654. (* L2: *)
  655. 085H, 0C9H, (* test ecx, ecx *)
  656. 075H, 0E7H, (* jne L1 *)
  657. 031H, 0C0H, (* xor eax, eax *)
  658. (* L3: *)
  659. 05BH, (* pop ebx *)
  660. 05FH, (* pop edi *)
  661. 05EH, (* pop esi *)
  662. 05DH, (* pop ebp *)
  663. 0C2H, 00CH, 000H (* ret 12 *)
  664. )
  665. RETURN 0
  666. END strncmp;
  667. PROCEDURE [stdcall] strncmpw (a, b, n: INTEGER): INTEGER;
  668. BEGIN
  669. SYSTEM.CODE(
  670. 056H, (* push esi *)
  671. 057H, (* push edi *)
  672. 053H, (* push ebx *)
  673. 08BH, 075H, 008H, (* mov esi, dword[ebp + 8]; esi <- a *)
  674. 08BH, 07DH, 00CH, (* mov edi, dword[ebp + 12]; edi <- b *)
  675. 08BH, 05DH, 010H, (* mov ebx, dword[ebp + 16]; ebx <- n *)
  676. 031H, 0C9H, (* xor ecx, ecx *)
  677. 031H, 0D2H, (* xor edx, edx *)
  678. 0B8H,
  679. 000H, 000H, 000H, 080H, (* mov eax, minint *)
  680. (* L1: *)
  681. 085H, 0DBH, (* test ebx, ebx *)
  682. 07EH, 01BH, (* jle L3 *)
  683. 066H, 08BH, 00EH, (* mov cx, word[esi] *)
  684. 066H, 08BH, 017H, (* mov dx, word[edi] *)
  685. 046H, (* inc esi *)
  686. 046H, (* inc esi *)
  687. 047H, (* inc edi *)
  688. 047H, (* inc edi *)
  689. 04BH, (* dec ebx *)
  690. 039H, 0D1H, (* cmp ecx, edx *)
  691. 074H, 006H, (* je L2 *)
  692. 089H, 0C8H, (* mov eax, ecx *)
  693. 029H, 0D0H, (* sub eax, edx *)
  694. 0EBH, 006H, (* jmp L3 *)
  695. (* L2: *)
  696. 085H, 0C9H, (* test ecx, ecx *)
  697. 075H, 0E3H, (* jne L1 *)
  698. 031H, 0C0H, (* xor eax, eax *)
  699. (* L3: *)
  700. 05BH, (* pop ebx *)
  701. 05FH, (* pop edi *)
  702. 05EH, (* pop esi *)
  703. 05DH, (* pop ebp *)
  704. 0C2H, 00CH, 000H (* ret 12 *)
  705. )
  706. RETURN 0
  707. END strncmpw;
  708. PROCEDURE [stdcall] _strcmp* (op, len2, str2, len1, str1: INTEGER): BOOLEAN;
  709. VAR
  710. res: INTEGER;
  711. bRes: BOOLEAN;
  712. c: CHAR;
  713. BEGIN
  714. res := strncmp(str1, str2, MIN(len1, len2));
  715. IF res = minint THEN
  716. IF len1 > len2 THEN
  717. SYSTEM.GET(str1 + len2, c);
  718. res := ORD(c)
  719. ELSIF len1 < len2 THEN
  720. SYSTEM.GET(str2 + len1, c);
  721. res := -ORD(c)
  722. ELSE
  723. res := 0
  724. END
  725. END;
  726. CASE op OF
  727. |0: bRes := res = 0
  728. |1: bRes := res # 0
  729. |2: bRes := res < 0
  730. |3: bRes := res <= 0
  731. |4: bRes := res > 0
  732. |5: bRes := res >= 0
  733. END
  734. RETURN bRes
  735. END _strcmp;
  736. PROCEDURE [stdcall] _strcmpw* (op, len2, str2, len1, str1: INTEGER): BOOLEAN;
  737. VAR
  738. res: INTEGER;
  739. bRes: BOOLEAN;
  740. c: WCHAR;
  741. BEGIN
  742. res := strncmpw(str1, str2, MIN(len1, len2));
  743. IF res = minint THEN
  744. IF len1 > len2 THEN
  745. SYSTEM.GET(str1 + len2 * 2, c);
  746. res := ORD(c)
  747. ELSIF len1 < len2 THEN
  748. SYSTEM.GET(str2 + len1 * 2, c);
  749. res := -ORD(c)
  750. ELSE
  751. res := 0
  752. END
  753. END;
  754. CASE op OF
  755. |0: bRes := res = 0
  756. |1: bRes := res # 0
  757. |2: bRes := res < 0
  758. |3: bRes := res <= 0
  759. |4: bRes := res > 0
  760. |5: bRes := res >= 0
  761. END
  762. RETURN bRes
  763. END _strcmpw;
  764. PROCEDURE PCharToStr (pchar: INTEGER; VAR s: ARRAY OF CHAR);
  765. VAR
  766. c: CHAR;
  767. i: INTEGER;
  768. BEGIN
  769. i := 0;
  770. REPEAT
  771. SYSTEM.GET(pchar, c);
  772. s[i] := c;
  773. INC(pchar);
  774. INC(i)
  775. UNTIL c = 0X
  776. END PCharToStr;
  777. PROCEDURE IntToStr (x: INTEGER; VAR str: ARRAY OF CHAR);
  778. VAR
  779. i, a: INTEGER;
  780. BEGIN
  781. i := 0;
  782. a := x;
  783. REPEAT
  784. INC(i);
  785. a := a DIV 10
  786. UNTIL a = 0;
  787. str[i] := 0X;
  788. REPEAT
  789. DEC(i);
  790. str[i] := CHR(x MOD 10 + ORD("0"));
  791. x := x DIV 10
  792. UNTIL x = 0
  793. END IntToStr;
  794. PROCEDURE append (VAR s1: ARRAY OF CHAR; s2: ARRAY OF CHAR);
  795. VAR
  796. n1, n2: INTEGER;
  797. BEGIN
  798. n1 := LENGTH(s1);
  799. n2 := LENGTH(s2);
  800. ASSERT(n1 + n2 < LEN(s1));
  801. SYSTEM.MOVE(SYSTEM.ADR(s2[0]), SYSTEM.ADR(s1[n1]), n2);
  802. s1[n1 + n2] := 0X
  803. END append;
  804. PROCEDURE [stdcall] _error* (modnum, _module, err, line: INTEGER);
  805. VAR
  806. s, temp: ARRAY 1024 OF CHAR;
  807. BEGIN
  808. CASE err OF
  809. | 1: s := "assertion failure"
  810. | 2: s := "NIL dereference"
  811. | 3: s := "bad divisor"
  812. | 4: s := "NIL procedure call"
  813. | 5: s := "type guard error"
  814. | 6: s := "index out of range"
  815. | 7: s := "invalid CASE"
  816. | 8: s := "array assignment error"
  817. | 9: s := "CHR out of range"
  818. |10: s := "WCHR out of range"
  819. |11: s := "BYTE out of range"
  820. END;
  821. append(s, API.eol + "module: "); PCharToStr(_module, temp); append(s, temp);
  822. append(s, API.eol + "line: "); IntToStr(line, temp); append(s, temp);
  823. API.DebugMsg(SYSTEM.ADR(s[0]), name);
  824. API.exit_thread(0)
  825. END _error;
  826. PROCEDURE [stdcall] _isrec* (t0, t1, r: INTEGER): INTEGER;
  827. BEGIN
  828. SYSTEM.GET(t0 + t1 + types, t0)
  829. RETURN t0 MOD 2
  830. END _isrec;
  831. PROCEDURE [stdcall] _is* (t0, p: INTEGER): INTEGER;
  832. BEGIN
  833. IF p # 0 THEN
  834. SYSTEM.GET(p - WORD, p);
  835. SYSTEM.GET(t0 + p + types, p)
  836. END
  837. RETURN p MOD 2
  838. END _is;
  839. PROCEDURE [stdcall] _guardrec* (t0, t1: INTEGER): INTEGER;
  840. BEGIN
  841. SYSTEM.GET(t0 + t1 + types, t0)
  842. RETURN t0 MOD 2
  843. END _guardrec;
  844. PROCEDURE [stdcall] _guard* (t0, p: INTEGER): INTEGER;
  845. BEGIN
  846. SYSTEM.GET(p, p);
  847. IF p # 0 THEN
  848. SYSTEM.GET(p - WORD, p);
  849. SYSTEM.GET(t0 + p + types, p)
  850. ELSE
  851. p := 1
  852. END
  853. RETURN p MOD 2
  854. END _guard;
  855. PROCEDURE [stdcall] _dllentry* (hinstDLL, fdwReason, lpvReserved: INTEGER): INTEGER;
  856. RETURN API.dllentry(hinstDLL, fdwReason, lpvReserved)
  857. END _dllentry;
  858. PROCEDURE [stdcall] _sofinit*;
  859. BEGIN
  860. API.sofinit
  861. END _sofinit;
  862. PROCEDURE [stdcall] _exit* (code: INTEGER);
  863. BEGIN
  864. API.exit(code)
  865. END _exit;
  866. PROCEDURE [stdcall] _init* (modname: INTEGER; tcount, _types: INTEGER; code, param: INTEGER);
  867. VAR
  868. t0, t1, i, j: INTEGER;
  869. BEGIN
  870. SYSTEM.CODE(09BH, 0DBH, 0E3H); (* finit *)
  871. API.init(param, code);
  872. types := API._NEW(tcount * tcount + SYSTEM.SIZE(INTEGER));
  873. ASSERT(types # 0);
  874. FOR i := 0 TO tcount - 1 DO
  875. FOR j := 0 TO tcount - 1 DO
  876. t0 := i; t1 := j;
  877. WHILE (t1 # 0) & (t1 # t0) DO
  878. SYSTEM.GET(_types + t1 * WORD, t1)
  879. END;
  880. SYSTEM.PUT8(i * tcount + j + types, ORD(t0 = t1))
  881. END
  882. END;
  883. name := modname
  884. END _init;
  885. END RTL.
  886. $END