b6bb3d2c62
git-svn-id: svn://kolibrios.org@7107 a494cfbc-eb01-0410-851d-a64ba20cac60
1988 lines
48 KiB
Plaintext
1988 lines
48 KiB
Plaintext
(*
|
|
Copyright 2016, 2017 Anton Krotov
|
|
|
|
This file is part of Compiler.
|
|
|
|
Compiler is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
Compiler is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Compiler. If not, see <http://www.gnu.org/licenses/>.
|
|
*)
|
|
|
|
MODULE X86;
|
|
|
|
IMPORT UTILS, sys := SYSTEM, SCAN, ELF;
|
|
|
|
CONST
|
|
|
|
ADIM* = 5;
|
|
|
|
lxPlus = 51; lxMinus = 52; lxMult = 53; lxSlash = 54;
|
|
lxEQ = 71; lxNE = 72; lxLT = 73; lxGT = 74; lxLE = 75; lxGE = 76;
|
|
|
|
TINTEGER = 1; TREAL = 2; TLONGREAL = 3; TCHAR = 4; TSET = 5; TBOOLEAN = 6; TVOID = 7;
|
|
TNIL = 8; TCARD16 = 9; TSTRING = 10; TARRAY = 11; TRECORD = 12; TPOINTER = 13; TPROC = 14;
|
|
|
|
stABS* = 1; stODD* = 2; stLEN* = 3; stLSL* = 4; stASR* = 5; stROR* = 6; stFLOOR* = 7;
|
|
stFLT* = 8; stORD* = 9; stCHR* = 10; stLONG* = 11; stSHORT* = 12; stINC* = 13;
|
|
stDEC* = 14; stINCL* = 15; stEXCL* = 16; stCOPY* = 17; stNEW* = 18; stASSERT* = 19;
|
|
stPACK* = 20; stUNPK* = 21; stDISPOSE* = 22; stFABS* = 23; stINC1* = 24;
|
|
stDEC1* = 25; stASSERT1* = 26; stUNPK1* = 27; stPACK1* = 28; stLSR* = 29;
|
|
stLENGTH* = 30; stMIN* = 31; stMAX* = 32;
|
|
|
|
sysMOVE* = 108;
|
|
|
|
JMP* = 0E9X; CALL = 0E8X;
|
|
JE = 84X; JNE = 85X; JLE = 8EX; JGE = 8DX; JG = 8FX; JL = 8CX;
|
|
|
|
JCMD = 1; LCMD = 2; GCMD = 3; OCMD = 4; ECMD = 5;
|
|
PUSHEAX = 6; PUSHECX = 7; PUSHEDX = 8; POPEAX = 9; POPECX = 10; POPEDX = 11;
|
|
ICMP1 = 13; ICMP2 = 14;
|
|
|
|
defcall = 0; stdcall = 1; cdecl = 2; winapi = 3;
|
|
|
|
_rset* = 0; _inset* = 1; _saverec* = 2; _length* = 3; _checktype* = 4; _strcmp* = 5;
|
|
_lstrcmp* = 6; _rstrcmp* = 7; _savearr* = 8; _newrec* = 9; _disprec* = 10; _arrayidx* = 11;
|
|
_arrayrot* = 12; _assrt* = 13; _strcopy* = 14; _arrayidx1* = 15; _init* = 16; _close* = 17; _halt* = 18;
|
|
ASSRT = 19; hInstance = 20; SELFNAME = 21; RTABLE = 22;LoadLibrary = 23; GetProcAddress = 24;
|
|
Exports = 25; szSTART = 26; START = 27; szversion = 28; _floor = 29; HALT = 30;
|
|
|
|
FREGS = 8;
|
|
|
|
TYPE
|
|
|
|
ASMLINE* = POINTER TO RECORD (UTILS.rITEM)
|
|
cmd, clen, varadr, adr, tcmd, codeadr: INTEGER; short: BOOLEAN
|
|
END;
|
|
|
|
TFLT = ARRAY 2 OF INTEGER;
|
|
|
|
TIDX* = ARRAY ADIM OF INTEGER;
|
|
|
|
SECTIONNAME = ARRAY 8 OF CHAR;
|
|
|
|
SECTION = RECORD
|
|
name: SECTIONNAME;
|
|
size, adr, sizealign, OAPfile, reserved6, reserved7, reserved8, attrflags: INTEGER
|
|
END;
|
|
|
|
HEADER = RECORD
|
|
msdos: ARRAY 180 OF CHAR;
|
|
typecomp, seccount: sys.CARD16;
|
|
time, reserved1, reserved2: INTEGER;
|
|
PEoptsize, infflags, PEfile, compver: sys.CARD16;
|
|
codesize, datasize, initdatasize, startadr,
|
|
codeadr, rdataadr, loadadr, secalign, filealign,
|
|
oldestver, version, oldestverNT, reserved3,
|
|
filesize, headersize, dllcrc: INTEGER;
|
|
UI, reserved4: sys.CARD16;
|
|
stksize, stkalloc, heapsize, heapalloc, reserved5, structcount: INTEGER;
|
|
structs: ARRAY 16 OF RECORD adr, size: INTEGER END;
|
|
sections: ARRAY 3 OF SECTION
|
|
END;
|
|
|
|
COFFHEADER = RECORD
|
|
Machine: sys.CARD16;
|
|
NumberOfSections: sys.CARD16;
|
|
TimeDateStamp,
|
|
PointerToSymbolTable,
|
|
NumberOfSymbols: INTEGER;
|
|
SizeOfOptionalHeader,
|
|
Characteristics: sys.CARD16;
|
|
text, data, bss: SECTION
|
|
END;
|
|
|
|
KOSHEADER = RECORD
|
|
menuet01: ARRAY 8 OF CHAR;
|
|
ver, start, size, mem, sp, param, path: INTEGER
|
|
END;
|
|
|
|
ETABLE = RECORD
|
|
reserved1, time, reserved2, dllnameoffset, firstnum, adrcount,
|
|
namecount, arradroffset, arrnameptroffset, arrnumoffset: INTEGER;
|
|
arradr, arrnameptr: ARRAY 10000H OF INTEGER;
|
|
arrnum: ARRAY 10000H OF sys.CARD16;
|
|
text: ARRAY 1000000 OF CHAR;
|
|
textlen, size: INTEGER
|
|
END;
|
|
|
|
RELOC = RECORD
|
|
Page, Size: INTEGER;
|
|
reloc: ARRAY 1024 OF sys.CARD16
|
|
END;
|
|
|
|
VAR asmlist: UTILS.LIST; start: ASMLINE; dll, con, gui, kos, elf, obj: BOOLEAN;
|
|
Lcount, reccount, topstk: INTEGER; recarray: ARRAY 2048 OF INTEGER; current*: ASMLINE;
|
|
callstk: ARRAY 1024, 2 OF ASMLINE; OutFile: UTILS.STRING;
|
|
Code: ARRAY 4000000 OF CHAR; ccount: INTEGER; Data: ARRAY 1000000 OF CHAR; dcount: INTEGER;
|
|
Labels: ARRAY 200000 OF INTEGER; rdata: ARRAY 400H OF INTEGER; Header: HEADER; etable: ETABLE;
|
|
ExecName: UTILS.STRING; LoadAdr: INTEGER; Reloc: ARRAY 200000 OF CHAR; rcount: INTEGER;
|
|
RtlProc: ARRAY 20 OF INTEGER; OutFilePos: INTEGER; RelocSection: SECTION;
|
|
fpu*: INTEGER; isfpu: BOOLEAN; maxfpu: INTEGER; fpucmd: ASMLINE;
|
|
kosexp: ARRAY 65536 OF RECORD Name: SCAN.NODE; Adr, NameLabel: INTEGER END; kosexpcount: INTEGER;
|
|
|
|
PROCEDURE AddRtlProc*(idx, proc: INTEGER);
|
|
BEGIN
|
|
RtlProc[idx] := proc
|
|
END AddRtlProc;
|
|
|
|
PROCEDURE IntToCard16(i: INTEGER): sys.CARD16;
|
|
VAR w: sys.CARD16;
|
|
BEGIN
|
|
sys.GET(sys.ADR(i), w)
|
|
RETURN w
|
|
END IntToCard16;
|
|
|
|
PROCEDURE CopyStr(VAR Dest: ARRAY OF CHAR; Source: ARRAY OF CHAR; VAR di: INTEGER; si: INTEGER);
|
|
BEGIN
|
|
DEC(di);
|
|
REPEAT
|
|
INC(di);
|
|
Dest[di] := Source[si];
|
|
INC(si)
|
|
UNTIL Dest[di] = 0X
|
|
END CopyStr;
|
|
|
|
PROCEDURE exch(VAR a, b: INTEGER);
|
|
VAR c: INTEGER;
|
|
BEGIN
|
|
c := a;
|
|
a := b;
|
|
b := c
|
|
END exch;
|
|
|
|
PROCEDURE Sort(VAR NamePtr, Adr: ARRAY OF INTEGER; Text: ARRAY OF CHAR; LB, RB: INTEGER);
|
|
VAR L, R: INTEGER;
|
|
|
|
PROCEDURE strle(s1, s2: INTEGER): BOOLEAN;
|
|
VAR S1, S2: ARRAY 256 OF CHAR; i: INTEGER;
|
|
BEGIN
|
|
i := 0;
|
|
CopyStr(S1, Text, i, s1);
|
|
i := 0;
|
|
CopyStr(S2, Text, i, s2)
|
|
RETURN S1 <= S2
|
|
END strle;
|
|
|
|
BEGIN
|
|
IF LB < RB THEN
|
|
L := LB;
|
|
R := RB;
|
|
REPEAT
|
|
WHILE (L < RB) & strle(NamePtr[L], NamePtr[LB]) DO
|
|
INC(L)
|
|
END;
|
|
WHILE (R > LB) & strle(NamePtr[LB], NamePtr[R]) DO
|
|
DEC(R)
|
|
END;
|
|
IF L < R THEN
|
|
exch(NamePtr[L], NamePtr[R]);
|
|
exch(Adr[L], Adr[R])
|
|
END
|
|
UNTIL L >= R;
|
|
IF R > LB THEN
|
|
exch(NamePtr[LB], NamePtr[R]);
|
|
exch(Adr[LB], Adr[R]);
|
|
Sort(NamePtr, Adr, Text, LB, R - 1)
|
|
END;
|
|
Sort(NamePtr, Adr, Text, R + 1, RB)
|
|
END
|
|
END Sort;
|
|
|
|
PROCEDURE PackExport(Name: ARRAY OF CHAR);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
Sort(etable.arrnameptr, etable.arradr, etable.text, 0, etable.namecount - 1);
|
|
FOR i := 0 TO etable.namecount - 1 DO
|
|
etable.arrnum[i] := IntToCard16(i)
|
|
END;
|
|
etable.size := 40 + etable.adrcount * 4 + etable.namecount * 6;
|
|
etable.arradroffset := 40;
|
|
etable.arrnameptroffset := 40 + etable.adrcount * 4;
|
|
etable.arrnumoffset := etable.arrnameptroffset + etable.namecount * 4;
|
|
etable.dllnameoffset := etable.size + etable.textlen;
|
|
CopyStr(etable.text, Name, etable.textlen, 0);
|
|
INC(etable.textlen);
|
|
FOR i := 0 TO etable.namecount - 1 DO
|
|
etable.arrnameptr[i] := etable.arrnameptr[i] + etable.size
|
|
END;
|
|
etable.size := etable.size + etable.textlen
|
|
END PackExport;
|
|
|
|
PROCEDURE ProcExport*(Number: INTEGER; Name: SCAN.NODE; NameLabel: INTEGER);
|
|
BEGIN
|
|
IF dll THEN
|
|
etable.arradr[etable.adrcount] := Number;
|
|
INC(etable.adrcount);
|
|
etable.arrnameptr[etable.namecount] := etable.textlen;
|
|
INC(etable.namecount);
|
|
CopyStr(etable.text, Name.Name, etable.textlen, 0);
|
|
INC(etable.textlen)
|
|
ELSIF obj THEN
|
|
kosexp[kosexpcount].Name := Name;
|
|
kosexp[kosexpcount].Adr := Number;
|
|
kosexp[kosexpcount].NameLabel := NameLabel;
|
|
INC(kosexpcount)
|
|
END
|
|
END ProcExport;
|
|
|
|
PROCEDURE Err(code: INTEGER);
|
|
BEGIN
|
|
CASE code OF
|
|
|1: UTILS.ErrMsg(67); UTILS.OutString(OutFile)
|
|
|2: UTILS.ErrMsg(69); UTILS.OutString(OutFile)
|
|
ELSE
|
|
END;
|
|
UTILS.Ln;
|
|
UTILS.HALT(1)
|
|
END Err;
|
|
|
|
PROCEDURE Align*(n, m: INTEGER): INTEGER;
|
|
RETURN n + (m - n MOD m) MOD m
|
|
END Align;
|
|
|
|
PROCEDURE PutReloc(R: RELOC);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
sys.PUT(sys.ADR(Reloc[rcount]), R.Page);
|
|
INC(rcount, 4);
|
|
sys.PUT(sys.ADR(Reloc[rcount]), R.Size);
|
|
INC(rcount, 4);
|
|
FOR i := 0 TO ASR(R.Size - 8, 1) - 1 DO
|
|
sys.PUT(sys.ADR(Reloc[rcount]), R.reloc[i]);
|
|
INC(rcount, 2)
|
|
END
|
|
END PutReloc;
|
|
|
|
PROCEDURE InitArray(VAR adr: INTEGER; chars: UTILS.STRING);
|
|
VAR i, x, n: INTEGER;
|
|
BEGIN
|
|
n := LEN(chars) - 1;
|
|
i := 0;
|
|
WHILE (i < n) & (chars[i] # 0X) DO
|
|
x := SCAN.hex(chars[i]) * 16 + SCAN.hex(chars[i + 1]);
|
|
sys.PUT(adr, CHR(x));
|
|
INC(adr);
|
|
INC(i, 2)
|
|
END
|
|
END InitArray;
|
|
|
|
PROCEDURE WriteF(F, A, N: INTEGER);
|
|
BEGIN
|
|
IF UTILS.Write(F, A, N) # N THEN
|
|
Err(2)
|
|
END
|
|
END WriteF;
|
|
|
|
PROCEDURE Write(A, N: INTEGER);
|
|
BEGIN
|
|
sys.MOVE(A, OutFilePos, N);
|
|
OutFilePos := OutFilePos + N
|
|
END Write;
|
|
|
|
PROCEDURE Fill(n: INTEGER; c: CHAR);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
FOR i := 1 TO n DO
|
|
Write(sys.ADR(c), 1)
|
|
END
|
|
END Fill;
|
|
|
|
PROCEDURE SetSection(VAR Section: SECTION; name: SECTIONNAME; size, adr, sizealign, OAPfile, attrflags: INTEGER);
|
|
BEGIN
|
|
Section.name := name;
|
|
Section.size := size;
|
|
Section.adr := adr;
|
|
Section.sizealign := sizealign;
|
|
Section.OAPfile := OAPfile;
|
|
Section.attrflags := attrflags;
|
|
END SetSection;
|
|
|
|
PROCEDURE WritePE(FName: ARRAY OF CHAR; stksize, codesize, datasize, rdatasize, gsize: INTEGER);
|
|
CONST textattr = 60000020H; rdataattr = 40000040H; dataattr = 0C0000040H; relocattr = 42000040H;
|
|
VAR i, F, adr, acodesize, compver, version, stkalloc, heapsize, heapalloc, filesize, filebuf: INTEGER;
|
|
cur: ASMLINE;
|
|
BEGIN
|
|
|
|
compver := 0;
|
|
version := 0;
|
|
stkalloc := stksize;
|
|
heapsize := 100000H;
|
|
heapalloc := 100000H;
|
|
acodesize := Align(codesize, 1000H) + 1000H;
|
|
adr := sys.ADR(rdata);
|
|
filesize := acodesize + Align(rdatasize, 1000H) + Align(datasize, 1000H) + Align(rcount, 1000H);
|
|
|
|
InitArray(adr, "5000000040000000000000003400000000000000000000006200000000000000");
|
|
InitArray(adr, "0000000000000000000000000000000000000000500000004000000000000000");
|
|
InitArray(adr, "A4014C6F61644C6962726172794100001F0147657450726F6341646472657373");
|
|
InitArray(adr, "00006B65726E656C33322E646C6C0000");
|
|
|
|
rdata[ 0] := acodesize + 50H;
|
|
rdata[ 1] := acodesize + 40H;
|
|
rdata[ 3] := acodesize + 34H;
|
|
rdata[ 6] := acodesize + 62H;
|
|
rdata[ 7] := acodesize;
|
|
rdata[13] := acodesize + 50H;
|
|
rdata[14] := acodesize + 40H;
|
|
|
|
adr := sys.ADR(Header.msdos);
|
|
InitArray(adr, "4D5A90000300000004000000FFFF0000B8000000000000004000000000000000");
|
|
InitArray(adr, "00000000000000000000000000000000000000000000000000000000B0000000");
|
|
InitArray(adr, "0E1FBA0E00B409CD21B8014CCD21546869732070726F6772616D2063616E6E6F");
|
|
InitArray(adr, "742062652072756E20696E20444F53206D6F64652E0D0D0A2400000000000000");
|
|
InitArray(adr, "5DCF9F8719AEF1D419AEF1D419AEF1D497B1E2D413AEF1D4E58EE3D418AEF1D4");
|
|
InitArray(adr, "5269636819AEF1D4000000000000000050450000");
|
|
Header.typecomp := IntToCard16(014CH);
|
|
IF dll THEN
|
|
Header.seccount := IntToCard16(0004H);
|
|
Header.infflags := IntToCard16(210EH)
|
|
ELSE
|
|
Header.seccount := IntToCard16(0003H);
|
|
Header.infflags := IntToCard16(010FH)
|
|
END;
|
|
Header.time := UTILS.Date;
|
|
Header.PEoptsize := IntToCard16(00E0H);
|
|
Header.PEfile := IntToCard16(010BH);
|
|
Header.compver := IntToCard16(compver);
|
|
Header.codesize := Align(codesize, 200H);
|
|
Header.datasize := Align(datasize + gsize, 200H) + Align(rdatasize, 200H) + Align(rcount, 200H);
|
|
Header.startadr := 1000H;
|
|
Header.codeadr := 1000H;
|
|
Header.rdataadr := Header.codeadr + Align(codesize, 1000H);
|
|
Header.loadadr := LoadAdr;
|
|
Header.secalign := 1000H;
|
|
Header.filealign := 0200H;
|
|
Header.oldestver := 0004H;
|
|
Header.version := version;
|
|
Header.oldestverNT := 0004H;
|
|
Header.filesize := Align(codesize, 1000H) + Align(datasize + gsize, 1000H) + Align(rdatasize, 1000H) + Align(rcount, 1000H) + 1000H;
|
|
Header.headersize := 0400H;
|
|
Header.UI := IntToCard16(ORD(con) + 2);
|
|
Header.stksize := stksize;
|
|
Header.stkalloc := stkalloc;
|
|
Header.heapsize := heapsize;
|
|
Header.heapalloc := heapalloc;
|
|
Header.structcount := 10H;
|
|
IF dll THEN
|
|
Header.structs[0].adr := Header.rdataadr + 0DAH;
|
|
Header.structs[0].size := etable.size
|
|
END;
|
|
|
|
Header.structs[1].adr := Header.rdataadr + 0CH;
|
|
Header.structs[1].size := 28H;
|
|
Header.structs[12].adr := Header.rdataadr;
|
|
Header.structs[12].size := 0CH;
|
|
|
|
SetSection(Header.sections[0], ".text", codesize, 1000H, Align(codesize, 200H), 400H, textattr);
|
|
SetSection(Header.sections[1], ".rdata", rdatasize, Align(codesize, 1000H) + 1000H, Align(rdatasize, 200H),
|
|
Align(codesize, 200H) + 400H, rdataattr);
|
|
SetSection(Header.sections[2], ".data", datasize + gsize, Align(codesize, 1000H) + Align(rdatasize, 1000H) + 1000H,
|
|
Align(datasize, 200H), Align(codesize, 200H) + Align(rdatasize, 200H) + 400H, dataattr);
|
|
|
|
IF dll THEN
|
|
SetSection(RelocSection, ".reloc", rcount, Header.sections[2].adr + Align(datasize + gsize, 1000H), Align(rcount, 200H),
|
|
Header.sections[2].OAPfile + Align(datasize, 200H), relocattr);
|
|
Header.structs[5].adr := RelocSection.adr;
|
|
Header.structs[5].size := rcount
|
|
END;
|
|
|
|
F := UTILS.CreateF(FName);
|
|
IF F = 0 THEN
|
|
Err(1)
|
|
END;
|
|
OutFilePos := UTILS.GetMem(filesize);
|
|
filebuf := OutFilePos;
|
|
UTILS.MemErr(OutFilePos = 0);
|
|
|
|
Write(sys.ADR(Header), sys.SIZE(HEADER));
|
|
IF dll THEN
|
|
Write(sys.ADR(RelocSection), sys.SIZE(SECTION));
|
|
Fill(Align(sys.SIZE(HEADER) + sys.SIZE(SECTION), 200H) - (sys.SIZE(HEADER) + sys.SIZE(SECTION)), 0X)
|
|
ELSE
|
|
Fill(Align(sys.SIZE(HEADER), 200H) - sys.SIZE(HEADER), 0X)
|
|
END;
|
|
|
|
cur := asmlist.First(ASMLINE);
|
|
WHILE cur # NIL DO
|
|
Write(sys.ADR(Code[cur.cmd]), cur.clen);
|
|
cur := cur.Next(ASMLINE)
|
|
END;
|
|
Fill(Align(codesize, 200H) - codesize, 0X);
|
|
Write(sys.ADR(rdata), 0DAH);
|
|
IF dll THEN
|
|
etable.time := Header.time;
|
|
Write(sys.ADR(etable), 40);
|
|
Write(sys.ADR(etable.arradr), etable.adrcount * 4);
|
|
Write(sys.ADR(etable.arrnameptr), etable.namecount * 4);
|
|
Write(sys.ADR(etable.arrnum), etable.namecount * 2);
|
|
Write(sys.ADR(etable.text), etable.textlen)
|
|
END;
|
|
Fill(Align(rdatasize, 200H) - rdatasize, 0X);
|
|
Write(sys.ADR(Data), datasize);
|
|
Fill(Align(datasize, 200H) - datasize, 0X);
|
|
IF dll THEN
|
|
Write(sys.ADR(Reloc), rcount);
|
|
Fill(Align(rcount, 200H) - rcount, 0X)
|
|
END;
|
|
WriteF(F, filebuf, OutFilePos - filebuf);
|
|
UTILS.CloseF(F)
|
|
END WritePE;
|
|
|
|
PROCEDURE New;
|
|
VAR nov: ASMLINE;
|
|
BEGIN
|
|
NEW(nov);
|
|
UTILS.MemErr(nov = NIL);
|
|
nov.cmd := ccount;
|
|
UTILS.Insert(asmlist, nov, current);
|
|
current := current.Next(ASMLINE)
|
|
END New;
|
|
|
|
PROCEDURE Empty(varadr: INTEGER);
|
|
BEGIN
|
|
New;
|
|
current.clen := 0;
|
|
current.tcmd := ECMD;
|
|
current.varadr := varadr
|
|
END Empty;
|
|
|
|
PROCEDURE OutByte(byte: INTEGER);
|
|
BEGIN
|
|
New;
|
|
current.clen := 1;
|
|
Code[ccount] := CHR(byte);
|
|
INC(ccount)
|
|
END OutByte;
|
|
|
|
PROCEDURE OutInt(int: INTEGER);
|
|
BEGIN
|
|
New;
|
|
current.clen := 4;
|
|
sys.PUT(sys.ADR(Code[ccount]), int);
|
|
INC(ccount, 4)
|
|
END OutInt;
|
|
|
|
PROCEDURE PushEAX;
|
|
BEGIN
|
|
OutByte(50H);
|
|
current.tcmd := PUSHEAX
|
|
END PushEAX;
|
|
|
|
PROCEDURE PushECX;
|
|
BEGIN
|
|
OutByte(51H);
|
|
current.tcmd := PUSHECX
|
|
END PushECX;
|
|
|
|
PROCEDURE PushEDX;
|
|
BEGIN
|
|
OutByte(52H);
|
|
current.tcmd := PUSHEDX
|
|
END PushEDX;
|
|
|
|
PROCEDURE PopEAX;
|
|
BEGIN
|
|
OutByte(58H);
|
|
current.tcmd := POPEAX
|
|
END PopEAX;
|
|
|
|
PROCEDURE PopECX;
|
|
BEGIN
|
|
OutByte(59H);
|
|
current.tcmd := POPECX
|
|
END PopECX;
|
|
|
|
PROCEDURE PopEDX;
|
|
BEGIN
|
|
OutByte(5AH);
|
|
current.tcmd := POPEDX
|
|
END PopEDX;
|
|
|
|
PROCEDURE OutCode(cmd: UTILS.STRING);
|
|
VAR a, b: INTEGER;
|
|
BEGIN
|
|
New;
|
|
a := sys.ADR(Code[ccount]);
|
|
b := a;
|
|
InitArray(a, cmd);
|
|
ccount := a - b + ccount;
|
|
current.clen := a - b
|
|
END OutCode;
|
|
|
|
PROCEDURE Del*(last: ASMLINE);
|
|
BEGIN
|
|
last.Next := current.Next;
|
|
IF current = asmlist.Last THEN
|
|
asmlist.Last := last
|
|
END;
|
|
current := last
|
|
END Del;
|
|
|
|
PROCEDURE NewLabel*(): INTEGER;
|
|
BEGIN
|
|
INC(Lcount)
|
|
RETURN Lcount
|
|
END NewLabel;
|
|
|
|
PROCEDURE PushCall*(asmline: ASMLINE);
|
|
BEGIN
|
|
New;
|
|
callstk[topstk][0] := asmline;
|
|
callstk[topstk][1] := current;
|
|
INC(topstk)
|
|
END PushCall;
|
|
|
|
PROCEDURE Param*;
|
|
BEGIN
|
|
current := callstk[topstk - 1][0]
|
|
END Param;
|
|
|
|
PROCEDURE EndCall*;
|
|
BEGIN
|
|
current := callstk[topstk - 1][1];
|
|
DEC(topstk)
|
|
END EndCall;
|
|
|
|
PROCEDURE Init*(UI: INTEGER);
|
|
VAR nov: ASMLINE;
|
|
BEGIN
|
|
dcount := 4;
|
|
dll := UI = 1;
|
|
gui := UI = 2;
|
|
con := UI = 3;
|
|
kos := UI = 4;
|
|
elf := UI = 5;
|
|
obj := UI = 6;
|
|
Lcount := HALT;
|
|
asmlist := UTILS.CreateList();
|
|
NEW(nov);
|
|
UTILS.MemErr(nov = NIL);
|
|
UTILS.Push(asmlist, nov);
|
|
current := nov;
|
|
END Init;
|
|
|
|
PROCEDURE datastr(str: UTILS.STRING);
|
|
VAR i, n: INTEGER;
|
|
BEGIN
|
|
i := 0;
|
|
n := LEN(str);
|
|
WHILE (i < n) & (str[i] # 0X) DO
|
|
Data[dcount] := str[i];
|
|
INC(dcount);
|
|
INC(i)
|
|
END;
|
|
Data[dcount] := 0X;
|
|
INC(dcount)
|
|
END datastr;
|
|
|
|
PROCEDURE dataint(n: INTEGER);
|
|
BEGIN
|
|
sys.PUT(sys.ADR(Data[dcount]), n);
|
|
INC(dcount, 4)
|
|
END dataint;
|
|
|
|
PROCEDURE jmp*(jamp: CHAR; label: INTEGER);
|
|
VAR n: INTEGER;
|
|
BEGIN
|
|
New;
|
|
CASE jamp OF
|
|
|JMP, CALL:
|
|
n := 5
|
|
|JE, JLE, JGE, JG, JL, JNE:
|
|
Code[ccount] := 0FX;
|
|
INC(ccount);
|
|
n := 6
|
|
ELSE
|
|
END;
|
|
current.clen := n;
|
|
Code[ccount] := jamp;
|
|
INC(ccount);
|
|
current.codeadr := sys.ADR(Code[ccount]);
|
|
current.varadr := sys.ADR(Labels[label]);
|
|
current.tcmd := JCMD;
|
|
current.short := TRUE;
|
|
INC(ccount, 4)
|
|
END jmp;
|
|
|
|
PROCEDURE jmplong(jamp: CHAR; label: INTEGER);
|
|
BEGIN
|
|
jmp(jamp, label);
|
|
current.short := FALSE
|
|
END jmplong;
|
|
|
|
PROCEDURE Label*(label: INTEGER);
|
|
BEGIN
|
|
New;
|
|
current.varadr := sys.ADR(Labels[label]);
|
|
current.tcmd := LCMD
|
|
END Label;
|
|
|
|
PROCEDURE CmdN(Number: INTEGER);
|
|
BEGIN
|
|
New;
|
|
current.clen := 4;
|
|
current.codeadr := sys.ADR(Code[ccount]);
|
|
current.varadr := sys.ADR(Labels[Number]);
|
|
current.tcmd := OCMD;
|
|
INC(ccount, 4)
|
|
END CmdN;
|
|
|
|
PROCEDURE IntByte(bytecode, intcode: UTILS.STRING; n: INTEGER);
|
|
BEGIN
|
|
IF (n <= 127) & (n >= -128) THEN
|
|
OutCode(bytecode);
|
|
OutByte(n)
|
|
ELSE
|
|
OutCode(intcode);
|
|
OutInt(n)
|
|
END
|
|
END IntByte;
|
|
|
|
PROCEDURE DropFpu*(long: BOOLEAN);
|
|
BEGIN
|
|
IF long THEN
|
|
OutCode("83EC08DD1C24")
|
|
ELSE
|
|
OutCode("83EC04D91C24")
|
|
END;
|
|
DEC(fpu)
|
|
END DropFpu;
|
|
|
|
PROCEDURE AfterRet(func, float: BOOLEAN; callconv, parsize: INTEGER);
|
|
BEGIN
|
|
IF callconv = cdecl THEN
|
|
OutCode("81C4");
|
|
OutInt(parsize)
|
|
END;
|
|
IF func THEN
|
|
IF float THEN
|
|
OutCode("83EC08DD1C24")
|
|
ELSE
|
|
PushEAX
|
|
END
|
|
END
|
|
END AfterRet;
|
|
|
|
PROCEDURE FpuSave(local: INTEGER);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
IF fpu > maxfpu THEN
|
|
maxfpu := fpu
|
|
END;
|
|
FOR i := 1 TO fpu DO
|
|
IntByte("DD5D", "DD9D", -local - i * 8)
|
|
END
|
|
END FpuSave;
|
|
|
|
PROCEDURE Incfpu;
|
|
BEGIN
|
|
IF fpu >= FREGS THEN
|
|
UTILS.ErrMsgPos(SCAN.coord.line, SCAN.coord.col, 97);
|
|
UTILS.HALT(1)
|
|
END;
|
|
INC(fpu);
|
|
isfpu := TRUE
|
|
END Incfpu;
|
|
|
|
PROCEDURE FpuLoad(local: INTEGER; float: BOOLEAN);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
FOR i := fpu TO 1 BY -1 DO
|
|
IntByte("DD45", "DD85", -local - i * 8)
|
|
END;
|
|
IF float THEN
|
|
Incfpu;
|
|
OutCode("DD042483C408")
|
|
END
|
|
END FpuLoad;
|
|
|
|
PROCEDURE Call*(proc: INTEGER; func, float: BOOLEAN; callconv, ccall, bases, level, parsize, local: INTEGER);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
IF ccall # 0 THEN
|
|
FOR i := level TO level - bases + ORD(ccall = 1) + 1 BY -1 DO
|
|
IntByte("FF75", "FFB5", 4 * i + 4)
|
|
END;
|
|
IF ccall = 1 THEN
|
|
OutByte(55H)
|
|
END
|
|
END;
|
|
FpuSave(local);
|
|
jmplong(CALL, proc);
|
|
AfterRet(func, float, callconv, parsize);
|
|
FpuLoad(local, func & float)
|
|
END Call;
|
|
|
|
PROCEDURE CallRTL(Proc: INTEGER);
|
|
BEGIN
|
|
New;
|
|
current.clen := 5;
|
|
Code[ccount] := CALL;
|
|
INC(ccount);
|
|
current.codeadr := sys.ADR(Code[ccount]);
|
|
current.varadr := sys.ADR(RtlProc[Proc]);
|
|
current.tcmd := JCMD;
|
|
INC(ccount, 4)
|
|
END CallRTL;
|
|
|
|
PROCEDURE PushInt*(n: INTEGER);
|
|
BEGIN
|
|
OutByte(68H);
|
|
CmdN(n)
|
|
END PushInt;
|
|
|
|
PROCEDURE Prolog*(exename: UTILS.STRING);
|
|
BEGIN
|
|
ExecName := exename;
|
|
Labels[hInstance] := -dcount;
|
|
dataint(0);
|
|
Labels[SELFNAME] := -dcount;
|
|
datastr(exename);
|
|
Label(START);
|
|
IF dll THEN
|
|
OutCode("558BEC837D0C007507");
|
|
CallRTL(_close);
|
|
OutCode("EB06837D0C017409B801000000C9C20C00")
|
|
ELSIF obj THEN
|
|
OutCode("558BEC")
|
|
END;
|
|
start := asmlist.Last(ASMLINE)
|
|
END Prolog;
|
|
|
|
PROCEDURE AddRec*(base: INTEGER);
|
|
BEGIN
|
|
INC(reccount);
|
|
recarray[reccount] := base
|
|
END AddRec;
|
|
|
|
PROCEDURE CmpOpt(inv: BOOLEAN): INTEGER;
|
|
VAR cur: ASMLINE; c: INTEGER;
|
|
BEGIN
|
|
c := ORD(Code[current.Prev.Prev(ASMLINE).cmd]);
|
|
IF inv THEN
|
|
IF ODD(c) THEN
|
|
DEC(c)
|
|
ELSE
|
|
INC(c)
|
|
END
|
|
END;
|
|
cur := current;
|
|
REPEAT
|
|
cur.tcmd := 0;
|
|
cur.clen := 0;
|
|
cur := cur.Prev(ASMLINE)
|
|
UNTIL cur.tcmd = ICMP1;
|
|
cur.tcmd := 0;
|
|
cur.clen := 0
|
|
RETURN c - 16
|
|
END CmpOpt;
|
|
|
|
PROCEDURE ifwh*(L: INTEGER);
|
|
VAR c: INTEGER;
|
|
BEGIN
|
|
IF current.Prev(ASMLINE).tcmd = ICMP2 THEN
|
|
c := CmpOpt(TRUE);
|
|
OutCode("5A583BC2");
|
|
jmp(CHR(c), L)
|
|
ELSE
|
|
PopECX;
|
|
OutCode("85C9");
|
|
jmp(JE, L)
|
|
END
|
|
END ifwh;
|
|
|
|
PROCEDURE PushConst*(Number: INTEGER);
|
|
BEGIN
|
|
IntByte("6A", "68", Number);
|
|
current.Prev(ASMLINE).varadr := Number
|
|
END PushConst;
|
|
|
|
PROCEDURE IfWhile*(L: INTEGER; orop: BOOLEAN);
|
|
VAR c, L1: INTEGER;
|
|
BEGIN
|
|
L1 := NewLabel();
|
|
IF current.Prev(ASMLINE).tcmd = ICMP2 THEN
|
|
c := CmpOpt(orop);
|
|
OutCode("5A583BC2");
|
|
jmp(CHR(c), L1);
|
|
PushConst(ORD(orop))
|
|
ELSE
|
|
PopECX;
|
|
OutCode("85C9");
|
|
IF orop THEN
|
|
jmp(JE, L1)
|
|
ELSE
|
|
jmp(JNE, L1)
|
|
END;
|
|
PushECX
|
|
END;
|
|
jmp(JMP, L);
|
|
Label(L1)
|
|
END IfWhile;
|
|
|
|
PROCEDURE newrec*;
|
|
BEGIN
|
|
CallRTL(_newrec)
|
|
END newrec;
|
|
|
|
PROCEDURE disprec*;
|
|
BEGIN
|
|
CallRTL(_disprec)
|
|
END disprec;
|
|
|
|
PROCEDURE String*(Number, Len: INTEGER; str: UTILS.STRING);
|
|
BEGIN
|
|
Labels[Number] := -dcount;
|
|
IF Len > 1 THEN
|
|
datastr(str)
|
|
ELSIF Len = 1 THEN
|
|
dataint(ORD(str[0]))
|
|
ELSE
|
|
dataint(0)
|
|
END
|
|
END String;
|
|
|
|
PROCEDURE InsertFpuInit;
|
|
VAR t: ASMLINE;
|
|
BEGIN
|
|
IF isfpu THEN
|
|
t := current;
|
|
current := fpucmd;
|
|
IF maxfpu > 0 THEN
|
|
OutCode("83EC");
|
|
OutByte(maxfpu * 8)
|
|
END;
|
|
OutCode("DBE3");
|
|
current := t
|
|
END
|
|
END InsertFpuInit;
|
|
|
|
PROCEDURE ProcBeg*(Number, Local: INTEGER; Module: BOOLEAN);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
IF Module THEN
|
|
OutCode("EB0C");
|
|
Label(Number + 3);
|
|
PushInt(Number + 2);
|
|
jmplong(JMP, HALT);
|
|
Label(Number + 1)
|
|
ELSE
|
|
Label(Number)
|
|
END;
|
|
OutCode("558BEC");
|
|
IF Local > 12 THEN
|
|
IntByte("83EC", "81EC", Local);
|
|
OutCode("8BD733C08BFCB9");
|
|
OutInt(ASR(Local, 2));
|
|
OutCode("9CFCF3AB8BFA9D")
|
|
ELSE
|
|
FOR i := 4 TO Local BY 4 DO
|
|
OutCode("6A00")
|
|
END
|
|
END;
|
|
fpucmd := current;
|
|
fpu := 0;
|
|
maxfpu := 0;
|
|
isfpu := FALSE
|
|
END ProcBeg;
|
|
|
|
PROCEDURE Leave*;
|
|
BEGIN
|
|
OutByte(0C9H);
|
|
InsertFpuInit
|
|
END Leave;
|
|
|
|
PROCEDURE ProcEnd*(Number, Param: INTEGER; func, float: BOOLEAN);
|
|
BEGIN
|
|
IF func & ~float THEN
|
|
PopEAX
|
|
END;
|
|
OutByte(0C9H);
|
|
IF Param = 0 THEN
|
|
OutByte(0C3H)
|
|
ELSE
|
|
OutByte(0C2H);
|
|
OutByte(Param MOD 256);
|
|
OutByte(ASR(Param, 8))
|
|
END;
|
|
InsertFpuInit
|
|
END ProcEnd;
|
|
|
|
PROCEDURE Module*(Name: UTILS.STRING; Number: INTEGER);
|
|
BEGIN
|
|
String(Number + 2, LENGTH(Name), Name);
|
|
jmplong(JMP, Number + 1)
|
|
END Module;
|
|
|
|
PROCEDURE Asm*(s: UTILS.STRING);
|
|
BEGIN
|
|
OutCode(s)
|
|
END Asm;
|
|
|
|
PROCEDURE GlobalAdr*(offset: INTEGER);
|
|
BEGIN
|
|
OutByte(0BAH);
|
|
OutInt(offset);
|
|
current.codeadr := sys.ADR(Code[ccount - 4]);
|
|
current.tcmd := GCMD;
|
|
PushEDX
|
|
END GlobalAdr;
|
|
|
|
PROCEDURE Mono*(Number: INTEGER);
|
|
BEGIN
|
|
PopEDX;
|
|
PushInt(Number)
|
|
END Mono;
|
|
|
|
PROCEDURE StrMono*;
|
|
BEGIN
|
|
PopEDX;
|
|
OutCode("6A02");
|
|
PushEDX
|
|
END StrMono;
|
|
|
|
PROCEDURE Not*;
|
|
BEGIN
|
|
PopECX;
|
|
OutCode("85C90F94C1");
|
|
PushECX
|
|
END Not;
|
|
|
|
PROCEDURE NegSet*;
|
|
BEGIN
|
|
OutCode("F71424")
|
|
END NegSet;
|
|
|
|
PROCEDURE Int*(Op: INTEGER);
|
|
BEGIN
|
|
PopEDX;
|
|
CASE Op OF
|
|
|lxPlus: OutCode("011424")
|
|
|lxMinus: OutCode("291424")
|
|
|lxMult: OutCode("58F7EA"); PushEAX
|
|
ELSE
|
|
END
|
|
END Int;
|
|
|
|
PROCEDURE Set*(Op: INTEGER);
|
|
BEGIN
|
|
PopEDX;
|
|
OutByte(58H);
|
|
CASE Op OF
|
|
|lxPlus: OutByte(0BH)
|
|
|lxMinus: OutCode("F7D223")
|
|
|lxMult: OutByte(23H)
|
|
|lxSlash: OutByte(33H)
|
|
ELSE
|
|
END;
|
|
OutByte(0C2H);
|
|
PushEAX
|
|
END Set;
|
|
|
|
PROCEDURE Setfpu*(newfpu: INTEGER);
|
|
BEGIN
|
|
fpu := newfpu
|
|
END Setfpu;
|
|
|
|
PROCEDURE PushFlt*(x: LONGREAL);
|
|
VAR f: TFLT; L: INTEGER;
|
|
BEGIN
|
|
sys.PUT(sys.ADR(f), x);
|
|
Incfpu;
|
|
IF x = 0.0D0 THEN
|
|
OutCode("D9EE")
|
|
ELSIF x = 1.0D0 THEN
|
|
OutCode("D9E8")
|
|
ELSE
|
|
L := NewLabel();
|
|
Labels[L] := -dcount;
|
|
dataint(f[0]);
|
|
dataint(f[1]);
|
|
OutByte(0BAH);
|
|
CmdN(L);
|
|
OutCode("DD02")
|
|
END
|
|
END PushFlt;
|
|
|
|
PROCEDURE farith*(op: INTEGER);
|
|
VAR n: INTEGER;
|
|
BEGIN
|
|
OutByte(0DEH);
|
|
CASE op OF
|
|
|lxPlus: n := 0C1H
|
|
|lxMinus: n := 0E9H
|
|
|lxMult: n := 0C9H
|
|
|lxSlash: n := 0F9H
|
|
ELSE
|
|
END;
|
|
OutByte(n);
|
|
DEC(fpu)
|
|
END farith;
|
|
|
|
PROCEDURE fcmp*(Op: INTEGER);
|
|
VAR n: INTEGER;
|
|
BEGIN
|
|
OutCode("33C9DED9DFE09E0F");
|
|
CASE Op OF
|
|
|lxEQ: n := 94H
|
|
|lxNE: n := 95H
|
|
|lxLT: n := 97H
|
|
|lxGT: n := 92H
|
|
|lxLE: n := 93H
|
|
|lxGE: n := 96H
|
|
ELSE
|
|
END;
|
|
DEC(fpu, 2);
|
|
OutByte(n);
|
|
OutByte(0C1H);
|
|
PushECX
|
|
END fcmp;
|
|
|
|
PROCEDURE fneg*;
|
|
BEGIN
|
|
OutCode("D9E0")
|
|
END fneg;
|
|
|
|
PROCEDURE OnError*(n: INTEGER);
|
|
BEGIN
|
|
OutByte(68H);
|
|
OutInt(LSL(UTILS.Line, 4) + n);
|
|
jmplong(JMP, UTILS.Unit + 3)
|
|
END OnError;
|
|
|
|
PROCEDURE idivmod*(opmod: BOOLEAN);
|
|
BEGIN
|
|
PopECX;
|
|
IF opmod THEN
|
|
OutCode("58E32E538BD833D9C1FB1F8BD0C1FA1F83F9FF750C3D0000008075055B6A00EB1AF7F985DB740685D2740203D15B52EB0A")
|
|
ELSE
|
|
OutCode("58E32C538BD833D9C1FB1F8BD0C1FA1F83F9FF750B3D0000008075045B50EB19F7F985DB740585D27401485B50EB0A")
|
|
END;
|
|
OnError(8)
|
|
END idivmod;
|
|
|
|
PROCEDURE rset*;
|
|
BEGIN
|
|
CallRTL(_rset);
|
|
PushEAX
|
|
END rset;
|
|
|
|
PROCEDURE inset*;
|
|
BEGIN
|
|
CallRTL(_inset);
|
|
PushEAX
|
|
END inset;
|
|
|
|
PROCEDURE Dup*;
|
|
BEGIN
|
|
PopEDX;
|
|
PushEDX;
|
|
PushEDX
|
|
END Dup;
|
|
|
|
PROCEDURE Inclusion*(Op: INTEGER);
|
|
BEGIN
|
|
PopEDX;
|
|
PopEAX;
|
|
IF Op = lxLE THEN
|
|
PushEDX
|
|
ELSE
|
|
PushEAX
|
|
END;
|
|
OutCode("0BC25933C8E3046A00EB026A01")
|
|
END Inclusion;
|
|
|
|
PROCEDURE NegInt*;
|
|
BEGIN
|
|
OutCode("F71C24")
|
|
END NegInt;
|
|
|
|
PROCEDURE CmpInt*(Op: INTEGER);
|
|
VAR n: INTEGER;
|
|
BEGIN
|
|
OutCode("33C95A583BC20F"); current.tcmd := ICMP1;
|
|
CASE Op OF
|
|
|lxEQ: n := 94H
|
|
|lxNE: n := 95H
|
|
|lxLT: n := 9CH
|
|
|lxGT: n := 9FH
|
|
|lxLE: n := 9EH
|
|
|lxGE: n := 9DH
|
|
ELSE
|
|
END;
|
|
OutByte(n);
|
|
OutByte(0C1H); current.tcmd := ICMP2;
|
|
PushECX;
|
|
END CmpInt;
|
|
|
|
PROCEDURE CallVar*(func, float: BOOLEAN; callconv, parsize, local: INTEGER);
|
|
BEGIN
|
|
PopEDX;
|
|
OutCode("8B1285D2750A");
|
|
OnError(2);
|
|
FpuSave(local);
|
|
OutCode("FFD2");
|
|
AfterRet(func, float, callconv, parsize);
|
|
FpuLoad(local, func & float)
|
|
END CallVar;
|
|
|
|
PROCEDURE LocalAdr*(offset, bases: INTEGER);
|
|
BEGIN
|
|
IF bases = 0 THEN
|
|
Empty(offset);
|
|
OutCode("8BD5")
|
|
ELSE
|
|
IntByte("8B55", "8B95", 4 * bases + 4)
|
|
END;
|
|
IntByte("83C2", "81C2", offset);
|
|
PushEDX;
|
|
IF bases = 0 THEN
|
|
Empty(offset)
|
|
END
|
|
END LocalAdr;
|
|
|
|
PROCEDURE Field*(offset: INTEGER);
|
|
BEGIN
|
|
IF offset # 0 THEN
|
|
IntByte("830424", "810424", offset)
|
|
END
|
|
END Field;
|
|
|
|
PROCEDURE DerefType*(n: INTEGER);
|
|
BEGIN
|
|
IntByte("8B5424", "8B9424", n);
|
|
OutCode("FF72FC")
|
|
END DerefType;
|
|
|
|
PROCEDURE Guard*(T: INTEGER; Check: BOOLEAN);
|
|
BEGIN
|
|
IF Check THEN
|
|
PopEAX;
|
|
OutCode("85C074");
|
|
IF T <= 127 THEN
|
|
OutByte(9)
|
|
ELSE
|
|
OutByte(12)
|
|
END;
|
|
PushEAX
|
|
END;
|
|
PushConst(T);
|
|
PushEAX;
|
|
CallRTL(_checktype);
|
|
IF Check THEN
|
|
PushEAX
|
|
ELSE
|
|
OutCode("85C0750A");
|
|
OnError(3)
|
|
END
|
|
END Guard;
|
|
|
|
PROCEDURE StProc*(proc: INTEGER);
|
|
BEGIN
|
|
CASE proc OF
|
|
|stINC: PopEDX; OutCode("590111")
|
|
|stDEC: PopEDX; OutCode("592911")
|
|
|stINC1: PopEDX; OutCode("FF02")
|
|
|stDEC1: PopEDX; OutCode("FF0A")
|
|
|stINCL: PopEDX; OutCode("580910")
|
|
|stEXCL: PopEDX; OutCode("582110")
|
|
|stPACK: OutCode("DB04245A5ADD02D9FDDD1A"); isfpu := TRUE
|
|
|stPACK1: OutCode("DB04245A5AD902D9FDD91A"); isfpu := TRUE
|
|
|stUNPK: PopEDX; OutCode("59DD01D9F4DD19DB1A"); isfpu := TRUE
|
|
|stUNPK1: PopEDX; OutCode("59D901D9F4D919DB1A"); isfpu := TRUE
|
|
|stCOPY: CallRTL(_strcopy)
|
|
|sysMOVE: CallRTL(_savearr)
|
|
ELSE
|
|
END
|
|
END StProc;
|
|
|
|
PROCEDURE Assert*(proc, assrt: INTEGER);
|
|
BEGIN
|
|
PopEDX;
|
|
OutCode("85D2751368");
|
|
OutInt(UTILS.Line * 16 + 1);
|
|
PushInt(UTILS.Unit + 2);
|
|
IF proc = stASSERT THEN
|
|
OutCode("6A026A")
|
|
ELSE
|
|
OutCode("6A016A")
|
|
END;
|
|
OutByte(assrt);
|
|
jmplong(JMP, ASSRT)
|
|
END Assert;
|
|
|
|
PROCEDURE StFunc*(func: INTEGER);
|
|
BEGIN
|
|
CASE func OF
|
|
|stABS: PopEDX; OutCode("85D27D02F7DA"); PushEDX
|
|
|stFABS: OutCode("D9E1")
|
|
|stFLT: OutCode("DB0424"); PopEAX; Incfpu;
|
|
|stFLOOR: jmplong(CALL, _floor); PushEAX; DEC(fpu)
|
|
|stODD: OutCode("83242401")
|
|
|stROR: PopECX; OutCode("58D3C8"); PushEAX
|
|
|stASR: PopECX; OutCode("58D3F8"); PushEAX
|
|
|stLSL: PopECX; OutCode("58D3E0"); PushEAX
|
|
|stLSR: PopECX; OutCode("58D3E8"); PushEAX
|
|
|stORD: PopEDX; OutCode("85D274036A015A"); PushEDX;
|
|
|stMIN: PopEDX; OutCode("3914247E025852");
|
|
|stMAX: PopEDX; OutCode("3B14247E025852");
|
|
|stLENGTH: CallRTL(_length); PushEAX
|
|
ELSE
|
|
END
|
|
END StFunc;
|
|
|
|
PROCEDURE Load*(T: INTEGER);
|
|
VAR lastcmd: ASMLINE; offset: INTEGER;
|
|
|
|
PROCEDURE del;
|
|
BEGIN
|
|
lastcmd.tcmd := 0;
|
|
offset := lastcmd.varadr;
|
|
lastcmd := lastcmd.Prev(ASMLINE);
|
|
WHILE lastcmd.tcmd # ECMD DO
|
|
lastcmd.clen := 0;
|
|
lastcmd.tcmd := 0;
|
|
lastcmd := lastcmd.Prev(ASMLINE)
|
|
END;
|
|
lastcmd.tcmd := 0
|
|
END del;
|
|
|
|
BEGIN
|
|
lastcmd := current;
|
|
CASE T OF
|
|
|TINTEGER, TSET, TPOINTER, TPROC:
|
|
IF lastcmd.tcmd = ECMD THEN
|
|
del;
|
|
IntByte("8B55", "8B95", offset);
|
|
PushEDX
|
|
ELSE
|
|
PopEDX;
|
|
OutCode("FF32")
|
|
END
|
|
|TCHAR, TBOOLEAN:
|
|
IF lastcmd.tcmd = ECMD THEN
|
|
del;
|
|
OutCode("0FB6");
|
|
IntByte("55", "95", offset);
|
|
PushEDX
|
|
ELSE
|
|
PopEDX;
|
|
OutCode("0FB60A");
|
|
PushECX
|
|
END
|
|
|TLONGREAL:
|
|
IF lastcmd.tcmd = ECMD THEN
|
|
del;
|
|
IntByte("DD45", "DD85", offset)
|
|
ELSE
|
|
PopEDX;
|
|
OutCode("DD02")
|
|
END;
|
|
Incfpu
|
|
|TREAL:
|
|
IF lastcmd.tcmd = ECMD THEN
|
|
del;
|
|
IntByte("D945", "D985", offset)
|
|
ELSE
|
|
PopEDX;
|
|
OutCode("D902")
|
|
END;
|
|
Incfpu
|
|
|TCARD16:
|
|
IF lastcmd.tcmd = ECMD THEN
|
|
del;
|
|
OutCode("33D2668B");
|
|
IntByte("55", "95", offset);
|
|
PushEDX
|
|
ELSE
|
|
PopEDX;
|
|
OutCode("33C9668B0A");
|
|
PushECX
|
|
END
|
|
ELSE
|
|
END
|
|
END Load;
|
|
|
|
PROCEDURE Save*(T: INTEGER);
|
|
BEGIN
|
|
CASE T OF
|
|
|TINTEGER, TSET, TPOINTER, TPROC:
|
|
PopEDX;
|
|
OutCode("588910")
|
|
|TCHAR, TSTRING, TBOOLEAN:
|
|
PopEDX;
|
|
OutCode("588810")
|
|
|TCARD16:
|
|
PopEDX;
|
|
OutCode("58668910")
|
|
|TLONGREAL:
|
|
PopEDX;
|
|
OutCode("DD1A");
|
|
DEC(fpu)
|
|
|TREAL:
|
|
PopEDX;
|
|
OutCode("D91A");
|
|
DEC(fpu)
|
|
|TRECORD:
|
|
CallRTL(_saverec);
|
|
OutCode("85C0750A");
|
|
OnError(4)
|
|
|TARRAY:
|
|
CallRTL(_savearr)
|
|
ELSE
|
|
END
|
|
END Save;
|
|
|
|
PROCEDURE OpenArray*(A: TIDX; n: INTEGER);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
PopEDX;
|
|
FOR i := n - 1 TO 0 BY -1 DO
|
|
PushConst(A[i])
|
|
END;
|
|
PushEDX
|
|
END OpenArray;
|
|
|
|
PROCEDURE OpenIdx*(n: INTEGER);
|
|
BEGIN
|
|
OutByte(54H);
|
|
IF n > 1 THEN
|
|
PushConst(n);
|
|
CallRTL(_arrayidx)
|
|
ELSE
|
|
CallRTL(_arrayidx1)
|
|
END;
|
|
PopEDX;
|
|
OutCode("85D2750A");
|
|
OnError(5);
|
|
PushEDX;
|
|
END OpenIdx;
|
|
|
|
PROCEDURE FixIdx*(len, size: INTEGER);
|
|
BEGIN
|
|
PopEDX;
|
|
IntByte("5983FA", "5981FA", len);
|
|
OutCode("720A");
|
|
OnError(5);
|
|
IF size > 1 THEN
|
|
IntByte("6BD2", "69D2", size)
|
|
END;
|
|
OutCode("03D1");
|
|
PushEDX
|
|
END FixIdx;
|
|
|
|
PROCEDURE Idx*;
|
|
BEGIN
|
|
PopEDX;
|
|
PopECX;
|
|
OutCode("03D1");
|
|
PushEDX
|
|
END Idx;
|
|
|
|
PROCEDURE DupLoadCheck*;
|
|
BEGIN
|
|
PopEDX;
|
|
OutCode("528B125285D2750A");
|
|
OnError(6)
|
|
END DupLoadCheck;
|
|
|
|
PROCEDURE DupLoad*;
|
|
BEGIN
|
|
PopEDX;
|
|
OutCode("528B12");
|
|
PushEDX;
|
|
END DupLoad;
|
|
|
|
PROCEDURE CheckNIL*;
|
|
BEGIN
|
|
PopEDX;
|
|
OutCode("85D2750A");
|
|
OnError(6);
|
|
PushEDX;
|
|
END CheckNIL;
|
|
|
|
PROCEDURE ExtArray*(A: TIDX; n, m: INTEGER);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
FOR i := n - 1 TO 0 BY -1 DO
|
|
PushConst(A[i])
|
|
END;
|
|
OutByte(54H);
|
|
PushConst(n);
|
|
PushConst(m);
|
|
CallRTL(_arrayrot)
|
|
END ExtArray;
|
|
|
|
PROCEDURE ADR*(dim: INTEGER);
|
|
BEGIN
|
|
IF dim > 0 THEN
|
|
PopEDX;
|
|
OutCode("83C4");
|
|
OutByte(dim * 4);
|
|
PushEDX
|
|
END
|
|
END ADR;
|
|
|
|
PROCEDURE Len*(dim: INTEGER);
|
|
BEGIN
|
|
PopEDX;
|
|
IF dim < 0 THEN
|
|
PushConst(-dim)
|
|
ELSIF dim > 1 THEN
|
|
PopEDX;
|
|
OutCode("83C4");
|
|
OutByte((dim - 1) * 4);
|
|
PushEDX
|
|
END
|
|
END Len;
|
|
|
|
PROCEDURE For*(inc: BOOLEAN; VAR LBeg, LEnd: INTEGER);
|
|
BEGIN
|
|
LEnd := NewLabel();
|
|
LBeg := NewLabel();
|
|
Label(LBeg);
|
|
OutCode("8B14248B4424043910");
|
|
IF inc THEN
|
|
jmp(JG, LEnd)
|
|
ELSE
|
|
jmp(JL, LEnd)
|
|
END
|
|
END For;
|
|
|
|
PROCEDURE NextFor*(step, LBeg, LEnd: INTEGER);
|
|
BEGIN
|
|
OutCode("8B542404");
|
|
IF step = 1 THEN
|
|
OutCode("FF02")
|
|
ELSIF step = -1 THEN
|
|
OutCode("FF0A")
|
|
ELSE
|
|
IntByte("8302", "8102", step)
|
|
END;
|
|
jmp(JMP, LBeg);
|
|
Label(LEnd);
|
|
OutCode("83C408")
|
|
END NextFor;
|
|
|
|
PROCEDURE CaseLabel*(a, b, LBeg: INTEGER);
|
|
VAR L: INTEGER;
|
|
BEGIN
|
|
L := NewLabel();
|
|
IntByte("83FA", "81FA", a);
|
|
IF a = b THEN
|
|
jmp(JNE, L)
|
|
ELSE
|
|
jmp(JL, L);
|
|
IntByte("83FA", "81FA", b);
|
|
jmp(JG, L)
|
|
END;
|
|
jmp(JMP, LBeg);
|
|
Label(L)
|
|
END CaseLabel;
|
|
|
|
PROCEDURE Drop*;
|
|
BEGIN
|
|
PopEDX
|
|
END Drop;
|
|
|
|
PROCEDURE strcmp*(Op, LR: INTEGER);
|
|
BEGIN
|
|
CASE Op OF
|
|
|lxEQ: PushConst(0)
|
|
|lxNE: PushConst(1)
|
|
|lxLT: PushConst(2)
|
|
|lxGT: PushConst(3)
|
|
|lxLE: PushConst(4)
|
|
|lxGE: PushConst(5)
|
|
ELSE
|
|
END;
|
|
CASE LR OF
|
|
|-1: CallRTL(_lstrcmp)
|
|
| 0: CallRTL(_strcmp)
|
|
| 1: CallRTL(_rstrcmp)
|
|
ELSE
|
|
END;
|
|
PushEAX
|
|
END strcmp;
|
|
|
|
PROCEDURE Optimization;
|
|
VAR cur: ASMLINE; flag: BOOLEAN;
|
|
BEGIN
|
|
cur := asmlist.First(ASMLINE);
|
|
WHILE cur # NIL DO
|
|
flag := FALSE;
|
|
CASE cur.tcmd OF
|
|
|PUSHEAX:
|
|
flag := cur.Next(ASMLINE).tcmd = POPEAX
|
|
|PUSHECX:
|
|
flag := cur.Next(ASMLINE).tcmd = POPECX
|
|
|PUSHEDX:
|
|
flag := cur.Next(ASMLINE).tcmd = POPEDX
|
|
ELSE
|
|
END;
|
|
IF flag THEN
|
|
cur.clen := 0;
|
|
cur.tcmd := 0;
|
|
cur := cur.Next(ASMLINE);
|
|
cur.clen := 0;
|
|
cur.tcmd := 0
|
|
END;
|
|
cur := cur.Next(ASMLINE)
|
|
END
|
|
END Optimization;
|
|
|
|
PROCEDURE WriteKOS(FName: ARRAY OF CHAR; stk, size, datasize, gsize: INTEGER; obj: BOOLEAN);
|
|
CONST strsize = 2048;
|
|
VAR Header: KOSHEADER; F, i, filesize, filebuf, a, sec, adr, size2: INTEGER; cur: ASMLINE;
|
|
Coff: COFFHEADER; sym: ARRAY 18 * 4 OF CHAR; FileName: UTILS.STRING;
|
|
BEGIN
|
|
F := UTILS.CreateF(FName);
|
|
IF F <= 0 THEN
|
|
Err(1)
|
|
END;
|
|
OutFilePos := UTILS.GetMem(Align(size, 4) + datasize + 1000H);
|
|
filebuf := OutFilePos;
|
|
UTILS.MemErr(OutFilePos = 0);
|
|
|
|
IF ~obj THEN
|
|
Header.menuet01 := "MENUET01";
|
|
Header.ver := 1;
|
|
Header.start := sys.SIZE(KOSHEADER);
|
|
Header.size := Align(size, 4) + datasize;
|
|
Header.mem := Header.size + stk + gsize + strsize * 2 + 1000H;
|
|
Header.sp := Header.size + gsize + stk;
|
|
Header.param := Header.sp;
|
|
Header.path := Header.param + strsize;
|
|
|
|
Write(sys.ADR(Header), sys.SIZE(KOSHEADER));
|
|
|
|
cur := asmlist.First(ASMLINE);
|
|
WHILE cur # NIL DO
|
|
Write(sys.ADR(Code[cur.cmd]), cur.clen);
|
|
cur := cur.Next(ASMLINE)
|
|
END;
|
|
Fill(Align(size, 4) - size, 0X);
|
|
Write(sys.ADR(Data), datasize);
|
|
WriteF(F, filebuf, OutFilePos - filebuf)
|
|
|
|
ELSE
|
|
|
|
size2 := size;
|
|
size := Align(size, 4) - sys.SIZE(KOSHEADER);
|
|
Coff.Machine := IntToCard16(014CH);
|
|
Coff.NumberOfSections := IntToCard16(3);
|
|
Coff.TimeDateStamp := UTILS.Date;
|
|
Coff.SizeOfOptionalHeader := IntToCard16(0);
|
|
Coff.Characteristics := IntToCard16(0184H);
|
|
|
|
Coff.text.name := ".flat";
|
|
Coff.text.size := 0;
|
|
Coff.text.adr := 0;
|
|
Coff.text.sizealign := size;
|
|
Coff.text.OAPfile := 8CH;
|
|
Coff.text.reserved6 := size + datasize + 8CH;
|
|
Coff.text.reserved7 := 0;
|
|
Coff.text.attrflags := 40300020H;
|
|
|
|
Coff.data.name := ".data";
|
|
Coff.data.size := 0;
|
|
Coff.data.adr := 0;
|
|
Coff.data.sizealign := datasize;
|
|
Coff.data.OAPfile := size + 8CH;
|
|
Coff.data.reserved6 := 0;
|
|
Coff.data.reserved7 := 0;
|
|
Coff.data.reserved8 := 0;
|
|
Coff.data.attrflags := 0C0300040H;
|
|
|
|
Coff.bss.name := ".bss";
|
|
Coff.bss.size := 0;
|
|
Coff.bss.adr := 0;
|
|
Coff.bss.sizealign := gsize;
|
|
Coff.bss.OAPfile := 0;
|
|
Coff.bss.reserved6 := 0;
|
|
Coff.bss.reserved7 := 0;
|
|
Coff.bss.reserved8 := 0;
|
|
Coff.bss.attrflags := 0C03000C0H;
|
|
|
|
size := Align(size2, 4);
|
|
rcount := 0;
|
|
cur := asmlist.First(ASMLINE);
|
|
WHILE cur # NIL DO
|
|
IF cur.tcmd IN {OCMD, GCMD} THEN
|
|
sys.GET(sys.ADR(Code[cur.cmd]), a);
|
|
IF a < size THEN
|
|
a := a - sys.SIZE(KOSHEADER);
|
|
sec := 1
|
|
ELSIF a < size + datasize THEN
|
|
a := a - size;
|
|
sec := 2
|
|
ELSE
|
|
a := a - size - datasize;
|
|
sec := 3
|
|
END;
|
|
sys.PUT(sys.ADR(Code[cur.cmd]), a);
|
|
sys.PUT(sys.ADR(Reloc[rcount]), cur.adr - sys.SIZE(KOSHEADER));
|
|
INC(rcount, 4);
|
|
sys.PUT(sys.ADR(Reloc[rcount]), sec);
|
|
INC(rcount, 4);
|
|
sys.PUT(sys.ADR(Reloc[rcount]), 06X); INC(rcount);
|
|
sys.PUT(sys.ADR(Reloc[rcount]), 00X); INC(rcount);
|
|
END;
|
|
Write(sys.ADR(Code[cur.cmd]), cur.clen);
|
|
cur := cur.Next(ASMLINE)
|
|
END;
|
|
size := size2;
|
|
Fill(Align(size, 4) - size2, 0X);
|
|
Write(sys.ADR(Data), datasize);
|
|
Coff.text.reserved8 := rcount DIV 10;
|
|
Coff.PointerToSymbolTable := Coff.text.reserved6 + rcount;
|
|
Coff.NumberOfSymbols := 4;
|
|
|
|
WriteF(F, sys.ADR(Coff), sys.SIZE(COFFHEADER));
|
|
WriteF(F, filebuf, OutFilePos - filebuf);
|
|
WriteF(F, sys.ADR(Reloc), rcount);
|
|
|
|
adr := sys.ADR(sym);
|
|
InitArray(adr, "4558504F52545300000000000100000002002E666C617400000000000000010000000300");
|
|
InitArray(adr, "2E64617461000000000000000200000003002E6273730000000000000000030000000300");
|
|
sys.PUT(sys.ADR(sym) + 8, Labels[Exports] - sys.SIZE(KOSHEADER));
|
|
|
|
WriteF(F, sys.ADR(sym), LEN(sym));
|
|
i := 4;
|
|
WriteF(F, sys.ADR(i), 4)
|
|
END;
|
|
UTILS.CloseF(F)
|
|
END WriteKOS;
|
|
|
|
PROCEDURE WriteELF(FName: ARRAY OF CHAR; code, data, glob: INTEGER);
|
|
VAR F, delta, filebuf: INTEGER; cur: ASMLINE; bytes: ARRAY 817H + 55FH + 4900 OF CHAR;
|
|
|
|
PROCEDURE Add(offset: INTEGER);
|
|
VAR m: INTEGER;
|
|
BEGIN
|
|
sys.GET(sys.ADR(bytes[offset]), m);
|
|
sys.PUT(sys.ADR(bytes[offset]), m + delta)
|
|
END Add;
|
|
|
|
PROCEDURE Sub(offset: INTEGER);
|
|
VAR m: INTEGER;
|
|
BEGIN
|
|
sys.GET(sys.ADR(bytes[offset]), m);
|
|
sys.PUT(sys.ADR(bytes[offset]), m - delta)
|
|
END Sub;
|
|
|
|
PROCEDURE Add8(a1, a2, a3, a4, a5, a6, a7, a8: INTEGER);
|
|
BEGIN
|
|
Add(a1); Add(a2); Add(a3); Add(a4);
|
|
Add(a5); Add(a6); Add(a7); Add(a8)
|
|
END Add8;
|
|
|
|
BEGIN
|
|
sys.MOVE(ELF.get(), sys.ADR(bytes[0]), ELF.size);
|
|
|
|
DEC(code, 13);
|
|
|
|
delta := Align(data, 1000H) - 100000H;
|
|
Add8(0020H, 00A4H, 00A8H, 0258H, 02B8H, 0308H, 0494H, 049CH);
|
|
Add8(04A4H, 0679H, 0681H, 06A4H, 06B0H, 06BAH, 0703H, 0762H);
|
|
Add8(0774H, 0786H, 0819H, 0823H, 17C5H, 17E5H, 17E9H, 1811H);
|
|
Add8(1839H, 1861H, 1889H, 1A25H, 1A95H, 1AA5H, 1C05H, 1C55H);
|
|
Add(1CE5H); Add(1D09H); Add(1D15H); Add(1D25H); Add(1D35H); Add(1D55H);
|
|
|
|
delta := Align(glob, 1000H) - 3200000H;
|
|
Add(00A8H); Add(17EDH); Add(1C09H); Add(1D25H);
|
|
|
|
delta := Align(code, 1000H) - 100000H;
|
|
Add8(0020H, 0084H, 0088H, 0098H, 009CH, 00A0H, 00B8H, 00BCH);
|
|
Add8(00C0H, 0118H, 011CH, 0120H, 0258H, 0278H, 02B8H, 0308H);
|
|
Add8(048CH, 0494H, 049CH, 04A4H, 04ACH, 04B4H, 04BCH, 04C4H);
|
|
Add8(04CCH, 04D4H, 04DCH, 04E4H, 04ECH, 04F4H, 04FCH, 0504H);
|
|
Add8(050CH, 0514H, 052BH, 0544H, 054EH, 0554H, 055EH, 056EH);
|
|
Add8(057EH, 058EH, 059EH, 05AEH, 05BEH, 05CEH, 05DEH, 05EEH);
|
|
Add8(05FEH, 060EH, 061EH, 062EH, 064CH, 0651H, 0679H, 0681H);
|
|
Add8(0686H, 068CH, 06A4H, 06ABH, 06B0H, 06BAH, 06D7H, 06EBH);
|
|
Add8(0703H, 0762H, 0774H, 0786H, 0819H, 0823H, 0828H, 082DH);
|
|
Add8(1635H, 1655H, 1659H, 167DH, 1681H, 16A5H, 16A9H, 16CDH);
|
|
Add8(16D1H, 16F5H, 16F9H, 171DH, 1721H, 1745H, 1749H, 176DH);
|
|
Add8(1771H, 1795H, 1799H, 17BDH, 17C1H, 17E5H, 17E9H, 1811H);
|
|
Add8(1839H, 1861H, 1889H, 1985H, 1995H, 19A5H, 19B5H, 19C5H);
|
|
Add8(19D5H, 19E5H, 19F5H, 1A05H, 1A15H, 1A25H, 1A55H, 1A65H);
|
|
Add8(1A75H, 1A95H, 1AA5H, 1AD5H, 1AE5H, 1AF5H, 1B05H, 1B25H);
|
|
Add8(1B35H, 1B45H, 1B55H, 1B65H, 1B75H, 1BB5H, 1BC5H, 1BE5H);
|
|
Add8(1C05H, 1C15H, 1C55H, 1C75H, 1CA5H, 1CB5H, 1CE5H, 1D05H);
|
|
Add8(1D15H, 1D25H, 1D35H, 1D55H, 1D75H, 1D89H, 08DEH, 08E8H);
|
|
Sub(0845H); Sub(087BH); Sub(0916H); Add(0C52H); Add(0C8AH); Add(0D0AH);
|
|
|
|
OutFilePos := UTILS.GetMem(code + data + 8000H);
|
|
filebuf := OutFilePos;
|
|
UTILS.MemErr(OutFilePos = 0);
|
|
|
|
Write(sys.ADR(bytes), 817H);
|
|
Fill(2DDH, 90X);
|
|
cur := asmlist.First(ASMLINE);
|
|
WHILE cur # NIL DO
|
|
Write(sys.ADR(Code[cur.cmd]), cur.clen);
|
|
cur := cur.Next(ASMLINE)
|
|
END;
|
|
Fill(Align(code, 1000H) - code, 90X);
|
|
Write(sys.ADR(bytes[817H]), 55FH);
|
|
Write(sys.ADR(Data), data);
|
|
Fill(Align(data, 1000H) - data, 0X);
|
|
Write(sys.ADR(bytes[817H + 55FH + 55FH]), 0DC5H);
|
|
|
|
F := UTILS.CreateF(FName);
|
|
IF F <= 0 THEN
|
|
Err(1)
|
|
END;
|
|
WriteF(F, filebuf, OutFilePos - filebuf);
|
|
UTILS.CloseF(F)
|
|
END WriteELF;
|
|
|
|
PROCEDURE DelProc*(beg, end: ASMLINE);
|
|
BEGIN
|
|
WHILE beg # end DO
|
|
beg.clen := 0;
|
|
beg.tcmd := 0;
|
|
beg := beg.Next(ASMLINE)
|
|
END;
|
|
beg.clen := 0;
|
|
beg.tcmd := 0
|
|
END DelProc;
|
|
|
|
PROCEDURE FixLabels*(FName: ARRAY OF CHAR; stk, gsize, glob: INTEGER);
|
|
VAR size, asize, i, rdatasize, RCount, n, temp, temp2, temp3: INTEGER; cur: ASMLINE; R: RELOC; c: CHAR;
|
|
BEGIN
|
|
dcount := Align(dcount, 4);
|
|
IF dll THEN
|
|
LoadAdr := 10000000H;
|
|
PackExport(ExecName)
|
|
ELSIF con OR gui THEN
|
|
LoadAdr := 400000H
|
|
ELSIF kos OR obj THEN
|
|
LoadAdr := sys.SIZE(KOSHEADER)
|
|
ELSIF elf THEN
|
|
LoadAdr := 134514420 + 1024;
|
|
INC(gsize, 1024)
|
|
END;
|
|
|
|
IF dll OR con OR gui THEN
|
|
rdatasize := 0DAH + etable.size;
|
|
size := 1000H + LoadAdr;
|
|
ELSIF kos OR elf OR obj THEN
|
|
rdatasize := 0;
|
|
size := LoadAdr
|
|
END;
|
|
|
|
Optimization;
|
|
temp2 := size;
|
|
cur := asmlist.First(ASMLINE);
|
|
WHILE cur # NIL DO
|
|
cur.adr := size;
|
|
IF cur.tcmd = LCMD THEN
|
|
sys.PUT(cur.varadr, size)
|
|
END;
|
|
size := size + cur.clen;
|
|
cur := cur.Next(ASMLINE)
|
|
END;
|
|
|
|
size := temp2;
|
|
cur := asmlist.First(ASMLINE);
|
|
WHILE cur # NIL DO
|
|
cur.adr := size;
|
|
IF cur.tcmd = LCMD THEN
|
|
sys.PUT(cur.varadr, size)
|
|
ELSIF (cur.tcmd = JCMD) & cur.short THEN
|
|
sys.GET(cur.varadr, i);
|
|
temp3 := i - cur.Next(ASMLINE).adr;
|
|
IF (-131 <= temp3) & (temp3 <= 123) THEN
|
|
sys.GET(cur(ASMLINE).codeadr - 1, c);
|
|
IF c = JMP THEN
|
|
sys.PUT(cur(ASMLINE).codeadr - 1, 0EBX)
|
|
ELSE (*JE, JNE, JLE, JGE, JG, JL*)
|
|
sys.PUT(cur(ASMLINE).codeadr - 2, ORD(c) - 16);
|
|
sys.PUT(cur(ASMLINE).codeadr - 1, temp3);
|
|
DEC(cur(ASMLINE).codeadr)
|
|
END;
|
|
cur.clen := 2
|
|
END
|
|
END;
|
|
size := size + cur.clen;
|
|
cur := cur.Next(ASMLINE)
|
|
END;
|
|
|
|
IF dll OR con OR gui THEN
|
|
asize := Align(size, 1000H)
|
|
ELSIF kos OR obj THEN
|
|
asize := Align(size, 4)
|
|
ELSIF elf THEN
|
|
asize := 134514420 + 6508 + Align(size - 13 - LoadAdr, 1000H)
|
|
END;
|
|
|
|
FOR i := 0 TO Lcount DO
|
|
IF Labels[i] < 0 THEN
|
|
Labels[i] := -Labels[i] + asize + Align(rdatasize, 1000H)
|
|
END
|
|
END;
|
|
|
|
temp := dcount;
|
|
IF elf THEN
|
|
asize := asize + Align(dcount, 1000H) + 64 + 1024;
|
|
sys.PUT(sys.ADR(Code[glob + 1]), asize - 1024);
|
|
dcount := 0
|
|
END;
|
|
|
|
IF dll THEN
|
|
asize := asize - LoadAdr + 0DAH;
|
|
FOR i := 0 TO etable.namecount - 1 DO
|
|
etable.arradr[i] := Labels[etable.arradr[i]] - LoadAdr;
|
|
etable.arrnameptr[i] := etable.arrnameptr[i] + asize
|
|
END;
|
|
etable.arradroffset := etable.arradroffset + asize;
|
|
etable.arrnameptroffset := etable.arrnameptroffset + asize;
|
|
etable.arrnumoffset := etable.arrnumoffset + asize;
|
|
etable.dllnameoffset := etable.dllnameoffset + asize;
|
|
asize := asize + LoadAdr - 0DAH
|
|
END;
|
|
IF dll OR con OR gui THEN
|
|
Labels[LoadLibrary] := asize + 4;
|
|
Labels[GetProcAddress] := asize;
|
|
R.Page := 0;
|
|
R.Size := 0;
|
|
RCount := 0;
|
|
END;
|
|
cur := asmlist.First(ASMLINE);
|
|
|
|
FOR i := 0 TO LEN(RtlProc) - 1 DO
|
|
RtlProc[i] := Labels[RtlProc[i]]
|
|
END;
|
|
|
|
temp3 := asize + Align(rdatasize, 1000H) + dcount;
|
|
WHILE cur # NIL DO
|
|
CASE cur.tcmd OF
|
|
|JCMD:
|
|
sys.GET(cur.varadr, i);
|
|
sys.PUT(cur.codeadr, i - cur.Next(ASMLINE).adr)
|
|
|GCMD:
|
|
sys.GET(cur.codeadr, i);
|
|
sys.PUT(cur.codeadr, i + temp3)
|
|
|OCMD:
|
|
sys.MOVE(cur.varadr, cur.codeadr, 4)
|
|
ELSE
|
|
END;
|
|
IF dll & (cur.tcmd IN {GCMD, OCMD}) THEN
|
|
n := cur.adr - LoadAdr;
|
|
IF ASR(n, 12) = ASR(R.Page, 12) THEN
|
|
R.reloc[RCount] := IntToCard16(n MOD 1000H + 3000H);
|
|
INC(RCount);
|
|
INC(R.Size, 2)
|
|
ELSE
|
|
IF R.Size # 0 THEN
|
|
PutReloc(R)
|
|
END;
|
|
R.Page := ASR(n, 12) * 1000H;
|
|
R.Size := 10;
|
|
R.reloc[0] := IntToCard16(n MOD 1000H + 3000H);
|
|
RCount := 1
|
|
END
|
|
END;
|
|
cur := cur.Next(ASMLINE)
|
|
END;
|
|
IF R.Size # 0 THEN
|
|
PutReloc(R)
|
|
END;
|
|
IF dll OR con OR gui THEN
|
|
WritePE(FName, stk, size - 1000H - LoadAdr, dcount, rdatasize, gsize)
|
|
ELSIF kos OR obj THEN
|
|
WriteKOS(FName, Align(stk, 4), size, dcount, gsize, obj)
|
|
ELSIF elf THEN
|
|
WriteELF(FName, size - LoadAdr, temp, gsize)
|
|
END
|
|
END FixLabels;
|
|
|
|
PROCEDURE OutStringZ(str: ARRAY OF CHAR);
|
|
VAR i: INTEGER;
|
|
BEGIN
|
|
New;
|
|
current.clen := LENGTH(str);
|
|
FOR i := 0 TO current.clen - 1 DO
|
|
Code[ccount] := str[i];
|
|
INC(ccount)
|
|
END;
|
|
Code[ccount] := 0X;
|
|
INC(ccount);
|
|
INC(current.clen)
|
|
END OutStringZ;
|
|
|
|
PROCEDURE Epilog*(gsize: INTEGER; FName: ARRAY OF CHAR; stk: INTEGER);
|
|
VAR i, glob: INTEGER;
|
|
BEGIN
|
|
glob := 0;
|
|
gsize := Align(gsize, 4) + 4;
|
|
COPY(FName, OutFile);
|
|
Labels[RTABLE] := -dcount;
|
|
dataint(recarray[0]);
|
|
FOR i := 1 TO reccount DO
|
|
dataint(recarray[i])
|
|
END;
|
|
current := start;
|
|
IF con OR gui OR dll THEN
|
|
PushInt(LoadLibrary);
|
|
PushInt(GetProcAddress);
|
|
OutCode("5859FF31FF3054")
|
|
ELSIF elf THEN
|
|
OutCode("6800000000");
|
|
glob := current.cmd;
|
|
ELSIF kos OR obj THEN
|
|
OutByte(54H)
|
|
END;
|
|
GlobalAdr(0);
|
|
PushConst(ASR(gsize, 2));
|
|
PushInt(RTABLE);
|
|
PushInt(SELFNAME);
|
|
CallRTL(_init);
|
|
current := asmlist.Last(ASMLINE);
|
|
IF dll THEN
|
|
OutCode("B801000000C9C20C00")
|
|
END;
|
|
IF obj THEN
|
|
OutCode("B801000000C9C20000")
|
|
END;
|
|
OutCode("EB05");
|
|
Label(ASSRT);
|
|
CallRTL(_assrt);
|
|
OutCode("EB09");
|
|
Label(HALT);
|
|
OutCode("6A006A00");
|
|
CallRTL(_assrt);
|
|
OutCode("6A00");
|
|
CallRTL(_halt);
|
|
Label(_floor);
|
|
OutCode("83EC06D93C2466812424FFF366810C24FFF7D92C2483C402D9FCDB1C2458C3");
|
|
IF obj THEN
|
|
Label(Exports);
|
|
CmdN(szSTART); CmdN(START);
|
|
CmdN(szversion); OutInt(stk);
|
|
FOR i := 0 TO kosexpcount - 1 DO
|
|
CmdN(kosexp[i].NameLabel); CmdN(kosexp[i].Adr)
|
|
END;
|
|
OutInt(0);
|
|
Label(szSTART); OutStringZ("lib_init");
|
|
Label(szversion); OutStringZ("version");
|
|
FOR i := 0 TO kosexpcount - 1 DO
|
|
Label(kosexp[i].NameLabel);
|
|
OutStringZ(kosexp[i].Name.Name)
|
|
END
|
|
END;
|
|
FixLabels(FName, stk, gsize, glob)
|
|
END Epilog;
|
|
|
|
END X86. |