From ecd0c53ca19a0c7cc9c7220a61c7e24560c78e61 Mon Sep 17 00:00:00 2001 From: Pavol Marko Date: Wed, 5 May 2004 18:45:35 +0000 Subject: [PATCH] added files for JIT on windows --- amxmodx/JIT/jits.asm | 2281 ++++++++++++++++++++++++++++++++++++++++++ amxmodx/JIT/jits.def | 10 + amxmodx/JIT/jits.exp | Bin 0 -> 942 bytes amxmodx/JIT/jits.lib | Bin 0 -> 14032 bytes 4 files changed, 2291 insertions(+) create mode 100755 amxmodx/JIT/jits.asm create mode 100755 amxmodx/JIT/jits.def create mode 100755 amxmodx/JIT/jits.exp create mode 100755 amxmodx/JIT/jits.lib diff --git a/amxmodx/JIT/jits.asm b/amxmodx/JIT/jits.asm new file mode 100755 index 00000000..d5229284 --- /dev/null +++ b/amxmodx/JIT/jits.asm @@ -0,0 +1,2281 @@ +; JIT.ASM: Just-In-Time compiler for the Abstract Machine of the "Small" language +; (C) 1999-2000, Marc Peter; beta version; provided AS IS WITHOUT ANY WARRANTIES + +; I reached >155 million instr./sec on my AMD K6-2/366 with the Hanoi "bench" +; (27 disks, no output, DOS4/GW under Win95) with this implementation of the +; JIT compiler. + +; NOTE 1: +; There is only one pass implemented in this version. This means there is no +; way of knowing the size of the compiled code before it has actually been com- +; piled. So the only chance the caller has, is to count the number of opcodes +; (in amx_BrowseRelocate()) and multiply this count with a "safe" factor to +; obtain a size value big enough to hold the entire code (and data, including +; the stack and heap, after adding their sizes). Afterwards it can realloc() +; this memory block to the actually needed smaller size. + +; NOTE 2: +; The compiler destroys the opcode addresses of the given source by storing the +; respective compiled code's addresses there for the final address relocation +; step. + +; NOTE 3: +; During execution of the compiled code with amx_exec_asm() the x86 processor's +; stack is switched into the data section of the abstract machine. This means +; that there should always be enough memory left between HEA and STK to provide +; stack space for occurring interrupts! (see the STACKRESERVE variable) + +; NOTE 4: +; Although the Small compiler doesn't generate the LCTRL, SCTRL and CALL.I +; instructions, I have to tell that they don't work as expected in a JIT +; compiled program, because there is no easy way of transforming AMX code +; addresses and JIT translated ones. This might be fixed in a future version. +; +; CALLING CONVENTIONS (by Thiadmer Riemersma) +; This version is the JIT that uses the "stack calling convention". In the +; original implementation, this meant __cdecl; both for the calling convention +; for the _asm_runJIT routine itself as for the callback functions. +; The current release supports __stdcall for the callback functions; to +; use it, you need to assemble the file with STDECL defined (Since STDCALL is +; a reserved word on the assembler, I had to choose a different name for the +; macro, hence STDECL.) + +; Revision History +;------------------ +; 6 march 2004 by Thiadmer Riemersma +; Corrected a bug in OP_FILL, where a cell preceding the array would +; be overwritten (zero'ed out). This bug was brought to my attention +; by Robert Daniels. +; 22 december 2003 by Thiadmer Riemersma (TR) +; Added the SYMTAG and SYSCALL.D opcodes (these are not really supported; +; SYMTAG is a no-op). +; Support __stdcall calling convention for for the native function "hook" +; function (the __cdecl calling convention is also still supported). +; 14 October 2002 by Thiadmer Riemersma (TR) +; Corrected the amx_s structure. The _hlw field was missing, which caused +; errors for arguments to native functions that were passed by reference. +; 2002/08/05 TR +; * store the status of the abstract machine in the AMX structure upon +; return, so that the machine can be restarted (OP_SLEEP) +; * added OP_NOP (for alignment, it is ignored by the JIT) +; * make sure the JIT does not crash when we NULL is passed for the +; return value +; 2000/03/03 MP +; * _amx_opcodelist is equipped with an underscore, again 8-P +; * added SRANGE as a no-op, so debugging info doesn't upset the JIT +; compiler anymore +; * added note about LCTRL, SCTRL and CALL.I +; 2000/03/02 MP +; * made JIT support __cdecl calling conventions +; * removed some rather unnecessary pops in the epilog of amx_exec_asm +; * changed the template for CALL into a DB byte sequence (tasm 4.1 +; didn't like the immediate value) +; 1999/12/07 MP +; * fixed crash caused by JIT compiler not saving registers +; 1999/08/06 MP - design change: closer to the "iron" with native stack +; * The JIT compiler now generates relocatable code for case tables by +; setting FORCERELOCATABLE = 1. +; * removed all debug hook code +; * exchanged meaning of ESP and ESI in asm_exec(): now low-level calls/ +; pushes/pops are possible +; * removed the run-time functions for the CALL, CALL_I and RET op-codes, +; they are now inline +; * All these changes gained around 80% performance increase for the +; hanoi bench. +; 1999/08/05 MP +; * fixed OP_LINE in the case of NODBGCALLS==1, where no compiled address +; was stored for the LINE byte code (I.e. SWITCH would jump to totally +; wrong addresses.) The same fix was applied to OP_FILL, OP_FILE and +; OP_SCTRL (for the no-op case). +; 1999/08/04 MP +; * updated with 4 new opcodes (SRANGE does nothing at the moment; 2dim. +; arrays have not been tested.) +; * hacked relocation code to support absoulute addresses for CASETBL +; (This assumes that no generated address will be greater than +; 0x7fffffff. Bit no. 31 is used as flag for absolute addresses.) +; * The run-time function for SWITCH uses a (hopefully) faster algorithm +; to compute the destination address: It searches backwards now. +; 1999/07/08 MP - initial revision + +.386 +.MODEL FLAT + +IFDEF @Version ; for Microsoft MASM 6.x + OPTION OLDSTRUCTS + OPTION M510 +ENDIF + +; +; Controls generation of LINE op-codes: 0 - no, all other values - yes +; Beware that, if set to 1, this makes the compiled code slower by a factor of +; up to 3! +; +ALLOWOPLINE = 0 + +; +; If this is set to 1 the JIT generates relocatable code for case tables, too. +; If set to 0, a faster variant for switch (using absolute addresses) is +; generated. I consider setting it to 0 a bad idea. +; +FORCERELOCATABLE = 1 + +; +; Determines how much memory should be reserved for occurring interrupts. +; (If my memory serves me right, DOS4/G(W) provides a stack of 512 bytes +; for interrupts that occur in real mode and are promoted to protected mode.) +; This value _MUST_ be greater than 64 (for AMX needs) and should be at least +; 128 (to serve interrupts). +; +STACKRESERVE = 512 + +; +; This variable controls the generation of memory range checks at run-time. +; You should set this to 0, only when you are sure that there are no range +; violations in your Small programs and you really need those 5% speed gain. +; +DORUNTIMECHECKS = 1 + + +amx_s STRUC + _base DD ? + _dataseg DD ? + _callback DD ? + _debug DD ? + _cip DD ? + _frm DD ? + _hea DD ? + _hlw DD ? + _stk DD ? + _stp DD ? + _flags DD ? + _curline DD ? + _curfile DD ? + _dbgcode DD ? + _dbgaddr DD ? + _dbgparam DD ? + _dbgname DD ? + _usertags DD 4 DUP (?) ; 4 = AMX_USERNUM (#define'd in amx.h) + _userdata DD 4 DUP (?) ; 4 = AMX_USERNUM (#define'd in amx.h) + _error DD ? + _pri DD ? + _alt DD ? + _reset_stk DD ? + _reset_hea DD ? + _syscall_d DD ? + ; the two fields below are for the JIT; they do not exist in + ; the non-JIT version of the abstract machine + _reloc_size DD ? ; memory block for relocations + _code_size DD ? ; memory size of the native code +amx_s ENDS + + +amxhead_s STRUC + _size DD ? ; size of the "file" + _magic DW ? ; signature + _file_version DB ? ;file format version + _amx_version DB ? ; required version of the AMX + _h_flags DW ? + _defsize DW ? ; size of one public/native function entry + _cod DD ? ; initial value of COD - code block + _dat DD ? ; initial value of DAT - data block + _h_hea DD ? ; initial value of HEA - start of the heap + _h_stp DD ? ; initial value of STP - stack top + _h_cip DD ? ; initial value of CIP - the instruction pointer + _publics DD ? ; offset to the "public functions" table + _natives DD ? ; offset to the "native functions" table + _libraries DD ? ; offset to the "library" table + _pubvars DD ? ; offset to the "public variables" table + _tags DD ? ; offset to the "public tagnames" table +amxhead_s ENDS + + + AMX_ERR_NONE = 0 + AMX_ERR_EXIT = 1 + AMX_ERR_ASSERT = 2 + AMX_ERR_STACKERR = 3 + AMX_ERR_BOUNDS = 4 + AMX_ERR_MEMACCESS = 5 + AMX_ERR_INVINSTR = 6 + AMX_ERR_STACKLOW = 7 + AMX_ERR_HEAPLOW = 8 + AMX_ERR_CALLBACK = 9 + AMX_ERR_NATIVE = 10 + AMX_ERR_DIVIDE = 11 ; MP: added for catching divide errors + AMX_ERR_SLEEP = 12 ; (TR) go into sleep mode + + AMX_ERR_MEMORY = 16 + AMX_ERR_FORMAT = 17 + AMX_ERR_VERSION = 18 + AMX_ERR_NOTFOUND = 19 + AMX_ERR_INDEX = 20 + AMX_ERR_DEBUG = 21 + AMX_ERR_INIT = 22 + AMX_ERR_USERDATA = 23 + AMX_ERR_INIT_JIT = 24 + AMX_ERR_PARAMS = 25 + AMX_ERR_DOMAIN = 26 + + DBG_INIT = 0 + DBG_FILE = 1 + DBG_LINE = 2 + DBG_SYMBOL = 3 + DBG_CLRSYM = 4 + DBG_CALL = 5 + DBG_RETURN = 6 + DBG_TERMINATE = 7 + DBG_SRANGE = 8 + DBG_SYMTAG = 9 + + AMX_FLAG_CHAR16 = 0001h ; characters are 16-bit + AMX_FLAG_DEBUG = 0002h ; symbolic info. available + AMX_FLAG_BROWSE = 4000h + AMX_FLAG_RELOC = 8000h ; jump/call addresses relocated + +; +; #define PUSH(v) ( stk-=sizeof(cell), *(cell *)(data+(int)stk)=v ) +; +_PUSH MACRO v + push DWORD ptr v + ENDM + +; +; #define POP(v) ( v=*(cell *)(data+(int)stk), stk+=sizeof(cell) ) +; +_POP MACRO v + pop DWORD ptr v + ENDM + +; +; For determining the biggest native code section generated for ONE Small +; opcode. (See the following macro and the PUBLIC function getMaxCodeSize().) +; +MAXCODESIZE = 0 + +; This is the work horse of the whole JIT: It actually copies the code. +; +GO_ON MACRO from, to, opsize ;opsize may be 4, 8 or 12 (default=4) + mov esi,OFFSET from ;get source address of JIT code + CODESIZE = (to-from) + mov ecx,CODESIZE ;get number of bytes to copy + ;all the following could (should?) be in a sub-routine... + ;determining the maximum size of a single code snippet + IF MAXCODESIZE LT CODESIZE + MAXCODESIZE = CODESIZE + ENDIF + mov [ebx],edi ;store address for jump-correction + IFB ;add operand size to the source ptr + add ebx,4 + ELSE + add ebx,opsize + ENDIF + rep movsb + cmp ebx,[end_code] + jae code_gen_done + jmp DWORD ptr [ebx] ;go on with the next opcode + ENDM + +; +; Modify the argument of an x86 instruction with the Small opcode's parameter +; before copying the code. +; +putval MACRO where + mov eax,[ebx+4] + mov DWORD ptr [where],eax + ENDM + +; +; Add an entry to the table of addresses which have to be relocated after the +; code compilation is done. +; +RELOC MACRO adr, dest + mov ebp,[reloc_num] + IFB + mov eax,[ebx+4] + ELSE + lea eax,[dest] + ENDIF + mov [edx+ebp],eax ; write absolute destination + lea eax,[edi+adr] + mov [edx+ebp+4],eax ; write address of jump operand + add DWORD ptr [reloc_num],8 + ENDM + +_DROPARGS MACRO n ; (TR) remove function arguments from the stack + IFNDEF STDECL ; (for __cdecl calling convention only) + add esp,n + ENDIF + ENDM + + +.CODE + + + PUBLIC _asm_runJIT + PUBLIC _amx_exec_asm + PUBLIC _getMaxCodeSize + + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +; ; +; void asm_runJIT( AMX_HEADER *amxh, JumpAddressArray *jumps, void *dest ) ; +; eax edx ebx ; +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +; asm_runJIT() assumes that the code of this module is allready browsed and +; relocated for the JIT compiler. It also assumes that both the jumps array and +; the dest memory block are large enough to hold all the data it has to write +; to them, as well as that the prefix (header) has already been copied to dest. + +_asm_runJIT PROC + push ebp + push ebx + push edi + push esi + + mov eax,[esp+20] ; get amxh + mov edx,[esp+24] ; get jumps array + mov ebx,[esp+28] ; get destination + + mov [amxhead],eax ; save pointer to AMX_HEADER struct + mov ecx,[eax+_cod] ; get offset of start of code + mov eax,[eax+_dat] ; offset of start of data = end of code + mov edi,ecx + add ecx,[amxhead] ; compute the real pointer + add eax,[amxhead] ; dito + add edi,ebx ; get write pointer into EDI + mov [compiled_code],ebx + mov [end_code],eax ; Store end-of-code address, so JIT + ; compiler knows when to stop. + mov DWORD ptr [reloc_num],0 ; init the index into the jumps array + + mov ebx,ecx + jmp DWORD ptr [ecx] ; start compiling + + ; The compiler will jump back here when code generation is complete. + +code_gen_done: ; Now copy the data section. + mov ebp,[amxhead] ; get source AMX_HEADER start address + add edi,3 ; DAT follows directly after COD + and edi,0fffffffch ; align it on a DWORD boundary + push edi ; save data start pointer + mov esi,[end_code] ; get start of data segment + mov ecx,[ebp]._h_hea + sub ecx,[ebp]._dat ; compute length of array to copy + rep movsb ; copy the data + + ; Now adjust the register values in the compiled AMX_HEADER. + ; COD stays the same since the size of AMX_HEADER doesn't change in + ; compiled mode. + mov ebx,[compiled_code] ; get compiled AMX's header address + pop esi ; recall data start pointer + sub esi,ebx ; DAT = size of code + size of prefix + mov [ebx]._dat,esi ; write corrected DAT register + + ;HEA and STP are already relative to DAT, so we don't need to fix them. + + ; Now the calls/jumps in the compiled code have to be relocated. + sub ecx,ecx ; reset offset into relocation table + cmp ecx,[reloc_num] + jae reloc_code_done ; if there's nothing to fix, skip this part + reloc_code_loop: + mov eax,[edx+ecx] ; get destination address + mov edi,[edx+ecx+4] ; determine where to write the relocated value + add ecx,8 ; set pointer to next entry in relocation table + add edi,4 ; base address from where the offset is taken +IF FORCERELOCATABLE EQ 0 + ;MP: hack to suport absolute addresses for the CASETBL instruction + test eax,80000000h ; check whether it is an absolute address + pushf + and eax,7fffffffh ; clear the flag bit for absolute addresses + popf + mov eax,[eax] ; translate into compiled absolute address + jne write_reloc ; leave out the subtraction if absolute +ELSE + mov eax,[eax] ; translate into compiled absolute address +ENDIF + sub eax,edi ; make a relative offset + write_reloc: + mov [edi-4],eax ; write the relocated address + cmp ecx,[reloc_num] + jb reloc_code_loop + +reloc_code_done: + ; Relocate the addresses in the AMX_HEADER structure. (CIP and publics) + add ebp,[ebp]._cod ; make all addresses relative to COD, not base + mov eax,[ebx]._h_cip + add eax,ebp ; get absolute source CIP + mov eax,[eax] ; translate CIP to compiled address + sub eax,ebx ; make it relative to base + sub eax,[ebx]._cod ; and now relative to COD + mov [ebx]._h_cip,eax; store relocated CIP + mov edi,[ebx]._publics + sub esi,esi + mov ecx,[ebx]._natives + sub ecx,edi ; ECX = _natives - _publics = public table size + mov si,[ebx]._defsize + or ecx,ecx + jz reloc_done ; If there are no publics, we are done. + reloc_publics_loop: + mov eax,[ebx+edi] ; get public function offset + add eax,ebp ; make it a source address + mov eax,[eax] ; translate to compiled address + sub eax,ebx ; make it an offset relative to base + sub eax,[ebx]._cod ; and now relative to COD + mov [ebx+edi],eax ; write corrected address back + add edi,esi ; step to next public function entry + sub ecx,esi + ja reloc_publics_loop + +reloc_done: + mov eax,0 + pop esi + pop edi + pop ebx + pop ebp + ret + +OP_LOAD_PRI: +;nop; + putval j_load_pri+2 + GO_ON j_load_pri, OP_LOAD_ALT, 8 + + j_load_pri: + mov eax,[edi+12345678h] + +OP_LOAD_ALT: +;nop; + putval j_load_alt+2 + GO_ON j_load_alt, OP_LOAD_S_PRI, 8 + + j_load_alt: + mov edx,[edi+12345678h] + +;good +OP_LOAD_S_PRI: +;nop; + putval j_load_s_pri+2 + GO_ON j_load_s_pri, OP_LOAD_S_ALT, 8 + + j_load_s_pri: + mov eax,[ebx+12345678h] + +;good +OP_LOAD_S_ALT: +;nop; + putval j_load_s_alt+2 + GO_ON j_load_s_alt, OP_LOAD_I, 8 + + j_load_s_alt: + mov edx,[ebx+12345678h] + +OP_LOAD_I: +;nop; + GO_ON j_load_i, OP_LODB_I + + j_load_i: +IF DORUNTIMECHECKS NE 0 + CALL [verify_adr_eax] +ENDIF + mov eax,[edi+eax] + +OP_LODB_I: +;nop; + mov eax,[ebx+4] + mov eax,DWORD ptr [(lodb_and-4)+eax*4] + mov DWORD ptr [j_lodb_i_sm+1],eax ;modify AND instruction + GO_ON j_lodb_i, OP_LREF_PRI, 8 + + j_lodb_i: +IF DORUNTIMECHECKS NE 0 + call [verify_adr_eax] +ENDIF + mov eax,[edi+eax] ;subject to misalignment stalls + j_lodb_i_sm: + and eax,12345678h + +OP_LREF_PRI: +;nop; + putval j_lref_pri+2 + GO_ON j_lref_pri, OP_LREF_ALT, 8 + + j_lref_pri: + mov eax,[edi+12345678h] + mov eax,[edi+eax] + +OP_LREF_ALT: +;nop; + putval j_lref_alt+2 + GO_ON j_lref_alt, OP_LREF_S_PRI, 8 + + j_lref_alt: + mov edx,[edi+12345678h] + mov edx,[edi+edx] + +OP_LREF_S_PRI: +;nop; + putval j_lref_s_pri+2 + GO_ON j_lref_s_pri, OP_LREF_S_ALT, 8 + + j_lref_s_pri: + mov eax,[ebx+12345678h] + mov eax,[edi+eax] + +OP_LREF_S_ALT: +;nop; + putval j_lref_s_alt+2 + GO_ON j_lref_s_alt, OP_CONST_PRI, 8 + + j_lref_s_alt: + mov edx,[ebx+12345678h] + mov edx,[edi+edx] + +;good +OP_CONST_PRI: +;nop; + putval j_const_pri+1 + GO_ON j_const_pri, OP_CONST_ALT, 8 + + j_const_pri: + mov eax,12345678h + +;good +OP_CONST_ALT: +;nop; + putval j_const_alt+1 + GO_ON j_const_alt, OP_ADDR_PRI, 8 + + j_const_alt: + mov edx,12345678h + +;good +OP_ADDR_PRI: +;nop; + putval j_addr_pri+1 + GO_ON j_addr_pri, OP_ADDR_ALT, 8 + + j_addr_pri: + mov eax,12345678h + add eax,frm + +;good +OP_ADDR_ALT: +;nop; + putval j_addr_alt+1 + GO_ON j_addr_alt, OP_STOR_PRI, 8 + + j_addr_alt: + mov edx,12345678h + add edx,frm + +OP_STOR_PRI: +;nop; + putval j_stor_pri+2 + GO_ON j_stor_pri, OP_STOR_ALT, 8 + + j_stor_pri: + mov [edi+12345678h],eax + +OP_STOR_ALT: +;nop; + putval j_stor_alt+2 + GO_ON j_stor_alt, OP_STOR_S_PRI, 8 + + j_stor_alt: + mov [edi+12345678h],edx + +;good +OP_STOR_S_PRI: +;nop; + putval j_stor_s_pri+2 + GO_ON j_stor_s_pri, OP_STOR_S_ALT, 8 + + j_stor_s_pri: + mov [ebx+12345678h],eax + +;good +OP_STOR_S_ALT: +;nop; + putval j_stor_s_alt+2 + GO_ON j_stor_s_alt, OP_STOR_I, 8 + + j_stor_s_alt: + mov [ebx+12345678h],edx + +;good +OP_STOR_I: +;nop; + GO_ON j_stor_i, OP_STRB_I + + j_stor_i: +IF DORUNTIMECHECKS NE 0 + call [verify_adr_edx] +ENDIF + mov [edi+edx],eax + +OP_STRB_I: +;nop; + mov eax,[ebx+4] + cmp eax,1 + jne strb_not1byte + GO_ON j_strb_i_1b, strb_not1byte, 8 + j_strb_i_1b: +IF DORUNTIMECHECKS NE 0 + call [verify_adr_edx] +ENDIF + mov [edi+edx],al + + strb_not1byte: + cmp eax,4 + je strb_4byte + GO_ON j_strb_i_2b, strb_4byte, 8 + j_strb_i_2b: +IF DORUNTIMECHECKS NE 0 + call [verify_adr_edx] +ENDIF + mov [edi+edx],ax + + strb_4byte: + GO_ON j_strb_i_4b, OP_SREF_PRI, 8 + j_strb_i_4b: +IF DORUNTIMECHECKS NE 0 + call [verify_adr_edx] +ENDIF + mov [edi+edx],eax + +OP_SREF_PRI: +;nop; + putval j_sref_pri+2 + GO_ON j_sref_pri, OP_SREF_ALT, 8 + + j_sref_pri: + mov ebp,[edi+12345678h] + mov [edi+ebp],eax + +OP_SREF_ALT: +;nop; + putval j_sref_alt+2 + GO_ON j_sref_alt, OP_SREF_S_PRI, 8 + + j_sref_alt: + mov ebp,[edi+12345678h] + mov [edi+ebp],edx + +OP_SREF_S_PRI: +;nop; + putval j_sref_s_pri+2 + GO_ON j_sref_s_pri, OP_SREF_S_ALT, 8 + + j_sref_s_pri: + mov ebp,[ebx+12345678h] + mov [edi+ebp],eax + +OP_SREF_S_ALT: +;nop; + putval j_sref_s_alt+2 + GO_ON j_sref_s_alt, OP_LIDX, 8 + + j_sref_s_alt: + mov ebp,[ebx+12345678h] + mov [edi+ebp],edx + +;good +OP_LIDX: +;nop; + GO_ON j_lidx, OP_LIDX_B + + j_lidx: + lea eax,[edx+4*eax] +IF DORUNTIMECHECKS NE 0 + call [verify_adr_eax] +ENDIF + mov eax,[edi+eax] + +OP_LIDX_B: +;nop; + mov al,[ebx+4] + mov BYTE ptr [j_lidx_b+2],al + GO_ON j_lidx_b, OP_IDXADDR, 8 + + j_lidx_b: + shl eax,12h + add eax,edx +IF DORUNTIMECHECKS NE 0 + call [verify_adr_eax] +ENDIF + mov eax,[edi+eax] + +;good +OP_IDXADDR: +;nop; + GO_ON j_idxaddr, OP_IDXADDR_B + + j_idxaddr: + lea eax,[edx+4*eax] + +OP_IDXADDR_B: +;nop; + mov al,[ebx+4] + mov BYTE ptr [j_idxaddr_b+2],al + GO_ON j_idxaddr_b, OP_ALIGN_PRI, 8 + + j_idxaddr_b: + shl eax,12h + add eax,edx + +OP_ALIGN_PRI: +;nop; + mov eax,4 + sub eax,[ebx+4] + mov DWORD ptr [j_align_pri+1],eax + GO_ON j_align_pri, OP_ALIGN_ALT, 8 + + j_align_pri: + xor eax,12345678h + +OP_ALIGN_ALT: +;nop; + mov eax,4 + sub eax,[ebx+4] + mov DWORD ptr [j_align_alt+1],eax + GO_ON j_align_alt, OP_LCTRL, 8 + + j_align_alt: + xor edx,12345678h + +OP_LCTRL: +;nop; + mov eax,[ebx+4] + cmp eax,0 + jne lctrl_1 + GO_ON j_lctrl_0, lctrl_1, 8 + j_lctrl_0: + mov eax,code ; 1=COD + lctrl_1: + cmp eax,1 + jne lctrl_2 + GO_ON j_lctrl_1, lctrl_2, 8 + j_lctrl_1: + mov eax,edi ; 1=DAT + lctrl_2: + cmp eax,2 + jne lctrl_3 + GO_ON j_lctrl_2, lctrl_3, 8 + j_lctrl_2: + mov eax,hea ; 2=HEA + lctrl_3: + cmp eax,3 + jne lctrl_4 + GO_ON j_lctrl_3, lctrl_4, 8 + j_lctrl_3: + mov ebp,amx + mov eax,[ebp+_stp] + lctrl_4: + cmp eax,4 + jne lctrl_5 + GO_ON j_lctrl_4, lctrl_5, 8 + j_lctrl_4: + mov eax,esp ; 4=STK + sub eax,edi + lctrl_5: + cmp eax,5 + jne lctrl_6 + GO_ON j_lctrl_5, lctrl_6, 8 + j_lctrl_5: + mov eax,frm ; 5=FRM + lctrl_6: + mov DWORD ptr [j_lctrl_6+1],edi + GO_ON j_lctrl_6, OP_SCTRL, 8 + j_lctrl_6: + mov eax,12345678h ; 6=CIP + + +OP_SCTRL: +;nop; + mov eax,[ebx+4] + cmp eax,2 + jne sctrl_4 + GO_ON j_sctrl_2, sctrl_4, 8 + j_sctrl_2: + mov hea,eax ; 2=HEA + sctrl_4: + cmp eax,4 + jne sctrl_5 + GO_ON j_sctrl_4, sctrl_5, 8 + j_sctrl_4: + ;mov esp,eax ; 4=STK + ;add esp,edi ; relocate stack + lea esp,[eax+edi] + sctrl_5: + cmp eax,5 + jne sctrl_ignore + GO_ON j_sctrl_5, sctrl_ignore, 8 + j_sctrl_5: + mov ebx,eax ; 5=FRM + mov frm,eax + add ebx,edi ; relocate frame + sctrl_ignore: + mov [ebx],edi + add ebx,8 + jmp DWORD ptr [ebx] + +OP_MOVE_PRI: +;nop; + GO_ON j_move_pri, OP_MOVE_ALT + + j_move_pri: + mov eax,edx + +;good +OP_MOVE_ALT: +;nop; + GO_ON j_move_alt, OP_XCHG + + j_move_alt: + mov edx,eax + +OP_XCHG: +;nop; + GO_ON j_xchg, OP_PUSH_PRI + + j_xchg: ;one might use pushes/pops for pre-586's + xchg eax,edx + +;good +OP_PUSH_PRI: +;nop; + GO_ON j_push_pri, OP_PUSH_ALT + + j_push_pri: + _PUSH eax + +;good +OP_PUSH_ALT: +;nop; + GO_ON j_push_alt, OP_PUSH_R_PRI + + j_push_alt: + _PUSH edx + +OP_PUSH_R_PRI: +;nop; + putval j_push_r_pri+1 + GO_ON j_push_r_pri, OP_PUSH_C, 8 + + j_push_r_pri: + mov ecx,12345678h + j_push_loop: + _PUSH eax + loop j_push_loop + ;dec ecx + ;jnz j_push_loop + +;good +OP_PUSH_C: +;nop; + putval j_push_c+1 + GO_ON j_push_c, OP_PUSH, 8 + + j_push_c: + _PUSH 12345678h + +OP_PUSH: +;nop; + putval j_push+2 + GO_ON j_push, OP_PUSH_S, 8 + + j_push: + _PUSH [edi+12345678h] + +;good +OP_PUSH_S: +;nop; + putval j_push_s+2 + GO_ON j_push_s, OP_POP_PRI, 8 + + j_push_s: + _PUSH [ebx+12345678h] + +OP_POP_PRI: +;nop; + GO_ON j_pop_pri, OP_POP_ALT + + j_pop_pri: + _POP eax + +;good +OP_POP_ALT: +;nop; + GO_ON j_pop_alt, OP_STACK + + j_pop_alt: + _POP edx + +;good +OP_STACK: +;nop; + putval j_stack+4 + GO_ON j_stack, OP_HEAP, 8 + + j_stack: + mov edx,esp + add esp,12345678h + sub edx,edi +IF DORUNTIMECHECKS NE 0 + call [chk_marginstack] +ENDIF + +;good +OP_HEAP: +;nop; + putval j_heap_call-4 + GO_ON j_heap, OP_PROC, 8 + + j_heap: + mov edx,hea + add DWORD ptr hea,12345678h + j_heap_call: +IF DORUNTIMECHECKS NE 0 + call [chk_marginheap] +ENDIF +;good +OP_PROC: +;nop; + GO_ON j_proc, OP_RET + + j_proc: ;[STK] = FRM, STK = STK - cell size, FRM = STK + _PUSH frm ; push old frame (for RET/RETN) + mov frm,esp ; get new frame + mov ebx,esp ; already relocated + sub frm,edi ; relocate frame + +OP_RET: +;nop; + GO_ON j_ret, OP_RETN + + j_ret: + _POP ebx ; pop frame + mov frm,ebx + add ebx,edi + ret + ;call [jit_ret] + +;good +OP_RETN: +;nop; + GO_ON j_retn, OP_CALL + + j_retn: + jmp [jit_retn] + +;good +OP_CALL: +;nop; + RELOC 1 + GO_ON j_call, OP_CALL_I, 8 + + j_call: + ;call 12345678h ; tasm chokes on this out of a sudden + db 0e8h, 0, 0, 0, 0 + +OP_CALL_I: +;nop; + GO_ON j_call_i, OP_JUMP + + j_call_i: + call eax + +;good +OP_JUMP: +;nop; + RELOC 1 + GO_ON j_jump, OP_JREL, 8 + + j_jump: + DB 0e9h + DD 12345678h + +OP_JREL: +;nop; + mov eax,[ebx+4] + ; create an absolute address from the relative one + RELOC 1, eax+ebx+8 + GO_ON j_jump, OP_JREL, 8 + +;good +OP_JZER: +;nop; + RELOC 4 + GO_ON j_jzer, OP_JNZ, 8 + + j_jzer: + or eax,eax + DB 0fh, 84h, 0, 0, 0, 0 ;jz NEAR 0 (tasm sucks a bit) + +;good +OP_JNZ: +;nop; + RELOC 4 + GO_ON j_jnz, OP_JEQ, 8 + + j_jnz: + or eax,eax + DB 0fh, 85h, 0, 0, 0, 0 ;jnz NEAR 0 + +;good +OP_JEQ: +;nop; + RELOC 4 + GO_ON j_jeq, OP_JNEQ, 8 + + j_jeq: + cmp eax,edx + DB 0fh, 84h, 0, 0, 0, 0 ;je NEAR 0 (tasm sucks a bit) + +OP_JNEQ: +;nop; + RELOC 4 + GO_ON j_jneq, OP_JLESS, 8 + + j_jneq: + cmp eax,edx + DB 0fh, 85h, 0, 0, 0, 0 ;jne NEAR 0 (tasm sucks a bit) + +OP_JLESS: +;nop; + RELOC 4 + GO_ON j_jless, OP_JLEQ, 8 + + j_jless: + cmp eax,edx + DB 0fh, 82h, 0, 0, 0, 0 ;jb NEAR 0 (tasm sucks a bit) + +OP_JLEQ: +;nop; + RELOC 4 + GO_ON j_jleq, OP_JGRTR, 8 + + j_jleq: + cmp eax,edx + DB 0fh, 86h, 0, 0, 0, 0 ;jbe NEAR 0 (tasm sucks a bit) + +OP_JGRTR: +;nop; + RELOC 4 + GO_ON j_jgrtr, OP_JGEQ, 8 + + j_jgrtr: + cmp eax,edx + DB 0fh, 87h, 0, 0, 0, 0 ;ja NEAR 0 (tasm sucks a bit) + +OP_JGEQ: +;nop; + RELOC 4 + GO_ON j_jgeq, OP_JSLESS, 8 + + j_jgeq: + cmp eax,edx + DB 0fh, 83h, 0, 0, 0, 0 ;jae NEAR 0 (unsigned comparison) + +OP_JSLESS: +;nop; + RELOC 4 + GO_ON j_jsless, OP_JSLEQ, 8 + + j_jsless: + cmp eax,edx + DB 0fh, 8ch, 0, 0, 0, 0 ;jl NEAR 0 + +;good +OP_JSLEQ: +;nop; + RELOC 4 + GO_ON j_jsleq, OP_JSGRTR, 8 + + j_jsleq: + cmp eax,edx + DB 0fh, 8eh, 0, 0, 0, 0 ;jle NEAR 0 + +OP_JSGRTR: +;nop; + RELOC 4 + GO_ON j_jsgrtr, OP_JSGEQ, 8 + + j_jsgrtr: + cmp eax,edx + DB 0fh, 8Fh, 0, 0, 0, 0 ;jg NEAR 0 + +OP_JSGEQ: +;nop; + RELOC 4 + GO_ON j_jsgeq, OP_SHL, 8 + + j_jsgeq: + cmp eax,edx + DB 0fh, 8dh, 0, 0, 0, 0 ;jge NEAR 0 + +OP_SHL: +;nop; + GO_ON j_shl, OP_SHR + j_shl: + mov ecx,edx ; TODO: save ECX if used as special register + shl eax,cl + +OP_SHR: +;nop; + GO_ON j_shr, OP_SSHR + j_shr: + mov ecx,edx ; TODO: save ECX if used as special register + shr eax,cl + +OP_SSHR: +;nop; + GO_ON j_sshr, OP_SHL_C_PRI + j_sshr: + mov ecx,edx ; TODO: save ECX if used as special register + sar eax,cl + +OP_SHL_C_PRI: +;nop; + mov al,[ebx+4] + mov BYTE ptr [j_shl_c_pri+2],al + GO_ON j_shl_c_pri, OP_SHL_C_ALT, 8 + j_shl_c_pri: + shl eax,12h + +OP_SHL_C_ALT: +;nop; + mov al,[ebx+4] + mov BYTE ptr [j_shl_c_alt+2],al + GO_ON j_shl_c_alt, OP_SHR_C_PRI, 8 + j_shl_c_alt: + shl edx,12h + +OP_SHR_C_PRI: +;nop; + mov al,[ebx+4] + mov BYTE ptr [j_shr_c_pri+2],al + GO_ON j_shr_c_pri, OP_SHR_C_ALT, 8 + j_shr_c_pri: + shr eax,12h + +OP_SHR_C_ALT: +;nop; + mov al,[ebx+4] + mov BYTE ptr [j_shr_c_alt+2],al + GO_ON j_shr_c_alt, OP_SMUL, 8 + j_shr_c_alt: + shr edx,12h + +OP_SMUL: +;nop; + GO_ON j_smul, OP_SDIV + j_smul: + push edx + imul edx + pop edx + +;good +OP_SDIV: +;nop; + GO_ON j_sdiv, OP_SDIV_ALT + j_sdiv: + call [jit_sdiv] + +OP_SDIV_ALT: +;nop; + GO_ON j_sdiv_alt, OP_UMUL + j_sdiv_alt: + xchg eax,edx + call [jit_sdiv] + +OP_UMUL: +;nop; + GO_ON j_umul, OP_UDIV + j_umul: + push edx + mul edx + pop edx + +OP_UDIV: +;nop; + GO_ON j_udiv, OP_UDIV_ALT + j_udiv: + mov ebp,edx + sub edx,edx + call [chk_dividezero] + div ebp + +OP_UDIV_ALT: +;nop; + GO_ON j_udiv_alt, OP_ADD + j_udiv_alt: + mov ebp,eax + mov eax,edx + sub edx,edx + call [chk_dividezero] + div ebp + +;good +OP_ADD: +;nop; + GO_ON j_add, OP_SUB + j_add: + add eax,edx + +;good +OP_SUB: +;nop; + GO_ON j_sub, OP_SUB_ALT + j_sub: + sub eax,edx + +;good +OP_SUB_ALT: +;nop; + GO_ON j_sub_alt, OP_AND + j_sub_alt: + neg eax + add eax,edx + +OP_AND: +;nop; + GO_ON j_and, OP_OR + j_and: + and eax,edx + +OP_OR: +;nop; + GO_ON j_or, OP_XOR + j_or: + or eax,edx + +OP_XOR: +;nop; + GO_ON j_xor, OP_NOT + j_xor: + xor eax,edx + +OP_NOT: +;nop; + GO_ON j_not, OP_NEG + j_not: + neg eax ; sets CF iff EAX != 0 + sbb eax,eax ; EAX == -1 iff CF set (zero otherwise) + inc eax ; -1 => 0 and 0 => 1 + +OP_NEG: +;nop; + GO_ON j_neg, OP_INVERT + j_neg: + neg eax + +OP_INVERT: +;nop; + GO_ON j_invert, OP_ADD_C + j_invert: + not eax + +;good +OP_ADD_C: +;nop; + putval j_add_c+1 + GO_ON j_add_c, OP_SMUL_C, 8 + j_add_c: + add eax,12345678h + +;good +OP_SMUL_C: +;nop; + putval j_smul_c+3 + GO_ON j_smul_c, OP_ZERO_PRI, 8 + j_smul_c: + push edx + imul eax,12345678h + pop edx + +;good +OP_ZERO_PRI: +;nop; + GO_ON j_zero_pri, OP_ZERO_ALT + j_zero_pri: + sub eax,eax + +;good +OP_ZERO_ALT: +;nop; + GO_ON j_zero_alt, OP_ZERO + j_zero_alt: + sub edx,edx + +OP_ZERO: +;nop; + putval j_zero+2 + GO_ON j_zero, OP_ZERO_S, 8 + j_zero: + mov DWORD ptr [edi+12345678h],0 + +OP_ZERO_S: +;nop; + putval j_zero_s+2 + GO_ON j_zero_s, OP_SIGN_PRI, 8 + j_zero_s: + mov DWORD ptr [ebx+12345678h],0 + +OP_SIGN_PRI: +;nop; + GO_ON j_sign_pri, OP_SIGN_ALT + j_sign_pri: + shl eax,24 + sar eax,24 + +OP_SIGN_ALT: +;nop; + GO_ON j_sign_alt, OP_EQ + j_sign_alt: + shl edx,24 + sar edx,24 + +OP_EQ: +;nop; + GO_ON j_eq, OP_NEQ + j_eq: + cmp eax,edx ; PRI == ALT ? + mov eax,0 + sete al + +OP_NEQ: +;nop; + GO_ON j_neq, OP_LESS + j_neq: + cmp eax,edx ; PRI != ALT ? + mov eax,0 + setne al + +OP_LESS: +;nop; + GO_ON j_less, OP_LEQ + j_less: + cmp eax,edx ; PRI < ALT ? (unsigned) + mov eax,0 + setb al + +OP_LEQ: +;nop; + GO_ON j_leq, OP_GRTR + j_leq: + cmp eax,edx ; PRI <= ALT ? (unsigned) + mov eax,0 + setbe al + +OP_GRTR: +;nop; + GO_ON j_grtr, OP_GEQ + j_grtr: + cmp eax,edx ; PRI > ALT ? (unsigned) + mov eax,0 + seta al + +OP_GEQ: +;nop; + GO_ON j_geq, OP_SLESS + j_geq: + cmp eax,edx ; PRI >= ALT ? (unsigned) + mov eax,0 + setae al + +;good +OP_SLESS: +;nop; + GO_ON j_sless, OP_SLEQ + j_sless: + cmp eax,edx ; PRI < ALT ? (signed) + mov eax,0 + setl al + +OP_SLEQ: +;nop; + GO_ON j_sleq, OP_SGRTR + j_sleq: + cmp eax,edx ; PRI <= ALT ? (signed) + mov eax,0 + setle al + +OP_SGRTR: +;nop; + GO_ON j_sgrtr, OP_SGEQ + j_sgrtr: + cmp eax,edx ; PRI > ALT ? (signed) + mov eax,0 + setg al + +OP_SGEQ: +;nop; + GO_ON j_sgeq, OP_EQ_C_PRI + j_sgeq: + cmp eax,edx ; PRI >= ALT ? (signed) + mov eax,0 + setge al + +OP_EQ_C_PRI: +;nop; + putval j_eq_c_pri+1 + GO_ON j_eq_c_pri, OP_EQ_C_ALT, 8 + j_eq_c_pri: + cmp eax,12345678h ; PRI == value ? + mov eax,0 + sete al + +OP_EQ_C_ALT: +;nop; + putval j_eq_c_alt+4 + GO_ON j_eq_c_alt, OP_INC_PRI, 8 + j_eq_c_alt: + sub eax,eax + cmp edx,12345678h ; ALT == value ? + sete al + +OP_INC_PRI: +;nop; + GO_ON j_inc_pri, OP_INC_ALT + j_inc_pri: + inc eax + +OP_INC_ALT: +;nop; + GO_ON j_inc_alt, OP_INC + j_inc_alt: + inc edx + +OP_INC: +;nop; + putval j_inc+2 + GO_ON j_inc, OP_INC_S, 8 + j_inc: + inc DWORD ptr [edi+12345678h] + +;good +OP_INC_S: +;nop; + putval j_inc_s+2 + GO_ON j_inc_s, OP_INC_I, 8 + j_inc_s: + inc DWORD ptr [ebx+12345678h] + +OP_INC_I: +;nop; + GO_ON j_inc_i, OP_DEC_PRI + j_inc_i: + inc DWORD ptr [edi+eax] + +OP_DEC_PRI: +;nop; + GO_ON j_dec_pri, OP_DEC_ALT + j_dec_pri: + dec eax + +OP_DEC_ALT: +;nop; + GO_ON j_dec_alt, OP_DEC + j_dec_alt: + dec edx + +OP_DEC: +;nop; + putval j_dec+2 + GO_ON j_dec, OP_DEC_S, 8 + j_dec: + dec DWORD ptr [edi+12345678h] + +OP_DEC_S: +;nop; + putval j_dec_s+2 + GO_ON j_dec_s, OP_DEC_I, 8 + j_dec_s: + dec DWORD ptr [ebx+12345678h] + +OP_DEC_I: +;nop; + GO_ON j_dec_i, OP_MOVS + j_dec_i: + dec DWORD ptr [edi+eax] + +OP_MOVS: +;nop; + putval j_movs+1 + GO_ON j_movs, OP_CMPS, 8 + j_movs: + mov ecx,12345678h ;TODO: save ECX if used as special register + call [jit_movs] + +OP_CMPS: +;nop; + putval j_cmps+1 + GO_ON j_cmps, OP_FILL, 8 + j_cmps: + mov ecx,12345678h ;TODO: save ECX if used as special register + call [jit_cmps] + +OP_FILL: +;nop; + putval j_fill+1 + GO_ON j_fill, OP_HALT, 8 + j_fill: + mov ecx,12345678h ;TODO: save ECX if used as special register + call [jit_fill] + +;good +OP_HALT: +;nop; + putval j_halt_sm+1 + GO_ON j_halt, OP_BOUNDS, 8 + j_halt: + cmp DWORD PTR retval,0 + je j_halt_no_value + mov ebp,retval + mov [ebp],eax + j_halt_no_value: + j_halt_sm: + mov eax,12345678h + jmp [jit_return] + +;good +OP_BOUNDS: +;nop; + putval j_bounds+1 + GO_ON j_bounds, OP_SYSREQ_C, 8 + j_bounds: + mov ebp,12345678h + call [jit_bounds] + +;good +OP_SYSREQ_C: +;nop; + putval j_sysreq_c+1 + GO_ON j_sysreq_c, OP_SYSREQ_PRI, 8 + j_sysreq_c: + mov eax,12345678h ; get function number + j_sysreq: + call [jit_sysreq] + +OP_SYSREQ_PRI: +;nop; + GO_ON j_sysreq, OP_SYSREQ_PRI + +OP_FILE: ;opcode is simply ignored +;nop; + mov eax,[ebx+4] ;get size + mov [ebx],edi + lea ebx,[ebx+eax+8] ;move on to next opcode + cmp ebx,DWORD ptr [end_code] + jae code_gen_done + jmp DWORD ptr [ebx] ;go on with the next opcode + +OP_LINE: +;nop; +IF ALLOWOPLINE EQ 0 + mov [ebx],edi ; no line number support: ignore opcode + add ebx,12 ; move on to next opcode + cmp ebx,[end_code] + jae code_gen_done + jmp DWORD ptr [ebx] ; go on with the next opcode +ELSE + putval j_line+6 + mov eax,[ebx+8] + mov [j_line_sm],eax + GO_ON j_line, OP_SYMBOL, 12 + j_line: + call [jit_line] + DD 0 ; space for curline +j_line_sm DD 0 ; space for curfile +ENDIF + +OP_SYMBOL: ;ignored + mov [ebx],edi + mov eax,[ebx+4] ; get size + lea ebx,[ebx+eax+8] ; move on to next opcode + cmp ebx,[end_code] + jae code_gen_done + jmp DWORD ptr [ebx] ; go on with the next opcode + + +OP_SRANGE: ;ignored + mov [ebx],edi ; store relocated address + add ebx,12 ; move on to next opcode + cmp ebx,[end_code] + jae code_gen_done + jmp DWORD ptr [ebx] ; go on with the next opcode + + +;not tested +OP_JUMP_PRI: + GO_ON j_jump_pri, OP_SWITCH + + j_jump_pri: ; MP: This opcode makes sense only in con- + jmp [eax] ; junction with a possibility to get the + ; address of a code location... + + +;good +OP_SWITCH: + lea eax,[edi+6] ; The case table will be copied directly + neg eax ; after the run-time call to [jit_switch]. + and eax,3 ; We should align this table on a DWORD + mov ecx,eax ; boundary. + mov al,90h ; 90h = opcode of x86 NOP instruction + rep stosb ; Write the right number of NOPs. + mov [ebx],edi ; store address of SWITCH for relocation step + mov esi,OFFSET j_switch + mov ecx,6 + rep movsb ; copy the call instruction + mov esi,[ebx+4] ; get address of CASETBL instruction + add ebx,8 ; set instruction pointer to next opcode + add esi,4 ; point esi to first entry: (count, default adr) + mov ecx,[esi] ; get number of cases (excluding default) + inc ecx + mov ebp,[reloc_num] + j_case_loop: + mov eax,[esi] ; get case value + stosd ; write it + mov eax,[esi+4] ; get destination address +IF FORCERELOCATABLE EQ 0 + or eax,80000000h ; add flag for "absolute address" +ENDIF + mov [edx+ebp],eax ; write dest. adr. into relocation table + mov eax,[esi+4] ; get destination address (again) + add esi,8 ; set ESI to next case + mov [edx+ebp+4],edi ; write adr. to patch into relocation table + add ebp,8 ; promote relocation pointer + stosd ; write dest. adr. + dec ecx + jnz j_case_loop + mov DWORD ptr [reloc_num],ebp ; write back updated reloc_num + + jmp [ebx] ; GO_ON to next op-code + + j_switch: + call [jit_switch] + +;good +OP_CASETBL: ; compiles to nothing, SWITCH does all the work + mov eax,[ebx+4] ; get count of cases + lea ebx,[ebx+8*eax+(8+4)] ; adjust instruction pointer + jmp [ebx] ; GO_ON with next op-code + + +OP_SWAP_PRI: ; TR + GO_ON j_swap_pri, OP_SWAP_ALT + + j_swap_pri: + _POP ebp + _PUSH eax + mov eax,ebp + + +OP_SWAP_ALT: ; TR + GO_ON j_swap_alt, OP_PUSHADDR + + j_swap_alt: + _POP ebp + _PUSH edx + mov edx,ebp + + +OP_PUSHADDR: ; TR + putval j_pushaddr+1 + GO_ON j_pushaddr, OP_NOP, 8 + + j_pushaddr: + mov ebp,12345678h ;get address (offset from frame) + add ebp,frm + _PUSH ebp + + +OP_NOP: ; TR + GO_ON j_nop, OP_SYSREQ_D + j_nop: ; code alignment is ignored by the JIT + + +OP_SYSREQ_D: +;nop; + putval j_sysreq_d+1 + GO_ON j_sysreq_d, OP_SYMTAG, 8 + j_sysreq_d: + mov ebx,12345678h ; get function address + call [jit_sysreq_d] + + +OP_SYMTAG: ;ignored (TR) + mov [ebx],edi ; store relocated address + add ebx,8 ; move on to next opcode + cmp ebx,[end_code] + jae code_gen_done + jmp DWORD ptr [ebx] ; go on with the next opcode + + +OP_INVALID: ; break from the compiler with an error code + mov eax,AMX_ERR_INVINSTR + pop esi + pop edi + pop ecx + pop ebp + ret + + +_asm_runJIT ENDP + + +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; +; ; +;cell asm_exec( cell *regs, cell *retval, cell stp, cell hea ); +; eax edx ebx ecx ; +;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; + +_amx_exec_asm PROC + + push edi + push esi + push ebp + push ebx ; due to __cdecl + + ; __cdecl overhead + mov eax, [esp+20] ; get address of amx regs structure + mov edx, [esp+24] ; get address of retval + mov ebx, [esp+28] ; get stp + mov ecx, [esp+32] ; get hea + + sub esp,4*3 ; place for PRI, ALT & STK at SYSREQs + + push DWORD ptr [eax+28] ; store pointer to code segment + push DWORD ptr [eax+24] ; store pointer to AMX + push edx ; store address of retval + push ebx ; store STP + push ecx ; store HEA + push DWORD ptr[eax+20]; store FRM + + stk equ [esi+32] ; define some aliases to registers that will + alt equ [esi+28] ; be stored on the stack when the code is + pri equ [esi+24] ; actually beeing executed + code equ [esi+20] + amx equ [esi+16] + retval equ [esi+12] + stp equ [esi+8] + hea equ [esi+4] + frm equ [esi] ; FRM is NOT stored in ebp, FRM+DAT is being held + ; in ebx instead. + + mov edx,[eax+4] ; get ALT + mov ecx,[eax+8] ; get CIP + mov edi,[eax+12] ; get pointer to data segment + mov esi,[eax+16] ; get STK !!changed, now ECX free as counter!! + mov ebx,[eax+20] ; get FRM + mov eax,[eax] ; get PRI + add ebx,edi ; relocate frame + + add esi,edi ; ESP will contain DAT+STK + xchg esp,esi ; switch to AMX stack + + add stp,edi ; make STP absolute address for run-time checks + + _POP ebp ; AMX pseudo-return address, ignored + ; Call compiled code via CALL NEAR
+ call ecx + +return_to_caller: + cmp DWORD PTR retval,0 + je return_to_caller_no_value + mov ebp,retval + mov [ebp],eax ; provide return value + + return_to_caller_no_value: + mov eax,AMX_ERR_NONE + jmp _return + +_return_popstack: + add esp,4 ; Correct ESP, because we just come from a + ; runtime error checking routine. +_return: + ; store machine state + mov ecx,esp ; get STK into ECX + mov ebp,amx ; get amx into EBP + + sub ecx,edi ; correct STK + mov [ebp+_stk],ecx ; store values in AMX structure: STK, ... + mov ecx,hea ; ... HEA, ... + mov [ebp+_hea],ecx + mov ecx,ebx ; ... and FRM + sub ecx,edi ; (correct FRM) + mov [ebp+_frm],ecx + mov [ebp+_pri],eax ; also store PRI, ... + mov [ebp+_alt],edx ; ... and ALT + + ; return + sub stp,edi ; make STP relative to DAT again + xchg esp,esi ; switch back to caller's stack + + add esp,4*9 ; remove temporary data + + pop ebx ; restore registers that have to be preserved + pop ebp ; when using __cdecl convention + pop esi + pop edi + + ret + + +err_stack: + mov eax,AMX_ERR_STACKERR + jmp _return_popstack + +err_stacklow: + mov eax,AMX_ERR_STACKLOW + jmp _return_popstack + +_CHKMARGIN_STACK: ; some run-time check routines + cmp esp,stp + lea ebp,[esp-STACKRESERVE] + jg err_stacklow + sub ebp,edi + cmp hea,ebp + jg err_stack + ret + +err_heaplow: + mov eax,AMX_ERR_HEAPLOW + jmp _return_popstack + +_CHKMARGIN_HEAP: + cmp esp,stp + jg err_stacklow + cmp DWORD ptr hea,0 + jl err_heaplow + ret + +err_memaccess: + mov eax,AMX_ERR_MEMACCESS + jmp _return_popstack + +_VERIFYADDRESS_eax: ; used in load.i, store.i & lidx + cmp eax,stp + jae err_memaccess + cmp eax,hea + jb veax1 + lea ebp,[eax+edi] + cmp ebp,esp + jb err_memaccess + veax1: + ret + +_VERIFYADDRESS_edx: ; used in load.i, store.i & lidx + cmp edx,stp + jae err_memaccess + cmp edx,hea + jb vedx1 + lea ebp,[edx+edi] + cmp ebp,esp + jb err_memaccess + vedx1: + ret + +JIT_OP_SDIV: + mov ebp,edx + xor edx,eax ; Check signs of the operands. + cdq + js sdiv_fiddle ; If the signs of the operands are different + ; we'll have to fiddle around to achieve + ; proper rounding towards minus infinity. + or ebp,ebp ; check for divide by zero + jz err_divide + idiv ebp ; default behavior is right in the other cases + ret + + sdiv_fiddle: + or ebp,ebp ; check for divide by zero + jz err_divide + idiv ebp + or edx,edx + jz sdiv_goon ; If there's no remainder the result is correct + add edx,ebp ; else fix the result values. + dec eax ; Amazing, how simple this is... + sdiv_goon: + ret + + ALIGN 4 + +JIT_OP_RETN: + _POP ebx ; pop frame + _POP ecx ; get return address + + mov frm,ebx + _POP ebp + + add ebx,edi + add esp,ebp ; remove data from stack + + jmp ecx + + +JIT_OP_MOVS: ;length of block to copy is already in ECX + push edi + push esi + lea esi,[edi+eax] + lea edi,[edi+edx] + + push ecx ; I hope the blocks to copy are properly + shr ecx,2 ; aligned, so I don't do anything about that. + rep movsd + pop ecx + and ecx,3 + rep movsb + + pop esi + pop edi + ret + +JIT_OP_CMPS: ;length of block to compare is already in ECX + push edi + push esi + lea esi,[edi+edx] + lea edi,[edi+eax] + + xor eax,eax ; This is surely not the fastest way to do this + repe cmpsb ; but the most simple one. + je cmps1 + sbb eax,eax + sbb eax,0ffffffffh + cmps1: + pop esi + pop edi + ret + + +JIT_OP_FILL: ;length (in bytes) of block to fill is already in ECX + push edi + lea edi,[edi+edx] + + shr ecx,2 ;length in 32-bit cells + rep stosd ;the value to use is already in EAX + + pop edi + ret + +JIT_OP_BOUNDS: + cmp eax,0 + jl err_bounds + cmp eax,ebp + jg err_bounds + ret +err_bounds: + mov eax,AMX_ERR_BOUNDS + jmp _return_popstack + +_CHKDIVIDEZERO: + or ebp,ebp ; check for divide by zero + jz err_divide + ret +err_divide: + mov eax,AMX_ERR_DIVIDE + jmp _return_popstack + +JIT_OP_SYSREQ: + mov ecx,esp ; get STK into ECX + mov ebp,amx ; get amx into EBP + + sub ecx,edi ; correct STK + mov alt,edx ; save ALT + + mov [ebp+_stk],ecx ; store values in AMX structure: STK, + mov ecx,hea ; HEA, + mov ebx,frm ; and FRM + mov [ebp+_hea],ecx + mov [ebp+_frm],ebx + + lea ebx,pri ; 3rd param: addr. of retval + lea ecx,[esp+4] ; 4th param: parameter array + + xchg esp,esi ; switch to caller stack + + push ecx + push ebx + push eax ; 2nd param: function number + push ebp ; 1st param: amx + call [ebp+_callback] + _DROPARGS 16 ; remove args from stack + + xchg esp,esi ; switch back to AMX stack + cmp eax,AMX_ERR_NONE + jne _return_popstack; return error code, if any + + mov eax,pri ; get retval into eax (PRI) + mov edx,alt ; restore ALT + mov ebx,frm ; restore FRM + add ebx,edi ; relocate frame + ret + + +JIT_OP_SYSREQ_D: ; (TR) + mov ecx,esp ; get STK into ECX + mov ebp,amx ; get amx into EBP + + sub ecx,edi ; correct STK + mov alt,edx ; save ALT + + mov [ebp+_stk],ecx ; store values in AMX structure: STK, + mov ecx,hea ; HEA, + mov eax,frm ; and FRM + mov [ebp+_hea],ecx + mov [ebp+_frm],eax ; eax & ecx are invalid by now + + lea edx,[esp+4] ; 2nd param: parameter array + xchg esp,esi ; switch to caller stack + push edx + push ebp ; 1st param: amx + call ebx ; direct call + _DROPARGS 8 ; remove args from stack + xchg esp,esi ; switch back to AMX stack + mov ebp,amx ; get amx into EBP + cmp [ebp+_error],AMX_ERR_NONE + jne _return_popstack; return error code, if any + + ; return value is in eax (PRI) + mov edx,alt ; restore ALT + mov ebx,frm ; restore FRM + add ebx,edi ; relocate frame + ret + + +JIT_OP_LINE: + pop ecx ; get return address + mov ebp,amx + push eax + push edx + mov eax,[ecx] ; get curline + mov edx,[ecx+4] ; get curfile + add ecx,8 ; skip curline & curfile + mov [ebp+_curline],eax ; store curline + mov [ebp+_curfile],edx ; store curfile + + pop edx + pop eax + jmp ecx ; jump back + + +JIT_OP_SWITCH: + pop ebp ; pop return address = table address + mov ecx,[ebp] ; ECX = number of records + lea ebp,[ebp+ecx*8+8] ; set pointer _after_ LAST case + op_switch_loop: + cmp eax,[ebp-8] ; PRI == case label? + je op_switch_jump ; found, jump + sub ebp,8 ; position to preceding case + loop op_switch_loop ; check next case, or fall through + op_switch_jump: +IF FORCERELOCATABLE EQ 0 + jmp [ebp-4] ; jump to the case instructions +ELSE + add ebp,[ebp-4] ; add offset to make absolute adddress + jmp ebp +ENDIF + +_amx_exec_asm ENDP + +; +; The caller of asm_runJIT() can determine the maximum size of the compiled +; code by multiplying the result of this function by the number of opcodes in +; Small module. +; +; unsigned long getMaxCodeSize_(); +; +_getMaxCodeSize PROC + + mov eax,MAXCODESIZE + ret + +_getMaxCodeSize ENDP + + +IFNDEF @Version + ; Microsoft MASM 6.x gives the error message "Register assumed to + ; ERROR" when I put the code lables in the data segment. I cannot find + ; a proper way around it. +.DATA +ENDIF + ALIGN 4 ; This is essential to avoid misalignment stalls. + +end_code DD 0 ; pointer to the end of the source code + +compiled_code DD 0 ; pointer to compiled code (including preamble) + +amxhead DD 0 ; pointer to the AMX_HEADER struct (arg #1 to runJIT) + +reloc_num DD 0 ; counts the addresses in the relocation table (jumps) + +lodb_and DD 0ffh, 0ffffh, 0, 0ffffffffh + +; +; A list of the "run-time-library" functions that are called via indirect calls. +; So these calls don't have to be relocated. This gives also the possibility to +; replace some of these with shorter/faster non-debug or non-checking versions, +; without changing the compiled code. Instead this table could be changed... +; +verify_adr_eax DD _VERIFYADDRESS_eax +verify_adr_edx DD _VERIFYADDRESS_edx +chk_marginstack DD _CHKMARGIN_STACK +chk_marginheap DD _CHKMARGIN_HEAP +chk_dividezero DD _CHKDIVIDEZERO +jit_return DD _return +jit_retn DD JIT_OP_RETN +jit_sdiv DD JIT_OP_SDIV +jit_movs DD JIT_OP_MOVS +jit_cmps DD JIT_OP_CMPS +jit_fill DD JIT_OP_FILL +jit_bounds DD JIT_OP_BOUNDS +jit_sysreq DD JIT_OP_SYSREQ +jit_sysreq_d DD JIT_OP_SYSREQ_D +jit_line DD JIT_OP_LINE +jit_switch DD JIT_OP_SWITCH + + +; +; The table for the browser/relocator function. +; + PUBLIC _amx_opcodelist +_amx_opcodelist: + DD OP_INVALID + DD OP_LOAD_PRI + DD OP_LOAD_ALT + DD OP_LOAD_S_PRI + DD OP_LOAD_S_ALT + DD OP_LREF_PRI + DD OP_LREF_ALT + DD OP_LREF_S_PRI + DD OP_LREF_S_ALT + DD OP_LOAD_I + DD OP_LODB_I + DD OP_CONST_PRI + DD OP_CONST_ALT + DD OP_ADDR_PRI + DD OP_ADDR_ALT + DD OP_STOR_PRI + DD OP_STOR_ALT + DD OP_STOR_S_PRI + DD OP_STOR_S_ALT + DD OP_SREF_PRI + DD OP_SREF_ALT + DD OP_SREF_S_PRI + DD OP_SREF_S_ALT + DD OP_STOR_I + DD OP_STRB_I + DD OP_LIDX + DD OP_LIDX_B + DD OP_IDXADDR + DD OP_IDXADDR_B + DD OP_ALIGN_PRI + DD OP_ALIGN_ALT + DD OP_LCTRL + DD OP_SCTRL + DD OP_MOVE_PRI + DD OP_MOVE_ALT + DD OP_XCHG + DD OP_PUSH_PRI + DD OP_PUSH_ALT + DD OP_PUSH_R_PRI + DD OP_PUSH_C + DD OP_PUSH + DD OP_PUSH_S + DD OP_POP_PRI + DD OP_POP_ALT + DD OP_STACK + DD OP_HEAP + DD OP_PROC + DD OP_RET + DD OP_RETN + DD OP_CALL + DD OP_CALL_I + DD OP_JUMP + DD OP_JREL + DD OP_JZER + DD OP_JNZ + DD OP_JEQ + DD OP_JNEQ + DD OP_JLESS + DD OP_JLEQ + DD OP_JGRTR + DD OP_JGEQ + DD OP_JSLESS + DD OP_JSLEQ + DD OP_JSGRTR + DD OP_JSGEQ + DD OP_SHL + DD OP_SHR + DD OP_SSHR + DD OP_SHL_C_PRI + DD OP_SHL_C_ALT + DD OP_SHR_C_PRI + DD OP_SHR_C_ALT + DD OP_SMUL + DD OP_SDIV + DD OP_SDIV_ALT + DD OP_UMUL + DD OP_UDIV + DD OP_UDIV_ALT + DD OP_ADD + DD OP_SUB + DD OP_SUB_ALT + DD OP_AND + DD OP_OR + DD OP_XOR + DD OP_NOT + DD OP_NEG + DD OP_INVERT + DD OP_ADD_C + DD OP_SMUL_C + DD OP_ZERO_PRI + DD OP_ZERO_ALT + DD OP_ZERO + DD OP_ZERO_S + DD OP_SIGN_PRI + DD OP_SIGN_ALT + DD OP_EQ + DD OP_NEQ + DD OP_LESS + DD OP_LEQ + DD OP_GRTR + DD OP_GEQ + DD OP_SLESS + DD OP_SLEQ + DD OP_SGRTR + DD OP_SGEQ + DD OP_EQ_C_PRI + DD OP_EQ_C_ALT + DD OP_INC_PRI + DD OP_INC_ALT + DD OP_INC + DD OP_INC_S + DD OP_INC_I + DD OP_DEC_PRI + DD OP_DEC_ALT + DD OP_DEC + DD OP_DEC_S + DD OP_DEC_I + DD OP_MOVS + DD OP_CMPS + DD OP_FILL + DD OP_HALT + DD OP_BOUNDS + DD OP_SYSREQ_PRI + DD OP_SYSREQ_C + DD OP_FILE + DD OP_LINE + DD OP_SYMBOL + DD OP_SRANGE + DD OP_JUMP_PRI + DD OP_SWITCH + DD OP_CASETBL + DD OP_SWAP_PRI ; TR + DD OP_SWAP_ALT ; TR + DD OP_PUSHADDR ; TR + DD OP_NOP ; TR + DD OP_SYSREQ_D ; TR + DD OP_SYMTAG ; TR + +END \ No newline at end of file diff --git a/amxmodx/JIT/jits.def b/amxmodx/JIT/jits.def new file mode 100755 index 00000000..970881a5 --- /dev/null +++ b/amxmodx/JIT/jits.def @@ -0,0 +1,10 @@ +LIBRARY jits +DESCRIPTION "JIT for AMX Mod X" +EXPORTS + asm_runJIT @1 + getMaxCodeSize @2 + amx_exec_asm @3 + amx_opcodelist @4 +SECTIONS + .data READ WRITE + .code EXECUTE \ No newline at end of file diff --git a/amxmodx/JIT/jits.exp b/amxmodx/JIT/jits.exp new file mode 100755 index 0000000000000000000000000000000000000000..df1c0fcf2ca0f87b93ef682ce9f2da00b3607f05 GIT binary patch literal 942 zcma)4O;5sL4DLXkpIp3}$bka~GEhl0!~=wpfR2s8h- zfp=`*eK;T*oAp`ywCnozts*4gGy7Wp5CBrZMUX7h@C@G}d`C4=zsNz^M=j$cnIo&` zO)cU`&QTq#vn0E|NB;(WI7HQ?u=P8tH+G7%bmfnH!v;7 z9(#Zk8%-Xs8anhXuW8Jx80pw^?3VgZpo(yWNHs-BBT`QyWDqyGRY0WnMYurR;#LuH zk6V`fomy3IXlhBKndjO!F^0~P ztKpz~19vFd=G*ZhEEd#p{Dx?nNf^&n!^N|;(DRv5e>g>lcNa2#7M&(!p0tq(P>-05 zT0l92yA`Gg_bbc^Ojejvc!?M^phOS|&D;M^K=2y?xD8!=`#IQ7Bne1iqrYF}UkkvN zg4~O(2dzxFA8cho7_;3WTgnI++_!j*A2!;;;M=u|jb=-0>rAe7s_jNg*V>F$02oud Wio({S@JwMpsMovdHIplP1-=2rL9#&r literal 0 HcmV?d00001 diff --git a/amxmodx/JIT/jits.lib b/amxmodx/JIT/jits.lib new file mode 100755 index 0000000000000000000000000000000000000000..1652b7bd2a59e8c28302c4798483373dd7cb0261 GIT binary patch literal 14032 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 QFDBnCtMgs5z;BxU7pRbycmMzZ literal 0 HcmV?d00001