82d72daa76
git-svn-id: svn://kolibrios.org@7597 a494cfbc-eb01-0410-851d-a64ba20cac60
630 lines
15 KiB
Plaintext
630 lines
15 KiB
Plaintext
(*
|
|
BSD 2-Clause License
|
|
|
|
Copyright (c) 2018, 2019, Anton Krotov
|
|
All rights reserved.
|
|
*)
|
|
|
|
MODULE RTL;
|
|
|
|
IMPORT SYSTEM, API;
|
|
|
|
|
|
CONST
|
|
|
|
bit_depth* = 32;
|
|
maxint* = 7FFFFFFFH;
|
|
minint* = 80000000H;
|
|
|
|
DLL_PROCESS_ATTACH = 1;
|
|
DLL_THREAD_ATTACH = 2;
|
|
DLL_THREAD_DETACH = 3;
|
|
DLL_PROCESS_DETACH = 0;
|
|
|
|
SIZE_OF_DWORD = 4;
|
|
|
|
|
|
TYPE
|
|
|
|
DLL_ENTRY* = PROCEDURE (hinstDLL, fdwReason, lpvReserved: INTEGER);
|
|
|
|
|
|
VAR
|
|
|
|
name: INTEGER;
|
|
types: INTEGER;
|
|
|
|
dll: RECORD
|
|
process_detach,
|
|
thread_detach,
|
|
thread_attach: DLL_ENTRY
|
|
END;
|
|
|
|
|
|
PROCEDURE [stdcall] _move* (bytes, source, dest: INTEGER);
|
|
BEGIN
|
|
SYSTEM.CODE(
|
|
|
|
08BH, 045H, 008H, (* mov eax, dword [ebp + 8] *)
|
|
085H, 0C0H, (* test eax, eax *)
|
|
07EH, 019H, (* jle L *)
|
|
0FCH, (* cld *)
|
|
057H, (* push edi *)
|
|
056H, (* push esi *)
|
|
08BH, 075H, 00CH, (* mov esi, dword [ebp + 12] *)
|
|
08BH, 07DH, 010H, (* mov edi, dword [ebp + 16] *)
|
|
089H, 0C1H, (* mov ecx, eax *)
|
|
0C1H, 0E9H, 002H, (* shr ecx, 2 *)
|
|
0F3H, 0A5H, (* rep movsd *)
|
|
089H, 0C1H, (* mov ecx, eax *)
|
|
083H, 0E1H, 003H, (* and ecx, 3 *)
|
|
0F3H, 0A4H, (* rep movsb *)
|
|
05EH, (* pop esi *)
|
|
05FH (* pop edi *)
|
|
(* L: *)
|
|
)
|
|
END _move;
|
|
|
|
|
|
PROCEDURE [stdcall] _move2* (bytes, dest, source: INTEGER);
|
|
BEGIN
|
|
SYSTEM.CODE(
|
|
|
|
08BH, 045H, 008H, (* mov eax, dword [ebp + 8] *)
|
|
085H, 0C0H, (* test eax, eax *)
|
|
07EH, 019H, (* jle L *)
|
|
0FCH, (* cld *)
|
|
057H, (* push edi *)
|
|
056H, (* push esi *)
|
|
08BH, 075H, 010H, (* mov esi, dword [ebp + 16] *)
|
|
08BH, 07DH, 00CH, (* mov edi, dword [ebp + 12] *)
|
|
089H, 0C1H, (* mov ecx, eax *)
|
|
0C1H, 0E9H, 002H, (* shr ecx, 2 *)
|
|
0F3H, 0A5H, (* rep movsd *)
|
|
089H, 0C1H, (* mov ecx, eax *)
|
|
083H, 0E1H, 003H, (* and ecx, 3 *)
|
|
0F3H, 0A4H, (* rep movsb *)
|
|
05EH, (* pop esi *)
|
|
05FH (* pop edi *)
|
|
(* L: *)
|
|
)
|
|
END _move2;
|
|
|
|
|
|
PROCEDURE [stdcall] _arrcpy* (base_size, len_dst, dst, len_src, src: INTEGER): BOOLEAN;
|
|
VAR
|
|
res: BOOLEAN;
|
|
|
|
BEGIN
|
|
IF len_src > len_dst THEN
|
|
res := FALSE
|
|
ELSE
|
|
_move(len_src * base_size, src, dst);
|
|
res := TRUE
|
|
END
|
|
|
|
RETURN res
|
|
END _arrcpy;
|
|
|
|
|
|
PROCEDURE [stdcall] _strcpy* (chr_size, len_dst, dst, len_src, src: INTEGER);
|
|
BEGIN
|
|
_move(MIN(len_dst, len_src) * chr_size, src, dst)
|
|
END _strcpy;
|
|
|
|
|
|
PROCEDURE [stdcall] _strcpy2* (chr_size, len_src, src, len_dst, dst: INTEGER);
|
|
BEGIN
|
|
_move(MIN(len_dst, len_src) * chr_size, src, dst)
|
|
END _strcpy2;
|
|
|
|
|
|
PROCEDURE [stdcall] _rot* (VAR A: ARRAY OF INTEGER);
|
|
VAR
|
|
i, n, k: INTEGER;
|
|
|
|
BEGIN
|
|
|
|
k := LEN(A) - 1;
|
|
n := A[0];
|
|
i := 0;
|
|
WHILE i < k DO
|
|
A[i] := A[i + 1];
|
|
INC(i)
|
|
END;
|
|
A[k] := n
|
|
|
|
END _rot;
|
|
|
|
|
|
PROCEDURE [stdcall] _set2* (a, b: INTEGER): INTEGER;
|
|
VAR
|
|
res: INTEGER;
|
|
|
|
BEGIN
|
|
IF (a <= b) & (a <= 31) & (b >= 0) THEN
|
|
IF b > 31 THEN
|
|
b := 31
|
|
END;
|
|
IF a < 0 THEN
|
|
a := 0
|
|
END;
|
|
res := LSR(ASR(ROR(1, 1), b - a), 31 - b)
|
|
ELSE
|
|
res := 0
|
|
END
|
|
|
|
RETURN res
|
|
END _set2;
|
|
|
|
|
|
PROCEDURE [stdcall] _set* (b, a: INTEGER): INTEGER;
|
|
RETURN _set2(a, b)
|
|
END _set;
|
|
|
|
|
|
PROCEDURE [stdcall] divmod (a, b: INTEGER; VAR mod: INTEGER): INTEGER;
|
|
BEGIN
|
|
SYSTEM.CODE(
|
|
|
|
08BH, 045H, 008H, (* mov eax, dword [ebp + 8] *)
|
|
08BH, 04DH, 00CH, (* mov ecx, dword [ebp + 12] *)
|
|
031H, 0D2H, (* xor edx, edx *)
|
|
085H, 0C0H, (* test eax, eax *)
|
|
07DH, 002H, (* jge L1 *)
|
|
0F7H, 0D2H, (* not edx *)
|
|
(* L1: *)
|
|
0F7H, 0F9H, (* idiv ecx *)
|
|
08BH, 04DH, 010H, (* mov ecx, dword [ebp + 16] *)
|
|
089H, 011H, (* mov dword [ecx], edx *)
|
|
0C9H, (* leave *)
|
|
0C2H, 00CH, 000H (* ret 12 *)
|
|
)
|
|
|
|
RETURN 0
|
|
END divmod;
|
|
|
|
|
|
PROCEDURE div_ (x, y: INTEGER): INTEGER;
|
|
VAR
|
|
div, mod: INTEGER;
|
|
|
|
BEGIN
|
|
div := divmod(x, y, mod);
|
|
IF (mod # 0) & ((x < 0) & (y > 0) OR (x > 0) & (y < 0)) THEN
|
|
DEC(div)
|
|
END
|
|
|
|
RETURN div
|
|
END div_;
|
|
|
|
|
|
PROCEDURE mod_ (x, y: INTEGER): INTEGER;
|
|
VAR
|
|
div, mod: INTEGER;
|
|
|
|
BEGIN
|
|
div := divmod(x, y, mod);
|
|
IF (mod # 0) & ((x < 0) & (y > 0) OR (x > 0) & (y < 0)) THEN
|
|
INC(mod, y)
|
|
END
|
|
|
|
RETURN mod
|
|
END mod_;
|
|
|
|
|
|
PROCEDURE [stdcall] _div* (b, a: INTEGER): INTEGER;
|
|
RETURN div_(a, b)
|
|
END _div;
|
|
|
|
|
|
PROCEDURE [stdcall] _div2* (a, b: INTEGER): INTEGER;
|
|
RETURN div_(a, b)
|
|
END _div2;
|
|
|
|
|
|
PROCEDURE [stdcall] _mod* (b, a: INTEGER): INTEGER;
|
|
RETURN mod_(a, b)
|
|
END _mod;
|
|
|
|
|
|
PROCEDURE [stdcall] _mod2* (a, b: INTEGER): INTEGER;
|
|
RETURN mod_(a, b)
|
|
END _mod2;
|
|
|
|
|
|
PROCEDURE [stdcall] _new* (t, size: INTEGER; VAR ptr: INTEGER);
|
|
BEGIN
|
|
ptr := API._NEW(size);
|
|
IF ptr # 0 THEN
|
|
SYSTEM.PUT(ptr, t);
|
|
INC(ptr, SIZE_OF_DWORD)
|
|
END
|
|
END _new;
|
|
|
|
|
|
PROCEDURE [stdcall] _dispose* (VAR ptr: INTEGER);
|
|
BEGIN
|
|
IF ptr # 0 THEN
|
|
ptr := API._DISPOSE(ptr - SIZE_OF_DWORD)
|
|
END
|
|
END _dispose;
|
|
|
|
|
|
PROCEDURE strncmp (a, b, n: INTEGER): INTEGER;
|
|
VAR
|
|
A, B: CHAR;
|
|
res: INTEGER;
|
|
|
|
BEGIN
|
|
res := 0;
|
|
WHILE n > 0 DO
|
|
SYSTEM.GET(a, A); INC(a);
|
|
SYSTEM.GET(b, B); INC(b);
|
|
DEC(n);
|
|
IF A # B THEN
|
|
res := ORD(A) - ORD(B);
|
|
n := 0
|
|
ELSIF A = 0X THEN
|
|
n := 0
|
|
END
|
|
END
|
|
RETURN res
|
|
END strncmp;
|
|
|
|
|
|
PROCEDURE strncmpw (a, b, n: INTEGER): INTEGER;
|
|
VAR
|
|
A, B: WCHAR;
|
|
res: INTEGER;
|
|
|
|
BEGIN
|
|
res := 0;
|
|
WHILE n > 0 DO
|
|
SYSTEM.GET(a, A); INC(a, 2);
|
|
SYSTEM.GET(b, B); INC(b, 2);
|
|
DEC(n);
|
|
IF A # B THEN
|
|
res := ORD(A) - ORD(B);
|
|
n := 0
|
|
ELSIF A = 0X THEN
|
|
n := 0
|
|
END
|
|
END
|
|
RETURN res
|
|
END strncmpw;
|
|
|
|
|
|
PROCEDURE [stdcall] _length* (len, str: INTEGER): INTEGER;
|
|
BEGIN
|
|
SYSTEM.CODE(
|
|
|
|
08BH, 045H, 00CH, (* mov eax, dword [ebp + 0Ch] *)
|
|
08BH, 04DH, 008H, (* mov ecx, dword [ebp + 08h] *)
|
|
048H, (* dec eax *)
|
|
(* L1: *)
|
|
040H, (* inc eax *)
|
|
080H, 038H, 000H, (* cmp byte [eax], 0 *)
|
|
074H, 003H, (* jz L2 *)
|
|
0E2H, 0F8H, (* loop L1 *)
|
|
040H, (* inc eax *)
|
|
(* L2: *)
|
|
02BH, 045H, 00CH, (* sub eax, dword [ebp + 0Ch] *)
|
|
0C9H, (* leave *)
|
|
0C2H, 008H, 000H (* ret 08h *)
|
|
)
|
|
|
|
RETURN 0
|
|
END _length;
|
|
|
|
|
|
PROCEDURE [stdcall] _lengthw* (len, str: INTEGER): INTEGER;
|
|
BEGIN
|
|
SYSTEM.CODE(
|
|
|
|
08BH, 045H, 00CH, (* mov eax, dword [ebp + 0Ch] *)
|
|
08BH, 04DH, 008H, (* mov ecx, dword [ebp + 08h] *)
|
|
048H, (* dec eax *)
|
|
048H, (* dec eax *)
|
|
(* L1: *)
|
|
040H, (* inc eax *)
|
|
040H, (* inc eax *)
|
|
066H, 083H, 038H, 000H, (* cmp word [eax], 0 *)
|
|
074H, 004H, (* jz L2 *)
|
|
0E2H, 0F6H, (* loop L1 *)
|
|
040H, (* inc eax *)
|
|
040H, (* inc eax *)
|
|
(* L2: *)
|
|
02BH, 045H, 00CH, (* sub eax, dword [ebp + 0Ch] *)
|
|
0D1H, 0E8H, (* shr eax, 1 *)
|
|
0C9H, (* leave *)
|
|
0C2H, 008H, 000H (* ret 08h *)
|
|
)
|
|
|
|
RETURN 0
|
|
END _lengthw;
|
|
|
|
|
|
PROCEDURE [stdcall] _strcmp* (op, len2, str2, len1, str1: INTEGER): BOOLEAN;
|
|
VAR
|
|
res: INTEGER;
|
|
bRes: BOOLEAN;
|
|
|
|
BEGIN
|
|
|
|
res := strncmp(str1, str2, MIN(len1, len2));
|
|
IF res = 0 THEN
|
|
res := _length(len1, str1) - _length(len2, str2)
|
|
END;
|
|
|
|
CASE op OF
|
|
|0: bRes := res = 0
|
|
|1: bRes := res # 0
|
|
|2: bRes := res < 0
|
|
|3: bRes := res <= 0
|
|
|4: bRes := res > 0
|
|
|5: bRes := res >= 0
|
|
END
|
|
|
|
RETURN bRes
|
|
END _strcmp;
|
|
|
|
|
|
PROCEDURE [stdcall] _strcmp2* (op, len1, str1, len2, str2: INTEGER): BOOLEAN;
|
|
RETURN _strcmp(op, len2, str2, len1, str1)
|
|
END _strcmp2;
|
|
|
|
|
|
PROCEDURE [stdcall] _strcmpw* (op, len2, str2, len1, str1: INTEGER): BOOLEAN;
|
|
VAR
|
|
res: INTEGER;
|
|
bRes: BOOLEAN;
|
|
|
|
BEGIN
|
|
|
|
res := strncmpw(str1, str2, MIN(len1, len2));
|
|
IF res = 0 THEN
|
|
res := _lengthw(len1, str1) - _lengthw(len2, str2)
|
|
END;
|
|
|
|
CASE op OF
|
|
|0: bRes := res = 0
|
|
|1: bRes := res # 0
|
|
|2: bRes := res < 0
|
|
|3: bRes := res <= 0
|
|
|4: bRes := res > 0
|
|
|5: bRes := res >= 0
|
|
END
|
|
|
|
RETURN bRes
|
|
END _strcmpw;
|
|
|
|
|
|
PROCEDURE [stdcall] _strcmpw2* (op, len1, str1, len2, str2: INTEGER): BOOLEAN;
|
|
RETURN _strcmpw(op, len2, str2, len1, str1)
|
|
END _strcmpw2;
|
|
|
|
|
|
PROCEDURE PCharToStr (pchar: INTEGER; VAR s: ARRAY OF CHAR);
|
|
VAR
|
|
c: CHAR;
|
|
i: INTEGER;
|
|
|
|
BEGIN
|
|
i := 0;
|
|
REPEAT
|
|
SYSTEM.GET(pchar, c);
|
|
s[i] := c;
|
|
INC(pchar);
|
|
INC(i)
|
|
UNTIL c = 0X
|
|
END PCharToStr;
|
|
|
|
|
|
PROCEDURE IntToStr (x: INTEGER; VAR str: ARRAY OF CHAR);
|
|
VAR
|
|
i, a, b: INTEGER;
|
|
c: CHAR;
|
|
|
|
BEGIN
|
|
|
|
i := 0;
|
|
REPEAT
|
|
str[i] := CHR(x MOD 10 + ORD("0"));
|
|
x := x DIV 10;
|
|
INC(i)
|
|
UNTIL x = 0;
|
|
|
|
a := 0;
|
|
b := i - 1;
|
|
WHILE a < b DO
|
|
c := str[a];
|
|
str[a] := str[b];
|
|
str[b] := c;
|
|
INC(a);
|
|
DEC(b)
|
|
END;
|
|
str[i] := 0X
|
|
END IntToStr;
|
|
|
|
|
|
PROCEDURE append (VAR s1: ARRAY OF CHAR; s2: ARRAY OF CHAR);
|
|
VAR
|
|
n1, n2, i, j: INTEGER;
|
|
BEGIN
|
|
n1 := LENGTH(s1);
|
|
n2 := LENGTH(s2);
|
|
|
|
ASSERT(n1 + n2 < LEN(s1));
|
|
|
|
i := 0;
|
|
j := n1;
|
|
WHILE i < n2 DO
|
|
s1[j] := s2[i];
|
|
INC(i);
|
|
INC(j)
|
|
END;
|
|
|
|
s1[j] := 0X
|
|
|
|
END append;
|
|
|
|
|
|
PROCEDURE [stdcall] _error* (module, err: INTEGER);
|
|
VAR
|
|
s, temp: ARRAY 1024 OF CHAR;
|
|
|
|
BEGIN
|
|
|
|
s := "";
|
|
CASE err MOD 16 OF
|
|
| 1: append(s, "assertion failure")
|
|
| 2: append(s, "NIL dereference")
|
|
| 3: append(s, "division by zero")
|
|
| 4: append(s, "NIL procedure call")
|
|
| 5: append(s, "type guard error")
|
|
| 6: append(s, "index out of range")
|
|
| 7: append(s, "invalid CASE")
|
|
| 8: append(s, "array assignment error")
|
|
| 9: append(s, "CHR out of range")
|
|
|10: append(s, "WCHR out of range")
|
|
|11: append(s, "BYTE out of range")
|
|
END;
|
|
|
|
append(s, API.eol);
|
|
|
|
append(s, "module: "); PCharToStr(module, temp); append(s, temp); append(s, API.eol);
|
|
append(s, "line: "); IntToStr(LSR(err, 4), temp); append(s, temp);
|
|
|
|
API.DebugMsg(SYSTEM.ADR(s[0]), name);
|
|
|
|
API.exit_thread(0)
|
|
END _error;
|
|
|
|
|
|
PROCEDURE [stdcall] _isrec* (t0, t1, r: INTEGER): BOOLEAN;
|
|
BEGIN
|
|
(* r IS t0 *)
|
|
|
|
WHILE (t1 # 0) & (t1 # t0) DO
|
|
SYSTEM.GET(types + t1 * SIZE_OF_DWORD, t1)
|
|
END
|
|
|
|
RETURN t1 = t0
|
|
END _isrec;
|
|
|
|
|
|
PROCEDURE [stdcall] _is* (t0, p: INTEGER): BOOLEAN;
|
|
VAR
|
|
t1: INTEGER;
|
|
|
|
BEGIN
|
|
(* p IS t0 *)
|
|
|
|
IF p # 0 THEN
|
|
DEC(p, SIZE_OF_DWORD);
|
|
SYSTEM.GET(p, t1);
|
|
WHILE (t1 # 0) & (t1 # t0) DO
|
|
SYSTEM.GET(types + t1 * SIZE_OF_DWORD, t1)
|
|
END
|
|
ELSE
|
|
t1 := -1
|
|
END
|
|
|
|
RETURN t1 = t0
|
|
END _is;
|
|
|
|
|
|
PROCEDURE [stdcall] _guardrec* (t0, t1: INTEGER): BOOLEAN;
|
|
BEGIN
|
|
(* r:t1 IS t0 *)
|
|
|
|
WHILE (t1 # 0) & (t1 # t0) DO
|
|
SYSTEM.GET(types + t1 * SIZE_OF_DWORD, t1)
|
|
END
|
|
|
|
RETURN t1 = t0
|
|
END _guardrec;
|
|
|
|
|
|
PROCEDURE [stdcall] _guard* (t0, p: INTEGER): BOOLEAN;
|
|
VAR
|
|
t1: INTEGER;
|
|
|
|
BEGIN
|
|
(* p IS t0 *)
|
|
SYSTEM.GET(p, p);
|
|
IF p # 0 THEN
|
|
DEC(p, SIZE_OF_DWORD);
|
|
SYSTEM.GET(p, t1);
|
|
WHILE (t1 # t0) & (t1 # 0) DO
|
|
SYSTEM.GET(types + t1 * SIZE_OF_DWORD, t1)
|
|
END
|
|
ELSE
|
|
t1 := t0
|
|
END
|
|
|
|
RETURN t1 = t0
|
|
END _guard;
|
|
|
|
|
|
PROCEDURE [stdcall] _dllentry* (hinstDLL, fdwReason, lpvReserved: INTEGER): INTEGER;
|
|
VAR
|
|
res: INTEGER;
|
|
|
|
BEGIN
|
|
CASE fdwReason OF
|
|
|DLL_PROCESS_ATTACH:
|
|
res := 1
|
|
|DLL_THREAD_ATTACH:
|
|
res := 0;
|
|
IF dll.thread_attach # NIL THEN
|
|
dll.thread_attach(hinstDLL, fdwReason, lpvReserved)
|
|
END
|
|
|DLL_THREAD_DETACH:
|
|
res := 0;
|
|
IF dll.thread_detach # NIL THEN
|
|
dll.thread_detach(hinstDLL, fdwReason, lpvReserved)
|
|
END
|
|
|DLL_PROCESS_DETACH:
|
|
res := 0;
|
|
IF dll.process_detach # NIL THEN
|
|
dll.process_detach(hinstDLL, fdwReason, lpvReserved)
|
|
END
|
|
ELSE
|
|
res := 0
|
|
END
|
|
|
|
RETURN res
|
|
END _dllentry;
|
|
|
|
|
|
PROCEDURE SetDll* (process_detach, thread_detach, thread_attach: DLL_ENTRY);
|
|
BEGIN
|
|
dll.process_detach := process_detach;
|
|
dll.thread_detach := thread_detach;
|
|
dll.thread_attach := thread_attach
|
|
END SetDll;
|
|
|
|
|
|
PROCEDURE [stdcall] _exit* (code: INTEGER);
|
|
BEGIN
|
|
API.exit(code)
|
|
END _exit;
|
|
|
|
|
|
PROCEDURE [stdcall] _init* (modname: INTEGER; typesc, _types: INTEGER; code, param: INTEGER);
|
|
BEGIN
|
|
SYSTEM.CODE(09BH, 0DBH, 0E3H); (* finit *)
|
|
API.init(param, code);
|
|
|
|
types := _types;
|
|
name := modname;
|
|
|
|
dll.process_detach := NIL;
|
|
dll.thread_detach := NIL;
|
|
dll.thread_attach := NIL;
|
|
END _init;
|
|
|
|
|
|
END RTL. |