compiler-lightning/include/lightning/jit_arm.h
2025-02-03 18:35:31 +13:00

201 lines
6.4 KiB
C

/*
* Copyright (C) 2012-2023 Free Software Foundation, Inc.
*
* This file is part of GNU lightning.
*
* GNU lightning is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation; either version 3, or (at your option)
* any later version.
*
* GNU lightning 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 Lesser General Public
* License for more details.
*
* Authors:
* Paulo Cesar Pereira de Andrade
*/
#ifndef _jit_arm_h
#define _jit_arm_h
#define JIT_HASH_CONSTS 0
#define JIT_NUM_OPERANDS 3
/*
* Types
*/
#define jit_swf_p() (jit_cpu.vfp == 0)
#define jit_hardfp_p() jit_cpu.abi
#define jit_ldrt_strt_p() jit_cpu.ldrt_strt
#define jit_post_index_p() jit_cpu.post_index
#define JIT_FP _R11
typedef enum {
#define jit_r(i) (_R4 + (i))
#define jit_r_num() 3
#define jit_v(i) (_R7 + (i))
#define jit_v_num() 3
#define jit_f(i) (jit_cpu.abi ? _D8 + ((i)<<1) : _D0 - ((i)<<1))
#define jit_f_num() 8
_R12, /* ip - temporary */
#define JIT_R0 _R4
#define JIT_R1 _R5
#define JIT_R2 _R6
_R4, /* r4 - variable */
_R5, /* r5 - variable */
_R6, /* r6 - variable */
#define JIT_V0 _R7
#define JIT_V1 _R8
#define JIT_V2 _R9
_R7, /* r7 - variable */
_R8, /* r8 - variable */
_R9, /* r9 - variable */
_R10, /* sl - stack limit */
_R11, /* fp - frame pointer */
_R13, /* sp - stack pointer */
_R14, /* lr - link register */
_R15, /* pc - program counter */
_R3, /* r3 - argument/result */
_R2, /* r2 - argument/result */
_R1, /* r1 - argument/result */
_R0, /* r0 - argument/result */
#define JIT_F0 (jit_hardfp_p() ? _D8 : _D0)
#define JIT_F1 (jit_hardfp_p() ? _D9 : _D1)
#define JIT_F2 (jit_hardfp_p() ? _D10 : _D2)
#define JIT_F3 (jit_hardfp_p() ? _D11 : _D3)
#define JIT_F4 (jit_hardfp_p() ? _D12 : _D4)
#define JIT_F5 (jit_hardfp_p() ? _D13 : _D5)
#define JIT_F6 (jit_hardfp_p() ? _D14 : _D6)
#define JIT_F7 (jit_hardfp_p() ? _D15 : _D7)
_S16, _D8 = _S16, _Q4 = _D8,
_S17,
_S18, _D9 = _S18,
_S19,
_S20, _D10 = _S20, _Q5 = _D10,
_S21,
_S22, _D11 = _S22,
_S23,
_S24, _D12 = _S24, _Q6 = _D12,
_S25,
_S26, _D13 = _S26,
_S27,
_S28, _D14 = _S28, _Q7 = _D14,
_S29,
_S30, _D15 = _S30,
_S31,
_S15,
_S14, _D7 = _S14,
_S13,
_S12, _D6 = _S12, _Q3 = _D6,
_S11,
_S10, _D5 = _S10,
_S9,
_S8, _D4 = _S8, _Q2 = _D4,
_S7,
_S6, _D3 = _S6,
_S5,
_S4, _D2 = _S4, _Q1 = _D2,
_S3,
_S2, _D1 = _S2,
_S1,
_S0, _D0 = _S0, _Q0 = _D0,
_NOREG,
#define JIT_NOREG _NOREG
} jit_reg_t;
typedef struct {
jit_uint32_t version : 4;
/* this field originally was only used for the 'e' in armv5te.
* it can also be used to force hardware division, if setting
* version to 7, telling it is armv7r or better. */
jit_uint32_t extend : 1;
/* only generate thumb instructions for thumb2 */
jit_uint32_t thumb : 1;
jit_uint32_t vfp : 3;
jit_uint32_t neon : 1;
jit_uint32_t abi : 2;
/* use strt+offset instead of str.w?
* on special cases it causes a SIGILL at least on qemu, probably
* due to some memory ordering constraint not being respected, so,
* disable by default */
jit_uint32_t ldrt_strt : 1;
/* assume functions called never match jit instruction set?
* that is libc, gmp, mpfr, etc functions are in thumb mode and jit
* is in arm mode, or the reverse, what may cause a crash upon return
* of that function if generating jit for a relative jump.
*/
/* Apparently a qemu 8.1.3 and possibly others bug, that treat
* ldrT Rt, [Rn, #+-<immN>]! and ldrT Rt, [Rn], #+/-<immN>
* identically, as a pre-index but the second one should adjust
* Rn after the load.
* The syntax for only offseting is ldrT Rt{, [Rn, #+/-<immN>}]
*/
jit_uint32_t post_index : 1;
jit_uint32_t exchange : 1;
/* By default assume cannot load unaligned data.
* A3.2.1
* Unaligned data access
* An ARMv7 implementation must support unaligned data accesses by
* some load and store instructions, as Table A3-1 shows. Software
* can set the SCTLR.A bit to control whether a misaligned access by
* one of these instructions causes an Alignment fault Data Abort
* exception.
* Table A3-1 Alignment requirements of load/store instructions
* Result if check fails when
* Instructions Alignment check SCTLR.A is 0 SCTLR.A is 1
* LDRB, LDREXB,
* LDRBT, LDRSB,
* LDRSBT, STRB,
* STREXB, STRBT,
* SWPB, TBB None - -
* LDRH, LDRHT,
* LDRSH, LDRSHT,
* STRH, STRHT,
* TBH Halfword Unaligned access Alignment fault
* LDREXH, STREXH Halfword Alignment fault Alignment fault
* LDR, LDRT,
* STR, STRT Word Unaligned access Alignment fault
* LDREX, STREX Word Alignment fault Alignment fault
* LDREXD, STREXD Doubleword Alignment fault Alignment fault
* All forms of
* LDM and STM,
* LDRD, RFE, SRS,
* STRD, SWP Word Alignment fault Alignment fault
* LDC, LDC2,
* STC, STC2 Word Alignment fault Alignment fault
* VLDM, VLDR,
* VPOP, VPUSH,
* VSTM, VSTR Word Alignment fault Alignment fault
* VLD1, VLD2,
* VLD3, VLD4,
* VST1, VST2,
* VST3, VST4,
* all with
* standard
* alignment (a) Element size Unaligned access Alignment fault
* VLD1, VLD2,
* VLD3, VLD4,
* VST1, VST2,
* VST3, VST4,
* all with
* @<align>
* specified (a) As specified by Alignment fault Alignment fault
* @<align>
*
* (a) These element and structure load/store instructions are only in
* the Advanced SIMD Extension to the ARMv7 ARM and Thumb instruction
* sets. ARMv7 does not support the pre-ARMv6 alignment model, so
* software cannot use that model with these instructions.
*/
jit_uint32_t unaligned : 1;
jit_uint32_t vfp_unaligned : 1;
} jit_cpu_t;
/*
* Initialization
*/
extern jit_cpu_t jit_cpu;
#endif /* _jit_arm_h */