diff options
author | Kimplul <kimi.h.kuparinen@gmail.com> | 2024-06-29 14:20:07 +0300 |
---|---|---|
committer | Kimplul <kimi.h.kuparinen@gmail.com> | 2024-06-29 14:20:07 +0300 |
commit | 49aa680ccdac46d1d2a7f9f250999b7ff7099548 (patch) | |
tree | 1de3bd5209feadfd147f7a05d1ac925f98b747b1 | |
parent | 29718f2e84478b296c3198ae6d35cfd5d79efb14 (diff) | |
download | ejit-49aa680ccdac46d1d2a7f9f250999b7ff7099548.tar.gz ejit-49aa680ccdac46d1d2a7f9f250999b7ff7099548.zip |
start adding tests
217 files changed, 11023 insertions, 192 deletions
@@ -1,4 +1,5 @@ deps.mk +tests.mk docs/output build ejit.o @@ -6,6 +6,12 @@ all: setup examples: setup $(MAKE) -f scripts/makefile examples +.PHONY: check +check: all + @echo -n > tests.mk + @./scripts/gen-tests $$(echo tests/*.c) + $(MAKE) -f tests/makefile check + # this kicks all unrecognised targets to the client script. # note that trying to compile individual files, e.g. # @@ -24,7 +30,7 @@ setup: @echo -n > deps.mk @./scripts/gen-deps -p EJIT -c COMPILE_EJIT -b ejit "$(EJIT_SOURCES)" -CLEANUP := build deps.mk ejit.o examples/exec examples/*.d +CLEANUP := build deps.mk tests.md ejit.o examples/exec examples/*.d tests/test-* CLEANUP_CMD := EJIT_SOURCES := @@ -32,7 +38,7 @@ include src/source.mk .PHONY: format format: - find src include -iname '*.[ch]' |\ + find src include tests -iname '*.[ch]' |\ xargs uncrustify -c uncrustify.conf --no-backup -F - .PHONY: license @@ -0,0 +1 @@ +./scripts/gen-tests: 8: tests/absr_d.c: Permission denied diff --git a/include/ejit/ejit.h b/include/ejit/ejit.h index 17a6034..895ce9b 100644 --- a/include/ejit/ejit.h +++ b/include/ejit/ejit.h @@ -27,14 +27,14 @@ enum ejit_type struct ejit_arg { union { - int8_t c; - uint8_t uc; - int16_t s; - uint16_t us; - int32_t i; - uint32_t ui; - int64_t l; - uint64_t ul; + int8_t i8; + uint8_t u8; + int16_t i16; + uint16_t u16; + int32_t i32; + uint32_t u32; + int64_t i64; + uint64_t u64; float f; double d; void *p; @@ -73,7 +73,7 @@ static inline bool ejit_float_type(enum ejit_type t) return false; } -static inline struct ejit_arg ejit_build_arg(enum ejit_type type, long x) +static inline struct ejit_arg ejit_build_arg(enum ejit_type type, uint64_t x) { assert(ejit_int_type(type)); @@ -81,14 +81,14 @@ static inline struct ejit_arg ejit_build_arg(enum ejit_type type, long x) a.type = type; switch (type) { - case EJIT_INT8: a.c = x; break; - case EJIT_INT16: a.s = x; break; - case EJIT_INT32: a.i = x; break; - case EJIT_INT64: a.l = x; break; - case EJIT_UINT8: a.uc = x; break; - case EJIT_UINT16: a.us = x; break; - case EJIT_UINT32: a.ui = x; break; - case EJIT_UINT64: a.ul = x; break; + case EJIT_INT8: a.i8 = x; break; + case EJIT_INT16: a.i16 = x; break; + case EJIT_INT32: a.i32 = x; break; + case EJIT_INT64: a.i64 = x; break; + case EJIT_UINT8: a.u8 = x; break; + case EJIT_UINT16: a.u16 = x; break; + case EJIT_UINT32: a.u32 = x; break; + case EJIT_UINT64: a.u64 = x; break; case EJIT_POINTER: a.p = (void *)x; break; default: abort(); } @@ -141,20 +141,29 @@ struct ejit_operand { enum ejit_type type; union { - long r; - double d; + long r; + double d; }; }; struct ejit_func; -struct ejit_func *ejit_create_func(enum ejit_type rtype, size_t argc, const struct ejit_operand args[argc]); +struct ejit_func *ejit_create_func(enum ejit_type rtype, size_t argc, + const struct ejit_operand args[argc]); void ejit_compile_func(struct ejit_func *f, size_t gpr, size_t fpr); -void ejit_select_compile_func(struct ejit_func *f, size_t gpr, size_t fpr, bool try_jit); +void ejit_select_compile_func(struct ejit_func *f, size_t gpr, size_t fpr, + bool try_jit); + +long ejit_run_func(struct ejit_func *f, size_t argc, + struct ejit_arg args[argc]); +double ejit_run_func_f(struct ejit_func *f, size_t argc, + struct ejit_arg args[argc]); -long ejit_run_func(struct ejit_func *f, size_t argc, struct ejit_arg args[argc]); -double ejit_run_func_f(struct ejit_func *f, size_t argc, struct ejit_arg args[argc]); +static inline double ejit_run_func_f_1(struct ejit_func *f, struct ejit_arg a0) +{ + return ejit_run_func_f(f, 1, &a0); +} void ejit_destroy_func(struct ejit_func *s); @@ -163,28 +172,81 @@ void ejit_destroy_func(struct ejit_func *s); #define EJIT_ARG_POINTER(x) ((struct ejit_arg){.p = (x), .type = EJIT_POINTER}) -#define EJIT_OPERAND_GPR(x, t) ((struct ejit_operand){ .kind = EJIT_OPERAND_GPR, .r = (long)(x), .type = (t)}) -#define EJIT_OPERAND_FPR(x, t) ((struct ejit_operand){ .kind = EJIT_OPERAND_FPR, .r = (long)(x) .type =(t)}) -#define EJIT_OPERAND_IMM(x, t) ((struct ejit_operand){ .kind = EJIT_OPERAND_IMM, .r = (long)(x), .type = (t)}) -#define EJIT_OPERAND_FLT(x, t) ((struct ejit_operand){ .kind = EJIT_OPERAND_FLT, .r = (double)(x), .type = (t)}) +#define EJIT_OPERAND_GPR(x, t) \ + ((struct ejit_operand){ .kind = EJIT_OPERAND_GPR, \ + .r = (long)(x), \ + .type = (t)}) +#define EJIT_OPERAND_FPR(x, t) \ + ((struct ejit_operand){ .kind = EJIT_OPERAND_FPR, \ + .r = (long)(x), \ + .type =(t)}) + +#define EJIT_OPERAND_IMM(x, t) \ + ((struct ejit_operand){ .kind = EJIT_OPERAND_IMM, \ + .r = (long)(x), \ + .type = (t)}) +#define EJIT_OPERAND_FLT(x, t) \ + ((struct ejit_operand){ .kind = EJIT_OPERAND_FLT, \ + .r = (double)(x), \ + .type = (t)}) + +static inline struct ejit_arg ejit_i8(int8_t a) +{ + return (struct ejit_arg){.i8 = a, .type = EJIT_INT8}; +} + +static inline struct ejit_arg ejit_i16(int16_t a) +{ + return (struct ejit_arg){.i16 = a, .type = EJIT_INT16}; +} + +static inline struct ejit_arg ejit_i32(int32_t a) +{ + return (struct ejit_arg){.i32 = a, .type = EJIT_INT32}; +} + +static inline struct ejit_arg ejit_double(double a) +{ + return (struct ejit_arg){.d = a, .type = EJIT_DOUBLE}; +} + +#define EJIT_ARG(x) \ + _Generic((x), \ + int8_t: ejit_i8, \ + int16_t: ejit_i16, \ + int32_t: ejit_i32, \ + int64_t: ejit_i64, \ + uint8_t: ejit_i8, \ + uint16_t: ejit_i16, \ + uint32_t: ejit_i32, \ + uint64_t: ejit_i64, \ + float: ejit_float, \ + double: ejit_double, \ + default: ejit_pointer, \ + )(x) typedef long (*ejit_escape_t)(size_t argc, const struct ejit_arg args[argc]); -typedef double (*ejit_escape_f_t)(size_t argc, const struct ejit_arg args[argc]); +typedef double (*ejit_escape_f_t)(size_t argc, + const struct ejit_arg args[argc]); struct ejit_label ejit_label(struct ejit_func *s); -void ejit_calli(struct ejit_func *s, struct ejit_func *f, size_t argc, const struct ejit_operand args[argc]); +void ejit_calli(struct ejit_func *s, struct ejit_func *f, size_t argc, + const struct ejit_operand args[argc]); -void ejit_calli_f(struct ejit_func *s, struct ejit_func *f, size_t argc, const struct ejit_operand args[argc]); +void ejit_calli_f(struct ejit_func *s, struct ejit_func *f, size_t argc, + const struct ejit_operand args[argc]); -void ejit_escapei(struct ejit_func *s, ejit_escape_t f, size_t argc, const struct ejit_operand args[argc]); +void ejit_escapei(struct ejit_func *s, ejit_escape_t f, size_t argc, + const struct ejit_operand args[argc]); -void ejit_escapei_f(struct ejit_func *s, ejit_escape_f_t f, size_t argc, const struct ejit_operand args[argc]); +void ejit_escapei_f(struct ejit_func *s, ejit_escape_f_t f, size_t argc, + const struct ejit_operand args[argc]); -void ejit_ret(struct ejit_func *s, struct ejit_gpr r0); -void ejit_ret_f(struct ejit_func *s, struct ejit_fpr r0); -void ejit_ret_i(struct ejit_func *s, long i); -void ejit_ret_fi(struct ejit_func *s, double f); +void ejit_retr(struct ejit_func *s, struct ejit_gpr r0); +void ejit_retr_f(struct ejit_func *s, struct ejit_fpr r0); +void ejit_reti(struct ejit_func *s, long i); +void ejit_reti_f(struct ejit_func *s, double f); void ejit_retval(struct ejit_func *s, struct ejit_gpr r0); void ejit_retval_f(struct ejit_func *s, struct ejit_fpr r0); @@ -192,7 +254,7 @@ void ejit_retval_f(struct ejit_func *s, struct ejit_fpr r0); /* move from r1 to r0 */ void ejit_movi(struct ejit_func *s, struct ejit_gpr r0, long i); void ejit_movr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1); -void ejit_movr_d(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1); +void ejit_movr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1); void ejit_ldi_8(struct ejit_func *s, struct ejit_gpr r0, void *p); void ejit_ldi_16(struct ejit_func *s, struct ejit_gpr r0, void *p); @@ -208,18 +270,28 @@ void ejit_ldi_f(struct ejit_func *s, struct ejit_fpr r0, void *p); void ejit_ldi_d(struct ejit_func *s, struct ejit_fpr r0, void *p); /* from r1 + o to r0 */ -void ejit_ldxi_8(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_ldxi_16(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_ldxi_32(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_ldxi_64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); - -void ejit_ldxi_u8(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_ldxi_u16(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_ldxi_u32(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_ldxi_u64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); - -void ejit_ldxi_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_gpr r1, long o); -void ejit_ldxi_d(struct ejit_func *s, struct ejit_fpr r0, struct ejit_gpr r1, long o); +void ejit_ldxi_8(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_ldxi_16(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_ldxi_32(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_ldxi_64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); + +void ejit_ldxi_u8(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_ldxi_u16(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_ldxi_u32(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_ldxi_u64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); + +void ejit_ldxi_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_gpr r1, + long o); +void ejit_ldxi_d(struct ejit_func *s, struct ejit_fpr r0, struct ejit_gpr r1, + long o); void ejit_sti_8(struct ejit_func *s, struct ejit_gpr r0, void *p); void ejit_sti_16(struct ejit_func *s, struct ejit_gpr r0, void *p); @@ -235,34 +307,55 @@ void ejit_sti_f(struct ejit_func *s, struct ejit_fpr r0, void *p); void ejit_sti_d(struct ejit_func *s, struct ejit_fpr r0, void *p); /* from r0 to r1 + o */ -void ejit_stxi_8(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_stxi_16(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_stxi_32(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); -void ejit_stxi_64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); - -void ejit_stxi_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_gpr r1, long o); -void ejit_stxi_d(struct ejit_func *s, struct ejit_fpr r0, struct ejit_gpr r1, long o); - -void ejit_addr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2); -void ejit_addr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, struct ejit_fpr r2); -void ejit_addi(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); - -void ejit_subr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2); -void ejit_subr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, struct ejit_fpr r2); -void ejit_subi(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o); - -void ejit_mulr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2); -void ejit_mulr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, struct ejit_fpr r2); - -void ejit_divr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2); -void ejit_divr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, struct ejit_fpr r2); +void ejit_stxi_8(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_stxi_16(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_stxi_32(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); +void ejit_stxi_64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); + +void ejit_stxi_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_gpr r1, + long o); +void ejit_stxi_d(struct ejit_func *s, struct ejit_fpr r0, struct ejit_gpr r1, + long o); + +void ejit_addr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2); +void ejit_addr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, + struct ejit_fpr r2); +void ejit_addi(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); + +void ejit_absr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1); + +void ejit_subr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2); +void ejit_subr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, + struct ejit_fpr r2); +void ejit_subi(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o); + +void ejit_mulr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2); +void ejit_mulr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, + struct ejit_fpr r2); + +void ejit_divr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2); +void ejit_divr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, + struct ejit_fpr r2); void ejit_negr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1); -void ejit_eqr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2); -void ejit_ltr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2); +void ejit_eqr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2); +void ejit_ltr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2); -struct ejit_reloc ejit_bltr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1); +struct ejit_reloc ejit_bltr(struct ejit_func *s, struct ejit_gpr r0, + struct ejit_gpr r1); struct ejit_reloc ejit_bnei(struct ejit_func *s, struct ejit_gpr r0, long o); struct ejit_reloc ejit_beqi(struct ejit_func *s, struct ejit_gpr r0, long o); struct ejit_reloc ejit_bgti(struct ejit_func *s, struct ejit_gpr r0, long o); diff --git a/scripts/gen-tests b/scripts/gen-tests new file mode 100755 index 0000000..934f13f --- /dev/null +++ b/scripts/gen-tests @@ -0,0 +1,27 @@ +#!/bin/sh + +PREFIX=JIT_TEST +for s in ${@} +do + exe="tests/test-jit-$(basename ${s%.*})" + dep="tests/${exe}.d" + + echo "${PREFIX} += ${exe}" >> tests.mk + echo "${dep}:" >> tests.mk + echo "-include ${dep}" >> tests.mk + echo "${exe}: ${s} ejit.o" >> tests.mk + echo " \$(COMPILE_JIT_TEST) ${s} ejit.o -o ${exe}" >> tests.mk +done + +PREFIX=BCODE_TESTS +for s in ${@} +do + exe="tests/test-bcode-$(basename ${s%.*})" + dep="tests/${exe}.d" + + echo "${PREFIX} += ${exe}" >> tests.mk + echo "${dep}:" >> tests.mk + echo "-include ${dep}" >> tests.mk + echo "${exe}: ${s} ejit.o" >> tests.mk + echo " \$(COMPILE_BCODE_TEST) ${s} ejit.o -o ${exe}" >> tests.mk +done diff --git a/src/common.h b/src/common.h index 1f2289d..c3442c6 100644 --- a/src/common.h +++ b/src/common.h @@ -57,6 +57,8 @@ enum ejit_opcode { ADDR_F, ADDI, + ABSR_F, + SUBR, SUBR_F, SUBI, @@ -88,10 +90,10 @@ enum ejit_opcode { CALLI, CALLI_F, - RET, - RET_I, - RET_F, - RET_FI, + RETR, + RETI, + RETR_F, + RETI_F, RETVAL, RETVAL_F, @@ -132,7 +134,7 @@ struct ejit_func { union interp_ret { - long r; + int64_t r; double d; }; @@ -142,9 +144,13 @@ struct interp_state { struct vec args; }; -union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg args[argc], struct interp_state *state, bool run, void ***labels_wb); +union interp_ret ejit_interp(struct ejit_func *f, size_t argc, + struct ejit_arg args[argc], + struct interp_state *state, bool run, + void ***labels_wb); -long ejit_run_interp(struct ejit_func *f, size_t argc, struct ejit_arg args[argc], struct interp_state *state); +int64_t ejit_run_interp(struct ejit_func *f, size_t argc, + struct ejit_arg args[argc], struct interp_state *state); bool ejit_compile(struct ejit_func *f); diff --git a/src/compile/compile.c b/src/compile/compile.c index c07a46a..c9a92a2 100644 --- a/src/compile/compile.c +++ b/src/compile/compile.c @@ -5,8 +5,8 @@ static void *alloc_arena(size_t size) { return mmap(NULL, size, - PROT_EXEC | PROT_READ | PROT_WRITE, - MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); + PROT_EXEC | PROT_READ | PROT_WRITE, + MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); } static void free_arena(void *arena, size_t size) @@ -27,7 +27,7 @@ static size_t frploc_count(struct ejit_func *f) static size_t stack_size(struct ejit_func *f) { return grploc_count(f) * sizeof(jit_uword_t) - + frploc_count(f) * sizeof(jit_float64_t); + + frploc_count(f) * sizeof(jit_float64_t); } static jit_off_t stack_loc(size_t l) @@ -40,7 +40,7 @@ static jit_off_t stack_loc_f(struct ejit_func *f, size_t l) { assert(l >= jit_vf_num()); return grploc_count(f) * sizeof(jit_uword_t) - + (l - jit_vf_num()) * sizeof(jit_float64_t); + + (l - jit_vf_num()) * sizeof(jit_float64_t); } @@ -84,14 +84,16 @@ static void compile_label(jit_state_t *j, size_t ii, struct vec *labels) vect_at(jit_addr_t, *labels, ii) = jit_address(j); } -static void compile_movi(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_movi(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t r = getreg(f, i.r0, 0); jit_movi(j, r, i.o); putloc(f, j, i.r0, r); } -static void compile_movr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_movr(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t to = getreg(f, i.r0, 0); jit_gpr_t from = getreg(f, i.r1, 1); @@ -99,7 +101,8 @@ static void compile_movr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i putloc(f, j, i.r0, to); } -static void compile_addr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_addr(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t dst = getreg(f, i.r0, 0); jit_gpr_t src0 = getloc(f, j, i.r1, 1); @@ -108,7 +111,8 @@ static void compile_addr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i putloc(f, j, i.r0, dst); } -static void compile_addi(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_addi(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t dst = getreg(f, i.r0, 0); jit_gpr_t src0 = getloc(f, j, i.r1, 1); @@ -116,7 +120,8 @@ static void compile_addi(struct ejit_func *f, jit_state_t *j, struct ejit_insn i putloc(f, j, i.r0, dst); } -static void compile_subr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_subr(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t dst = getreg(f, i.r0, 0); jit_gpr_t src0 = getloc(f, j, i.r1, 1); @@ -125,7 +130,8 @@ static void compile_subr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i putloc(f, j, i.r0, dst); } -static void compile_subi(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_subi(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t dst = getreg(f, i.r0, 0); jit_gpr_t src0 = getloc(f, j, i.r1, 1); @@ -133,14 +139,16 @@ static void compile_subi(struct ejit_func *f, jit_state_t *j, struct ejit_insn i putloc(f, j, i.r0, dst); } -static void compile_stxi64(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_stxi64(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t r0 = getloc(f, j, i.r0, 0); jit_gpr_t r1 = getloc(f, j, i.r1, 1); jit_stxi_l(j, i.o, r1, r0); } -static void compile_ldxiu64(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_ldxiu64(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t r0 = getreg(f, i.r0, 0); jit_gpr_t r1 = getloc(f, j, i.r1, 1); @@ -148,8 +156,10 @@ static void compile_ldxiu64(struct ejit_func *f, jit_state_t *j, struct ejit_ins putloc(f, j, i.r0, r0); } -static void compile_reg_cmp(struct ejit_func *f, jit_state_t *j, struct ejit_insn i, - jit_reloc_t (*bcomp)(jit_state_t *, jit_gpr_t, jit_gpr_t), long same) +static void compile_reg_cmp(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i, + jit_reloc_t (*bcomp)(jit_state_t *, jit_gpr_t, + jit_gpr_t), long same) { jit_gpr_t r0 = getreg(f, i.r0, 0); if (i.r1 == i.r2) { @@ -185,7 +195,8 @@ static void compile_ltr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) compile_reg_cmp(f, j, i, jit_bltr, 0); } -static void compile_bltr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i, struct vec *relocs) +static void compile_bltr(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i, struct vec *relocs) { jit_gpr_t c0 = getloc(f, j, i.r1, 0); jit_gpr_t c1 = getloc(f, j, i.r2, 1); @@ -194,7 +205,8 @@ static void compile_bltr(struct ejit_func *f, jit_state_t *j, struct ejit_insn i vect_append(struct reloc_helper, *relocs, &h); } -static void compile_beqi(struct ejit_func *f, jit_state_t *j, struct ejit_insn i, struct vec *relocs) +static void compile_beqi(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i, struct vec *relocs) { jit_gpr_t r1 = getloc(f, j, i.r1, 0); jit_reloc_t r = jit_beqi(j, r1, i.o); @@ -202,7 +214,8 @@ static void compile_beqi(struct ejit_func *f, jit_state_t *j, struct ejit_insn i vect_append(struct reloc_helper, *relocs, &h); } -static void compile_bnei(struct ejit_func *f, jit_state_t *j, struct ejit_insn i, struct vec *relocs) +static void compile_bnei(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i, struct vec *relocs) { jit_gpr_t r1 = getloc(f, j, i.r1, 0); jit_reloc_t r = jit_bnei(j, r1, i.o); @@ -210,7 +223,8 @@ static void compile_bnei(struct ejit_func *f, jit_state_t *j, struct ejit_insn i vect_append(struct reloc_helper, *relocs, &h); } -static void compile_bgti(struct ejit_func *f, jit_state_t *j, struct ejit_insn i, struct vec *relocs) +static void compile_bgti(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i, struct vec *relocs) { jit_gpr_t r1 = getloc(f, j, i.r1, 0); jit_reloc_t r = jit_bgti(j, r1, i.o); @@ -218,7 +232,8 @@ static void compile_bgti(struct ejit_func *f, jit_state_t *j, struct ejit_insn i vect_append(struct reloc_helper, *relocs, &h); } -static void compile_jmp(struct ejit_func *f, jit_state_t *j, struct ejit_insn i, struct vec *relocs) +static void compile_jmp(struct ejit_func *f, jit_state_t *j, struct ejit_insn i, + struct vec *relocs) { (void)(f); jit_reloc_t r = jit_jmp(j); @@ -226,7 +241,8 @@ static void compile_jmp(struct ejit_func *f, jit_state_t *j, struct ejit_insn i, vect_append(struct reloc_helper, *relocs, &h); } -static void compile_retval(struct ejit_func *f, jit_state_t *j, struct ejit_insn i) +static void compile_retval(struct ejit_func *f, jit_state_t *j, + struct ejit_insn i) { jit_gpr_t r0 = getreg(f, i.r0, 0); jit_retval(j, r0); @@ -256,14 +272,14 @@ static enum jit_operand_abi jit_abi_from(enum ejit_type t) static size_t arg_offsetof(enum ejit_type t) { switch (t) { - case EJIT_INT8: return offsetof(struct ejit_arg, c); - case EJIT_INT16: return offsetof(struct ejit_arg, s); - case EJIT_INT32: return offsetof(struct ejit_arg, i); - case EJIT_INT64: return offsetof(struct ejit_arg, l); - case EJIT_UINT8: return offsetof(struct ejit_arg, uc); - case EJIT_UINT16: return offsetof(struct ejit_arg, us); - case EJIT_UINT32: return offsetof(struct ejit_arg, ui); - case EJIT_UINT64: return offsetof(struct ejit_arg, ul); + case EJIT_INT8: return offsetof(struct ejit_arg, i8); + case EJIT_INT16: return offsetof(struct ejit_arg, i16); + case EJIT_INT32: return offsetof(struct ejit_arg, i32); + case EJIT_INT64: return offsetof(struct ejit_arg, i64); + case EJIT_UINT8: return offsetof(struct ejit_arg, u8); + case EJIT_UINT16: return offsetof(struct ejit_arg, u16); + case EJIT_UINT32: return offsetof(struct ejit_arg, u32); + case EJIT_UINT64: return offsetof(struct ejit_arg, u64); case EJIT_POINTER: return offsetof(struct ejit_arg, p); case EJIT_FLOAT: return offsetof(struct ejit_arg, f); case EJIT_DOUBLE: return offsetof(struct ejit_arg, d); @@ -282,7 +298,8 @@ static jit_off_t arg_offset(struct ejit_insn i) static jit_off_t type_offset(struct ejit_insn i) { - return (sizeof(struct ejit_arg) * i.r0) + offsetof(struct ejit_arg, type); + return (sizeof(struct ejit_arg) * i.r0) + offsetof(struct ejit_arg, + type); } static void fixup_operands(struct vec *operands, size_t fixup) @@ -297,7 +314,8 @@ static void fixup_operands(struct vec *operands, size_t fixup) } } -static void compile_imm_call(jit_state_t *j, struct vec *src, struct vec *dst, void *addr, size_t argc, jit_operand_t args[argc]) +static void compile_imm_call(jit_state_t *j, struct vec *src, struct vec *dst, + void *addr, size_t argc, jit_operand_t args[argc]) { /* each move is type + arg, so twofold */ size_t movec = vec_len(src) / 2; @@ -313,7 +331,8 @@ static void compile_imm_call(jit_state_t *j, struct vec *src, struct vec *dst, v vec_reset(dst); } -static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, size_t size) +static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, + size_t size) { jit_begin(j, arena, size); size_t gprs = f->gpr >= jit_v_num() ? jit_v_num() : f->gpr; @@ -323,8 +342,8 @@ static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, /* very important, argc we don't really do anything with but JIR_R1 * contains the argument stack! */ jit_load_args_2(j, - jit_operand_gpr(JIT_OPERAND_ABI_WORD, JIT_R0), - jit_operand_gpr(JIT_OPERAND_ABI_POINTER, JIT_R1)); + jit_operand_gpr(JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr(JIT_OPERAND_ABI_POINTER, JIT_R1)); size_t stack = jit_align_stack(j, stack_size(f)); @@ -367,24 +386,29 @@ static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, case JMP: compile_jmp(f, j, i, &relocs); break; case ARG: { - jit_operand_t type = jit_operand_imm(JIT_OPERAND_ABI_WORD, i.r1); + jit_operand_t type = + jit_operand_imm(JIT_OPERAND_ABI_WORD, i.r1); jit_operand_t arg; if (i.r0 < jit_v_num()) { /* regular register */ - arg = jit_operand_gpr(jit_abi_from(i.r1), jit_v(i.r2)); + arg = jit_operand_gpr(jit_abi_from(i.r1), + jit_v(i.r2)); } else { /* stack location, note that we'll fix up the SP * offset before doing the actual call */ - arg = jit_operand_mem(jit_abi_from(i.r1), JIT_SP, stack_loc(i.r0)); + arg = jit_operand_mem(jit_abi_from(i.r1), + JIT_SP, stack_loc(i.r0)); } vec_append(&src, &type); vec_append(&src, &arg); jit_operand_t to[2] = { - jit_operand_mem(JIT_OPERAND_ABI_WORD, JIT_SP, type_offset(i)), - jit_operand_mem(jit_abi_from(i.r1), JIT_SP, arg_offset(i)) + jit_operand_mem(JIT_OPERAND_ABI_WORD, JIT_SP, + type_offset(i)), + jit_operand_mem(jit_abi_from(i.r1), JIT_SP, + arg_offset(i)) }; vec_append(&dst, &to[0]); @@ -394,7 +418,8 @@ static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, case ESCAPEI: { jit_operand_t args[2] = { - jit_operand_imm(JIT_OPERAND_ABI_WORD, vec_len(&src) / 2), + jit_operand_imm(JIT_OPERAND_ABI_WORD, + vec_len(&src) / 2), jit_operand_gpr(JIT_OPERAND_ABI_POINTER, JIT_SP) }; compile_imm_call(j, &src, &dst, (void *)i.o, 2, args); @@ -404,7 +429,8 @@ static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, case CALLI: { jit_operand_t args[3] = { jit_operand_imm(JIT_OPERAND_ABI_POINTER, i.o), - jit_operand_imm(JIT_OPERAND_ABI_WORD, vec_len(&src) / 2), + jit_operand_imm(JIT_OPERAND_ABI_WORD, + vec_len(&src) / 2), jit_operand_gpr(JIT_OPERAND_ABI_POINTER, JIT_SP) }; compile_imm_call(j, &src, &dst, ejit_run_func, 3, args); @@ -412,7 +438,7 @@ static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, } case RETVAL: compile_retval(f, j, i); break; - case RET: { + case RETR: { jit_gpr_t r = getloc(f, j, i.r0, 0); /* R0 won't get overwritten by jit_leave_jit_abi */ jit_movr(j, JIT_R0, r); @@ -422,7 +448,7 @@ static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, break; } - case RET_I: { + case RETI: { jit_shrink_stack(j, stack); jit_leave_jit_abi(j, gprs, fprs, frame); jit_reti(j, i.o); @@ -440,19 +466,21 @@ static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, case PARAM: { /* move from argument stack to location */ jit_operand_t from = jit_operand_mem( - jit_abi_from(i.r1), - JIT_R1, - arg_offset(i) - ); + jit_abi_from(i.r1), + JIT_R1, + arg_offset(i) + ); jit_operand_t to; if (i.r0 < jit_v_num()) { /* regular register */ - to = jit_operand_gpr(jit_abi_from(i.r1), jit_v(i.r2)); + to = jit_operand_gpr(jit_abi_from(i.r1), + jit_v(i.r2)); } else { /* stack location */ - to = jit_operand_mem(jit_abi_from(i.r1), JIT_SP, stack_loc(i.r2)); + to = jit_operand_mem(jit_abi_from(i.r1), JIT_SP, + stack_loc(i.r2)); } vec_append(&src, &from); @@ -474,7 +502,8 @@ static size_t compile_fn_body(struct ejit_func *f, jit_state_t *j, void *arena, } foreach_vec(ri, relocs) { - struct reloc_helper h = vect_at(struct reloc_helper, relocs, ri); + struct reloc_helper h = vect_at(struct reloc_helper, relocs, + ri); jit_addr_t a = vect_at(jit_addr_t, labels, h.to); jit_reloc_t r = h.r; @@ -5,31 +5,36 @@ #include "common.h" -static void emit_insn_i(struct ejit_func *f, enum ejit_opcode op, size_t r0, size_t r1, long o) +static void emit_insn_i(struct ejit_func *f, enum ejit_opcode op, size_t r0, + size_t r1, long o) { struct ejit_insn i = {.op = op, .r0 = r0, .r1 = r1, .o = o}; vec_append(&f->insns, &i); } -static void emit_insn_r(struct ejit_func *f, enum ejit_opcode op, size_t r0, size_t r1, size_t r2) +static void emit_insn_r(struct ejit_func *f, enum ejit_opcode op, size_t r0, + size_t r1, size_t r2) { struct ejit_insn i = {.op = op, .r0 = r0, .r1 = r1, .r2 = r2}; vec_append(&f->insns, &i); } -static void emit_insn_p(struct ejit_func *f, enum ejit_opcode op, size_t r0, size_t r1, void *p) +static void emit_insn_p(struct ejit_func *f, enum ejit_opcode op, size_t r0, + size_t r1, void *p) { struct ejit_insn i = {.op = op, .r0 = r0, .r1 = r1, .p = p}; vec_append(&f->insns, &i); } -static void emit_insn_f(struct ejit_func *f, enum ejit_opcode op, size_t r0, size_t r1, double d) +static void emit_insn_f(struct ejit_func *f, enum ejit_opcode op, size_t r0, + size_t r1, double d) { struct ejit_insn i = {.op = op, .r0 = r0, .r1 = r1, .d = d}; vec_append(&f->insns, &i); } -struct ejit_func *ejit_create_func(enum ejit_type rtype, size_t argc, const struct ejit_operand args[argc]) +struct ejit_func *ejit_create_func(enum ejit_type rtype, size_t argc, + const struct ejit_operand args[argc]) { struct ejit_func *f = malloc(sizeof(struct ejit_func)); assert(f); @@ -43,8 +48,10 @@ struct ejit_func *ejit_create_func(enum ejit_type rtype, size_t argc, const stru for (size_t i = 0; i < argc; ++i) { switch (args[i].kind) { - case EJIT_OPERAND_GPR: emit_insn_r(f, PARAM, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_FPR: emit_insn_r(f, PARAM_F, i, args[i].type, args[i].r); break; + case EJIT_OPERAND_GPR: emit_insn_r(f, PARAM, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_FPR: emit_insn_r(f, PARAM_F, i, args[i].type, + args[i].r); break; default: abort(); } } @@ -58,7 +65,8 @@ void ejit_compile_func(struct ejit_func *f, size_t gpr, size_t fpr) ejit_select_compile_func(f, gpr, fpr, true); } -void ejit_select_compile_func(struct ejit_func *f, size_t gpr, size_t fpr, bool try_jit) +void ejit_select_compile_func(struct ejit_func *f, size_t gpr, size_t fpr, + bool try_jit) { /* emit a final end instruction in case user didn't do a return */ emit_insn_i(f, END, 0, 0, 0); @@ -109,14 +117,19 @@ void ejit_patch(struct ejit_func *f, struct ejit_reloc r, struct ejit_label l) vect_at(struct ejit_insn, f->insns, r.insn) = i; } -void ejit_calli(struct ejit_func *s, struct ejit_func *f, size_t argc, const struct ejit_operand args[argc]) +void ejit_calli(struct ejit_func *s, struct ejit_func *f, size_t argc, + const struct ejit_operand args[argc]) { for (size_t i = 0; i < argc; ++i) { switch (args[i].kind) { - case EJIT_OPERAND_GPR: emit_insn_r(s, ARG, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_FPR: emit_insn_r(s, ARG_F, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_IMM: emit_insn_i(s, ARG_I, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_FLT: emit_insn_f(s, ARG_FI, i, args[i].type, args[i].d); break; + case EJIT_OPERAND_GPR: emit_insn_r(s, ARG, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_FPR: emit_insn_r(s, ARG_F, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_IMM: emit_insn_i(s, ARG_I, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_FLT: emit_insn_f(s, ARG_FI, i, args[i].type, + args[i].d); break; default: abort(); } } @@ -124,14 +137,19 @@ void ejit_calli(struct ejit_func *s, struct ejit_func *f, size_t argc, const str emit_insn_p(s, CALLI, 0, 0, f); } -void ejit_escapei(struct ejit_func *s, ejit_escape_t f, size_t argc, const struct ejit_operand args[argc]) +void ejit_escapei(struct ejit_func *s, ejit_escape_t f, size_t argc, + const struct ejit_operand args[argc]) { for (size_t i = 0; i < argc; ++i) { switch (args[i].kind) { - case EJIT_OPERAND_GPR: emit_insn_r(s, ARG, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_FPR: emit_insn_r(s, ARG_F, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_IMM: emit_insn_r(s, ARG_I, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_FLT: emit_insn_f(s, ARG_FI, i, args[i].type, args[i].d); break; + case EJIT_OPERAND_GPR: emit_insn_r(s, ARG, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_FPR: emit_insn_r(s, ARG_F, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_IMM: emit_insn_r(s, ARG_I, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_FLT: emit_insn_f(s, ARG_FI, i, args[i].type, + args[i].d); break; default: abort(); } } @@ -139,14 +157,19 @@ void ejit_escapei(struct ejit_func *s, ejit_escape_t f, size_t argc, const struc emit_insn_p(s, ESCAPEI, 0, 0, f); } -void ejit_escapei_f(struct ejit_func *s, ejit_escape_f_t f, size_t argc, const struct ejit_operand args[argc]) +void ejit_escapei_f(struct ejit_func *s, ejit_escape_f_t f, size_t argc, + const struct ejit_operand args[argc]) { for (size_t i = 0; i < argc; ++i) { switch (args[i].kind) { - case EJIT_OPERAND_GPR: emit_insn_r(s, ARG, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_FPR: emit_insn_r(s, ARG_F, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_IMM: emit_insn_r(s, ARG_I, i, args[i].type, args[i].r); break; - case EJIT_OPERAND_FLT: emit_insn_f(s, ARG_FI, i, args[i].type, args[i].d); break; + case EJIT_OPERAND_GPR: emit_insn_r(s, ARG, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_FPR: emit_insn_r(s, ARG_F, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_IMM: emit_insn_r(s, ARG_I, i, args[i].type, + args[i].r); break; + case EJIT_OPERAND_FLT: emit_insn_f(s, ARG_FI, i, args[i].type, + args[i].d); break; default: abort(); } } @@ -159,72 +182,87 @@ void ejit_retval(struct ejit_func *s, struct ejit_gpr r0) emit_insn_i(s, RETVAL, r0.r, 0, 0); } -void ejit_stxi_64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o) +void ejit_stxi_64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o) { emit_insn_i(s, STXI64, r0.r, r1.r, o); } -void ejit_ldxi_u64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o) +void ejit_ldxi_u64(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o) { emit_insn_i(s, LDXIU64, r0.r, r1.r, o); } -void ejit_ret(struct ejit_func *s, struct ejit_gpr r0) +void ejit_retr(struct ejit_func *s, struct ejit_gpr r0) { - emit_insn_r(s, RET, r0.r, 0, 0); + emit_insn_r(s, RETR, r0.r, 0, 0); } -void ejit_ret_f(struct ejit_func *s, struct ejit_fpr r0) +void ejit_retr_f(struct ejit_func *s, struct ejit_fpr r0) { - emit_insn_r(s, RET_F, r0.f, 0, 0); + emit_insn_r(s, RETR_F, r0.f, 0, 0); } -void ejit_ret_i(struct ejit_func *s, long i) +void ejit_reti(struct ejit_func *s, long i) { - emit_insn_i(s, RET_I, 0, 0, i); + emit_insn_i(s, RETI, 0, 0, i); } -void ejit_ret_fi(struct ejit_func *s, double f) +void ejit_reti_f(struct ejit_func *s, double f) { - emit_insn_f(s, RET_FI, 0, 0, f); + emit_insn_f(s, RETI_F, 0, 0, f); } -void ejit_addr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2) +void ejit_addr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2) { emit_insn_r(s, ADDR, r0.r, r1.r, r2.r); } -void ejit_addr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, struct ejit_fpr r2) +void ejit_addr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, + struct ejit_fpr r2) { emit_insn_r(s, ADDR_F, r0.f, r1.f, r2.f); } -void ejit_addi(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o) +void ejit_addi(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o) { emit_insn_i(s, ADDI, r0.r, r1.r, o); } -void ejit_subr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2) +void ejit_absr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1) +{ + emit_insn_i(s, ABSR_F, r0.f, r1.f, 0); +} + +void ejit_subr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2) { emit_insn_r(s, SUBR, r0.r, r1.r, r2.r); } -void ejit_subr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, struct ejit_fpr r2) +void ejit_subr_f(struct ejit_func *s, struct ejit_fpr r0, struct ejit_fpr r1, + struct ejit_fpr r2) { emit_insn_r(s, SUBR_F, r0.f, r1.f, r2.f); } -void ejit_subi(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, long o) +void ejit_subi(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + long o) { emit_insn_i(s, SUBI, r0.r, r1.r, o); } -void ejit_mulr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2) +void ejit_mulr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2) { emit_insn_r(s, MULR, r0.r, r1.r, r2.r); } -void ejit_divr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2) +void ejit_divr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2) { emit_insn_r(s, DIVR, r0.r, r1.r, r2.r); } @@ -247,17 +285,20 @@ void ejit_movr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1) emit_insn_i(s, MOVR, r0.r, r1.r, 0); } -void ejit_eqr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2) +void ejit_eqr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2) { emit_insn_r(s, EQR, r0.r, r1.r, r2.r); } -void ejit_ltr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, struct ejit_gpr r2) +void ejit_ltr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1, + struct ejit_gpr r2) { emit_insn_r(s, LTR, r0.r, r1.r, r2.r); } -struct ejit_reloc ejit_bltr(struct ejit_func *s, struct ejit_gpr r0, struct ejit_gpr r1) +struct ejit_reloc ejit_bltr(struct ejit_func *s, struct ejit_gpr r0, + struct ejit_gpr r1) { size_t addr = vec_len(&s->insns); emit_insn_r(s, BLTR, 0, r0.r, r1.r); @@ -296,7 +337,7 @@ static struct interp_state create_interp_state() { struct interp_state state; state.gprs = vec_create(sizeof(long)); - state.gprs = vec_create(sizeof(double)); + state.fprs = vec_create(sizeof(double)); state.args = vec_create(sizeof(struct ejit_arg)); return state; } @@ -308,7 +349,8 @@ static void destroy_interp_state(struct interp_state state) vec_destroy(&state.args); } -long ejit_run_interp(struct ejit_func *f, size_t argc, struct ejit_arg args[argc], struct interp_state *state) +long ejit_run_interp(struct ejit_func *f, size_t argc, + struct ejit_arg args[argc], struct interp_state *state) { assert(f->gpr && "trying to run a function that hasn't been compiled"); assert(f->rtype == EJIT_VOID || ejit_int_type(f->rtype)); @@ -331,7 +373,8 @@ long ejit_run_func(struct ejit_func *f, size_t argc, struct ejit_arg args[argc]) return r; } -double ejit_run_func_f(struct ejit_func *f, size_t argc, struct ejit_arg args[argc]) +double ejit_run_func_f(struct ejit_func *f, size_t argc, + struct ejit_arg args[argc]) { assert(f->fpr && "trying to run a function that hasn't been compiled"); assert(ejit_float_type(f->rtype)); diff --git a/src/interp.c b/src/interp.c index ed7e59c..3395537 100644 --- a/src/interp.c +++ b/src/interp.c @@ -1,8 +1,12 @@ #include <ejit/ejit.h> +#include <math.h> #include "common.h" -union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg args[argc], struct interp_state *state, bool run, void ***labels_wb) +union interp_ret ejit_interp(struct ejit_func *f, size_t argc, + struct ejit_arg args[argc], + struct interp_state *state, bool run, + void ***labels_wb) { static void *labels[OPCODE_COUNT] = { [MOVI] = &&MOVI, @@ -13,6 +17,8 @@ union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg a [ADDR_F] = &&ADDR_F, [ADDI] = &&ADDI, + [ABSR_F] = &&ABSR_F, + [SUBR] = &&SUBR, [SUBR_F] = &&SUBR_F, [SUBI] = &&SUBI, @@ -33,10 +39,10 @@ union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg a [JMP] = &&JMP, - [RET] = &&RET, - [RET_I] = &&RET_I, - [RET_F] = &&RET_F, - [RET_FI] = &&RET_FI, + [RETR] = &&RETR, + [RETI] = &&RETI, + [RETR_F] = &&RETR_F, + [RETI_F] = &&RETI_F, [RETVAL] = &&RETVAL, @@ -76,10 +82,10 @@ union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg a /* retval is kind of an unfortunate extra bit of state to keep track of, * but having call and return value separated is pretty convenient for * void calls so I guess I don't mind? */ - long retval = 0; double retval_f = 0.; + int64_t retval = 0; double retval_f = 0.; size_t pc = 0; -#define DO(x) x: { struct ejit_insn i = insns[pc]; (void)i; +#define DO(x) x : { struct ejit_insn i = insns[pc]; (void)i; #define JUMP(a) goto *insns[pc = a].addr; #define DISPATCH() } goto *insns[++pc].addr; @@ -116,6 +122,10 @@ union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg a gpr[i.r0] = gpr[i.r1] + i.o; DISPATCH(); + DO(ABSR_F); + fpr[i.r0] = fabs(fpr[i.r1]); + DISPATCH(); + DO(SUBR); gpr[i.r0] = gpr[i.r1] - gpr[i.r2]; DISPATCH(); @@ -187,7 +197,7 @@ union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg a DISPATCH(); DO(PARAM); - gpr[i.r2] = args[i.r0].l; + gpr[i.r2] = args[i.r0].u64; DISPATCH(); DO(PARAM_F); @@ -217,7 +227,8 @@ union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg a DO(CALLI); struct ejit_func *f = i.p; size_t argc = vec_len(&state->args) - prev_argc; - struct ejit_arg *args = ((struct ejit_arg *)state->args.buf) + prev_argc; + struct ejit_arg *args = ((struct ejit_arg *)state->args.buf) + + prev_argc; retval = ejit_run_interp(f, argc, args, state); @@ -232,7 +243,8 @@ union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg a DO(ESCAPEI); ejit_escape_t f = i.p; size_t argc = vec_len(&state->args) - prev_argc; - struct ejit_arg *args = ((struct ejit_arg *)state->args.buf) + prev_argc; + struct ejit_arg *args = ((struct ejit_arg *)state->args.buf) + + prev_argc; retval = f(argc, args); @@ -241,22 +253,22 @@ union interp_ret ejit_interp(struct ejit_func *f, size_t argc, struct ejit_arg a /* dispatch is technically unnecessary for returns, but keep it for * symmetry */ - DO(RET); + DO(RETR); retval = gpr[i.r0]; goto out_int; DISPATCH(); - DO(RET_I); + DO(RETI); retval = i.o; goto out_int; DISPATCH(); - DO(RET_F); + DO(RETR_F); retval_f = fpr[i.r0]; goto out_float; DISPATCH(); - DO(RET_FI); + DO(RETI_F); retval_f = i.d; goto out_float; DISPATCH(); diff --git a/tests/absr_f.c b/tests/absr_f.c new file mode 100644 index 0000000..cc070b6 --- /dev/null +++ b/tests/absr_f.c @@ -0,0 +1,19 @@ +#include <ejit/ejit.h> +#include <assert.h> +#include "do_jit.h" + +int main() +{ + struct ejit_operand operands[1] = { + EJIT_OPERAND_FPR(0, EJIT_DOUBLE) + }; + struct ejit_func *f = ejit_create_func(EJIT_DOUBLE, 1, operands); + ejit_absr_f(f, EJIT_FPR(0), EJIT_FPR(0)); + ejit_retr_f(f, EJIT_FPR(0)); + ejit_select_compile_func(f, 0, 1, do_jit); + + assert(ejit_run_func_f_1(f, ejit_double(0.0)) == 0.0); + assert(ejit_run_func_f_1(f, ejit_double(-0.0)) == 0.0); + assert(ejit_run_func_f_1(f, ejit_double(0.5)) == 0.5); + assert(ejit_run_func_f_1(f, ejit_double(-0.5)) == 0.5); +} diff --git a/tests/addi.c b/tests/addi.c new file mode 100644 index 0000000..27cdfab --- /dev/null +++ b/tests/addi.c @@ -0,0 +1,25 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_addi(j, JIT_R0, JIT_R0, 69); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + int (*f)(int) = ret; + ASSERT(f(42) == 111); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/addr.c b/tests/addr.c new file mode 100644 index 0000000..f524aa6 --- /dev/null +++ b/tests/addr.c @@ -0,0 +1,23 @@ +#include <ejit/ejit.h> + +int +main (int argc, char *argv[]) +{ + struct ejit_operand operands[2] = { + EJIT_OPERAND_GPR(0, EJIT_LONG), + EJIT_OPERAND_GPR(1, EJIT_LONG) + }; + + struct ejit_func *f = ejit_create_func(EJIT_LONG, 2, operands); + + ejit_addr(j, EJIT_GPR(0), EJIT_GPR(0), EJIT_GPR(1)); + ejit_retr(j, EJIT_GPR(0)); + + ejit_compile(f); + + struct ejit_arg args[2] = { + EJIT_ARG(42, EJIT_LONG), + EJIT_ARG(69, EJIT_LONG) + }; + ASSERT(ejit_run_func(f, 2, args) == 111); +} diff --git a/tests/addr_d.c b/tests/addr_d.c new file mode 100644 index 0000000..2aa9002 --- /dev/null +++ b/tests/addr_d.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_addr_d(j, JIT_F0, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + double (*f)(double, double) = ret; + ASSERT(f(42., 69.) == 111.); + ASSERT(f(42.5, 69.5) == 112.); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/addr_f.c b/tests/addr_f.c new file mode 100644 index 0000000..d27ce8b --- /dev/null +++ b/tests/addr_f.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_addr_f(j, JIT_F0, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + float (*f)(float, float) = ret; + ASSERT(f(42.f, 69.f) == 111.f); + ASSERT(f(42.5f, 69.5f) == 112.f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/addx.c b/tests/addx.c new file mode 100644 index 0000000..94b2f19 --- /dev/null +++ b/tests/addx.c @@ -0,0 +1,63 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_movi(j, JIT_R2, 0); + jit_addcr(j, JIT_R0, JIT_R0, JIT_R1); + jit_addxi(j, JIT_R2, JIT_R2, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R2); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0, 0) == 0); + +#if EJIT_WORDSIZE == 32 + /* carry */ + ASSERT(f(0xffffffff, 0xffffffff) == 1); + /* overflow */ + ASSERT(f(0x7fffffff, 1) == 0); + /* overflow */ + ASSERT(f(0x7fffffff, 0x7fffffff) == 0); + /* carry */ + ASSERT(f(0x7fffffff, 0x80000000) == 0); + /* carry+overflow */ + ASSERT(f(0x80000000, 0x80000000) == 1); +#else + /* nothing */ + ASSERT(f(0xffffffff, 0xffffffff) == 0); + /* nothing */ + ASSERT(f(0x7fffffff, 1) == 0); + /* nothing */ + ASSERT(f(0x7fffffff, 0x7fffffff) == 0); + /* nothing */ + ASSERT(f(0x7fffffff, 0x80000000) == 0); + /* nothing */ + ASSERT(f(0x80000000, 0x80000000) == 0); + /* carry */ + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 1); + /* overflow */ + ASSERT(f(0x7fffffffffffffff, 1) == 0); + /* overflow */ + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == 0); + /* overflow */ + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0); + /* carry+overflow */ + ASSERT(f(0x8000000000000000, 0x8000000000000000) == 1); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/andi.c b/tests/andi.c new file mode 100644 index 0000000..fa3551b --- /dev/null +++ b/tests/andi.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_andi(j, JIT_R0, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t) = ret; + + ASSERT(f(0x7fffffff) == 1); + ASSERT(f(0x80000000) == 0); +#if EJIT_WORDSIZE == 64 + ASSERT(f(0x7fffffffffffffff) == 1); + ASSERT(f(0x8000000000000000) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/andr.c b/tests/andr.c new file mode 100644 index 0000000..c4aaeaf --- /dev/null +++ b/tests/andr.c @@ -0,0 +1,48 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_andr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7fffffff, 1) == 1); + ASSERT(f(1, 0x7fffffff) == 1); + ASSERT(f(0x80000000, 1) == 0); + ASSERT(f(1, 0x80000000) == 0); + ASSERT(f(0x7fffffff, 0x80000000) == 0); + ASSERT(f(0x80000000, 0x7fffffff) == 0); + ASSERT(f(0x7fffffff, 0xffffffff) == 0x7fffffff); + ASSERT(f(0xffffffff, 0x7fffffff) == 0x7fffffff); + ASSERT(f(0xffffffff, 0xffffffff) == 0xffffffff); + ASSERT(f(0x7fffffff, 0) == 0); + ASSERT(f(0, 0x7fffffff) == 0); +#if EJIT_WORDSIZE == 64 + ASSERT(f(0x7fffffffffffffff, 1) == 1); + ASSERT(f(1, 0x7fffffffffffffff) == 1); + ASSERT(f(0x8000000000000000, 1) == 0); + ASSERT(f(1, 0x8000000000000000) == 0); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 0); + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == 0x7fffffffffffffff); + ASSERT(f(0xffffffffffffffff, 0x7fffffffffffffff) == 0x7fffffffffffffff); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 0xffffffffffffffff); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/beqi.c b/tests/beqi.c new file mode 100644 index 0000000..b366a8e --- /dev/null +++ b/tests/beqi.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_beqi(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 1); + ASSERT(f(1) == 0); + ASSERT(f(-1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/beqr.c b/tests/beqr.c new file mode 100644 index 0000000..0cc4476 --- /dev/null +++ b/tests/beqr.c @@ -0,0 +1,32 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_beqr(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 0); + ASSERT(f(1, 1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/beqr_d.c b/tests/beqr_d.c new file mode 100644 index 0000000..8d61e6e --- /dev/null +++ b/tests/beqr_d.c @@ -0,0 +1,35 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_beqr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 0); + ASSERT(f(1, 1) == 1); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/beqr_f.c b/tests/beqr_f.c new file mode 100644 index 0000000..7463ed6 --- /dev/null +++ b/tests/beqr_f.c @@ -0,0 +1,35 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_beqr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 0); + ASSERT(f(1, 1) == 1); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bgei.c b/tests/bgei.c new file mode 100644 index 0000000..e0b8e1a --- /dev/null +++ b/tests/bgei.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bgei(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 1); + ASSERT(f(1) == 1); + ASSERT(f(-1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bgei_u.c b/tests/bgei_u.c new file mode 100644 index 0000000..d2997ff --- /dev/null +++ b/tests/bgei_u.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bgei_u(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 1); + ASSERT(f(1) == 1); + ASSERT(f(-1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bger.c b/tests/bger.c new file mode 100644 index 0000000..651a9e8 --- /dev/null +++ b/tests/bger.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bger(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bger_d.c b/tests/bger_d.c new file mode 100644 index 0000000..6a9cea3 --- /dev/null +++ b/tests/bger_d.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bger_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bger_f.c b/tests/bger_f.c new file mode 100644 index 0000000..7c4974f --- /dev/null +++ b/tests/bger_f.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bger_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bger_u.c b/tests/bger_u.c new file mode 100644 index 0000000..05038d5 --- /dev/null +++ b/tests/bger_u.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bger_u(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bgti.c b/tests/bgti.c new file mode 100644 index 0000000..79ea2d4 --- /dev/null +++ b/tests/bgti.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bgti(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(-1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bgti_u.c b/tests/bgti_u.c new file mode 100644 index 0000000..69325df --- /dev/null +++ b/tests/bgti_u.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bgti_u(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(-1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bgtr.c b/tests/bgtr.c new file mode 100644 index 0000000..c6c1fda --- /dev/null +++ b/tests/bgtr.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bgtr(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bgtr_d.c b/tests/bgtr_d.c new file mode 100644 index 0000000..aa9b23a --- /dev/null +++ b/tests/bgtr_d.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bgtr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bgtr_f.c b/tests/bgtr_f.c new file mode 100644 index 0000000..6245644 --- /dev/null +++ b/tests/bgtr_f.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bgtr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bgtr_u.c b/tests/bgtr_u.c new file mode 100644 index 0000000..53373f0 --- /dev/null +++ b/tests/bgtr_u.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bgtr_u(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/blei.c b/tests/blei.c new file mode 100644 index 0000000..46ce461 --- /dev/null +++ b/tests/blei.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_blei(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 1); + ASSERT(f(1) == 0); + ASSERT(f(-1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/blei_u.c b/tests/blei_u.c new file mode 100644 index 0000000..f8ed875 --- /dev/null +++ b/tests/blei_u.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_blei_u(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 1); + ASSERT(f(1) == 0); + ASSERT(f(-1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bler.c b/tests/bler.c new file mode 100644 index 0000000..cf835e0 --- /dev/null +++ b/tests/bler.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bler(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bler_d.c b/tests/bler_d.c new file mode 100644 index 0000000..f67df1a --- /dev/null +++ b/tests/bler_d.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bler_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bler_f.c b/tests/bler_f.c new file mode 100644 index 0000000..ff18190 --- /dev/null +++ b/tests/bler_f.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bler_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bler_u.c b/tests/bler_u.c new file mode 100644 index 0000000..87ac322 --- /dev/null +++ b/tests/bler_u.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bler_u(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bltgtr_d.c b/tests/bltgtr_d.c new file mode 100644 index 0000000..d549d08 --- /dev/null +++ b/tests/bltgtr_d.c @@ -0,0 +1,36 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bltgtr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 1); + ASSERT(f(1, 1) == 0); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); + ASSERT(f(0.0/0.0, 0.0/0.0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bltgtr_f.c b/tests/bltgtr_f.c new file mode 100644 index 0000000..6f4e6ac --- /dev/null +++ b/tests/bltgtr_f.c @@ -0,0 +1,36 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bltgtr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 1); + ASSERT(f(1, 1) == 0); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); + ASSERT(f(0.0/0.0, 0.0/0.0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/blti.c b/tests/blti.c new file mode 100644 index 0000000..dd9c907 --- /dev/null +++ b/tests/blti.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_blti(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 0); + ASSERT(f(-1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/blti_u.c b/tests/blti_u.c new file mode 100644 index 0000000..13582c3 --- /dev/null +++ b/tests/blti_u.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_blti_u(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 0); + ASSERT(f(-1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bltr.c b/tests/bltr.c new file mode 100644 index 0000000..b5c7976 --- /dev/null +++ b/tests/bltr.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bltr(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bltr_d.c b/tests/bltr_d.c new file mode 100644 index 0000000..d19be55 --- /dev/null +++ b/tests/bltr_d.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bltr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bltr_f.c b/tests/bltr_f.c new file mode 100644 index 0000000..35eb77b --- /dev/null +++ b/tests/bltr_f.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bltr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bltr_u.c b/tests/bltr_u.c new file mode 100644 index 0000000..d0a9f39 --- /dev/null +++ b/tests/bltr_u.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bltr_u(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bmci.c b/tests/bmci.c new file mode 100644 index 0000000..e8310e0 --- /dev/null +++ b/tests/bmci.c @@ -0,0 +1,29 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bmci(j, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 1); + ASSERT(f(1) == 0); + ASSERT(f(-1) == 0); + ASSERT(f(2) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bmcr.c b/tests/bmcr.c new file mode 100644 index 0000000..3dbc98a --- /dev/null +++ b/tests/bmcr.c @@ -0,0 +1,35 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bmcr(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 1); + ASSERT(f(1, 1) == 0); + ASSERT(f(1, -1) == 0); + ASSERT(f(-1, 1) == 0); + ASSERT(f(-1, -1) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bmsi.c b/tests/bmsi.c new file mode 100644 index 0000000..aaad68a --- /dev/null +++ b/tests/bmsi.c @@ -0,0 +1,29 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bmsi(j, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(-1) == 1); + ASSERT(f(2) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bmsr.c b/tests/bmsr.c new file mode 100644 index 0000000..d471fdf --- /dev/null +++ b/tests/bmsr.c @@ -0,0 +1,35 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bmsr(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 0); + ASSERT(f(1, 1) == 1); + ASSERT(f(1, -1) == 1); + ASSERT(f(-1, 1) == 1); + ASSERT(f(-1, -1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bnei.c b/tests/bnei.c new file mode 100644 index 0000000..3e9e376 --- /dev/null +++ b/tests/bnei.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bnei(j, JIT_R0, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(-1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bner.c b/tests/bner.c new file mode 100644 index 0000000..d9b2305 --- /dev/null +++ b/tests/bner.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bner(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bner_d.c b/tests/bner_d.c new file mode 100644 index 0000000..b45deed --- /dev/null +++ b/tests/bner_d.c @@ -0,0 +1,36 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bner_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 1); + ASSERT(f(1, 1) == 0); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); + ASSERT(f(0.0/0.0, 0.0/0.0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bner_f.c b/tests/bner_f.c new file mode 100644 index 0000000..f57bbc4 --- /dev/null +++ b/tests/bner_f.c @@ -0,0 +1,36 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bner_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 1); + ASSERT(f(1, 1) == 0); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); + ASSERT(f(0.0/0.0, 0.0/0.0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/boaddi.c b/tests/boaddi.c new file mode 100644 index 0000000..e0fed81 --- /dev/null +++ b/tests/boaddi.c @@ -0,0 +1,41 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_boaddi(j, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_patch_here(j, r); + jit_movi(j, JIT_R0, overflowed); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(-1) == 0); + ASSERT(f(0) == 1); + ASSERT(f(1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x7fffffff) == overflowed); + ASSERT(f(0x80000000) == 0x80000001); + ASSERT(f(0xffffffff) == 0); +#else + ASSERT(f(0x7fffffffffffffff) == overflowed); + ASSERT(f(0x8000000000000000) == 0x8000000000000001); + ASSERT(f(0xffffffffffffffff) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/boaddi_u.c b/tests/boaddi_u.c new file mode 100644 index 0000000..b6ffa89 --- /dev/null +++ b/tests/boaddi_u.c @@ -0,0 +1,41 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_boaddi_u(j, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_patch_here(j, r); + jit_movi(j, JIT_R0, overflowed); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(-1) == overflowed); + ASSERT(f(0) == 1); + ASSERT(f(1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x7fffffff) == 0x80000000); + ASSERT(f(0x80000000) == 0x80000001); + ASSERT(f(0xffffffff) == overflowed); +#else + ASSERT(f(0x7fffffffffffffff) == 0x8000000000000000); + ASSERT(f(0x8000000000000000) == 0x8000000000000001); + ASSERT(f(0xffffffffffffffff) == overflowed); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/boaddr.c b/tests/boaddr.c new file mode 100644 index 0000000..69668f7 --- /dev/null +++ b/tests/boaddr.c @@ -0,0 +1,51 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_boaddr(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_patch_here(j, r); + jit_movi(j, JIT_R0, overflowed); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(1, 1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 0xffffffff) == -2); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0x7fffffff, 1) == overflowed); + ASSERT(f(0x7fffffff, 0x7fffffff) == overflowed); + ASSERT(f(0x7fffffff, 0x80000000) == -1); + ASSERT(f(0x80000000, 0x80000000) == overflowed); +#else + ASSERT(f(0xffffffff, 0xffffffff) == 0xffffffffull + 0xffffffffull); + ASSERT(f(0x7fffffff, 1) == 0x80000000); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0x7fffffffull + 0x7fffffffull); + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + ASSERT(f(0x80000000, 0x80000000) == 0x100000000); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == -2); + ASSERT(f(0x7fffffffffffffff, 1) == overflowed); + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == overflowed); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == -1); + ASSERT(f(0x8000000000000000, 0x8000000000000000) == overflowed); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/boaddr_u.c b/tests/boaddr_u.c new file mode 100644 index 0000000..1483471 --- /dev/null +++ b/tests/boaddr_u.c @@ -0,0 +1,51 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_boaddr_u(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_patch_here(j, r); + jit_movi(j, JIT_R0, overflowed); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(1, 1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 0xffffffff) == overflowed); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0x7fffffff, 1) == 0x80000000); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0x7fffffffu + 0x7fffffffu); + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + ASSERT(f(0x80000000, 0x80000000) == overflowed); +#else + ASSERT(f(0xffffffff, 0xffffffff) == 0xffffffffull + 0xffffffffull); + ASSERT(f(0x7fffffff, 1) == 0x80000000); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0x7fffffffull + 0x7fffffffull); + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + ASSERT(f(0x80000000, 0x80000000) == 0x100000000); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == overflowed); + ASSERT(f(0x7fffffffffffffff, 1) == 0x8000000000000000); + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == -2); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == -1); + ASSERT(f(0x8000000000000000, 0x8000000000000000) == overflowed); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bordr_d.c b/tests/bordr_d.c new file mode 100644 index 0000000..ca44db7 --- /dev/null +++ b/tests/bordr_d.c @@ -0,0 +1,36 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bordr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 1); + ASSERT(f(1, 1) == 1); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); + ASSERT(f(0.0/0.0, 0.0/0.0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bordr_f.c b/tests/bordr_f.c new file mode 100644 index 0000000..a3db62e --- /dev/null +++ b/tests/bordr_f.c @@ -0,0 +1,36 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bordr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 1); + ASSERT(f(1, 1) == 1); + + ASSERT(f(0, 0.0/0.0) == 0); + ASSERT(f(0.0/0.0, 0) == 0); + ASSERT(f(0.0/0.0, 0.0/0.0) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bosubi.c b/tests/bosubi.c new file mode 100644 index 0000000..b1bbc77 --- /dev/null +++ b/tests/bosubi.c @@ -0,0 +1,41 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bosubi(j, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_patch_here(j, r); + jit_movi(j, JIT_R0, overflowed); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(-1) == -2); + ASSERT(f(0) == -1); + ASSERT(f(1) == 0); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x7fffffff) == 0x7ffffffe); + ASSERT(f(0x80000000) == overflowed); + ASSERT(f(0x80000001) == 0x80000000); +#else + ASSERT(f(0x7fffffffffffffff) == 0x7ffffffffffffffe); + ASSERT(f(0x8000000000000000) == overflowed); + ASSERT(f(0x8000000000000001) == 0x8000000000000000); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bosubi_u.c b/tests/bosubi_u.c new file mode 100644 index 0000000..c04b971 --- /dev/null +++ b/tests/bosubi_u.c @@ -0,0 +1,37 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bosubi_u(j, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_patch_here(j, r); + jit_movi(j, JIT_R0, overflowed); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(-1) == -2); + ASSERT(f(0) == overflowed); + ASSERT(f(1) == 0); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x80000000) == 0x7fffffff); +#else + ASSERT(f(0x8000000000000000) == 0x7fffffffffffffff); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bosubr.c b/tests/bosubr.c new file mode 100644 index 0000000..01ea9fe --- /dev/null +++ b/tests/bosubr.c @@ -0,0 +1,48 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bosubr(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_patch_here(j, r); + jit_movi(j, JIT_R0, overflowed); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == -1); + ASSERT(f(1, 1) == 0); + ASSERT(f(1, -1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 0xffffffff) == 0); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0x7fffffff, 1) == 0x7ffffffe); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0); + ASSERT(f(0x80000000, 0x7fffffff) == overflowed); + ASSERT(f(0x7fffffff, 0x80000000) == overflowed); + ASSERT(f(0x80000000, 0x80000000) == 0); +#else + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == 0); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == overflowed); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == overflowed); + ASSERT(f(0x8000000000000000, 0x8000000000000000) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bosubr_u.c b/tests/bosubr_u.c new file mode 100644 index 0000000..0f6d5b7 --- /dev/null +++ b/tests/bosubr_u.c @@ -0,0 +1,47 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bosubr_u(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_patch_here(j, r); + jit_movi(j, JIT_R0, overflowed); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(1, 1) == 0); + ASSERT(f(0, 1) == overflowed); + ASSERT(f(1, 0) == 1); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 0xffffffff) == 0); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0x7fffffff, 1) == 0x7ffffffe); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0); + ASSERT(f(0x7fffffff, 0x80000000) == overflowed); + ASSERT(f(0x80000000, 0x80000000) == 0); +#else + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 0); + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == 0); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == overflowed); + ASSERT(f(0x8000000000000000, 0x8000000000000000) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bswapr_ui.c b/tests/bswapr_ui.c new file mode 100644 index 0000000..9c4b42b --- /dev/null +++ b/tests/bswapr_ui.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_bswapr_ui(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(0x12345678) == 0x78563412); +#if EJIT_WORDSIZE > 32 + ASSERT(f(0xff12345678) == 0x78563412); + ASSERT(f(0xff00000000) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bswapr_ul.c b/tests/bswapr_ul.c new file mode 100644 index 0000000..03136d8 --- /dev/null +++ b/tests/bswapr_ul.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_bswapr_ul(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(0x12345678) == 0x7856341200000000); + ASSERT(f(0xff12345678) == 0x78563412ff000000); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bswapr_us.c b/tests/bswapr_us.c new file mode 100644 index 0000000..a2c223b --- /dev/null +++ b/tests/bswapr_us.c @@ -0,0 +1,24 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_bswapr_us(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(0x12345678) == 0x7856); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/buneqr_d.c b/tests/buneqr_d.c new file mode 100644 index 0000000..e0ed6e9 --- /dev/null +++ b/tests/buneqr_d.c @@ -0,0 +1,35 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_buneqr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 0); + ASSERT(f(1, 1) == 1); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/buneqr_f.c b/tests/buneqr_f.c new file mode 100644 index 0000000..4277f86 --- /dev/null +++ b/tests/buneqr_f.c @@ -0,0 +1,35 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_buneqr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 0); + ASSERT(f(1, 1) == 1); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bunger_d.c b/tests/bunger_d.c new file mode 100644 index 0000000..e65f136 --- /dev/null +++ b/tests/bunger_d.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bunger_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bunger_f.c b/tests/bunger_f.c new file mode 100644 index 0000000..0fa542b --- /dev/null +++ b/tests/bunger_f.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bunger_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bungtr_d.c b/tests/bungtr_d.c new file mode 100644 index 0000000..0985696 --- /dev/null +++ b/tests/bungtr_d.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bungtr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bungtr_f.c b/tests/bungtr_f.c new file mode 100644 index 0000000..94a9170 --- /dev/null +++ b/tests/bungtr_f.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bungtr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 1); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 1); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bunler_d.c b/tests/bunler_d.c new file mode 100644 index 0000000..decd314 --- /dev/null +++ b/tests/bunler_d.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bunler_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bunler_f.c b/tests/bunler_f.c new file mode 100644 index 0000000..3b40f95 --- /dev/null +++ b/tests/bunler_f.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bunler_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 1); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bunltr_d.c b/tests/bunltr_d.c new file mode 100644 index 0000000..9ce8d63 --- /dev/null +++ b/tests/bunltr_d.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bunltr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bunltr_f.c b/tests/bunltr_f.c new file mode 100644 index 0000000..d86ce1a --- /dev/null +++ b/tests/bunltr_f.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bunltr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 1); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 1); + ASSERT(f(0, -1) == 0); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bunordr_d.c b/tests/bunordr_d.c new file mode 100644 index 0000000..13ed8a6 --- /dev/null +++ b/tests/bunordr_d.c @@ -0,0 +1,36 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_reloc_t r = jit_bunordr_d(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(double, double) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 0); + ASSERT(f(1, 1) == 0); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); + ASSERT(f(0.0/0.0, 0.0/0.0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bunordr_f.c b/tests/bunordr_f.c new file mode 100644 index 0000000..663d988 --- /dev/null +++ b/tests/bunordr_f.c @@ -0,0 +1,36 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_reloc_t r = jit_bunordr_f(j, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 0); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_reti(j, 1); + + jit_word_t (*f)(float, float) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == 0); + ASSERT(f(1, 0) == 0); + ASSERT(f(-1, 0) == 0); + ASSERT(f(0, -1) == 0); + ASSERT(f(1, 1) == 0); + + ASSERT(f(0, 0.0/0.0) == 1); + ASSERT(f(0.0/0.0, 0) == 1); + ASSERT(f(0.0/0.0, 0.0/0.0) == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bxaddi.c b/tests/bxaddi.c new file mode 100644 index 0000000..282b1f1 --- /dev/null +++ b/tests/bxaddi.c @@ -0,0 +1,39 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bxaddi(j, JIT_R0, 1); + jit_movi(j, JIT_R0, overflowed); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(-1) == 0); + ASSERT(f(0) == 1); + ASSERT(f(1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x7fffffff) == overflowed); + ASSERT(f(0x80000000) == 0x80000001); + ASSERT(f(0xffffffff) == 0); +#else + ASSERT(f(0x7fffffffffffffff) == overflowed); + ASSERT(f(0x8000000000000000) == 0x8000000000000001); + ASSERT(f(0xffffffffffffffff) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bxaddi_u.c b/tests/bxaddi_u.c new file mode 100644 index 0000000..dc3140e --- /dev/null +++ b/tests/bxaddi_u.c @@ -0,0 +1,39 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bxaddi_u(j, JIT_R0, 1); + jit_movi(j, JIT_R0, overflowed); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(-1) == overflowed); + ASSERT(f(0) == 1); + ASSERT(f(1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x7fffffff) == 0x80000000); + ASSERT(f(0x80000000) == 0x80000001); + ASSERT(f(0xffffffff) == overflowed); +#else + ASSERT(f(0x7fffffffffffffff) == 0x8000000000000000); + ASSERT(f(0x8000000000000000) == 0x8000000000000001); + ASSERT(f(0xffffffffffffffff) == overflowed); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bxaddr.c b/tests/bxaddr.c new file mode 100644 index 0000000..472906d --- /dev/null +++ b/tests/bxaddr.c @@ -0,0 +1,49 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bxaddr(j, JIT_R0, JIT_R1); + jit_movi(j, JIT_R0, overflowed); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(1, 1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 0xffffffff) == -2); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0x7fffffff, 1) == overflowed); + ASSERT(f(0x7fffffff, 0x7fffffff) == overflowed); + ASSERT(f(0x7fffffff, 0x80000000) == -1); + ASSERT(f(0x80000000, 0x80000000) == overflowed); +#else + ASSERT(f(0xffffffff, 0xffffffff) == 0xffffffffull + 0xffffffffull); + ASSERT(f(0x7fffffff, 1) == 0x80000000); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0x7fffffffull + 0x7fffffffull); + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + ASSERT(f(0x80000000, 0x80000000) == 0x100000000); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == -2); + ASSERT(f(0x7fffffffffffffff, 1) == overflowed); + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == overflowed); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == -1); + ASSERT(f(0x8000000000000000, 0x8000000000000000) == overflowed); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bxaddr_u.c b/tests/bxaddr_u.c new file mode 100644 index 0000000..d40e0ff --- /dev/null +++ b/tests/bxaddr_u.c @@ -0,0 +1,49 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bxaddr_u(j, JIT_R0, JIT_R1); + jit_movi(j, JIT_R0, overflowed); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(1, 1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 0xffffffff) == overflowed); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0x7fffffff, 1) == 0x80000000); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0x7fffffffu + 0x7fffffffu); + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + ASSERT(f(0x80000000, 0x80000000) == overflowed); +#else + ASSERT(f(0xffffffff, 0xffffffff) == 0xffffffffull + 0xffffffffull); + ASSERT(f(0x7fffffff, 1) == 0x80000000); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0x7fffffffull + 0x7fffffffull); + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + ASSERT(f(0x80000000, 0x80000000) == 0x100000000); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == overflowed); + ASSERT(f(0x7fffffffffffffff, 1) == 0x8000000000000000); + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == -2); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == -1); + ASSERT(f(0x8000000000000000, 0x8000000000000000) == overflowed); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bxsubi.c b/tests/bxsubi.c new file mode 100644 index 0000000..e6224bf --- /dev/null +++ b/tests/bxsubi.c @@ -0,0 +1,39 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bxsubi(j, JIT_R0, 1); + jit_movi(j, JIT_R0, overflowed); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(-1) == -2); + ASSERT(f(0) == -1); + ASSERT(f(1) == 0); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x7fffffff) == 0x7ffffffe); + ASSERT(f(0x80000000) == overflowed); + ASSERT(f(0x80000001) == 0x80000000); +#else + ASSERT(f(0x7fffffffffffffff) == 0x7ffffffffffffffe); + ASSERT(f(0x8000000000000000) == overflowed); + ASSERT(f(0x8000000000000001) == 0x8000000000000000); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bxsubi_u.c b/tests/bxsubi_u.c new file mode 100644 index 0000000..6c1d1f3 --- /dev/null +++ b/tests/bxsubi_u.c @@ -0,0 +1,35 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_bxsubi_u(j, JIT_R0, 1); + jit_movi(j, JIT_R0, overflowed); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(-1) == -2); + ASSERT(f(0) == overflowed); + ASSERT(f(1) == 0); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x80000000) == 0x7fffffff); +#else + ASSERT(f(0x8000000000000000) == 0x7fffffffffffffff); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bxsubr.c b/tests/bxsubr.c new file mode 100644 index 0000000..50f5dbb --- /dev/null +++ b/tests/bxsubr.c @@ -0,0 +1,46 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bxsubr(j, JIT_R0, JIT_R1); + jit_movi(j, JIT_R0, overflowed); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(0, 1) == -1); + ASSERT(f(1, 1) == 0); + ASSERT(f(1, -1) == 2); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 0xffffffff) == 0); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0x7fffffff, 1) == 0x7ffffffe); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0); + ASSERT(f(0x80000000, 0x7fffffff) == overflowed); + ASSERT(f(0x7fffffff, 0x80000000) == overflowed); + ASSERT(f(0x80000000, 0x80000000) == 0); +#else + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == 0); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == overflowed); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == overflowed); + ASSERT(f(0x8000000000000000, 0x8000000000000000) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/bxsubr_u.c b/tests/bxsubr_u.c new file mode 100644 index 0000000..771ea3c --- /dev/null +++ b/tests/bxsubr_u.c @@ -0,0 +1,45 @@ +#include "test.h" + +static const jit_word_t overflowed = 0xcabba9e5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_reloc_t r = jit_bxsubr_u(j, JIT_R0, JIT_R1); + jit_movi(j, JIT_R0, overflowed); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t, jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0, 0) == 0); + ASSERT(f(1, 1) == 0); + ASSERT(f(0, 1) == overflowed); + ASSERT(f(1, 0) == 1); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 0xffffffff) == 0); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0x7fffffff, 1) == 0x7ffffffe); + ASSERT(f(0x7fffffff, 0x7fffffff) == 0); + ASSERT(f(0x7fffffff, 0x80000000) == overflowed); + ASSERT(f(0x80000000, 0x80000000) == 0); +#else + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 0); + ASSERT(f(0x7fffffffffffffff, 0x7fffffffffffffff) == 0); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == overflowed); + ASSERT(f(0x8000000000000000, 0x8000000000000000) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/call_10.c b/tests/call_10.c new file mode 100644 index 0000000..533acda --- /dev/null +++ b/tests/call_10.c @@ -0,0 +1,64 @@ +#include "test.h" + +static int32_t f(int32_t a, int32_t b, int32_t c, int32_t d, int32_t e, + int32_t f, int32_t g, int32_t h, int32_t i, int32_t j) { + ASSERT(a == 0); + ASSERT(b == 1); + ASSERT(c == 2); + ASSERT(d == 3); + ASSERT(e == 4); + ASSERT(f == 5); + ASSERT(g == 6); + ASSERT(h == 7); + ASSERT(i == 8); + ASSERT(j == 9); + return 42; +} + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0)); + + jit_operand_t args[10] = { + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 0 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 1 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 2 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 3 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 4 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 5 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 6 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 7 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 8 * sizeof(int32_t)), + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, + 9 * sizeof(int32_t)) + }; + jit_calli(j, f, 10, args); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + size_t size = 0; + void* ret = jit_end(j, &size); + + int32_t (*f)(int32_t*) = ret; + + int32_t iargs[10] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 }; + ASSERT(f(iargs) == 42); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/call_double.c b/tests/call_double.c new file mode 100644 index 0000000..1138655 --- /dev/null +++ b/tests/call_double.c @@ -0,0 +1,38 @@ +#include "test.h" + +static double f(int32_t a, double b) { + return b + a; +} + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_operand_t args[2] = { + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, 0), + jit_operand_mem(JIT_OPERAND_ABI_DOUBLE, JIT_R1, 0) + }; + jit_calli(j, f, 2, args); + jit_retval_d(j, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + double (*f)(int32_t*, double*) = ret; + + double d = 22.0f; + int32_t i = 20; + ASSERT(f(&i, &d) == 42.0f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/call_float.c b/tests/call_float.c new file mode 100644 index 0000000..9534e4b --- /dev/null +++ b/tests/call_float.c @@ -0,0 +1,38 @@ +#include "test.h" + +static float f(int32_t a, float b) { + return b + a; +} + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_operand_t args[2] = { + jit_operand_mem(JIT_OPERAND_ABI_INT32, JIT_R0, 0), + jit_operand_mem(JIT_OPERAND_ABI_FLOAT, JIT_R1, 0) + }; + jit_calli(j, f, 2, args); + jit_retval_f(j, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + float (*f)(int32_t*, float*) = ret; + + float d = 22.0f; + int32_t i = 20; + ASSERT(f(&i, &d) == 42.0f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/callee_9.c b/tests/callee_9.c new file mode 100644 index 0000000..9d4e217 --- /dev/null +++ b/tests/callee_9.c @@ -0,0 +1,68 @@ +#include "test.h" + +struct args +{ + int8_t a; + int16_t b; + int32_t c; + jit_word_t d; + uint16_t e; + float f; + double g; + float h; +}; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 3, 0, 0); + + jit_operand_t args[9] = { + jit_operand_gpr(JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr(JIT_OPERAND_ABI_INT8, JIT_R1), + jit_operand_gpr(JIT_OPERAND_ABI_INT16, JIT_R2), + jit_operand_gpr(JIT_OPERAND_ABI_INT32, JIT_V0), + jit_operand_gpr(JIT_OPERAND_ABI_WORD, JIT_V1), + jit_operand_gpr(JIT_OPERAND_ABI_UINT16, JIT_V2), + jit_operand_fpr(JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr(JIT_OPERAND_ABI_DOUBLE, JIT_F1), + jit_operand_fpr(JIT_OPERAND_ABI_FLOAT, JIT_F2), + }; + jit_load_args(j, 9, args); + jit_stxi_c(j, offsetof(struct args, a), JIT_R0, JIT_R1); // a + jit_stxi_s(j, offsetof(struct args, b), JIT_R0, JIT_R2); // b + jit_stxi_i(j, offsetof(struct args, c), JIT_R0, JIT_V0); // c + jit_stxi(j, offsetof(struct args, d), JIT_R0, JIT_V1);// d + jit_stxi_s(j, offsetof(struct args, e), JIT_R0, JIT_V2); // e + jit_stxi_f(j, offsetof(struct args, f), JIT_R0, JIT_F0); // f + jit_stxi_d(j, offsetof(struct args, g), JIT_R0, JIT_F1); // g + jit_stxi_f(j, offsetof(struct args, h), JIT_R0, JIT_F2); // h + + jit_leave_jit_abi(j, 3, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + struct args* (*f)(struct args*, int8_t, int16_t, int32_t, jit_word_t, + uint16_t, float, double, float) = ret; + + struct args in = { 0, 1, 2, 3, 4, 5, 6, 7 }; + struct args out; + ASSERT(f(&out, in.a, in.b, in.c, in.d, in.e, in.f, in.g, in.h) == &out); + ASSERT(in.a == out.a); + ASSERT(in.b == out.b); + ASSERT(in.c == out.c); + ASSERT(in.d == out.d); + ASSERT(in.e == out.e); + ASSERT(in.f == out.f); + ASSERT(in.g == out.g); + ASSERT(in.h == out.h); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/cas_atomic.c b/tests/cas_atomic.c new file mode 100644 index 0000000..a382eae --- /dev/null +++ b/tests/cas_atomic.c @@ -0,0 +1,33 @@ +#include "test.h" + +static long data[] = { 0x12121212, 0x00000000, 0x34343434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_3(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_INT32, JIT_R1), + jit_operand_gpr (JIT_OPERAND_ABI_INT32, JIT_R2)); + + jit_cas_atomic(j, JIT_R0, JIT_R0, JIT_R1, JIT_R2); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, int32_t, int32_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34343434); + f(&data[1], 0, 0x0f0f0f0f); + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x0f0f0f0f); + ASSERT(data[2] == 0x34343434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/comr.c b/tests/comr.c new file mode 100644 index 0000000..3cfacac --- /dev/null +++ b/tests/comr.c @@ -0,0 +1,41 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_comr(j, JIT_R0, JIT_R0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0) == 0xffffffff); + ASSERT(f(1) == 0xfffffffe); + ASSERT(f(0xffffffff) == 0); + ASSERT(f(0x80000000) == 0x7fffffff); + ASSERT(f(0x7fffffff) == 0x80000000); + ASSERT(f(0x80000001) == 0x7ffffffe); +#else + ASSERT(f(0) == 0xffffffffffffffff); + ASSERT(f(1) == 0xfffffffffffffffe); + ASSERT(f(0xffffffff) == 0xffffffff00000000); + ASSERT(f(0x80000000) == 0xffffffff7fffffff); + ASSERT(f(0x7fffffff) == 0xffffffff80000000); + ASSERT(f(0x80000001) == 0xffffffff7ffffffe); + ASSERT(f(0xffffffffffffffff) == 0); + ASSERT(f(0x8000000000000000) == 0x7fffffffffffffff); + ASSERT(f(0x7fffffffffffffff) == 0x8000000000000000); + ASSERT(f(0x8000000000000001) == 0x7ffffffffffffffe); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/divr.c b/tests/divr.c new file mode 100644 index 0000000..c68fcaa --- /dev/null +++ b/tests/divr.c @@ -0,0 +1,60 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_divr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7fffffff, 1) == 0x7fffffff); + ASSERT(f(1, 0x7fffffff) == 0); + ASSERT(f(0x80000000, 1) == 0x80000000); + ASSERT(f(1, 0x80000000) == 0); + ASSERT(f(0x7fffffff, 2) == 0x3fffffff); + ASSERT(f(2, 0x7fffffff) == 0); + ASSERT(f(2, 0x80000000) == 0); + ASSERT(f(0x7fffffff, 0x80000000) == 0); + ASSERT(f(0, 0x7fffffff) == 0); + ASSERT(f(0xffffffff, 0xffffffff) == 1); +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x80000000, 2) == 0xc0000000); + ASSERT(f(0x80000000, 0x7fffffff) == 0xffffffff); + ASSERT(f(0x7fffffff, 0xffffffff) == 0x80000001); + ASSERT(f(0xffffffff, 0x7fffffff) == 0); +#else + ASSERT(f(0x80000000, 2) == 0x40000000); + ASSERT(f(0x80000000, 0x7fffffff) == 1); + ASSERT(f(0x7fffffff, 0xffffffff) == 0); + ASSERT(f(0xffffffff, 0x7fffffff) == 2); + ASSERT(f(0x7fffffffffffffff, 1) == 0x7fffffffffffffff); + ASSERT(f(1, 0x7fffffffffffffff) == 0); + ASSERT(f(0x8000000000000000, 1) == 0x8000000000000000); + ASSERT(f(1, 0x8000000000000000) == 0); + ASSERT(f(0x7fffffffffffffff, 2) == 0x3fffffffffffffff); + ASSERT(f(2, 0x7fffffffffffffff) == 0); + ASSERT(f(0x8000000000000000, 2) == 0xc000000000000000); + ASSERT(f(2, 0x8000000000000000) == 0); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 0xffffffffffffffff); + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == 0x8000000000000001); + ASSERT(f(0xffffffffffffffff, 0x7fffffffffffffff) == 0); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 1); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/divr_d.c b/tests/divr_d.c new file mode 100644 index 0000000..41dd6c3 --- /dev/null +++ b/tests/divr_d.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_divr_d(j, JIT_F0, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + double (*f)(double, double) = ret; + ASSERT(f(-0.5f, 0.5f) == -1.0f); + ASSERT(f(1.25f, 0.5f) == 2.5f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/divr_f.c b/tests/divr_f.c new file mode 100644 index 0000000..085ecff --- /dev/null +++ b/tests/divr_f.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_divr_f(j, JIT_F0, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + float (*f)(float, float) = ret; + ASSERT(f(-0.5f, 0.5f) == -1.0f); + ASSERT(f(1.25f, 0.5f) == 2.5f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/divr_u.c b/tests/divr_u.c new file mode 100644 index 0000000..65d0ae2 --- /dev/null +++ b/tests/divr_u.c @@ -0,0 +1,55 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_divr_u(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7fffffff, 1) == 0x7fffffff); + ASSERT(f(1, 0x7fffffff) == 0); + ASSERT(f(0x80000000, 1) == 0x80000000); + ASSERT(f(1, 0x80000000) == 0); + ASSERT(f(0x7fffffff, 2) == 0x3fffffff); + ASSERT(f(2, 0x7fffffff) == 0); + ASSERT(f(0x80000000, 2) == 0x40000000); + ASSERT(f(2, 0x80000000) == 0); + ASSERT(f(0x7fffffff, 0x80000000) == 0); + ASSERT(f(0x80000000, 0x7fffffff) == 1); + ASSERT(f(0, 0x7fffffff) == 0); + ASSERT(f(0x7fffffff, 0xffffffff) == 0); + ASSERT(f(0xffffffff, 0x7fffffff) == 2); + ASSERT(f(0xffffffff, 0xffffffff) == 1); +#if EJIT_WORDSIZE != 32 + ASSERT(f(0x7fffffffffffffff, 1) == 0x7fffffffffffffff); + ASSERT(f(1, 0x7fffffffffffffff) == 0); + ASSERT(f(0x8000000000000000, 1) == 0x8000000000000000); + ASSERT(f(1, 0x8000000000000000) == 0); + ASSERT(f(0x7fffffffffffffff, 2) == 0x3fffffffffffffff); + ASSERT(f(2, 0x7fffffffffffffff) == 0); + ASSERT(f(0x8000000000000000, 2) == 0x4000000000000000); + ASSERT(f(2, 0x8000000000000000) == 0); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 1); + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == 0); + ASSERT(f(0xffffffffffffffff, 0x7fffffffffffffff) == 2); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 1); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/do_jit.h b/tests/do_jit.h new file mode 100644 index 0000000..d0d4921 --- /dev/null +++ b/tests/do_jit.h @@ -0,0 +1,10 @@ +#ifndef DOJIT_H +#define DOJIT_H + +#ifdef FORCE_BCODE +static bool do_jit = false; +#else +static bool do_jit = true; +#endif + +#endif /* DOJIT_H */ diff --git a/tests/extr_c.c b/tests/extr_c.c new file mode 100644 index 0000000..ff652a7 --- /dev/null +++ b/tests/extr_c.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_extr_c(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(0xf) == 0xf); + ASSERT(f(0xff) == -1); + ASSERT(f(0xfff) == -1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_d.c b/tests/extr_d.c new file mode 100644 index 0000000..ac8b2f9 --- /dev/null +++ b/tests/extr_d.c @@ -0,0 +1,25 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_extr_d(j, JIT_F0, JIT_R0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0.0); + ASSERT(f(1) == 1.0); + ASSERT(f(-100) == -100.0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_d_f.c b/tests/extr_d_f.c new file mode 100644 index 0000000..87ad4a8 --- /dev/null +++ b/tests/extr_d_f.c @@ -0,0 +1,26 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_extr_d_f(j, JIT_F0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(double) = jit_end(j, NULL); + + ASSERT(f(0.0) == 0.0f); + ASSERT(f(0.5) == 0.5f); + ASSERT(f(1.0 / 0.0) == 1.0f / 0.0f); + ASSERT(f(1.25) == 1.25f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_f.c b/tests/extr_f.c new file mode 100644 index 0000000..7721127 --- /dev/null +++ b/tests/extr_f.c @@ -0,0 +1,25 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_extr_f(j, JIT_F0, JIT_R0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0.0f); + ASSERT(f(1) == 1.0f); + ASSERT(f(-100) == -100.0f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_f_d.c b/tests/extr_f_d.c new file mode 100644 index 0000000..8a9d38c --- /dev/null +++ b/tests/extr_f_d.c @@ -0,0 +1,26 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_extr_f_d(j, JIT_F0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(float) = jit_end(j, NULL); + + ASSERT(f(0.0f) == 0.0); + ASSERT(f(0.5f) == 0.5); + ASSERT(f(1.0f / 0.0f) == 1.0 / 0.0); + ASSERT(f(1.25f) == 1.25); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_i.c b/tests/extr_i.c new file mode 100644 index 0000000..cf3a2e1 --- /dev/null +++ b/tests/extr_i.c @@ -0,0 +1,30 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_extr_i(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(0xfffffff) == 0xfffffff); + ASSERT(f(0xffffffff) == -1); + ASSERT(f(0xfffffffff) == -1); + ASSERT(f(0xf00000000) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_s.c b/tests/extr_s.c new file mode 100644 index 0000000..08ea42e --- /dev/null +++ b/tests/extr_s.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_extr_s(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(0xfff) == 0xfff); + ASSERT(f(0xffff) == -1); + ASSERT(f(0xfffff) == -1); + ASSERT(f(0xf0000) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_uc.c b/tests/extr_uc.c new file mode 100644 index 0000000..063962b --- /dev/null +++ b/tests/extr_uc.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_extr_uc(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(0xff) == 0xff); + ASSERT(f(0xfff) == 0xff); + ASSERT(f(0xf00) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_ui.c b/tests/extr_ui.c new file mode 100644 index 0000000..457e8f8 --- /dev/null +++ b/tests/extr_ui.c @@ -0,0 +1,29 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_extr_ui(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(0xffffffff) == 0xffffffff); + ASSERT(f(0xfffffffff) == 0xffffffff); + ASSERT(f(0xf00000000) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/extr_us.c b/tests/extr_us.c new file mode 100644 index 0000000..61374cc --- /dev/null +++ b/tests/extr_us.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_extr_us(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); + ASSERT(f(1) == 1); + ASSERT(f(0xffff) == 0xffff); + ASSERT(f(0xfffff) == 0xffff); + ASSERT(f(0xf0000) == 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/jmp0.c b/tests/jmp0.c new file mode 100644 index 0000000..0f86fe6 --- /dev/null +++ b/tests/jmp0.c @@ -0,0 +1,24 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_reloc_t r = jit_jmp(j); + jit_patch_here(j, r); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + ASSERT(f(42) == 42); + ASSERT(f(-1) == -1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/jmp_table.c b/tests/jmp_table.c new file mode 100644 index 0000000..2ea14cd --- /dev/null +++ b/tests/jmp_table.c @@ -0,0 +1,61 @@ +#include "test.h" + +#define NTARGETS ((size_t) 4) + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0)); + + jit_reloc_t default_target = jit_bgei_u(j, JIT_R0, NTARGETS); + + // FIXME: need ldxr with word stride, then can eliminate lshi. + jit_lshi(j, JIT_R0, JIT_R0, sizeof(intptr_t) == 4 ? 2 : 3); + jit_reloc_t table = jit_mov_addr(j, JIT_R1); + jit_ldxr(j, JIT_R1, JIT_R1, JIT_R0); + jit_jmpr(j, JIT_R1); + + jit_begin_data (j, (NTARGETS + 1) * sizeof(intptr_t)); + jit_align(j, sizeof(intptr_t)); + jit_patch_here(j, table); + jit_reloc_t targets[NTARGETS]; + jit_reloc_t tails[NTARGETS]; + for (size_t i = 0; i < NTARGETS; i++) { + targets[i] = jit_emit_addr(j); + } + jit_end_data (j); + + for (size_t i = 0; i < NTARGETS; i++) { + jit_patch_here(j, targets[i]); + jit_movi(j, JIT_R0, i * i); + tails[i] = jit_jmp(j); + } + + jit_patch_here(j, default_target); + jit_movi(j, JIT_R0, 42); + for (int i = 0; i < NTARGETS; i++) { + jit_patch_here(j, tails[i]); + } + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + for (int i = -2; i < ((int) NTARGETS) + 2; i++) { + if (i < 0) { + ASSERT(f(i) == 42); + } else if (i < NTARGETS) { + ASSERT(f(i) == i * i); + } else { + ASSERT(f(i) == 42); + } + } +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/jmpi.c b/tests/jmpi.c new file mode 100644 index 0000000..cb12789 --- /dev/null +++ b/tests/jmpi.c @@ -0,0 +1,41 @@ +#include "test.h" + +void *tail; + +static void *target; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + jit_enter_jit_abi(j, 0, 0, 0); + jit_movi(j, JIT_R0, 42); + jit_jmpi(j, target); + // Unreachable. + jit_breakpoint(j); + int (*f)(void) = jit_end(j, NULL); + ASSERT(f() == 42); +} + +// Make the tail-call target via a separate main_helper because probably the new +// arena will be allocated farther away, forcing nonlocal jumps. +static void +make_target(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + // Tail call target assumes tail caller called enter_jit_abi with compatible + // parameters. + target = jit_address(j); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + jit_end(j, NULL); + + main_helper(0, NULL, run_test); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, make_target); +} diff --git a/tests/jmpi_local.c b/tests/jmpi_local.c new file mode 100644 index 0000000..1131997 --- /dev/null +++ b/tests/jmpi_local.c @@ -0,0 +1,25 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + + jit_reloc_t r = jit_jmp (j); + jit_reti (j, 0); + jit_pointer_t addr = jit_address (j); + jit_reti (j, 1); + jit_patch_here (j, r); + jit_jmpi (j, addr); + jit_reti (j, 2); + + int (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == 1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/jmpr.c b/tests/jmpr.c new file mode 100644 index 0000000..45111f9 --- /dev/null +++ b/tests/jmpr.c @@ -0,0 +1,25 @@ +#include "test.h" + +static int tail(void) { + return 42; +} + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0)); + jit_leave_jit_abi(j, 0, 0, align); + + jit_jmpr(j, JIT_R0); + + int (*f)(void*) = jit_end(j, NULL); + ASSERT(f(tail) == 42); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_c.c b/tests/ldi_c.c new file mode 100644 index 0000000..c6539c0 --- /dev/null +++ b/tests/ldi_c.c @@ -0,0 +1,24 @@ +#include "test.h" + +static uint8_t data[] = { 0xff, 0x00, 0x42 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_c(j, JIT_R0, &data[0]); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == -1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_d.c b/tests/ldi_d.c new file mode 100644 index 0000000..c7574dc --- /dev/null +++ b/tests/ldi_d.c @@ -0,0 +1,24 @@ +#include "test.h" + +static double data = -1.5; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_d(j, JIT_F0, &data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == data); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_f.c b/tests/ldi_f.c new file mode 100644 index 0000000..d5cd095 --- /dev/null +++ b/tests/ldi_f.c @@ -0,0 +1,24 @@ +#include "test.h" + +static float data = -1.5f; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_f(j, JIT_F0, &data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == data); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_i.c b/tests/ldi_i.c new file mode 100644 index 0000000..72035d4 --- /dev/null +++ b/tests/ldi_i.c @@ -0,0 +1,24 @@ +#include "test.h" + +static uint32_t data = 0xffffffff; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_i(j, JIT_R0, &data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == -1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_l.c b/tests/ldi_l.c new file mode 100644 index 0000000..58a8099 --- /dev/null +++ b/tests/ldi_l.c @@ -0,0 +1,26 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint64_t data = 0xffffffffffffffff; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_l(j, JIT_R0, &data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == -1); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_s.c b/tests/ldi_s.c new file mode 100644 index 0000000..7d10106 --- /dev/null +++ b/tests/ldi_s.c @@ -0,0 +1,24 @@ +#include "test.h" + +static uint16_t data = 0xffff; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_s(j, JIT_R0, &data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == -1); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_uc.c b/tests/ldi_uc.c new file mode 100644 index 0000000..d5c80ac --- /dev/null +++ b/tests/ldi_uc.c @@ -0,0 +1,24 @@ +#include "test.h" + +static uint8_t data[] = { 0xff, 0x00, 0x42 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_uc(j, JIT_R0, data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == 0xff); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_ui.c b/tests/ldi_ui.c new file mode 100644 index 0000000..bfccc30 --- /dev/null +++ b/tests/ldi_ui.c @@ -0,0 +1,26 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint32_t data[] = { 0xffffffff, 0x00000000, 0x42424242 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_ui(j, JIT_R0, data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == data[0]); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldi_us.c b/tests/ldi_us.c new file mode 100644 index 0000000..970db9c --- /dev/null +++ b/tests/ldi_us.c @@ -0,0 +1,24 @@ +#include "test.h" + +static uint16_t data[] = { 0xffff, 0x0000, 0x4242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_ldi_us(j, JIT_R0, data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == data[0]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_atomic.c b/tests/ldr_atomic.c new file mode 100644 index 0000000..339ba56 --- /dev/null +++ b/tests/ldr_atomic.c @@ -0,0 +1,28 @@ +#include "test.h" + +static long data[] = { 0x0f0f0f0f, 0x00000000, 0x42424242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + /* atm does not test for actual atomicity, just that no segfaults etc happen */ + jit_ldr_atomic(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == 0x0f0f0f0f); + ASSERT(f(&data[1]) == 0); + ASSERT(f(&data[2]) == 0x42424242); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_c.c b/tests/ldr_c.c new file mode 100644 index 0000000..a630123 --- /dev/null +++ b/tests/ldr_c.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint8_t data[] = { 0xff, 0x00, 0x42 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_c(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == -1); + ASSERT(f(&data[1]) == 0); + ASSERT(f(&data[2]) == 0x42); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_d.c b/tests/ldr_d.c new file mode 100644 index 0000000..6a4c1bd --- /dev/null +++ b/tests/ldr_d.c @@ -0,0 +1,27 @@ +#include "test.h" + +static double data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_d(j, JIT_F0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == data[0]); + ASSERT(f(&data[1]) == data[1]); + ASSERT(f(&data[2]) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_f.c b/tests/ldr_f.c new file mode 100644 index 0000000..4ea978d --- /dev/null +++ b/tests/ldr_f.c @@ -0,0 +1,27 @@ +#include "test.h" + +static float data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_f(j, JIT_F0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == data[0]); + ASSERT(f(&data[1]) == data[1]); + ASSERT(f(&data[2]) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_i.c b/tests/ldr_i.c new file mode 100644 index 0000000..a7a4cd0 --- /dev/null +++ b/tests/ldr_i.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint32_t data[] = { 0xffffffff, 0x00000000, 0x42424242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_i(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == -1); + ASSERT(f(&data[1]) == 0); + ASSERT(f(&data[2]) == 0x42424242); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_l.c b/tests/ldr_l.c new file mode 100644 index 0000000..d689d47 --- /dev/null +++ b/tests/ldr_l.c @@ -0,0 +1,29 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint64_t data[] = { 0xffffffffffffffff, 0, 0x4242424212345678 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_l(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == -1); + ASSERT(f(&data[1]) == 0); + ASSERT(f(&data[2]) == data[2]); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_s.c b/tests/ldr_s.c new file mode 100644 index 0000000..cbb72ab --- /dev/null +++ b/tests/ldr_s.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint16_t data[] = { 0xffff, 0x0000, 0x4242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_s(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == -1); + ASSERT(f(&data[1]) == 0); + ASSERT(f(&data[2]) == 0x4242); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_uc.c b/tests/ldr_uc.c new file mode 100644 index 0000000..65d56cf --- /dev/null +++ b/tests/ldr_uc.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint8_t data[] = { 0xff, 0x00, 0x42 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_uc(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == 0xff); + ASSERT(f(&data[1]) == 0); + ASSERT(f(&data[2]) == 0x42); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_ui.c b/tests/ldr_ui.c new file mode 100644 index 0000000..92ace34 --- /dev/null +++ b/tests/ldr_ui.c @@ -0,0 +1,29 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint32_t data[] = { 0xffffffff, 0x00000000, 0x42424242 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_ui(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == data[0]); + ASSERT(f(&data[1]) == data[1]); + ASSERT(f(&data[2]) == data[2]); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldr_us.c b/tests/ldr_us.c new file mode 100644 index 0000000..5d1a67d --- /dev/null +++ b/tests/ldr_us.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint16_t data[] = { 0xffff, 0x0000, 0x4242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1)); + + jit_ldr_us(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*) = jit_end(j, NULL); + + ASSERT(f(&data[0]) == data[0]); + ASSERT(f(&data[1]) == data[1]); + ASSERT(f(&data[2]) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_c.c b/tests/ldxi_c.c new file mode 100644 index 0000000..cc40ddb --- /dev/null +++ b/tests/ldxi_c.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint8_t data[] = { 0xff, 0x00, 0x42 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ldxi_c(j, JIT_R0, JIT_R0, (uintptr_t)&data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == -1); + ASSERT(f(1) == 0); + ASSERT(f(2) == 0x42); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_d.c b/tests/ldxi_d.c new file mode 100644 index 0000000..a93e582 --- /dev/null +++ b/tests/ldxi_d.c @@ -0,0 +1,27 @@ +#include "test.h" + +static double data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ldxi_d(j, JIT_F0, JIT_R0, (uintptr_t)data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == data[0]); + ASSERT(f(8) == data[1]); + ASSERT(f(16) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_f.c b/tests/ldxi_f.c new file mode 100644 index 0000000..015d816 --- /dev/null +++ b/tests/ldxi_f.c @@ -0,0 +1,27 @@ +#include "test.h" + +static float data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ldxi_f(j, JIT_F0, JIT_R0, (uintptr_t)data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == data[0]); + ASSERT(f(4) == data[1]); + ASSERT(f(8) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_i.c b/tests/ldxi_i.c new file mode 100644 index 0000000..7e53001 --- /dev/null +++ b/tests/ldxi_i.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint32_t data[] = { 0xffffffff, 0x00000000, 0x42424242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0)); + + jit_ldxi_i(j, JIT_R0, JIT_R0, (uintptr_t)data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == -1); + ASSERT(f(4) == 0); + ASSERT(f(8) == 0x42424242); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_l.c b/tests/ldxi_l.c new file mode 100644 index 0000000..3a4454f --- /dev/null +++ b/tests/ldxi_l.c @@ -0,0 +1,29 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint64_t data[] = { 0xffffffffffffffff, 0, 0x4242424212345678 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ldxi_l(j, JIT_R0, JIT_R0, (uintptr_t)data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == -1); + ASSERT(f(8) == 0); + ASSERT(f(16) == data[2]); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_s.c b/tests/ldxi_s.c new file mode 100644 index 0000000..05125da --- /dev/null +++ b/tests/ldxi_s.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint16_t data[] = { 0xffff, 0x0000, 0x4242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ldxi_s(j, JIT_R0, JIT_R0, (uintptr_t)data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == -1); + ASSERT(f(2) == 0); + ASSERT(f(4) == 0x4242); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_uc.c b/tests/ldxi_uc.c new file mode 100644 index 0000000..dc82aa2 --- /dev/null +++ b/tests/ldxi_uc.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint8_t data[] = { 0xff, 0x00, 0x42 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ldxi_uc(j, JIT_R0, JIT_R0, (uintptr_t)data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0xff); + ASSERT(f(1) == 0); + ASSERT(f(2) == 0x42); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_ui.c b/tests/ldxi_ui.c new file mode 100644 index 0000000..4500507 --- /dev/null +++ b/tests/ldxi_ui.c @@ -0,0 +1,29 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint32_t data[] = { 0xffffffff, 0x00000000, 0x42424242 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ldxi_ui(j, JIT_R0, JIT_R0, (uintptr_t)data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == data[0]); + ASSERT(f(4) == data[1]); + ASSERT(f(8) == data[2]); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxi_us.c b/tests/ldxi_us.c new file mode 100644 index 0000000..db2d8c8 --- /dev/null +++ b/tests/ldxi_us.c @@ -0,0 +1,27 @@ +#include "test.h" + +static uint16_t data[] = { 0xffff, 0x0000, 0x4242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ldxi_us(j, JIT_R0, JIT_R0, (uintptr_t)data); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(0) == data[0]); + ASSERT(f(2) == data[1]); + ASSERT(f(4) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_c.c b/tests/ldxr_c.c new file mode 100644 index 0000000..b691b58 --- /dev/null +++ b/tests/ldxr_c.c @@ -0,0 +1,28 @@ +#include "test.h" + +static uint8_t data[] = { 0xff, 0x00, 0x42 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_c(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == -1); + ASSERT(f(data, 1) == 0); + ASSERT(f(data, 2) == 0x42); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_d.c b/tests/ldxr_d.c new file mode 100644 index 0000000..e8bb45e --- /dev/null +++ b/tests/ldxr_d.c @@ -0,0 +1,28 @@ +#include "test.h" + +static double data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_d(j, JIT_F0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == data[0]); + ASSERT(f(data, 8) == data[1]); + ASSERT(f(data, 16) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_f.c b/tests/ldxr_f.c new file mode 100644 index 0000000..c6dacba --- /dev/null +++ b/tests/ldxr_f.c @@ -0,0 +1,28 @@ +#include "test.h" + +static float data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_f(j, JIT_F0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == data[0]); + ASSERT(f(data, 4) == data[1]); + ASSERT(f(data, 8) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_i.c b/tests/ldxr_i.c new file mode 100644 index 0000000..e035438 --- /dev/null +++ b/tests/ldxr_i.c @@ -0,0 +1,28 @@ +#include "test.h" + +static uint32_t data[] = { 0xffffffff, 0x00000000, 0x42424242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_i(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == -1); + ASSERT(f(data, 4) == 0); + ASSERT(f(data, 8) == 0x42424242); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_l.c b/tests/ldxr_l.c new file mode 100644 index 0000000..6dcb5c7 --- /dev/null +++ b/tests/ldxr_l.c @@ -0,0 +1,30 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint64_t data[] = { 0xffffffffffffffff, 0, 0x4242424212345678 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_l(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == -1); + ASSERT(f(data, 8) == 0); + ASSERT(f(data, 16) == data[2]); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_s.c b/tests/ldxr_s.c new file mode 100644 index 0000000..cea155f --- /dev/null +++ b/tests/ldxr_s.c @@ -0,0 +1,28 @@ +#include "test.h" + +static uint16_t data[] = { 0xffff, 0x0000, 0x4242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_s(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == -1); + ASSERT(f(data, 2) == 0); + ASSERT(f(data, 4) == 0x4242); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_uc.c b/tests/ldxr_uc.c new file mode 100644 index 0000000..3b6ab3e --- /dev/null +++ b/tests/ldxr_uc.c @@ -0,0 +1,28 @@ +#include "test.h" + +static uint8_t data[] = { 0xff, 0x00, 0x42 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_uc(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == 0xff); + ASSERT(f(data, 1) == 0); + ASSERT(f(data, 2) == 0x42); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_ui.c b/tests/ldxr_ui.c new file mode 100644 index 0000000..eddb56b --- /dev/null +++ b/tests/ldxr_ui.c @@ -0,0 +1,30 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint32_t data[] = { 0xffffffff, 0x00000000, 0x42424242 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_ui(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == data[0]); + ASSERT(f(data, 4) == data[1]); + ASSERT(f(data, 8) == data[2]); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ldxr_us.c b/tests/ldxr_us.c new file mode 100644 index 0000000..d18c133 --- /dev/null +++ b/tests/ldxr_us.c @@ -0,0 +1,28 @@ +#include "test.h" + +static uint16_t data[] = { 0xffff, 0x0000, 0x4242 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_ldxr_us(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void*, jit_uword_t) = jit_end(j, NULL); + + ASSERT(f(data, 0) == data[0]); + ASSERT(f(data, 2) == data[1]); + ASSERT(f(data, 4) == data[2]); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/link-register.c b/tests/link-register.c new file mode 100644 index 0000000..9e5413b --- /dev/null +++ b/tests/link-register.c @@ -0,0 +1,35 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0)); + + jit_reloc_t call_tramp = jit_jmp (j); + + void *tramp = jit_address (j); + jit_pop_link_register (j); + jit_movr (j, JIT_R0, JIT_LR); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr (j, JIT_R0); + + jit_patch_here (j, call_tramp); + jit_jmpi_with_link (j, tramp); + + void *expected_link = jit_address_to_function_pointer (jit_address (j)); + + size_t size = 0; + void* ret = jit_end(j, &size); + + void* (*f)(void) = ret; + + ASSERT(f() == expected_link); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/lshi.c b/tests/lshi.c new file mode 100644 index 0000000..772c0ed --- /dev/null +++ b/tests/lshi.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_lshi(j, JIT_R0, JIT_R0, 31); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(-0x7f) == 0x80000000); +#else + ASSERT(f(-0x7f) == 0xffffffc080000000); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/lshr.c b/tests/lshr.c new file mode 100644 index 0000000..89b9084 --- /dev/null +++ b/tests/lshr.c @@ -0,0 +1,69 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_lshr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7f, 1) == 0xfe); + ASSERT(f(0x7fff, 2) == 0x1fffc); + ASSERT(f(0x81, 16) == 0x810000); + ASSERT(f(0xff, 15) == 0x7f8000); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xffffffff, 8) == 0xffffff00); + ASSERT(f(0x7fffffff, 3) == 0xfffffff8); + ASSERT(f(-0x7f, 31) == 0x80000000); + ASSERT(f(-0x7fff, 30) == 0x40000000); + ASSERT(f(-0x7fffffff, 29) == 0x20000000); + ASSERT(f(0x80000001, 28) == 0x10000000); + ASSERT(f(0x8001, 17) == 0x20000); + ASSERT(f(0x80000001, 18) == 0x40000); + ASSERT(f(-0xffff, 24) == 0x1000000); +#else + ASSERT(f(0xffffffff, 8) == 0xffffffff00); + ASSERT(f(0x7fffffff, 3) == 0x3fffffff8); + ASSERT(f(-0x7f, 31) == 0xffffffc080000000); + ASSERT(f(-0x7fff, 30) == 0xffffe00040000000); + ASSERT(f(-0x7fffffff, 29) == 0xf000000020000000); + ASSERT(f(0x80000001, 28) == 0x800000010000000); + ASSERT(f(0x8001, 17) == 0x100020000); + ASSERT(f(0x80000001, 18) == 0x2000000040000); + ASSERT(f(-0xffff, 24) == 0xffffff0001000000); + ASSERT(f(0x7f, 33) == 0xfe00000000); + ASSERT(f(0x7ffff, 34) == 0x1ffffc00000000); + ASSERT(f(0x7fffffff, 35) == 0xfffffff800000000); + ASSERT(f(-0x7f, 63) == 0x8000000000000000); + ASSERT(f(-0x7fff, 62) == 0x4000000000000000); + ASSERT(f(-0x7fffffff, 61) == 0x2000000000000000); + ASSERT(f(0x80000001, 60) == 0x1000000000000000); + ASSERT(f(0x81, 48) == 0x81000000000000); + ASSERT(f(0x8001, 49) == 0x2000000000000); + ASSERT(f(0x80000001, 40) == 0x10000000000); + ASSERT(f(0xff, 47) == 0x7f800000000000); + ASSERT(f(0xffff0001, 56) == 0x100000000000000); + ASSERT(f(0xffffffff, 40) == 0xffffff0000000000); + ASSERT(f(0x7fffffffff, 33) == 0xfffffffe00000000); + ASSERT(f(-0x7fffffffff, 63) == 0x8000000000000000); + ASSERT(f(0x8000000001, 48) == 0x1000000000000); + ASSERT(f(0xffffffffff, 47) == 0xffff800000000000); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/makefile b/tests/makefile new file mode 100644 index 0000000..3ab921a --- /dev/null +++ b/tests/makefile @@ -0,0 +1,32 @@ +include ./tests.mk + +LLVM ?= 0 +CROSS_COMPILE := +COMPILER != [ "$(LLVM)" != "0" ] \ + && echo clang --target="$(CROSS_COMPILE)" \ + || echo $(CROSS_COMPILE)gcc + +CFLAGS := -Wall -Wextra -O0 -g +INCLUDE_FLAGS := -I include +COMPILE_JIT_TEST := $(COMPILER) $(CFLAGS) $(INCLUDE_FLAGS) +COMPILE_BCODE_TEST := $(COMPILER) $(CFLAGS) $(INCLUDE_FLAGS) -DFORCE_BCODE=1 + +.PHONY: check +check: $(BCODE_TESTS) $(JIT_TESTS) + echo "Success!" + +.PHONY: check-bcode +check-bcode: + @echo "Running bytecode tests..." + @set -e; for test in $(BCODE_TESTS); do \ + echo "Testing: $$test"; \ + ./$$test; \ + done + +.PHONY: check-jit +check-jit: + @echo "Running jit tests..." + @set -e; for test in $(JIT_TESTS); do \ + echo "Testing: $$test"; \ + ./$$test; \ + done diff --git a/tests/mov_addr.c b/tests/mov_addr.c new file mode 100644 index 0000000..01a0e7e --- /dev/null +++ b/tests/mov_addr.c @@ -0,0 +1,25 @@ +#include "test.h" + +static uint64_t thing = 0x123456789abcdef0; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_patch_there(j, jit_mov_addr(j, JIT_R0), &thing); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + void* (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == &thing); + ASSERT(*(uint64_t*)f() == thing); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/movi.c b/tests/movi.c new file mode 100644 index 0000000..1282b6f --- /dev/null +++ b/tests/movi.c @@ -0,0 +1,22 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_movi(j, JIT_R0, 0xa500a500); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_uword_t (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == 0xa500a500); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/movi_d.c b/tests/movi_d.c new file mode 100644 index 0000000..650064b --- /dev/null +++ b/tests/movi_d.c @@ -0,0 +1,22 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_movi_d(j, JIT_F0, 3.14159); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == 3.14159); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/movi_f.c b/tests/movi_f.c new file mode 100644 index 0000000..82e2d0a --- /dev/null +++ b/tests/movi_f.c @@ -0,0 +1,22 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + + jit_movi_f(j, JIT_F0, 3.14159f); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(void) = jit_end(j, NULL); + + ASSERT(f() == 3.14159f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/mulr.c b/tests/mulr.c new file mode 100644 index 0000000..3f0a6f8 --- /dev/null +++ b/tests/mulr.c @@ -0,0 +1,64 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_mulr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7fffffff, 1) == 0x7fffffff); + ASSERT(f(1, 0x7fffffff) == 0x7fffffff); + ASSERT(f(0x80000000, 1) == 0x80000000); + ASSERT(f(1, 0x80000000) == 0x80000000); + ASSERT(f(0x7fffffff, 2) == 0xfffffffe); + ASSERT(f(2, 0x7fffffff) == 0xfffffffe); + ASSERT(f(0x7fffffff, 0) == 0); + ASSERT(f(0, 0x7fffffff) == 0); +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x80000000, 2) == 0); + ASSERT(f(2, 0x80000000) == 0); + ASSERT(f(0x7fffffff, 0x80000000) == 0x80000000); + ASSERT(f(0x80000000, 0x7fffffff) == 0x80000000); + ASSERT(f(0x7fffffff, 0xffffffff) == 0x80000001); + ASSERT(f(0xffffffff, 0x7fffffff) == 0x80000001); + ASSERT(f(0xffffffff, 0xffffffff) == 1); +#else + ASSERT(f(0x80000000, 2) == 0x100000000); + ASSERT(f(2, 0x80000000) == 0x100000000); + ASSERT(f(0x7fffffff, 0x80000000) == 0x3fffffff80000000); + ASSERT(f(0x80000000, 0x7fffffff) == 0x3fffffff80000000); + ASSERT(f(0x7fffffff, 0xffffffff) == 0x7ffffffe80000001); + ASSERT(f(0xffffffff, 0x7fffffff) == 0x7ffffffe80000001); + ASSERT(f(0xffffffff, 0xffffffff) == 0xfffffffe00000001); + ASSERT(f(0x7fffffffffffffff, 1) == 0x7fffffffffffffff); + ASSERT(f(1, 0x7fffffffffffffff) == 0x7fffffffffffffff); + ASSERT(f(0x8000000000000000, 1) == 0x8000000000000000); + ASSERT(f(1, 0x8000000000000000) == 0x8000000000000000); + ASSERT(f(0x7fffffffffffffff, 2) == 0xfffffffffffffffe); + ASSERT(f(2, 0x7fffffffffffffff) == 0xfffffffffffffffe); + ASSERT(f(0x8000000000000000, 2) == 0); + ASSERT(f(2, 0x8000000000000000) == 0); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0x8000000000000000); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 0x8000000000000000); + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == 0x8000000000000001); + ASSERT(f(0xffffffffffffffff, 0x7fffffffffffffff) == 0x8000000000000001); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 1); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/mulr_d.c b/tests/mulr_d.c new file mode 100644 index 0000000..5464170 --- /dev/null +++ b/tests/mulr_d.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_mulr_d(j, JIT_F0, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + double (*f)(double, double) = ret; + ASSERT(f(-0.5, 0.5) == -0.25); + ASSERT(f(0.25, 0.75) == 0.1875); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/mulr_f.c b/tests/mulr_f.c new file mode 100644 index 0000000..5e5f46a --- /dev/null +++ b/tests/mulr_f.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_mulr_f(j, JIT_F0, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + float (*f)(float, float) = ret; + ASSERT(f(-0.5f, 0.5f) == -0.25f); + ASSERT(f(0.25f, 0.75f) == 0.1875f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/negr.c b/tests/negr.c new file mode 100644 index 0000000..db65f3c --- /dev/null +++ b/tests/negr.c @@ -0,0 +1,39 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_negr(j, JIT_R0, JIT_R0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + + ASSERT(f(0) == 0); +#if EJIT_WORDSIZE == 32 + ASSERT(f(1) == 0xffffffff); + ASSERT(f(0xffffffff) == 1); + ASSERT(f(0x80000000) == 0x80000000); + ASSERT(f(0x7fffffff) == 0x80000001); + ASSERT(f(0x80000001) == 0x7fffffff); +#else + ASSERT(f(1) == 0xffffffffffffffff); + ASSERT(f(0xffffffff) == 0xffffffff00000001); + ASSERT(f(0x80000000) == 0xffffffff80000000); + ASSERT(f(0x7fffffff) == 0xffffffff80000001); + ASSERT(f(0x80000001) == 0xffffffff7fffffff); + ASSERT(f(0xffffffffffffffff) == 1); + ASSERT(f(0x8000000000000000) == 0x8000000000000000); + ASSERT(f(0x7fffffffffffffff) == 0x8000000000000001); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/negr_d.c b/tests/negr_d.c new file mode 100644 index 0000000..b8d4a00 --- /dev/null +++ b/tests/negr_d.c @@ -0,0 +1,26 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_negr_d(j, JIT_F0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(double) = jit_end(j, NULL); + + ASSERT(f(0.0) == -0.0); + ASSERT(f(0.5) == -0.5); + ASSERT(f(1.0 / 0.0) == -1.0 / 0.0); + ASSERT(f(-1.25) == 1.25); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/negr_f.c b/tests/negr_f.c new file mode 100644 index 0000000..08b6364 --- /dev/null +++ b/tests/negr_f.c @@ -0,0 +1,26 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_negr_f(j, JIT_F0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(float) = jit_end(j, NULL); + + ASSERT(f(0.0f) == -0.0f); + ASSERT(f(0.5f) == -0.5f); + ASSERT(f(1.0f / 0.0f) == -1.0f / 0.0f); + ASSERT(f(-1.25f) == 1.25f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/ori.c b/tests/ori.c new file mode 100644 index 0000000..27f7b2c --- /dev/null +++ b/tests/ori.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_ori(j, JIT_R0, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t) = ret; + + ASSERT(f(0x7fffffff) == 0x7fffffff); + ASSERT(f(0x80000000) == 0x80000001); +#if EJIT_WORDSIZE == 64 + ASSERT(f(0x7fffffffffffffff) == 0x7fffffffffffffff); + ASSERT(f(0x8000000000000000) == 0x8000000000000001); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/orr.c b/tests/orr.c new file mode 100644 index 0000000..af8e141 --- /dev/null +++ b/tests/orr.c @@ -0,0 +1,48 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_orr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7fffffff, 1) == 0x7fffffff); + ASSERT(f(1, 0x7fffffff) == 0x7fffffff); + ASSERT(f(0x80000000, 1) == 0x80000001); + ASSERT(f(1, 0x80000000) == 0x80000001); + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + ASSERT(f(0x80000000, 0x7fffffff) == 0xffffffff); + ASSERT(f(0x7fffffff, 0xffffffff) == 0xffffffff); + ASSERT(f(0xffffffff, 0x7fffffff) == 0xffffffff); + ASSERT(f(0xffffffff, 0xffffffff) == 0xffffffff); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0, 0x7fffffff) == 0x7fffffff); +#if EJIT_WORDSIZE == 64 + ASSERT(f(0x7fffffffffffffff, 1) == 0x7fffffffffffffff); + ASSERT(f(1, 0x7fffffffffffffff) == 0x7fffffffffffffff); + ASSERT(f(0x8000000000000000, 1) == 0x8000000000000001); + ASSERT(f(1, 0x8000000000000000) == 0x8000000000000001); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0xffffffffffffffff); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 0xffffffffffffffff); + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == 0xffffffffffffffff); + ASSERT(f(0xffffffffffffffff, 0x7fffffffffffffff) == 0xffffffffffffffff); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 0xffffffffffffffff); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/qdivr.c b/tests/qdivr.c new file mode 100644 index 0000000..c559802 --- /dev/null +++ b/tests/qdivr.c @@ -0,0 +1,45 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 3, 0, 0); + + jit_operand_t args[] = + { jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_V0) }; + jit_load_args(j, 4, args); + + jit_qdivr(j, JIT_V1, JIT_V2, JIT_R2, JIT_V0); + jit_str(j, JIT_R0, JIT_V1); + jit_str(j, JIT_R1, JIT_V2); + + jit_leave_jit_abi(j, 3, 0, align); + + jit_ret(j); + + size_t size = 0; + void* ret = jit_end(j, &size); + + void (*f)(jit_word_t*, jit_word_t*, jit_word_t, jit_word_t) = ret; + +#define QDIV(a, b, c, d) \ + do { \ + jit_word_t C = 0, D = 0; f(&C, &D, a, b); ASSERT(C == c); \ + ASSERT(D == d); \ + } while (0) + + QDIV(10, 3, 3, 1); + QDIV(-33, 9, -3, -6); + QDIV(-41, -7, 5, -6); + QDIV(65536, 4096, 16, 0); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/qdivr_u.c b/tests/qdivr_u.c new file mode 100644 index 0000000..df215ec --- /dev/null +++ b/tests/qdivr_u.c @@ -0,0 +1,43 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + + size_t align = jit_enter_jit_abi(j, 3, 0, 0); + + jit_operand_t args[] = + { jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_V0) }; + jit_load_args(j, 4, args); + + jit_qdivr_u(j, JIT_V1, JIT_V2, JIT_R2, JIT_V0); + jit_str(j, JIT_R0, JIT_V1); + jit_str(j, JIT_R1, JIT_V2); + + jit_leave_jit_abi(j, 3, 0, align); + + jit_ret(j); + + size_t size = 0; + void* ret = jit_end(j, &size); + + void (*f)(jit_word_t*, jit_word_t*, jit_word_t, jit_word_t) = ret; +#define QDIV(a, b, c, d) \ + do { \ + jit_word_t C = 0, D = 0; f(&C, &D, a, b); ASSERT(C == c); \ + ASSERT(D == d); \ + } while (0) + + QDIV(-1, -2, 1, 1); + QDIV(-2, -5, 1, 3); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/qmulr.c b/tests/qmulr.c new file mode 100644 index 0000000..b7aa5e8 --- /dev/null +++ b/tests/qmulr.c @@ -0,0 +1,59 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + + size_t align = jit_enter_jit_abi(j, 3, 0, 0); + + jit_operand_t args[] = + { jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_V0) }; + jit_load_args(j, 4, args); + + jit_qmulr(j, JIT_V1, JIT_V2, JIT_R2, JIT_V0); + jit_str(j, JIT_R0, JIT_V1); + jit_str(j, JIT_R1, JIT_V2); + + jit_leave_jit_abi(j, 3, 0, align); + + jit_ret(j); + + size_t size = 0; + void* ret = jit_end(j, &size); + + void (*f)(jit_word_t*, jit_word_t*, jit_word_t, jit_word_t) = ret; + +#define QMUL(a, b, c, d) \ + do { \ + jit_word_t C = 0, D = 0; f(&C, &D, a, b); ASSERT(C == c); \ + ASSERT(D == d); \ + } while (0) + + QMUL(-2, -1, 2, 0); + QMUL(0, -1, 0, 0); + QMUL(-1, 0, 0, 0); + QMUL(1, -1, -1, -1); +#if EJIT_WORDSIZE == 32 + QMUL(0x7ffff, 0x7ffff, 0xfff00001, 0x3f); + QMUL(0x80000000, -2, 0, 1); + QMUL(0x80000000, 2, 0, -1); + QMUL(0x80000001, 3, 0x80000003, -2); + QMUL(0x80000001, -3, 0x7ffffffd, 1); +#else + QMUL(0x7ffffffff, 0x7ffffffff, 0xfffffff000000001, 0x3f); + QMUL(0x8000000000000000, -2, 0, 1); + QMUL(0x8000000000000000, 2, 0, -1); + QMUL(0x8000000000000001, 3, 0x8000000000000003, -2); + QMUL(0x8000000000000001, -3, 0x7ffffffffffffffd, 1); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/qmulr_u.c b/tests/qmulr_u.c new file mode 100644 index 0000000..3840c92 --- /dev/null +++ b/tests/qmulr_u.c @@ -0,0 +1,47 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + + size_t align = jit_enter_jit_abi(j, 3, 0, 0); + + jit_operand_t args[] = + { jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R1), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_V0) }; + jit_load_args(j, 4, args); + + jit_qmulr_u(j, JIT_V1, JIT_V2, JIT_R2, JIT_V0); + jit_str(j, JIT_R0, JIT_V1); + jit_str(j, JIT_R1, JIT_V2); + + jit_leave_jit_abi(j, 3, 0, align); + + jit_ret(j); + + size_t size = 0; + void* ret = jit_end(j, &size); + + void (*f)(jit_word_t*, jit_word_t*, jit_word_t, jit_word_t) = ret; + +#define UQMUL(a, b, c, d) \ + do { \ + jit_word_t C = 0, D = 0; f(&C, &D, a, b); ASSERT(C == c); \ + ASSERT(D == d); \ + } while (0) + +#if EJIT_WORDSIZE == 32 + UQMUL(0xffffff, 0xffffff, 0xfe000001, 0xffff); +#else + UQMUL(0xffffffffff, 0xffffffffff, 0xfffffe0000000001, 0xffff); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/remr.c b/tests/remr.c new file mode 100644 index 0000000..cd5bf4f --- /dev/null +++ b/tests/remr.c @@ -0,0 +1,60 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_remr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7fffffff, 1) == 0); + ASSERT(f(1, 0x7fffffff) == 1); + ASSERT(f(0x80000000, 1) == 0); + ASSERT(f(1, 0x80000000) == 1); + ASSERT(f(0x7fffffff, 2) == 1); + ASSERT(f(2, 0x7fffffff) == 2); + ASSERT(f(0x80000000, 2) == 0); + ASSERT(f(2, 0x80000000) == 2); + ASSERT(f(0x7fffffff, 0x80000000) == 0x7fffffff); + ASSERT(f(0, 0x7fffffff) == 0); + ASSERT(f(0xffffffff, 0xffffffff) == 0); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x80000000, 0x7fffffff) == 0xffffffff); + ASSERT(f(0x7fffffff, 0xffffffff) == 0); + ASSERT(f(0xffffffff, 0x7fffffff) == 0xffffffff); +#else + ASSERT(f(0x80000000, 0x7fffffff) == 1); + ASSERT(f(0x7fffffff, 0xffffffff) == 0x7fffffff); + ASSERT(f(0xffffffff, 0x7fffffff) == 1); + ASSERT(f(0x7fffffffffffffff, 1) == 0); + ASSERT(f(1, 0x7fffffffffffffff) == 1); + ASSERT(f(0x8000000000000000, 1) == 0); + ASSERT(f(1, 0x8000000000000000) == 1); + ASSERT(f(0x7fffffffffffffff, 2) == 1); + ASSERT(f(2, 0x7fffffffffffffff) == 2); + ASSERT(f(0x8000000000000000, 2) == 0); + ASSERT(f(2, 0x8000000000000000) == 2); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0x7fffffffffffffff); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 0xffffffffffffffff); + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == 0); + ASSERT(f(0xffffffffffffffff, 0x7fffffffffffffff) == 0xffffffffffffffff); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/remr_u.c b/tests/remr_u.c new file mode 100644 index 0000000..ba015c1 --- /dev/null +++ b/tests/remr_u.c @@ -0,0 +1,56 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_remr_u(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7fffffff, 1) == 0); + ASSERT(f(1, 0x7fffffff) == 1); + ASSERT(f(0x80000000, 1) == 0); + ASSERT(f(1, 0x80000000) == 1); + ASSERT(f(0x7fffffff, 2) == 1); + ASSERT(f(2, 0x7fffffff) == 2); + ASSERT(f(0x80000000, 2) == 0); + ASSERT(f(2, 0x80000000) == 2); + ASSERT(f(0x7fffffff, 0x80000000) == 0x7fffffff); + ASSERT(f(0x80000000, 0x7fffffff) == 1); + ASSERT(f(0, 0x7fffffff) == 0); + ASSERT(f(0x7fffffff, 0xffffffff) == 0x7fffffff); + ASSERT(f(0xffffffff, 0x7fffffff) == 1); + ASSERT(f(0xffffffff, 0xffffffff) == 0); + +#if EJIT_WORDSIZE != 32 + ASSERT(f(0x7fffffffffffffff, 1) == 0); + ASSERT(f(1, 0x7fffffffffffffff) == 1); + ASSERT(f(0x8000000000000000, 1) == 0); + ASSERT(f(1, 0x8000000000000000) == 1); + ASSERT(f(0x7fffffffffffffff, 2) == 1); + ASSERT(f(2, 0x7fffffffffffffff) == 2); + ASSERT(f(0x8000000000000000, 2) == 0); + ASSERT(f(2, 0x8000000000000000) == 2); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0x7fffffffffffffff); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 1); + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == 0x7fffffffffffffff); + ASSERT(f(0xffffffffffffffff, 0x7fffffffffffffff) == 1); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/rshi.c b/tests/rshi.c new file mode 100644 index 0000000..ab5438a --- /dev/null +++ b/tests/rshi.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_rshi(j, JIT_R0, JIT_R0, 31); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x80000000) == -1); +#else + ASSERT(f(0x80000000) == 1); + ASSERT(f(0x8000000000000000) == 0xffffffff00000000); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/rshi_u.c b/tests/rshi_u.c new file mode 100644 index 0000000..aa02206 --- /dev/null +++ b/tests/rshi_u.c @@ -0,0 +1,28 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_rshi_u(j, JIT_R0, JIT_R0, 31); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + jit_word_t (*f)(jit_word_t) = jit_end(j, NULL); + +#if EJIT_WORDSIZE == 32 + ASSERT(f(0x80000000) == 1); +#else + ASSERT(f(0x80000000) == 1); + ASSERT(f(0x8000000000000000) == 0x100000000); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/rshr.c b/tests/rshr.c new file mode 100644 index 0000000..d14f64f --- /dev/null +++ b/tests/rshr.c @@ -0,0 +1,63 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_rshr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0xfe, 1) == 0x7f); + ASSERT(f(0x1fffc, 2) == 0x7fff); + ASSERT(f(0x40000000, 30) == 1); + ASSERT(f(0x20000000, 29) == 1); + ASSERT(f(0x10000000, 28) == 1); + ASSERT(f(0x810000, 16) == 0x81); + ASSERT(f(0x20000, 17) == 1); + ASSERT(f(0x40000, 18) == 1); + ASSERT(f(0x7f8000, 15) == 0xff); + ASSERT(f(0x1000000, 24) == 1); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xfffffff8, 3) == 0xffffffff); + ASSERT(f(0x80000000, 31) == 0xffffffff); + ASSERT(f(0xffffff00, 8) == 0xffffffff); +#else + ASSERT(f(0x3fffffff8, 3) == 0x7fffffff); + ASSERT(f(0xffffffc080000000, 31) == 0xffffffffffffff81); + ASSERT(f(0xffffff00, 8) == 0xffffff); + ASSERT(f(0xfe00000000, 33) == 0x7f); + ASSERT(f(0x1ffffc00000000, 34) == 0x7ffff); + ASSERT(f(0xfffffff800000000, 29) == 0xffffffffffffffc0); + ASSERT(f(0x8000000000000000, 63) == 0xffffffffffffffff); + ASSERT(f(0x4000000000000000, 62) == 1); + ASSERT(f(0x2000000000000000, 61) == 1); + ASSERT(f(0x1000000000000000, 60) == 1); + ASSERT(f(0x81000000000000, 48) == 0x81); + ASSERT(f(0x2000000000000, 49) == 1); + ASSERT(f(0x10000000000, 40) == 1); + ASSERT(f(0x7f800000000000, 47) == 0xff); + ASSERT(f(0x100000000000000, 56) == 1); + ASSERT(f(0xffffff0000000000, 40) == 0xffffffffffffffff); + ASSERT(f(0xfffffffe00000000, 33) == 0xffffffffffffffff); + ASSERT(f(0x8000000000000001, 63) == 0xffffffffffffffff); + ASSERT(f(0x1000000000000, 48) == 1); + ASSERT(f(0xffff800000000000, 47) == 0xffffffffffffffff); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/rshr_u.c b/tests/rshr_u.c new file mode 100644 index 0000000..082b9ed --- /dev/null +++ b/tests/rshr_u.c @@ -0,0 +1,62 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_rshr_u(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0xfe, 1) == 0x7f); + ASSERT(f(0x1fffc, 2) == 0x7fff); + ASSERT(f(0x80000000, 31) == 1); + ASSERT(f(0x40000000, 30) == 1); + ASSERT(f(0x20000000, 29) == 1); + ASSERT(f(0x10000000, 28) == 1); + ASSERT(f(0x810000, 16) == 0x81); + ASSERT(f(0x20000, 17) == 1); + ASSERT(f(0x40000, 18) == 1); + ASSERT(f(0x7f8000, 15) == 0xff); + ASSERT(f(0x1000000, 24) == 1); + ASSERT(f(0xffffff00, 8) == 0xffffff); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); +#if EJIT_WORDSIZE == 32 + ASSERT(f(0xfffffff8, 3) == 0x1fffffff); +#else + ASSERT(f(0x3fffffff8, 3) == 0x7fffffff); + ASSERT(f(0xffffffc080000000, 31) == 0x1ffffff81); + ASSERT(f(0xfe00000000, 33) == 0x7f); + ASSERT(f(0x1ffffc00000000, 34) == 0x7ffff); + ASSERT(f(0xfffffff800000000, 29) == 0x7ffffffc0); + ASSERT(f(0x8000000000000000, 63) == 1); + ASSERT(f(0x4000000000000000, 62) == 1); + ASSERT(f(0x2000000000000000, 61) == 1); + ASSERT(f(0x1000000000000000, 60) == 1); + ASSERT(f(0x81000000000000, 48) == 0x81); + ASSERT(f(0x2000000000000, 49) == 1); + ASSERT(f(0x10000000000, 40) == 1); + ASSERT(f(0x7f800000000000, 47) == 0xff); + ASSERT(f(0x100000000000000, 56) == 1); + ASSERT(f(0xffffff0000000000, 40) == 0xffffff); + ASSERT(f(0xfffffffe00000000, 33) == 0x7fffffff); + ASSERT(f(0x8000000000000001, 63) == 1); + ASSERT(f(0x1000000000000, 48) == 1); + ASSERT(f(0xffff800000000000, 47) == 0x1ffff); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/sqrtr_d.c b/tests/sqrtr_d.c new file mode 100644 index 0000000..f07b94f --- /dev/null +++ b/tests/sqrtr_d.c @@ -0,0 +1,25 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_sqrtr_d(j, JIT_F0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + double (*f)(double) = jit_end(j, NULL); + + ASSERT(f(0.0) == 0.0); + ASSERT(f(4.0) == 2.0); + ASSERT(f(-4.0) != f(-4.0)); // nan +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/sqrtr_f.c b/tests/sqrtr_f.c new file mode 100644 index 0000000..7735f68 --- /dev/null +++ b/tests/sqrtr_f.c @@ -0,0 +1,25 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_sqrtr_f(j, JIT_F0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + float (*f)(float) = jit_end(j, NULL); + + ASSERT(f(0.0) == 0.0); + ASSERT(f(4.0) == 2.0); + ASSERT(f(-4.0) != f(-4.0)); // nan +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/sti_c.c b/tests/sti_c.c new file mode 100644 index 0000000..e323b37 --- /dev/null +++ b/tests/sti_c.c @@ -0,0 +1,31 @@ +#include "test.h" + +static uint8_t data[] = { 0x12, 0x00, 0x34 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_INT8, JIT_R1)); + + jit_sti_c(j, &data[1], JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(int8_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34); + f(-1); + ASSERT(data[0] == 0x12); + ASSERT(data[1] == 0xff); + ASSERT(data[2] == 0x34); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/sti_d.c b/tests/sti_d.c new file mode 100644 index 0000000..9fce200 --- /dev/null +++ b/tests/sti_d.c @@ -0,0 +1,31 @@ +#include "test.h" + +static double data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_sti_d(j, &data[1], JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(double) = jit_end(j, NULL); + + ASSERT(data[0] == -1.0); + ASSERT(data[1] == 0.0); + ASSERT(data[2] == 0.5); + f(42.5); + ASSERT(data[0] == -1.0); + ASSERT(data[1] == 42.5); + ASSERT(data[2] == 0.5); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/sti_f.c b/tests/sti_f.c new file mode 100644 index 0000000..67a0c38 --- /dev/null +++ b/tests/sti_f.c @@ -0,0 +1,31 @@ +#include "test.h" + +static float data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_sti_f(j, &data[1], JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(float) = jit_end(j, NULL); + + ASSERT(data[0] == -1.0f); + ASSERT(data[1] == 0.0f); + ASSERT(data[2] == 0.5f); + f(42.5f); + ASSERT(data[0] == -1.0f); + ASSERT(data[1] == 42.5f); + ASSERT(data[2] == 0.5f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/sti_i.c b/tests/sti_i.c new file mode 100644 index 0000000..1f014fd --- /dev/null +++ b/tests/sti_i.c @@ -0,0 +1,31 @@ +#include "test.h" + +static uint32_t data[] = { 0x12121212, 0x00000000, 0x34343434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_INT32, JIT_R1)); + + jit_sti_i(j, &data[1], JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(int32_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34343434); + f(-1); + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0xffffffff); + ASSERT(data[2] == 0x34343434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/sti_l.c b/tests/sti_l.c new file mode 100644 index 0000000..36cca18 --- /dev/null +++ b/tests/sti_l.c @@ -0,0 +1,33 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint64_t data[] = { 0x1212121212121212, 0, 0x3434343434343434 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_INT64, JIT_R1)); + + jit_sti_l(j, &data[1], JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(int64_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x1212121212121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x3434343434343434); + f(-1); + ASSERT(data[0] == 0x1212121212121212); + ASSERT(data[1] == 0xffffffffffffffff); + ASSERT(data[2] == 0x3434343434343434); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/sti_s.c b/tests/sti_s.c new file mode 100644 index 0000000..00941ea --- /dev/null +++ b/tests/sti_s.c @@ -0,0 +1,31 @@ +#include "test.h" + +static uint16_t data[] = { 0x1212, 0x0000, 0x3434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_INT16, JIT_R1)); + + jit_sti_s(j, &data[1], JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(int16_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x1212); + ASSERT(data[1] == 0); + ASSERT(data[2] == 0x3434); + f(-1); + ASSERT(data[0] == 0x1212); + ASSERT(data[1] == 0xffff); + ASSERT(data[2] == 0x3434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/str_atomic.c b/tests/str_atomic.c new file mode 100644 index 0000000..5086078 --- /dev/null +++ b/tests/str_atomic.c @@ -0,0 +1,32 @@ +#include "test.h" + +static long data[] = { 0x12121212, 0x00000000, 0x34343434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_INT32, JIT_R1)); + + jit_str_atomic(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, int32_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34343434); + f(&data[1], 0x0f0f0f0f); + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x0f0f0f0f); + ASSERT(data[2] == 0x34343434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/str_c.c b/tests/str_c.c new file mode 100644 index 0000000..492cc73 --- /dev/null +++ b/tests/str_c.c @@ -0,0 +1,32 @@ +#include "test.h" + +static uint8_t data[] = { 0x12, 0x00, 0x34 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_INT8, JIT_R1)); + + jit_str_c(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, int8_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34); + f(&data[1], -1); + ASSERT(data[0] == 0x12); + ASSERT(data[1] == 0xff); + ASSERT(data[2] == 0x34); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/str_d.c b/tests/str_d.c new file mode 100644 index 0000000..41db4fd --- /dev/null +++ b/tests/str_d.c @@ -0,0 +1,32 @@ +#include "test.h" + +static double data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_str_d(j, JIT_R0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, double) = jit_end(j, NULL); + + ASSERT(data[0] == -1.0); + ASSERT(data[1] == 0.0); + ASSERT(data[2] == 0.5); + f(&data[1], 42.5); + ASSERT(data[0] == -1.0); + ASSERT(data[1] == 42.5); + ASSERT(data[2] == 0.5); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/str_f.c b/tests/str_f.c new file mode 100644 index 0000000..7b394ae --- /dev/null +++ b/tests/str_f.c @@ -0,0 +1,32 @@ +#include "test.h" + +static float data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_str_f(j, JIT_R0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, float) = jit_end(j, NULL); + + ASSERT(data[0] == -1.0f); + ASSERT(data[1] == 0.0f); + ASSERT(data[2] == 0.5f); + f(&data[1], 42.5f); + ASSERT(data[0] == -1.0f); + ASSERT(data[1] == 42.5f); + ASSERT(data[2] == 0.5f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/str_i.c b/tests/str_i.c new file mode 100644 index 0000000..39e88b9 --- /dev/null +++ b/tests/str_i.c @@ -0,0 +1,32 @@ +#include "test.h" + +static uint32_t data[] = { 0x12121212, 0x00000000, 0x34343434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_INT32, JIT_R1)); + + jit_str_i(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, int32_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34343434); + f(&data[1], -1); + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0xffffffff); + ASSERT(data[2] == 0x34343434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/str_l.c b/tests/str_l.c new file mode 100644 index 0000000..c3338ff --- /dev/null +++ b/tests/str_l.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint64_t data[] = { 0x1212121212121212, 0, 0x3434343434343434 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_INT64, JIT_R1)); + + jit_str_l(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, int64_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x1212121212121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x3434343434343434); + f(&data[1], -1); + ASSERT(data[0] == 0x1212121212121212); + ASSERT(data[1] == 0xffffffffffffffff); + ASSERT(data[2] == 0x3434343434343434); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/str_s.c b/tests/str_s.c new file mode 100644 index 0000000..d2e7e45 --- /dev/null +++ b/tests/str_s.c @@ -0,0 +1,32 @@ +#include "test.h" + +static uint16_t data[] = { 0x1212, 0x0000, 0x3434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_INT16, JIT_R1)); + + jit_str_s(j, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, int16_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x1212); + ASSERT(data[1] == 0); + ASSERT(data[2] == 0x3434); + f(&data[1], -1); + ASSERT(data[0] == 0x1212); + ASSERT(data[1] == 0xffff); + ASSERT(data[2] == 0x3434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxi_c.c b/tests/stxi_c.c new file mode 100644 index 0000000..828a9d4 --- /dev/null +++ b/tests/stxi_c.c @@ -0,0 +1,32 @@ +#include "test.h" + +static uint8_t data[] = { 0x12, 0x00, 0x34 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_INT8, JIT_R1)); + + jit_stxi_c(j, (uintptr_t)data, JIT_R2, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(jit_word_t, int8_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34); + f(1, -1); + ASSERT(data[0] == 0x12); + ASSERT(data[1] == 0xff); + ASSERT(data[2] == 0x34); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxi_d.c b/tests/stxi_d.c new file mode 100644 index 0000000..85b3a4a --- /dev/null +++ b/tests/stxi_d.c @@ -0,0 +1,32 @@ +#include "test.h" + +static double data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_stxi_d(j, (uintptr_t)data, JIT_R2, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(jit_word_t, double) = jit_end(j, NULL); + + ASSERT(data[0] == -1.0); + ASSERT(data[1] == 0.0); + ASSERT(data[2] == 0.5); + f(8, 42.5); + ASSERT(data[0] == -1.0); + ASSERT(data[1] == 42.5); + ASSERT(data[2] == 0.5); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxi_f.c b/tests/stxi_f.c new file mode 100644 index 0000000..7325b3c --- /dev/null +++ b/tests/stxi_f.c @@ -0,0 +1,32 @@ +#include "test.h" + +static float data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_stxi_f(j, (uintptr_t)data, JIT_R2, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(jit_word_t, float) = jit_end(j, NULL); + + ASSERT(data[0] == -1.0f); + ASSERT(data[1] == 0.0f); + ASSERT(data[2] == 0.5f); + f(4, 42.5f); + ASSERT(data[0] == -1.0f); + ASSERT(data[1] == 42.5f); + ASSERT(data[2] == 0.5f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxi_i.c b/tests/stxi_i.c new file mode 100644 index 0000000..37bfd8f --- /dev/null +++ b/tests/stxi_i.c @@ -0,0 +1,32 @@ +#include "test.h" + +static uint32_t data[] = { 0x12121212, 0x00000000, 0x34343434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_INT32, JIT_R1)); + + jit_stxi_i(j, (uintptr_t)data, JIT_R2, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(jit_word_t, int32_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34343434); + f(4, -1); + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0xffffffff); + ASSERT(data[2] == 0x34343434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxi_l.c b/tests/stxi_l.c new file mode 100644 index 0000000..a758fa9 --- /dev/null +++ b/tests/stxi_l.c @@ -0,0 +1,34 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint64_t data[] = { 0x1212121212121212, 0, 0x3434343434343434 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_INT64, JIT_R1)); + + jit_stxi_l(j, (uintptr_t)data, JIT_R2, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(jit_word_t, int64_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x1212121212121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x3434343434343434); + f(8, -1); + ASSERT(data[0] == 0x1212121212121212); + ASSERT(data[1] == 0xffffffffffffffff); + ASSERT(data[2] == 0x3434343434343434); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxi_s.c b/tests/stxi_s.c new file mode 100644 index 0000000..d3aebc3 --- /dev/null +++ b/tests/stxi_s.c @@ -0,0 +1,32 @@ +#include "test.h" + +static uint16_t data[] = { 0x1212, 0x0000, 0x3434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_INT16, JIT_R1)); + + jit_stxi_s(j, (uintptr_t)data, JIT_R2, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(jit_word_t, int16_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x1212); + ASSERT(data[1] == 0); + ASSERT(data[2] == 0x3434); + f(2, -1); + ASSERT(data[0] == 0x1212); + ASSERT(data[1] == 0xffff); + ASSERT(data[2] == 0x3434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxr_c.c b/tests/stxr_c.c new file mode 100644 index 0000000..329083b --- /dev/null +++ b/tests/stxr_c.c @@ -0,0 +1,33 @@ +#include "test.h" + +static uint8_t data[] = { 0x12, 0x00, 0x34 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_3(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_INT8, JIT_R1)); + + jit_stxr_c(j, JIT_R0, JIT_R2, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, jit_word_t, int8_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34); + f(data, 1, -1); + ASSERT(data[0] == 0x12); + ASSERT(data[1] == 0xff); + ASSERT(data[2] == 0x34); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxr_d.c b/tests/stxr_d.c new file mode 100644 index 0000000..ee6e86a --- /dev/null +++ b/tests/stxr_d.c @@ -0,0 +1,33 @@ +#include "test.h" + +static double data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_3(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_stxr_d(j, JIT_R0, JIT_R2, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, jit_word_t, double) = jit_end(j, NULL); + + ASSERT(data[0] == -1.0); + ASSERT(data[1] == 0.0); + ASSERT(data[2] == 0.5); + f(data, 8, 42.5); + ASSERT(data[0] == -1.0); + ASSERT(data[1] == 42.5); + ASSERT(data[2] == 0.5); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxr_f.c b/tests/stxr_f.c new file mode 100644 index 0000000..d5bf9bf --- /dev/null +++ b/tests/stxr_f.c @@ -0,0 +1,33 @@ +#include "test.h" + +static float data[] = { -1.0, 0.0, 0.5 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_3(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_stxr_f(j, JIT_R0, JIT_R2, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, jit_word_t, float) = jit_end(j, NULL); + + ASSERT(data[0] == -1.0f); + ASSERT(data[1] == 0.0f); + ASSERT(data[2] == 0.5f); + f(data, 4, 42.5f); + ASSERT(data[0] == -1.0f); + ASSERT(data[1] == 42.5f); + ASSERT(data[2] == 0.5f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxr_i.c b/tests/stxr_i.c new file mode 100644 index 0000000..5ee69e7 --- /dev/null +++ b/tests/stxr_i.c @@ -0,0 +1,33 @@ +#include "test.h" + +static uint32_t data[] = { 0x12121212, 0x00000000, 0x34343434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_3(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_INT32, JIT_R1)); + + jit_stxr_i(j, JIT_R0, JIT_R2, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, jit_word_t, int32_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34343434); + f(data, 4, -1); + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0xffffffff); + ASSERT(data[2] == 0x34343434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxr_l.c b/tests/stxr_l.c new file mode 100644 index 0000000..6562dda --- /dev/null +++ b/tests/stxr_l.c @@ -0,0 +1,35 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + static uint64_t data[] = { 0x1212121212121212, 0, 0x3434343434343434 }; + + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_3(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_INT64, JIT_R1)); + + jit_stxr_l(j, JIT_R0, JIT_R2, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, jit_word_t, int64_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x1212121212121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x3434343434343434); + f(data, 8, -1); + ASSERT(data[0] == 0x1212121212121212); + ASSERT(data[1] == 0xffffffffffffffff); + ASSERT(data[2] == 0x3434343434343434); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/stxr_s.c b/tests/stxr_s.c new file mode 100644 index 0000000..8505892 --- /dev/null +++ b/tests/stxr_s.c @@ -0,0 +1,33 @@ +#include "test.h" + +static uint16_t data[] = { 0x1212, 0x0000, 0x3434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_3(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R2), + jit_operand_gpr (JIT_OPERAND_ABI_INT16, JIT_R1)); + + jit_stxr_s(j, JIT_R0, JIT_R2, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, jit_word_t, int16_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x1212); + ASSERT(data[1] == 0); + ASSERT(data[2] == 0x3434); + f(data, 2, -1); + ASSERT(data[0] == 0x1212); + ASSERT(data[1] == 0xffff); + ASSERT(data[2] == 0x3434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/subr.c b/tests/subr.c new file mode 100644 index 0000000..e2c22bc --- /dev/null +++ b/tests/subr.c @@ -0,0 +1,26 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_subr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + int (*f)(int, int) = ret; + ASSERT(f(42, 69) == -27); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/subr_d.c b/tests/subr_d.c new file mode 100644 index 0000000..65b5ecd --- /dev/null +++ b/tests/subr_d.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F1)); + + jit_subr_d(j, JIT_F0, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_d(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + double (*f)(double, double) = ret; + ASSERT(f(42., 69.) == -27.); + ASSERT(f(42., 69.5) == -27.5); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/subr_f.c b/tests/subr_f.c new file mode 100644 index 0000000..3fdf071 --- /dev/null +++ b/tests/subr_f.c @@ -0,0 +1,27 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0), + jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F1)); + + jit_subr_f(j, JIT_F0, JIT_F0, JIT_F1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr_f(j, JIT_F0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + float (*f)(float, float) = ret; + ASSERT(f(42.f, 69.f) == -27.f); + ASSERT(f(42.0f, 69.5f) == -27.5f); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/subx.c b/tests/subx.c new file mode 100644 index 0000000..b2bb3c2 --- /dev/null +++ b/tests/subx.c @@ -0,0 +1,63 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_movi(j, JIT_R2, 0); + jit_subcr(j, JIT_R0, JIT_R0, JIT_R1); + jit_subxi(j, JIT_R2, JIT_R2, 0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R2); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0, 0) == 0); + +#if EJIT_WORDSIZE == 32 + /* carry */ + ASSERT(f(0x7fffffff, 0xffffffff) == 0xffffffff); + /* overflow */ + ASSERT(f(0x80000000, 1) == 0); + /* carry */ + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + /* overflow */ + ASSERT(f(0x80000000, 0x7fffffff) == 0); + /* carry+overflow */ + ASSERT(f(1, 0x80000000) == 0xffffffff); +#else + /* carry */ + ASSERT(f(0x7fffffff, 0xffffffff) == -1); + /* nothing */ + ASSERT(f(0x80000000, 1) == 0); + /* carry */ + ASSERT(f(0x7fffffff, 0x80000000) == -1); + /* nothing */ + ASSERT(f(0x80000000, 0x7fffffff) == 0); + /* carry */ + ASSERT(f(1, 0x80000000) == -1); + /* carry */ + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == -1); + /* overflow */ + ASSERT(f(0x8000000000000000, 1) == 0); + /* carry */ + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == -1); + /* overflow */ + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 0); + /* carry+overflow */ + ASSERT(f(1, 0x8000000000000000) == -1); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/swap_atomic.c b/tests/swap_atomic.c new file mode 100644 index 0000000..5803164 --- /dev/null +++ b/tests/swap_atomic.c @@ -0,0 +1,32 @@ +#include "test.h" + +static long data[] = { 0x12121212, 0x00000000, 0x34343434 }; + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_POINTER, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_INT32, JIT_R1)); + + jit_swap_atomic(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_ret(j); + + void (*f)(void*, int32_t) = jit_end(j, NULL); + + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x00); + ASSERT(data[2] == 0x34343434); + f(&data[1], 0x0f0f0f0f); + ASSERT(data[0] == 0x12121212); + ASSERT(data[1] == 0x0f0f0f0f); + ASSERT(data[2] == 0x34343434); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/test-bcode-absr_f b/tests/test-bcode-absr_f Binary files differnew file mode 100755 index 0000000..8e902b5 --- /dev/null +++ b/tests/test-bcode-absr_f diff --git a/tests/truncr_d_i.c b/tests/truncr_d_i.c new file mode 100644 index 0000000..09e163e --- /dev/null +++ b/tests/truncr_d_i.c @@ -0,0 +1,30 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_truncr_d_i(j, JIT_R0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + int (*f)(double) = jit_end(j, NULL); + + ASSERT(f(0.0) == 0); + ASSERT(f(-0.0) == 0); + ASSERT(f(0.5) == 0); + ASSERT(f(-0.5) == 0); + ASSERT(f(1.5) == 1); + ASSERT(f(-1.5) == -1); + ASSERT(f(2.5) == 2); + ASSERT(f(-2.5) == -2); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/truncr_d_l.c b/tests/truncr_d_l.c new file mode 100644 index 0000000..ee9b477 --- /dev/null +++ b/tests/truncr_d_l.c @@ -0,0 +1,32 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_DOUBLE, JIT_F0)); + + jit_truncr_d_l(j, JIT_R0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + int64_t (*f)(double) = jit_end(j, NULL); + + ASSERT(f(0.0) == 0); + ASSERT(f(-0.0) == 0); + ASSERT(f(0.5) == 0); + ASSERT(f(-0.5) == 0); + ASSERT(f(1.5) == 1); + ASSERT(f(-1.5) == -1); + ASSERT(f(2.5) == 2); + ASSERT(f(-2.5) == -2); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/truncr_f_i.c b/tests/truncr_f_i.c new file mode 100644 index 0000000..0314753 --- /dev/null +++ b/tests/truncr_f_i.c @@ -0,0 +1,30 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_truncr_f_i(j, JIT_R0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + int (*f)(float) = jit_end(j, NULL); + + ASSERT(f(0.0) == 0); + ASSERT(f(-0.0) == 0); + ASSERT(f(0.5) == 0); + ASSERT(f(-0.5) == 0); + ASSERT(f(1.5) == 1); + ASSERT(f(-1.5) == -1); + ASSERT(f(2.5) == 2); + ASSERT(f(-2.5) == -2); +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/truncr_f_l.c b/tests/truncr_f_l.c new file mode 100644 index 0000000..ad95744 --- /dev/null +++ b/tests/truncr_f_l.c @@ -0,0 +1,32 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ +#if EJIT_WORDSIZE > 32 + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_fpr (JIT_OPERAND_ABI_FLOAT, JIT_F0)); + + jit_truncr_f_l(j, JIT_R0, JIT_F0); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + int64_t (*f)(float) = jit_end(j, NULL); + + ASSERT(f(0.0) == 0); + ASSERT(f(-0.0) == 0); + ASSERT(f(0.5) == 0); + ASSERT(f(-0.5) == 0); + ASSERT(f(1.5) == 1); + ASSERT(f(-1.5) == -1); + ASSERT(f(2.5) == 2); + ASSERT(f(-2.5) == -2); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/xori.c b/tests/xori.c new file mode 100644 index 0000000..16197ce --- /dev/null +++ b/tests/xori.c @@ -0,0 +1,31 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_1(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0)); + + jit_xori(j, JIT_R0, JIT_R0, 1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t) = ret; + + ASSERT(f(0x7fffffff) == 0x7ffffffe); + ASSERT(f(0x80000000) == 0x80000001); +#if EJIT_WORDSIZE == 64 + ASSERT(f(0x7fffffffffffffff) == 0x7ffffffffffffffe); + ASSERT(f(0x8000000000000000) == 0x8000000000000001); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/tests/xorr.c b/tests/xorr.c new file mode 100644 index 0000000..123bfb7 --- /dev/null +++ b/tests/xorr.c @@ -0,0 +1,48 @@ +#include "test.h" + +static void +run_test(jit_state_t *j, uint8_t *arena_base, size_t arena_size) +{ + jit_begin(j, arena_base, arena_size); + size_t align = jit_enter_jit_abi(j, 0, 0, 0); + jit_load_args_2(j, jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R0), + jit_operand_gpr (JIT_OPERAND_ABI_WORD, JIT_R1)); + + jit_xorr(j, JIT_R0, JIT_R0, JIT_R1); + jit_leave_jit_abi(j, 0, 0, align); + jit_retr(j, JIT_R0); + + size_t size = 0; + void* ret = jit_end(j, &size); + + jit_word_t (*f)(jit_word_t, jit_word_t) = ret; + + ASSERT(f(0x7fffffff, 1) == 0x7ffffffe); + ASSERT(f(1, 0x7fffffff) == 0x7ffffffe); + ASSERT(f(0x80000000, 1) == 0x80000001); + ASSERT(f(1, 0x80000000) == 0x80000001); + ASSERT(f(0x7fffffff, 0x80000000) == 0xffffffff); + ASSERT(f(0x80000000, 0x7fffffff) == 0xffffffff); + ASSERT(f(0x7fffffff, 0xffffffff) == 0x80000000); + ASSERT(f(0xffffffff, 0x7fffffff) == 0x80000000); + ASSERT(f(0xffffffff, 0xffffffff) == 0); + ASSERT(f(0x7fffffff, 0) == 0x7fffffff); + ASSERT(f(0, 0x7fffffff) == 0x7fffffff); +#if EJIT_WORDSIZE == 64 + ASSERT(f(0x7fffffffffffffff, 1) == 0x7ffffffffffffffe); + ASSERT(f(1, 0x7fffffffffffffff) == 0x7ffffffffffffffe); + ASSERT(f(0x8000000000000000, 1) == 0x8000000000000001); + ASSERT(f(1, 0x8000000000000000) == 0x8000000000000001); + ASSERT(f(0x7fffffffffffffff, 0x8000000000000000) == 0xffffffffffffffff); + ASSERT(f(0x8000000000000000, 0x7fffffffffffffff) == 0xffffffffffffffff); + ASSERT(f(0x7fffffffffffffff, 0xffffffffffffffff) == 0x8000000000000000); + ASSERT(f(0xffffffffffffffff, 0x7fffffffffffffff) == 0x8000000000000000); + ASSERT(f(0xffffffffffffffff, 0xffffffffffffffff) == 0); +#endif +} + +int +main (int argc, char *argv[]) +{ + return main_helper(argc, argv, run_test); +} diff --git a/uncrustify.conf b/uncrustify.conf new file mode 100644 index 0000000..5e01d98 --- /dev/null +++ b/uncrustify.conf @@ -0,0 +1,3708 @@ +# Uncrustify-0.78.1_f + +# +# General options +# + +# The type of line endings. +# +# Default: auto +newlines = auto # lf/crlf/cr/auto + +# The original size of tabs in the input. +# +# Default: 8 +input_tab_size = 8 # unsigned number + +# The size of tabs in the output (only used if align_with_tabs=true). +# +# Default: 8 +output_tab_size = 8 # unsigned number + +# The ASCII value of the string escape char, usually 92 (\) or (Pawn) 94 (^). +# +# Default: 92 +string_escape_char = 92 # unsigned number + +# Alternate string escape char (usually only used for Pawn). +# Only works right before the quote char. +string_escape_char2 = 0 # unsigned number + +# Replace tab characters found in string literals with the escape sequence \t +# instead. +string_replace_tab_chars = false # true/false + +# Allow interpreting '>=' and '>>=' as part of a template in code like +# 'void f(list<list<B>>=val);'. If true, 'assert(x<0 && y>=3)' will be broken. +# Improvements to template detection may make this option obsolete. +tok_split_gte = false # true/false + +# Disable formatting of NL_CONT ('\\n') ended lines (e.g. multi-line macros). +disable_processing_nl_cont = false # true/false + +# Specify the marker used in comments to disable processing of part of the +# file. +# +# Default: *INDENT-OFF* +disable_processing_cmt = " *INDENT-OFF*" # string + +# Specify the marker used in comments to (re)enable processing in a file. +# +# Default: *INDENT-ON* +enable_processing_cmt = " *INDENT-ON*" # string + +# Enable parsing of digraphs. +enable_digraphs = false # true/false + +# Option to allow both disable_processing_cmt and enable_processing_cmt +# strings, if specified, to be interpreted as ECMAScript regular expressions. +# If true, a regex search will be performed within comments according to the +# specified patterns in order to disable/enable processing. +processing_cmt_as_regex = false # true/false + +# Add or remove the UTF-8 BOM (recommend 'remove'). +utf8_bom = ignore # ignore/add/remove/force/not_defined + +# If the file contains bytes with values between 128 and 255, but is not +# UTF-8, then output as UTF-8. +utf8_byte = false # true/false + +# Force the output encoding to UTF-8. +utf8_force = false # true/false + +# +# Spacing options +# + +# Add or remove space around non-assignment symbolic operators ('+', '/', '%', +# '<<', and so forth). +sp_arith = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around arithmetic operators '+' and '-'. +# +# Overrides sp_arith. +sp_arith_additive = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around assignment operator '=', '+=', etc. +sp_assign = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around '=' in C++11 lambda capture specifications. +# +# Overrides sp_assign. +sp_cpp_lambda_assign = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the capture specification of a C++11 lambda when +# an argument list is present, as in '[] <here> (int x){ ... }'. +sp_cpp_lambda_square_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the capture specification of a C++11 lambda with +# no argument list is present, as in '[] <here> { ... }'. +sp_cpp_lambda_square_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the opening parenthesis and before the closing +# parenthesis of a argument list of a C++11 lambda, as in +# '[]( <here> ){ ... }' +# with an empty list. +sp_cpp_lambda_argument_list_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the opening parenthesis and before the closing +# parenthesis of a argument list of a C++11 lambda, as in +# '[]( <here> int x <here> ){ ... }'. +sp_cpp_lambda_argument_list = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the argument list of a C++11 lambda, as in +# '[](int x) <here> { ... }'. +sp_cpp_lambda_paren_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a lambda body and its call operator of an +# immediately invoked lambda, as in '[]( ... ){ ... } <here> ( ... )'. +sp_cpp_lambda_fparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around assignment operator '=' in a prototype. +# +# If set to ignore, use sp_assign. +sp_assign_default = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before assignment operator '=', '+=', etc. +# +# Overrides sp_assign. +sp_before_assign = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after assignment operator '=', '+=', etc. +# +# Overrides sp_assign. +sp_after_assign = ignore # ignore/add/remove/force/not_defined + +# Add or remove space in 'enum {'. +# +# Default: add +sp_enum_brace = add # ignore/add/remove/force/not_defined + +# Add or remove space in 'NS_ENUM ('. +sp_enum_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around assignment '=' in enum. +sp_enum_assign = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before assignment '=' in enum. +# +# Overrides sp_enum_assign. +sp_enum_before_assign = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after assignment '=' in enum. +# +# Overrides sp_enum_assign. +sp_enum_after_assign = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around assignment ':' in enum. +sp_enum_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around preprocessor '##' concatenation operator. +# +# Default: add +sp_pp_concat = remove # ignore/add/remove/force/not_defined + +# Add or remove space after preprocessor '#' stringify operator. +# Also affects the '#@' charizing operator. +sp_pp_stringify = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before preprocessor '#' stringify operator +# as in '#define x(y) L#y'. +sp_before_pp_stringify = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around boolean operators '&&' and '||'. +sp_bool = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around compare operator '<', '>', '==', etc. +sp_compare = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside '(' and ')'. +sp_inside_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between nested parentheses, i.e. '((' vs. ') )'. +sp_paren_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between back-to-back parentheses, i.e. ')(' vs. ') ('. +sp_cparen_oparen = ignore # ignore/add/remove/force/not_defined + +# Whether to balance spaces inside nested parentheses. +sp_balance_nested_parens = false # true/false + +# Add or remove space between ')' and '{'. +sp_paren_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between nested braces, i.e. '{{' vs. '{ {'. +sp_brace_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before pointer star '*'. +sp_before_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before pointer star '*' that isn't followed by a +# variable name. If set to ignore, sp_before_ptr_star is used instead. +sp_before_unnamed_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before pointer star '*' that is followed by a qualifier. +# If set to ignore, sp_before_unnamed_ptr_star is used instead. +sp_before_qualifier_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before pointer star '*' that is followed by 'operator' keyword. +# If set to ignore, sp_before_unnamed_ptr_star is used instead. +sp_before_operator_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before pointer star '*' that is followed by +# a class scope (as in 'int *MyClass::method()') or namespace scope +# (as in 'int *my_ns::func()'). +# If set to ignore, sp_before_unnamed_ptr_star is used instead. +sp_before_scope_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before pointer star '*' that is followed by '::', +# as in 'int *::func()'. +# If set to ignore, sp_before_unnamed_ptr_star is used instead. +sp_before_global_scope_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a qualifier and a pointer star '*' that isn't +# followed by a variable name, as in '(char const *)'. If set to ignore, +# sp_before_ptr_star is used instead. +sp_qualifier_unnamed_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between pointer stars '*', as in 'int ***a;'. +sp_between_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between pointer star '*' and reference '&', as in 'int *& a;'. +sp_between_ptr_ref = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after pointer star '*', if followed by a word. +# +# Overrides sp_type_func. +sp_after_ptr_star = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after pointer caret '^', if followed by a word. +sp_after_ptr_block_caret = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after pointer star '*', if followed by a qualifier. +sp_after_ptr_star_qualifier = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after a pointer star '*', if followed by a function +# prototype or function definition. +# +# Overrides sp_after_ptr_star and sp_type_func. +sp_after_ptr_star_func = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after a pointer star '*' in the trailing return of a +# function prototype or function definition. +sp_after_ptr_star_trailing = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between the pointer star '*' and the name of the variable +# in a function pointer definition. +sp_ptr_star_func_var = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between the pointer star '*' and the name of the type +# in a function pointer type definition. +sp_ptr_star_func_type = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after a pointer star '*', if followed by an open +# parenthesis, as in 'void* (*)()'. +sp_ptr_star_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before a pointer star '*', if followed by a function +# prototype or function definition. If set to ignore, sp_before_ptr_star is +# used instead. +sp_before_ptr_star_func = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a qualifier and a pointer star '*' followed by +# the name of the function in a function prototype or definition, as in +# 'char const *foo()`. If set to ignore, sp_before_ptr_star is used instead. +sp_qualifier_ptr_star_func = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before a pointer star '*' in the trailing return of a +# function prototype or function definition. +sp_before_ptr_star_trailing = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a qualifier and a pointer star '*' in the +# trailing return of a function prototype or function definition, as in +# 'auto foo() -> char const *'. +sp_qualifier_ptr_star_trailing = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before a reference sign '&'. +sp_before_byref = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before a reference sign '&' that isn't followed by a +# variable name. If set to ignore, sp_before_byref is used instead. +sp_before_unnamed_byref = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after reference sign '&', if followed by a word. +# +# Overrides sp_type_func. +sp_after_byref = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after a reference sign '&', if followed by a function +# prototype or function definition. +# +# Overrides sp_after_byref and sp_type_func. +sp_after_byref_func = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before a reference sign '&', if followed by a function +# prototype or function definition. +sp_before_byref_func = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after a reference sign '&', if followed by an open +# parenthesis, as in 'char& (*)()'. +sp_byref_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between type and word. In cases where total removal of +# whitespace would be a syntax error, a value of 'remove' is treated the same +# as 'force'. +# +# This also affects some other instances of space following a type that are +# not covered by other options; for example, between the return type and +# parenthesis of a function type template argument, between the type and +# parenthesis of an array parameter, or between 'decltype(...)' and the +# following word. +# +# Default: force +sp_after_type = force # ignore/add/remove/force/not_defined + +# Add or remove space between 'decltype(...)' and word, +# brace or function call. +sp_after_decltype = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove space before the parenthesis in the D constructs +# 'template Foo(' and 'class Foo('. +sp_before_template_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'template' and '<'. +# If set to ignore, sp_before_angle is used. +sp_template_angle = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before '<'. +sp_before_angle = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside '<' and '>'. +sp_inside_angle = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside '<>'. +# if empty. +sp_inside_angle_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '>' and ':'. +sp_angle_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after '>'. +sp_after_angle = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '>' and '(' as found in 'new List<byte>(foo);'. +sp_angle_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '>' and '()' as found in 'new List<byte>();'. +sp_angle_paren_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '>' and a word as in 'List<byte> m;' or +# 'template <typename T> static ...'. +sp_angle_word = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '>' and '>' in '>>' (template stuff). +# +# Default: add +sp_angle_shift = add # ignore/add/remove/force/not_defined + +# (C++11) Permit removal of the space between '>>' in 'foo<bar<int> >'. Note +# that sp_angle_shift cannot remove the space without this option. +sp_permit_cpp11_shift = false # true/false + +# Add or remove space before '(' of control statements ('if', 'for', 'switch', +# 'while', etc.). +sp_before_sparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside '(' and ')' of control statements other than +# 'for'. +sp_inside_sparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after '(' of control statements other than 'for'. +# +# Overrides sp_inside_sparen. +sp_inside_sparen_open = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before ')' of control statements other than 'for'. +# +# Overrides sp_inside_sparen. +sp_inside_sparen_close = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside '(' and ')' of 'for' statements. +sp_inside_for = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after '(' of 'for' statements. +# +# Overrides sp_inside_for. +sp_inside_for_open = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before ')' of 'for' statements. +# +# Overrides sp_inside_for. +sp_inside_for_close = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '((' or '))' of control statements. +sp_sparen_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after ')' of control statements. +sp_after_sparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between ')' and '{' of control statements. +sp_sparen_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'do' and '{'. +sp_do_brace_open = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '}' and 'while'. +sp_brace_close_while = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'while' and '('. Overrides sp_before_sparen. +sp_while_paren_open = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove space between 'invariant' and '('. +sp_invariant_paren = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove space after the ')' in 'invariant (C) c'. +sp_after_invariant_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before empty statement ';' on 'if', 'for' and 'while'. +sp_special_semi = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before ';'. +# +# Default: remove +sp_before_semi = remove # ignore/add/remove/force/not_defined + +# Add or remove space before ';' in non-empty 'for' statements. +sp_before_semi_for = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before a semicolon of an empty left part of a for +# statement, as in 'for ( <here> ; ; )'. +sp_before_semi_for_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between the semicolons of an empty middle part of a for +# statement, as in 'for ( ; <here> ; )'. +sp_between_semi_for_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after ';', except when followed by a comment. +# +# Default: add +sp_after_semi = add # ignore/add/remove/force/not_defined + +# Add or remove space after ';' in non-empty 'for' statements. +# +# Default: force +sp_after_semi_for = force # ignore/add/remove/force/not_defined + +# Add or remove space after the final semicolon of an empty part of a for +# statement, as in 'for ( ; ; <here> )'. +sp_after_semi_for_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before '[' (except '[]'). +sp_before_square = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before '[' for a variable definition. +# +# Default: remove +sp_before_vardef_square = remove # ignore/add/remove/force/not_defined + +# Add or remove space before '[' for asm block. +sp_before_square_asm_block = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before '[]'. +sp_before_squares = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before C++17 structured bindings. +sp_cpp_before_struct_binding = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside a non-empty '[' and ']'. +sp_inside_square = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside '[]'. +# if empty. +sp_inside_square_empty = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space inside a non-empty Objective-C boxed array '@[' and +# ']'. If set to ignore, sp_inside_square is used. +sp_inside_square_oc_array = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after ',', i.e. 'a,b' vs. 'a, b'. +sp_after_comma = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before ',', i.e. 'a,b' vs. 'a ,b'. +# +# Default: remove +sp_before_comma = remove # ignore/add/remove/force/not_defined + +# (C#, Vala) Add or remove space between ',' and ']' in multidimensional array type +# like 'int[,,]'. +sp_after_mdatype_commas = ignore # ignore/add/remove/force/not_defined + +# (C#, Vala) Add or remove space between '[' and ',' in multidimensional array type +# like 'int[,,]'. +sp_before_mdatype_commas = ignore # ignore/add/remove/force/not_defined + +# (C#, Vala) Add or remove space between ',' in multidimensional array type +# like 'int[,,]'. +sp_between_mdatype_commas = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between an open parenthesis and comma, +# i.e. '(,' vs. '( ,'. +# +# Default: force +sp_paren_comma = force # ignore/add/remove/force/not_defined + +# Add or remove space between a type and ':'. +sp_type_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the variadic '...' when preceded by a +# non-punctuator. +# The value REMOVE will be overridden with FORCE +sp_after_ellipsis = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before the variadic '...' when preceded by a +# non-punctuator. +# The value REMOVE will be overridden with FORCE +sp_before_ellipsis = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a type and '...'. +sp_type_ellipsis = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a '*' and '...'. +sp_ptr_type_ellipsis = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between ')' and '...'. +sp_paren_ellipsis = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '&&' and '...'. +sp_byref_ellipsis = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between ')' and a qualifier such as 'const'. +sp_paren_qualifier = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between ')' and 'noexcept'. +sp_paren_noexcept = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after class ':'. +sp_after_class_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before class ':'. +sp_before_class_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after class constructor ':'. +# +# Default: add +sp_after_constr_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before class constructor ':'. +# +# Default: add +sp_before_constr_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before case ':'. +# +# Default: remove +sp_before_case_colon = remove # ignore/add/remove/force/not_defined + +# Add or remove space between 'operator' and operator sign. +sp_after_operator = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between the operator symbol and the open parenthesis, as +# in 'operator ++('. +sp_after_operator_sym = ignore # ignore/add/remove/force/not_defined + +# Overrides sp_after_operator_sym when the operator has no arguments, as in +# 'operator *()'. +sp_after_operator_sym_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after C/D cast, i.e. 'cast(int)a' vs. 'cast(int) a' or +# '(int)a' vs. '(int) a'. +sp_after_cast = ignore # ignore/add/remove/force/not_defined + +# Add or remove spaces inside cast parentheses. +sp_inside_paren_cast = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between the type and open parenthesis in a C++ cast, +# i.e. 'int(exp)' vs. 'int (exp)'. +sp_cpp_cast_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'sizeof' and '('. +sp_sizeof_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'sizeof' and '...'. +sp_sizeof_ellipsis = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'sizeof...' and '('. +sp_sizeof_ellipsis_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '...' and a parameter pack. +sp_ellipsis_parameter_pack = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a parameter pack and '...'. +sp_parameter_pack_ellipsis = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'decltype' and '('. +sp_decltype_paren = ignore # ignore/add/remove/force/not_defined + +# (Pawn) Add or remove space after the tag keyword. +sp_after_tag = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside enum '{' and '}'. +sp_inside_braces_enum = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside struct/union '{' and '}'. +sp_inside_braces_struct = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space inside Objective-C boxed dictionary '{' and '}' +sp_inside_braces_oc_dict = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after open brace in an unnamed temporary +# direct-list-initialization +# if statement is a brace_init_lst +# works only if sp_brace_brace is set to ignore. +sp_after_type_brace_init_lst_open = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before close brace in an unnamed temporary +# direct-list-initialization +# if statement is a brace_init_lst +# works only if sp_brace_brace is set to ignore. +sp_before_type_brace_init_lst_close = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside an unnamed temporary direct-list-initialization +# if statement is a brace_init_lst +# works only if sp_brace_brace is set to ignore +# works only if sp_before_type_brace_init_lst_close is set to ignore. +sp_inside_type_brace_init_lst = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside '{' and '}'. +sp_inside_braces = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside '{}'. +# if empty. +sp_inside_braces_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around trailing return operator '->'. +sp_trailing_return = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between return type and function name. A minimum of 1 +# is forced except for pointer return types. +sp_type_func = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between type and open brace of an unnamed temporary +# direct-list-initialization. +sp_type_brace_init_lst = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between function name and '(' on function declaration. +sp_func_proto_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between function name and '()' on function declaration +# if empty. +sp_func_proto_paren_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between function name and '(' with a typedef specifier. +sp_func_type_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between alias name and '(' of a non-pointer function type typedef. +sp_func_def_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between function name and '()' on function definition +# if empty. +sp_func_def_paren_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside empty function '()'. +# Overrides sp_after_angle unless use_sp_after_angle_always is set to true. +sp_inside_fparens = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside function '(' and ')'. +sp_inside_fparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside user functor '(' and ')'. +sp_func_call_user_inside_rparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside empty functor '()'. +# Overrides sp_after_angle unless use_sp_after_angle_always is set to true. +sp_inside_rparens = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside functor '(' and ')'. +sp_inside_rparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside the first parentheses in a function type, as in +# 'void (*x)(...)'. +sp_inside_tparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between the ')' and '(' in a function type, as in +# 'void (*x)(...)'. +sp_after_tparen_close = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between ']' and '(' when part of a function call. +sp_square_fparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between ')' and '{' of function. +sp_fparen_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between ')' and '{' of a function call in object +# initialization. +# +# Overrides sp_fparen_brace. +sp_fparen_brace_initializer = ignore # ignore/add/remove/force/not_defined + +# (Java) Add or remove space between ')' and '{{' of double brace initializer. +sp_fparen_dbrace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between function name and '(' on function calls. +sp_func_call_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between function name and '()' on function calls without +# parameters. If set to ignore (the default), sp_func_call_paren is used. +sp_func_call_paren_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between the user function name and '(' on function +# calls. You need to set a keyword to be a user function in the config file, +# like: +# set func_call_user tr _ i18n +sp_func_call_user_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside user function '(' and ')'. +sp_func_call_user_inside_fparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between nested parentheses with user functions, +# i.e. '((' vs. '( ('. +sp_func_call_user_paren_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a constructor/destructor and the open +# parenthesis. +sp_func_class_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a constructor without parameters or destructor +# and '()'. +sp_func_class_paren_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after 'return'. +# +# Default: force +sp_return = force # ignore/add/remove/force/not_defined + +# Add or remove space between 'return' and '('. +sp_return_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'return' and '{'. +sp_return_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '__attribute__' and '('. +sp_attribute_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'defined' and '(' in '#if defined (FOO)'. +sp_defined_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'throw' and '(' in 'throw (something)'. +sp_throw_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'throw' and anything other than '(' as in +# '@throw [...];'. +sp_after_throw = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'catch' and '(' in 'catch (something) { }'. +# If set to ignore, sp_before_sparen is used. +sp_catch_paren = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space between '@catch' and '(' +# in '@catch (something) { }'. If set to ignore, sp_catch_paren is used. +sp_oc_catch_paren = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space before Objective-C protocol list +# as in '@protocol Protocol<here><Protocol_A>' or '@interface MyClass : NSObject<here><MyProtocol>'. +sp_before_oc_proto_list = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space between class name and '(' +# in '@interface className(categoryName)<ProtocolName>:BaseClass' +sp_oc_classname_paren = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove space between 'version' and '(' +# in 'version (something) { }'. If set to ignore, sp_before_sparen is used. +sp_version_paren = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove space between 'scope' and '(' +# in 'scope (something) { }'. If set to ignore, sp_before_sparen is used. +sp_scope_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'super' and '(' in 'super (something)'. +# +# Default: remove +sp_super_paren = remove # ignore/add/remove/force/not_defined + +# Add or remove space between 'this' and '(' in 'this (something)'. +# +# Default: remove +sp_this_paren = remove # ignore/add/remove/force/not_defined + +# Add or remove space between a macro name and its definition. +sp_macro = remove # ignore/add/remove/force/not_defined + +# Add or remove space between a macro function ')' and its definition. +sp_macro_func = remove # ignore/add/remove/force/not_defined + +# Add or remove space between 'else' and '{' if on the same line. +sp_else_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '}' and 'else' if on the same line. +sp_brace_else = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '}' and the name of a typedef on the same line. +sp_brace_typedef = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before the '{' of a 'catch' statement, if the '{' and +# 'catch' are on the same line, as in 'catch (decl) <here> {'. +sp_catch_brace = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space before the '{' of a '@catch' statement, if the '{' +# and '@catch' are on the same line, as in '@catch (decl) <here> {'. +# If set to ignore, sp_catch_brace is used. +sp_oc_catch_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '}' and 'catch' if on the same line. +sp_brace_catch = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space between '}' and '@catch' if on the same line. +# If set to ignore, sp_brace_catch is used. +sp_oc_brace_catch = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'finally' and '{' if on the same line. +sp_finally_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between '}' and 'finally' if on the same line. +sp_brace_finally = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'try' and '{' if on the same line. +sp_try_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between get/set and '{' if on the same line. +sp_getset_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a variable and '{' for C++ uniform +# initialization. +sp_word_brace_init_lst = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between a variable and '{' for a namespace. +# +# Default: add +sp_word_brace_ns = add # ignore/add/remove/force/not_defined + +# Add or remove space before the '::' operator. +sp_before_dc = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the '::' operator. +sp_after_dc = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove around the D named array initializer ':' operator. +sp_d_array_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the '!' (not) unary operator. +# +# Default: remove +sp_not = remove # ignore/add/remove/force/not_defined + +# Add or remove space between two '!' (not) unary operators. +# If set to ignore, sp_not will be used. +sp_not_not = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the '~' (invert) unary operator. +# +# Default: remove +sp_inv = remove # ignore/add/remove/force/not_defined + +# Add or remove space after the '&' (address-of) unary operator. This does not +# affect the spacing after a '&' that is part of a type. +# +# Default: remove +sp_addr = remove # ignore/add/remove/force/not_defined + +# Add or remove space around the '.' or '->' operators. +# +# Default: remove +sp_member = remove # ignore/add/remove/force/not_defined + +# Add or remove space after the '*' (dereference) unary operator. This does +# not affect the spacing after a '*' that is part of a type. +# +# Default: remove +sp_deref = remove # ignore/add/remove/force/not_defined + +# Add or remove space after '+' or '-', as in 'x = -5' or 'y = +7'. +# +# Default: remove +sp_sign = remove # ignore/add/remove/force/not_defined + +# Add or remove space between '++' and '--' the word to which it is being +# applied, as in '(--x)' or 'y++;'. +# +# Default: remove +sp_incdec = remove # ignore/add/remove/force/not_defined + +# Add or remove space before a backslash-newline at the end of a line. +# +# Default: add +sp_before_nl_cont = add # ignore/add/remove/force/not_defined + +# (OC) Add or remove space after the scope '+' or '-', as in '-(void) foo;' +# or '+(int) bar;'. +sp_after_oc_scope = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space after the colon in message specs, +# i.e. '-(int) f:(int) x;' vs. '-(int) f: (int) x;'. +sp_after_oc_colon = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space before the colon in message specs, +# i.e. '-(int) f: (int) x;' vs. '-(int) f : (int) x;'. +sp_before_oc_colon = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space after the colon in immutable dictionary expression +# 'NSDictionary *test = @{@"foo" :@"bar"};'. +sp_after_oc_dict_colon = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space before the colon in immutable dictionary expression +# 'NSDictionary *test = @{@"foo" :@"bar"};'. +sp_before_oc_dict_colon = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space after the colon in message specs, +# i.e. '[object setValue:1];' vs. '[object setValue: 1];'. +sp_after_send_oc_colon = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space before the colon in message specs, +# i.e. '[object setValue:1];' vs. '[object setValue :1];'. +sp_before_send_oc_colon = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space after the (type) in message specs, +# i.e. '-(int)f: (int) x;' vs. '-(int)f: (int)x;'. +sp_after_oc_type = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space after the first (type) in message specs, +# i.e. '-(int) f:(int)x;' vs. '-(int)f:(int)x;'. +sp_after_oc_return_type = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space between '@selector' and '(', +# i.e. '@selector(msgName)' vs. '@selector (msgName)'. +# Also applies to '@protocol()' constructs. +sp_after_oc_at_sel = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space between '@selector(x)' and the following word, +# i.e. '@selector(foo) a:' vs. '@selector(foo)a:'. +sp_after_oc_at_sel_parens = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space inside '@selector' parentheses, +# i.e. '@selector(foo)' vs. '@selector( foo )'. +# Also applies to '@protocol()' constructs. +sp_inside_oc_at_sel_parens = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space before a block pointer caret, +# i.e. '^int (int arg){...}' vs. ' ^int (int arg){...}'. +sp_before_oc_block_caret = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space after a block pointer caret, +# i.e. '^int (int arg){...}' vs. '^ int (int arg){...}'. +sp_after_oc_block_caret = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space between the receiver and selector in a message, +# as in '[receiver selector ...]'. +sp_after_oc_msg_receiver = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space after '@property'. +sp_after_oc_property = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove space between '@synchronized' and the open parenthesis, +# i.e. '@synchronized(foo)' vs. '@synchronized (foo)'. +sp_after_oc_synchronized = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around the ':' in 'b ? t : f'. +sp_cond_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before the ':' in 'b ? t : f'. +# +# Overrides sp_cond_colon. +sp_cond_colon_before = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the ':' in 'b ? t : f'. +# +# Overrides sp_cond_colon. +sp_cond_colon_after = ignore # ignore/add/remove/force/not_defined + +# Add or remove space around the '?' in 'b ? t : f'. +sp_cond_question = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before the '?' in 'b ? t : f'. +# +# Overrides sp_cond_question. +sp_cond_question_before = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the '?' in 'b ? t : f'. +# +# Overrides sp_cond_question. +sp_cond_question_after = ignore # ignore/add/remove/force/not_defined + +# In the abbreviated ternary form '(a ?: b)', add or remove space between '?' +# and ':'. +# +# Overrides all other sp_cond_* options. +sp_cond_ternary_short = ignore # ignore/add/remove/force/not_defined + +# Fix the spacing between 'case' and the label. Only 'ignore' and 'force' make +# sense here. +sp_case_label = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove space around the D '..' operator. +sp_range = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after ':' in a Java/C++11 range-based 'for', +# as in 'for (Type var : <here> expr)'. +sp_after_for_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before ':' in a Java/C++11 range-based 'for', +# as in 'for (Type var <here> : expr)'. +sp_before_for_colon = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove space between 'extern' and '(' as in 'extern <here> (C)'. +sp_extern_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the opening of a C++ comment, as in '// <here> A'. +sp_cmt_cpp_start = ignore # ignore/add/remove/force/not_defined + +# remove space after the '//' and the pvs command '-V1234', +# only works with sp_cmt_cpp_start set to add or force. +sp_cmt_cpp_pvs = false # true/false + +# remove space after the '//' and the command 'lint', +# only works with sp_cmt_cpp_start set to add or force. +sp_cmt_cpp_lint = false # true/false + +# Add or remove space in a C++ region marker comment, as in '// <here> BEGIN'. +# A region marker is defined as a comment which is not preceded by other text +# (i.e. the comment is the first non-whitespace on the line), and which starts +# with either 'BEGIN' or 'END'. +# +# Overrides sp_cmt_cpp_start. +sp_cmt_cpp_region = ignore # ignore/add/remove/force/not_defined + +# If true, space added with sp_cmt_cpp_start will be added after Doxygen +# sequences like '///', '///<', '//!' and '//!<'. +sp_cmt_cpp_doxygen = false # true/false + +# If true, space added with sp_cmt_cpp_start will be added after Qt translator +# or meta-data comments like '//:', '//=', and '//~'. +sp_cmt_cpp_qttr = false # true/false + +# Add or remove space between #else or #endif and a trailing comment. +sp_endif_cmt = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after 'new', 'delete' and 'delete[]'. +sp_after_new = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between 'new' and '(' in 'new()'. +sp_between_new_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space between ')' and type in 'new(foo) BAR'. +sp_after_newop_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space inside parentheses of the new operator +# as in 'new(foo) BAR'. +sp_inside_newop_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after the open parenthesis of the new operator, +# as in 'new(foo) BAR'. +# +# Overrides sp_inside_newop_paren. +sp_inside_newop_paren_open = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before the close parenthesis of the new operator, +# as in 'new(foo) BAR'. +# +# Overrides sp_inside_newop_paren. +sp_inside_newop_paren_close = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before a trailing comment. +sp_before_tr_cmt = ignore # ignore/add/remove/force/not_defined + +# Number of spaces before a trailing comment. +sp_num_before_tr_cmt = 0 # unsigned number + +# Add or remove space before an embedded comment. +# +# Default: force +sp_before_emb_cmt = force # ignore/add/remove/force/not_defined + +# Number of spaces before an embedded comment. +# +# Default: 1 +sp_num_before_emb_cmt = 1 # unsigned number + +# Add or remove space after an embedded comment. +# +# Default: force +sp_after_emb_cmt = force # ignore/add/remove/force/not_defined + +# Number of spaces after an embedded comment. +# +# Default: 1 +sp_num_after_emb_cmt = 1 # unsigned number + +# (Java) Add or remove space between an annotation and the open parenthesis. +sp_annotation_paren = ignore # ignore/add/remove/force/not_defined + +# If true, vbrace tokens are dropped to the previous token and skipped. +sp_skip_vbrace_tokens = false # true/false + +# Add or remove space after 'noexcept'. +sp_after_noexcept = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after '_'. +sp_vala_after_translation = ignore # ignore/add/remove/force/not_defined + +# Add or remove space before a bit colon ':'. +sp_before_bit_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove space after a bit colon ':'. +sp_after_bit_colon = ignore # ignore/add/remove/force/not_defined + +# If true, a <TAB> is inserted after #define. +force_tab_after_define = false # true/false + +# +# Indenting options +# + +# The number of columns to indent per level. Usually 2, 3, 4, or 8. +# +# Default: 8 +indent_columns = 8 # unsigned number + +# Whether to ignore indent for the first continuation line. Subsequent +# continuation lines will still be indented to match the first. +indent_ignore_first_continue = false # true/false + +# The continuation indent. If non-zero, this overrides the indent of '(', '[' +# and '=' continuation indents. Negative values are OK; negative value is +# absolute and not increased for each '(' or '[' level. +# +# For FreeBSD, this is set to 4. +# Requires indent_ignore_first_continue=false. +indent_continue = 0 # number + +# The continuation indent, only for class header line(s). If non-zero, this +# overrides the indent of 'class' continuation indents. +# Requires indent_ignore_first_continue=false. +indent_continue_class_head = 0 # unsigned number + +# Whether to indent empty lines (i.e. lines which contain only spaces before +# the newline character). +indent_single_newlines = false # true/false + +# The continuation indent for func_*_param if they are true. If non-zero, this +# overrides the indent. +indent_param = 0 # unsigned number + +# How to use tabs when indenting code. +# +# 0: Spaces only +# 1: Indent with tabs to brace level, align with spaces (default) +# 2: Indent and align with tabs, using spaces when not on a tabstop +# +# Default: 1 +indent_with_tabs = 1 # unsigned number + +# Whether to indent comments that are not at a brace level with tabs on a +# tabstop. Requires indent_with_tabs=2. If false, will use spaces. +indent_cmt_with_tabs = false # true/false + +# Whether to indent strings broken by '\' so that they line up. +indent_align_string = false # true/false + +# The number of spaces to indent multi-line XML strings. +# Requires indent_align_string=true. +indent_xml_string = 0 # unsigned number + +# Spaces to indent '{' from level. +indent_brace = 0 # unsigned number + +# Whether braces are indented to the body level. +indent_braces = false # true/false + +# Whether to disable indenting function braces if indent_braces=true. +indent_braces_no_func = false # true/false + +# Whether to disable indenting class braces if indent_braces=true. +indent_braces_no_class = false # true/false + +# Whether to disable indenting struct braces if indent_braces=true. +indent_braces_no_struct = false # true/false + +# Whether to indent based on the size of the brace parent, +# i.e. 'if' => 3 spaces, 'for' => 4 spaces, etc. +indent_brace_parent = false # true/false + +# Whether to indent based on the open parenthesis instead of the open brace +# in '({\n'. +indent_paren_open_brace = false # true/false + +# (C#) Whether to indent the brace of a C# delegate by another level. +indent_cs_delegate_brace = false # true/false + +# (C#) Whether to indent a C# delegate (to handle delegates with no brace) by +# another level. +indent_cs_delegate_body = false # true/false + +# Whether to indent the body of a 'namespace'. +indent_namespace = false # true/false + +# Whether to indent only the first namespace, and not any nested namespaces. +# Requires indent_namespace=true. +indent_namespace_single_indent = false # true/false + +# The number of spaces to indent a namespace block. +# If set to zero, use the value indent_columns +indent_namespace_level = 0 # unsigned number + +# If the body of the namespace is longer than this number, it won't be +# indented. Requires indent_namespace=true. 0 means no limit. +indent_namespace_limit = 0 # unsigned number + +# Whether to indent only in inner namespaces (nested in other namespaces). +# Requires indent_namespace=true. +indent_namespace_inner_only = false # true/false + +# Whether the 'extern "C"' body is indented. +indent_extern = false # true/false + +# Whether the 'class' body is indented. +indent_class = false # true/false + +# Whether to ignore indent for the leading base class colon. +indent_ignore_before_class_colon = false # true/false + +# Additional indent before the leading base class colon. +# Negative values decrease indent down to the first column. +# Requires indent_ignore_before_class_colon=false and a newline break before +# the colon (see pos_class_colon and nl_class_colon) +indent_before_class_colon = 0 # number + +# Whether to indent the stuff after a leading base class colon. +indent_class_colon = false # true/false + +# Whether to indent based on a class colon instead of the stuff after the +# colon. Requires indent_class_colon=true. +indent_class_on_colon = false # true/false + +# Whether to ignore indent for a leading class initializer colon. +indent_ignore_before_constr_colon = false # true/false + +# Whether to indent the stuff after a leading class initializer colon. +indent_constr_colon = false # true/false + +# Virtual indent from the ':' for leading member initializers. +# +# Default: 2 +indent_ctor_init_leading = 2 # unsigned number + +# Virtual indent from the ':' for following member initializers. +# +# Default: 2 +indent_ctor_init_following = 2 # unsigned number + +# Additional indent for constructor initializer list. +# Negative values decrease indent down to the first column. +indent_ctor_init = 0 # number + +# Whether to indent 'if' following 'else' as a new block under the 'else'. +# If false, 'else\nif' is treated as 'else if' for indenting purposes. +indent_else_if = false # true/false + +# Amount to indent variable declarations after a open brace. +# +# <0: Relative +# >=0: Absolute +indent_var_def_blk = 0 # number + +# Whether to indent continued variable declarations instead of aligning. +indent_var_def_cont = true # true/false + +# How to indent continued shift expressions ('<<' and '>>'). +# Set align_left_shift=false when using this. +# 0: Align shift operators instead of indenting them (default) +# 1: Indent by one level +# -1: Preserve original indentation +indent_shift = 0 # number + +# Whether to force indentation of function definitions to start in column 1. +indent_func_def_force_col1 = false # true/false + +# Whether to indent continued function call parameters one indent level, +# rather than aligning parameters under the open parenthesis. +indent_func_call_param = false # true/false + +# Whether to indent continued function definition parameters one indent level, +# rather than aligning parameters under the open parenthesis. +indent_func_def_param = false # true/false + +# for function definitions, only if indent_func_def_param is false +# Allows to align params when appropriate and indent them when not +# behave as if it was true if paren position is more than this value +# if paren position is more than the option value +indent_func_def_param_paren_pos_threshold = 0 # unsigned number + +# Whether to indent continued function call prototype one indent level, +# rather than aligning parameters under the open parenthesis. +indent_func_proto_param = false # true/false + +# Whether to indent continued function call declaration one indent level, +# rather than aligning parameters under the open parenthesis. +indent_func_class_param = false # true/false + +# Whether to indent continued class variable constructors one indent level, +# rather than aligning parameters under the open parenthesis. +indent_func_ctor_var_param = false # true/false + +# Whether to indent continued template parameter list one indent level, +# rather than aligning parameters under the open parenthesis. +indent_template_param = false # true/false + +# Double the indent for indent_func_xxx_param options. +# Use both values of the options indent_columns and indent_param. +indent_func_param_double = false # true/false + +# Indentation column for standalone 'const' qualifier on a function +# prototype. +indent_func_const = 0 # unsigned number + +# Indentation column for standalone 'throw' qualifier on a function +# prototype. +indent_func_throw = 0 # unsigned number + +# How to indent within a macro followed by a brace on the same line +# This allows reducing the indent in macros that have (for example) +# `do { ... } while (0)` blocks bracketing them. +# +# true: add an indent for the brace on the same line as the macro +# false: do not add an indent for the brace on the same line as the macro +# +# Default: true +indent_macro_brace = true # true/false + +# The number of spaces to indent a continued '->' or '.'. +# Usually set to 0, 1, or indent_columns. +indent_member = 0 # unsigned number + +# Whether lines broken at '.' or '->' should be indented by a single indent. +# The indent_member option will not be effective if this is set to true. +indent_member_single = false # true/false + +# Spaces to indent single line ('//') comments on lines before code. +indent_single_line_comments_before = 0 # unsigned number + +# Spaces to indent single line ('//') comments on lines after code. +indent_single_line_comments_after = 0 # unsigned number + +# When opening a paren for a control statement (if, for, while, etc), increase +# the indent level by this value. Negative values decrease the indent level. +indent_sparen_extra = 0 # number + +# Whether to indent trailing single line ('//') comments relative to the code +# instead of trying to keep the same absolute column. +indent_relative_single_line_comments = false # true/false + +# Spaces to indent 'case' from 'switch'. Usually 0 or indent_columns. +# It might be wise to choose the same value for the option indent_case_brace. +indent_switch_case = 0 # unsigned number + +# Spaces to indent the body of a 'switch' before any 'case'. +# Usually the same as indent_columns or indent_switch_case. +indent_switch_body = 0 # unsigned number + +# Whether to ignore indent for '{' following 'case'. +indent_ignore_case_brace = false # true/false + +# Spaces to indent '{' from 'case'. By default, the brace will appear under +# the 'c' in case. Usually set to 0 or indent_columns. Negative values are OK. +# It might be wise to choose the same value for the option indent_switch_case. +indent_case_brace = 0 # number + +# indent 'break' with 'case' from 'switch'. +indent_switch_break_with_case = false # true/false + +# Whether to indent preprocessor statements inside of switch statements. +# +# Default: true +indent_switch_pp = true # true/false + +# Spaces to shift the 'case' line, without affecting any other lines. +# Usually 0. +indent_case_shift = 0 # unsigned number + +# Whether to align comments before 'case' with the 'case'. +# +# Default: true +indent_case_comment = true # true/false + +# Whether to indent comments not found in first column. +# +# Default: true +indent_comment = true # true/false + +# Whether to indent comments found in first column. +indent_col1_comment = false # true/false + +# Whether to indent multi string literal in first column. +indent_col1_multi_string_literal = false # true/false + +# Align comments on adjacent lines that are this many columns apart or less. +# +# Default: 3 +indent_comment_align_thresh = 3 # unsigned number + +# Whether to ignore indent for goto labels. +indent_ignore_label = false # true/false + +# How to indent goto labels. Requires indent_ignore_label=false. +# +# >0: Absolute column where 1 is the leftmost column +# <=0: Subtract from brace indent +# +# Default: 1 +indent_label = 1 # number + +# How to indent access specifiers that are followed by a +# colon. +# +# >0: Absolute column where 1 is the leftmost column +# <=0: Subtract from brace indent +# +# Default: 1 +indent_access_spec = 1 # number + +# Whether to indent the code after an access specifier by one level. +# If true, this option forces 'indent_access_spec=0'. +indent_access_spec_body = false # true/false + +# If an open parenthesis is followed by a newline, whether to indent the next +# line so that it lines up after the open parenthesis (not recommended). +indent_paren_nl = false # true/false + +# How to indent a close parenthesis after a newline. +# +# 0: Indent to body level (default) +# 1: Align under the open parenthesis +# 2: Indent to the brace level +# -1: Preserve original indentation +indent_paren_close = 0 # number + +# Whether to indent the open parenthesis of a function definition, +# if the parenthesis is on its own line. +indent_paren_after_func_def = false # true/false + +# Whether to indent the open parenthesis of a function declaration, +# if the parenthesis is on its own line. +indent_paren_after_func_decl = false # true/false + +# Whether to indent the open parenthesis of a function call, +# if the parenthesis is on its own line. +indent_paren_after_func_call = false # true/false + +# How to indent a comma when inside braces. +# 0: Indent by one level (default) +# 1: Align under the open brace +# -1: Preserve original indentation +indent_comma_brace = 0 # number + +# How to indent a comma when inside parentheses. +# 0: Indent by one level (default) +# 1: Align under the open parenthesis +# -1: Preserve original indentation +indent_comma_paren = 0 # number + +# How to indent a Boolean operator when inside parentheses. +# 0: Indent by one level (default) +# 1: Align under the open parenthesis +# -1: Preserve original indentation +indent_bool_paren = 0 # number + +# Whether to ignore the indentation of a Boolean operator when outside +# parentheses. +indent_ignore_bool = false # true/false + +# Whether to ignore the indentation of an arithmetic operator. +indent_ignore_arith = false # true/false + +# Whether to indent a semicolon when inside a for parenthesis. +# If true, aligns under the open for parenthesis. +indent_semicolon_for_paren = false # true/false + +# Whether to ignore the indentation of a semicolon outside of a 'for' +# statement. +indent_ignore_semicolon = false # true/false + +# Whether to align the first expression to following ones +# if indent_bool_paren=1. +indent_first_bool_expr = false # true/false + +# Whether to align the first expression to following ones +# if indent_semicolon_for_paren=true. +indent_first_for_expr = false # true/false + +# If an open square is followed by a newline, whether to indent the next line +# so that it lines up after the open square (not recommended). +indent_square_nl = false # true/false + +# (ESQL/C) Whether to preserve the relative indent of 'EXEC SQL' bodies. +indent_preserve_sql = false # true/false + +# Whether to ignore the indentation of an assignment operator. +indent_ignore_assign = false # true/false + +# Whether to align continued statements at the '='. If false or if the '=' is +# followed by a newline, the next line is indent one tab. +# +# Default: true +indent_align_assign = true # true/false + +# If true, the indentation of the chunks after a '=' sequence will be set at +# LHS token indentation column before '='. +indent_off_after_assign = false # true/false + +# Whether to align continued statements at the '('. If false or the '(' is +# followed by a newline, the next line indent is one tab. +# +# Default: true +indent_align_paren = true # true/false + +# (OC) Whether to indent Objective-C code inside message selectors. +indent_oc_inside_msg_sel = false # true/false + +# (OC) Whether to indent Objective-C blocks at brace level instead of usual +# rules. +indent_oc_block = false # true/false + +# (OC) Indent for Objective-C blocks in a message relative to the parameter +# name. +# +# =0: Use indent_oc_block rules +# >0: Use specified number of spaces to indent +indent_oc_block_msg = 0 # unsigned number + +# (OC) Minimum indent for subsequent parameters +indent_oc_msg_colon = 0 # unsigned number + +# (OC) Whether to prioritize aligning with initial colon (and stripping spaces +# from lines, if necessary). +# +# Default: true +indent_oc_msg_prioritize_first_colon = true # true/false + +# (OC) Whether to indent blocks the way that Xcode does by default +# (from the keyword if the parameter is on its own line; otherwise, from the +# previous indentation level). Requires indent_oc_block_msg=true. +indent_oc_block_msg_xcode_style = false # true/false + +# (OC) Whether to indent blocks from where the brace is, relative to a +# message keyword. Requires indent_oc_block_msg=true. +indent_oc_block_msg_from_keyword = false # true/false + +# (OC) Whether to indent blocks from where the brace is, relative to a message +# colon. Requires indent_oc_block_msg=true. +indent_oc_block_msg_from_colon = false # true/false + +# (OC) Whether to indent blocks from where the block caret is. +# Requires indent_oc_block_msg=true. +indent_oc_block_msg_from_caret = false # true/false + +# (OC) Whether to indent blocks from where the brace caret is. +# Requires indent_oc_block_msg=true. +indent_oc_block_msg_from_brace = false # true/false + +# When indenting after virtual brace open and newline add further spaces to +# reach this minimum indent. +indent_min_vbrace_open = 0 # unsigned number + +# Whether to add further spaces after regular indent to reach next tabstop +# when indenting after virtual brace open and newline. +indent_vbrace_open_on_tabstop = false # true/false + +# How to indent after a brace followed by another token (not a newline). +# true: indent all contained lines to match the token +# false: indent all contained lines to match the brace +# +# Default: true +indent_token_after_brace = true # true/false + +# Whether to indent the body of a C++11 lambda. +indent_cpp_lambda_body = false # true/false + +# How to indent compound literals that are being returned. +# true: add both the indent from return & the compound literal open brace +# (i.e. 2 indent levels) +# false: only indent 1 level, don't add the indent for the open brace, only +# add the indent for the return. +# +# Default: true +indent_compound_literal_return = true # true/false + +# (C#) Whether to indent a 'using' block if no braces are used. +# +# Default: true +indent_using_block = true # true/false + +# How to indent the continuation of ternary operator. +# +# 0: Off (default) +# 1: When the `if_false` is a continuation, indent it under the `if_true` branch +# 2: When the `:` is a continuation, indent it under `?` +indent_ternary_operator = 0 # unsigned number + +# Whether to indent the statements inside ternary operator. +indent_inside_ternary_operator = false # true/false + +# If true, the indentation of the chunks after a `return` sequence will be set at return indentation column. +indent_off_after_return = false # true/false + +# If true, the indentation of the chunks after a `return new` sequence will be set at return indentation column. +indent_off_after_return_new = false # true/false + +# If true, the tokens after return are indented with regular single indentation. By default (false) the indentation is after the return token. +indent_single_after_return = false # true/false + +# Whether to ignore indent and alignment for 'asm' blocks (i.e. assume they +# have their own indentation). +indent_ignore_asm_block = false # true/false + +# Don't indent the close parenthesis of a function definition, +# if the parenthesis is on its own line. +donot_indent_func_def_close_paren = false # true/false + +# +# Newline adding and removing options +# + +# Whether to collapse empty blocks between '{' and '}' except for functions. +# Use nl_collapse_empty_body_functions to specify how empty function braces +# should be formatted. +nl_collapse_empty_body = false # true/false + +# Whether to collapse empty blocks between '{' and '}' for functions only. +# If true, overrides nl_inside_empty_func. +nl_collapse_empty_body_functions = false # true/false + +# Don't split one-line braced assignments, as in 'foo_t f = { 1, 2 };'. +nl_assign_leave_one_liners = false # true/false + +# Don't split one-line braced statements inside a 'class xx { }' body. +nl_class_leave_one_liners = false # true/false + +# Don't split one-line enums, as in 'enum foo { BAR = 15 };' +nl_enum_leave_one_liners = false # true/false + +# Don't split one-line get or set functions. +nl_getset_leave_one_liners = false # true/false + +# (C#) Don't split one-line property get or set functions. +nl_cs_property_leave_one_liners = false # true/false + +# Don't split one-line function definitions, as in 'int foo() { return 0; }'. +# might modify nl_func_type_name +nl_func_leave_one_liners = false # true/false + +# Don't split one-line C++11 lambdas, as in '[]() { return 0; }'. +nl_cpp_lambda_leave_one_liners = false # true/false + +# Don't split one-line if/else statements, as in 'if(...) b++;'. +nl_if_leave_one_liners = false # true/false + +# Don't split one-line while statements, as in 'while(...) b++;'. +nl_while_leave_one_liners = false # true/false + +# Don't split one-line do statements, as in 'do { b++; } while(...);'. +nl_do_leave_one_liners = false # true/false + +# Don't split one-line for statements, as in 'for(...) b++;'. +nl_for_leave_one_liners = false # true/false + +# (OC) Don't split one-line Objective-C messages. +nl_oc_msg_leave_one_liner = false # true/false + +# (OC) Add or remove newline between method declaration and '{'. +nl_oc_mdef_brace = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove newline between Objective-C block signature and '{'. +nl_oc_block_brace = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove blank line before '@interface' statement. +nl_oc_before_interface = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove blank line before '@implementation' statement. +nl_oc_before_implementation = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove blank line before '@end' statement. +nl_oc_before_end = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove newline between '@interface' and '{'. +nl_oc_interface_brace = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove newline between '@implementation' and '{'. +nl_oc_implementation_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newlines at the start of the file. +nl_start_of_file = ignore # ignore/add/remove/force/not_defined + +# The minimum number of newlines at the start of the file (only used if +# nl_start_of_file is 'add' or 'force'). +nl_start_of_file_min = 0 # unsigned number + +# Add or remove newline at the end of the file. +nl_end_of_file = ignore # ignore/add/remove/force/not_defined + +# The minimum number of newlines at the end of the file (only used if +# nl_end_of_file is 'add' or 'force'). +nl_end_of_file_min = 0 # unsigned number + +# Add or remove newline between '=' and '{'. +nl_assign_brace = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove newline between '=' and '['. +nl_assign_square = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '[]' and '{'. +nl_tsquare_brace = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove newline after '= ['. Will also affect the newline before +# the ']'. +nl_after_square_assign = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between a function call's ')' and '{', as in +# 'list_for_each(item, &list) { }'. +nl_fcall_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'enum' and '{'. +nl_enum_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'enum' and 'class'. +nl_enum_class = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'enum class' and the identifier. +nl_enum_class_identifier = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'enum class' type and ':'. +nl_enum_identifier_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'enum class identifier :' and type. +nl_enum_colon_type = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'struct and '{'. +nl_struct_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'union' and '{'. +nl_union_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'if' and '{'. +nl_if_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '}' and 'else'. +nl_brace_else = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'else if' and '{'. If set to ignore, +# nl_if_brace is used instead. +nl_elseif_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'else' and '{'. +nl_else_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'else' and 'if'. +nl_else_if = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline before '{' opening brace +nl_before_opening_brace_func_class_def = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline before 'if'/'else if' closing parenthesis. +nl_before_if_closing_paren = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '}' and 'finally'. +nl_brace_finally = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'finally' and '{'. +nl_finally_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'try' and '{'. +nl_try_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between get/set and '{'. +nl_getset_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'for' and '{'. +nl_for_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline before the '{' of a 'catch' statement, as in +# 'catch (decl) <here> {'. +nl_catch_brace = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove newline before the '{' of a '@catch' statement, as in +# '@catch (decl) <here> {'. If set to ignore, nl_catch_brace is used. +nl_oc_catch_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '}' and 'catch'. +nl_brace_catch = ignore # ignore/add/remove/force/not_defined + +# (OC) Add or remove newline between '}' and '@catch'. If set to ignore, +# nl_brace_catch is used. +nl_oc_brace_catch = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '}' and ']'. +nl_brace_square = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '}' and ')' in a function invocation. +nl_brace_fparen = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'while' and '{'. +nl_while_brace = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove newline between 'scope (x)' and '{'. +nl_scope_brace = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove newline between 'unittest' and '{'. +nl_unittest_brace = ignore # ignore/add/remove/force/not_defined + +# (D) Add or remove newline between 'version (x)' and '{'. +nl_version_brace = ignore # ignore/add/remove/force/not_defined + +# (C#) Add or remove newline between 'using' and '{'. +nl_using_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between two open or close braces. Due to general +# newline/brace handling, REMOVE may not work. +nl_brace_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'do' and '{'. +nl_do_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '}' and 'while' of 'do' statement. +nl_brace_while = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'switch' and '{'. +nl_switch_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'synchronized' and '{'. +nl_synchronized_brace = ignore # ignore/add/remove/force/not_defined + +# Add a newline between ')' and '{' if the ')' is on a different line than the +# if/for/etc. +# +# Overrides nl_for_brace, nl_if_brace, nl_switch_brace, nl_while_switch and +# nl_catch_brace. +nl_multi_line_cond = false # true/false + +# Add a newline after '(' if an if/for/while/switch condition spans multiple +# lines +nl_multi_line_sparen_open = ignore # ignore/add/remove/force/not_defined + +# Add a newline before ')' if an if/for/while/switch condition spans multiple +# lines. Overrides nl_before_if_closing_paren if both are specified. +nl_multi_line_sparen_close = ignore # ignore/add/remove/force/not_defined + +# Force a newline in a define after the macro name for multi-line defines. +nl_multi_line_define = false # true/false + +# Whether to add a newline before 'case', and a blank line before a 'case' +# statement that follows a ';' or '}'. +nl_before_case = false # true/false + +# Whether to add a newline after a 'case' statement. +nl_after_case = false # true/false + +# Add or remove newline between a case ':' and '{'. +# +# Overrides nl_after_case. +nl_case_colon_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between ')' and 'throw'. +nl_before_throw = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'namespace' and '{'. +nl_namespace_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<...>' of a template class. +nl_template_class = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<...>' of a template class declaration. +# +# Overrides nl_template_class. +nl_template_class_decl = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<>' of a specialized class declaration. +# +# Overrides nl_template_class_decl. +nl_template_class_decl_special = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<...>' of a template class definition. +# +# Overrides nl_template_class. +nl_template_class_def = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<>' of a specialized class definition. +# +# Overrides nl_template_class_def. +nl_template_class_def_special = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<...>' of a template function. +nl_template_func = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<...>' of a template function +# declaration. +# +# Overrides nl_template_func. +nl_template_func_decl = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<>' of a specialized function +# declaration. +# +# Overrides nl_template_func_decl. +nl_template_func_decl_special = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<...>' of a template function +# definition. +# +# Overrides nl_template_func. +nl_template_func_def = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<>' of a specialized function +# definition. +# +# Overrides nl_template_func_def. +nl_template_func_def_special = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after 'template<...>' of a template variable. +nl_template_var = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'template<...>' and 'using' of a templated +# type alias. +nl_template_using = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'class' and '{'. +nl_class_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline before or after (depending on pos_class_comma, +# may not be IGNORE) each',' in the base class list. +nl_class_init_args = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after each ',' in the constructor member +# initialization. Related to nl_constr_colon, pos_constr_colon and +# pos_constr_comma. +nl_constr_init_args = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline before first element, after comma, and after last +# element, in 'enum'. +nl_enum_own_lines = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between return type and function name in a function +# definition. +# might be modified by nl_func_leave_one_liners +nl_func_type_name = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between return type and function name inside a class +# definition. If set to ignore, nl_func_type_name or nl_func_proto_type_name +# is used instead. +nl_func_type_name_class = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between class specification and '::' +# in 'void A::f() { }'. Only appears in separate member implementation (does +# not appear with in-line implementation). +nl_func_class_scope = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between function scope and name, as in +# 'void A :: <here> f() { }'. +nl_func_scope_name = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between return type and function name in a prototype. +nl_func_proto_type_name = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between a function name and the opening '(' in the +# declaration. +nl_func_paren = ignore # ignore/add/remove/force/not_defined + +# Overrides nl_func_paren for functions with no parameters. +nl_func_paren_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between a function name and the opening '(' in the +# definition. +nl_func_def_paren = ignore # ignore/add/remove/force/not_defined + +# Overrides nl_func_def_paren for functions with no parameters. +nl_func_def_paren_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between a function name and the opening '(' in the +# call. +nl_func_call_paren = ignore # ignore/add/remove/force/not_defined + +# Overrides nl_func_call_paren for functions with no parameters. +nl_func_call_paren_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after '(' in a function declaration. +nl_func_decl_start = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after '(' in a function definition. +nl_func_def_start = ignore # ignore/add/remove/force/not_defined + +# Overrides nl_func_decl_start when there is only one parameter. +nl_func_decl_start_single = ignore # ignore/add/remove/force/not_defined + +# Overrides nl_func_def_start when there is only one parameter. +nl_func_def_start_single = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline after '(' in a function declaration if '(' and ')' +# are in different lines. If false, nl_func_decl_start is used instead. +nl_func_decl_start_multi_line = false # true/false + +# Whether to add a newline after '(' in a function definition if '(' and ')' +# are in different lines. If false, nl_func_def_start is used instead. +nl_func_def_start_multi_line = false # true/false + +# Add or remove newline after each ',' in a function declaration. +nl_func_decl_args = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after each ',' in a function definition. +nl_func_def_args = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline after each ',' in a function call. +nl_func_call_args = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline after each ',' in a function declaration if '(' +# and ')' are in different lines. If false, nl_func_decl_args is used instead. +nl_func_decl_args_multi_line = false # true/false + +# Whether to add a newline after each ',' in a function definition if '(' +# and ')' are in different lines. If false, nl_func_def_args is used instead. +nl_func_def_args_multi_line = false # true/false + +# Add or remove newline before the ')' in a function declaration. +nl_func_decl_end = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline before the ')' in a function definition. +nl_func_def_end = ignore # ignore/add/remove/force/not_defined + +# Overrides nl_func_decl_end when there is only one parameter. +nl_func_decl_end_single = ignore # ignore/add/remove/force/not_defined + +# Overrides nl_func_def_end when there is only one parameter. +nl_func_def_end_single = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline before ')' in a function declaration if '(' and ')' +# are in different lines. If false, nl_func_decl_end is used instead. +nl_func_decl_end_multi_line = false # true/false + +# Whether to add a newline before ')' in a function definition if '(' and ')' +# are in different lines. If false, nl_func_def_end is used instead. +nl_func_def_end_multi_line = false # true/false + +# Add or remove newline between '()' in a function declaration. +nl_func_decl_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '()' in a function definition. +nl_func_def_empty = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between '()' in a function call. +nl_func_call_empty = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline after '(' in a function call, +# has preference over nl_func_call_start_multi_line. +nl_func_call_start = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline before ')' in a function call. +nl_func_call_end = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline after '(' in a function call if '(' and ')' are in +# different lines. +nl_func_call_start_multi_line = false # true/false + +# Whether to add a newline after each ',' in a function call if '(' and ')' +# are in different lines. +nl_func_call_args_multi_line = false # true/false + +# Whether to add a newline before ')' in a function call if '(' and ')' are in +# different lines. +nl_func_call_end_multi_line = false # true/false + +# Whether to respect nl_func_call_XXX option in case of closure args. +nl_func_call_args_multi_line_ignore_closures = false # true/false + +# Whether to add a newline after '<' of a template parameter list. +nl_template_start = false # true/false + +# Whether to add a newline after each ',' in a template parameter list. +nl_template_args = false # true/false + +# Whether to add a newline before '>' of a template parameter list. +nl_template_end = false # true/false + +# (OC) Whether to put each Objective-C message parameter on a separate line. +# See nl_oc_msg_leave_one_liner. +nl_oc_msg_args = false # true/false + +# (OC) Minimum number of Objective-C message parameters before applying nl_oc_msg_args. +nl_oc_msg_args_min_params = 0 # unsigned number + +# (OC) Max code width of Objective-C message before applying nl_oc_msg_args. +nl_oc_msg_args_max_code_width = 0 # unsigned number + +# Add or remove newline between function signature and '{'. +nl_fdef_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between function signature and '{', +# if signature ends with ')'. Overrides nl_fdef_brace. +nl_fdef_brace_cond = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between C++11 lambda signature and '{'. +nl_cpp_ldef_brace = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'return' and the return expression. +nl_return_expr = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline between 'throw' and the throw expression. +nl_throw_expr = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline after semicolons, except in 'for' statements. +nl_after_semicolon = false # true/false + +# (Java) Add or remove newline between the ')' and '{{' of the double brace +# initializer. +nl_paren_dbrace_open = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline after the type in an unnamed temporary +# direct-list-initialization, better: +# before a direct-list-initialization. +nl_type_brace_init_lst = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline after the open brace in an unnamed temporary +# direct-list-initialization. +nl_type_brace_init_lst_open = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline before the close brace in an unnamed temporary +# direct-list-initialization. +nl_type_brace_init_lst_close = ignore # ignore/add/remove/force/not_defined + +# Whether to add a newline before '{'. +nl_before_brace_open = false # true/false + +# Whether to add a newline after '{'. +nl_after_brace_open = false # true/false + +# Whether to add a newline between the open brace and a trailing single-line +# comment. Requires nl_after_brace_open=true. +nl_after_brace_open_cmt = false # true/false + +# Whether to add a newline after a virtual brace open with a non-empty body. +# These occur in un-braced if/while/do/for statement bodies. +nl_after_vbrace_open = false # true/false + +# Whether to add a newline after a virtual brace open with an empty body. +# These occur in un-braced if/while/do/for statement bodies. +nl_after_vbrace_open_empty = false # true/false + +# Whether to add a newline after '}'. Does not apply if followed by a +# necessary ';'. +nl_after_brace_close = false # true/false + +# Whether to add a newline after a virtual brace close, +# as in 'if (foo) a++; <here> return;'. +nl_after_vbrace_close = false # true/false + +# Add or remove newline between the close brace and identifier, +# as in 'struct { int a; } <here> b;'. Affects enumerations, unions and +# structures. If set to ignore, uses nl_after_brace_close. +nl_brace_struct_var = ignore # ignore/add/remove/force/not_defined + +# Whether to alter newlines in '#define' macros. +nl_define_macro = false # true/false + +# Whether to alter newlines between consecutive parenthesis closes. The number +# of closing parentheses in a line will depend on respective open parenthesis +# lines. +nl_squeeze_paren_close = false # true/false + +# Whether to remove blanks after '#ifxx' and '#elxx', or before '#elxx' and +# '#endif'. Does not affect top-level #ifdefs. +nl_squeeze_ifdef = false # true/false + +# Makes the nl_squeeze_ifdef option affect the top-level #ifdefs as well. +nl_squeeze_ifdef_top_level = false # true/false + +# Add or remove blank line before 'if'. +nl_before_if = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line after 'if' statement. Add/Force work only if the +# next token is not a closing brace. +nl_after_if = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line before 'for'. +nl_before_for = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line after 'for' statement. +nl_after_for = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line before 'while'. +nl_before_while = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line after 'while' statement. +nl_after_while = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line before 'switch'. +nl_before_switch = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line after 'switch' statement. +nl_after_switch = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line before 'synchronized'. +nl_before_synchronized = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line after 'synchronized' statement. +nl_after_synchronized = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line before 'do'. +nl_before_do = ignore # ignore/add/remove/force/not_defined + +# Add or remove blank line after 'do/while' statement. +nl_after_do = ignore # ignore/add/remove/force/not_defined + +# Ignore nl_before_{if,for,switch,do,synchronized} if the control +# statement is immediately after a case statement. +# if nl_before_{if,for,switch,do} is set to remove, this option +# does nothing. +nl_before_ignore_after_case = false # true/false + +# Whether to put a blank line before 'return' statements, unless after an open +# brace. +nl_before_return = false # true/false + +# Whether to put a blank line after 'return' statements, unless followed by a +# close brace. +nl_after_return = false # true/false + +# Whether to put a blank line before a member '.' or '->' operators. +nl_before_member = ignore # ignore/add/remove/force/not_defined + +# (Java) Whether to put a blank line after a member '.' or '->' operators. +nl_after_member = ignore # ignore/add/remove/force/not_defined + +# Whether to double-space commented-entries in 'struct'/'union'/'enum'. +nl_ds_struct_enum_cmt = false # true/false + +# Whether to force a newline before '}' of a 'struct'/'union'/'enum'. +# (Lower priority than eat_blanks_before_close_brace.) +nl_ds_struct_enum_close_brace = false # true/false + +# Add or remove newline before or after (depending on pos_class_colon) a class +# colon, as in 'class Foo <here> : <or here> public Bar'. +nl_class_colon = ignore # ignore/add/remove/force/not_defined + +# Add or remove newline around a class constructor colon. The exact position +# depends on nl_constr_init_args, pos_constr_colon and pos_constr_comma. +nl_constr_colon = ignore # ignore/add/remove/force/not_defined + +# Whether to collapse a two-line namespace, like 'namespace foo\n{ decl; }' +# into a single line. If true, prevents other brace newline rules from turning +# such code into four lines. If true, it also preserves one-liner namespaces. +nl_namespace_two_to_one_liner = false # true/false + +# Whether to remove a newline in simple unbraced if statements, turning them +# into one-liners, as in 'if(b)\n i++;' => 'if(b) i++;'. +nl_create_if_one_liner = false # true/false + +# Whether to remove a newline in simple unbraced for statements, turning them +# into one-liners, as in 'for (...)\n stmt;' => 'for (...) stmt;'. +nl_create_for_one_liner = false # true/false + +# Whether to remove a newline in simple unbraced while statements, turning +# them into one-liners, as in 'while (expr)\n stmt;' => 'while (expr) stmt;'. +nl_create_while_one_liner = false # true/false + +# Whether to collapse a function definition whose body (not counting braces) +# is only one line so that the entire definition (prototype, braces, body) is +# a single line. +nl_create_func_def_one_liner = false # true/false + +# Whether to split one-line simple list definitions into three lines by +# adding newlines, as in 'int a[12] = { <here> 0 <here> };'. +nl_create_list_one_liner = false # true/false + +# Whether to split one-line simple unbraced if statements into two lines by +# adding a newline, as in 'if(b) <here> i++;'. +nl_split_if_one_liner = false # true/false + +# Whether to split one-line simple unbraced for statements into two lines by +# adding a newline, as in 'for (...) <here> stmt;'. +nl_split_for_one_liner = false # true/false + +# Whether to split one-line simple unbraced while statements into two lines by +# adding a newline, as in 'while (expr) <here> stmt;'. +nl_split_while_one_liner = false # true/false + +# Don't add a newline before a cpp-comment in a parameter list of a function +# call. +donot_add_nl_before_cpp_comment = false # true/false + +# +# Blank line options +# + +# The maximum number of consecutive newlines (3 = 2 blank lines). +nl_max = 0 # unsigned number + +# The maximum number of consecutive newlines in a function. +nl_max_blank_in_func = 0 # unsigned number + +# The number of newlines inside an empty function body. +# This option overrides eat_blanks_after_open_brace and +# eat_blanks_before_close_brace, but is ignored when +# nl_collapse_empty_body_functions=true +nl_inside_empty_func = 0 # unsigned number + +# The number of newlines before a function prototype. +nl_before_func_body_proto = 0 # unsigned number + +# The number of newlines before a multi-line function definition. Where +# applicable, this option is overridden with eat_blanks_after_open_brace=true +nl_before_func_body_def = 0 # unsigned number + +# The number of newlines before a class constructor/destructor prototype. +nl_before_func_class_proto = 0 # unsigned number + +# The number of newlines before a class constructor/destructor definition. +nl_before_func_class_def = 0 # unsigned number + +# The number of newlines after a function prototype. +nl_after_func_proto = 0 # unsigned number + +# The number of newlines after a function prototype, if not followed by +# another function prototype. +nl_after_func_proto_group = 0 # unsigned number + +# The number of newlines after a class constructor/destructor prototype. +nl_after_func_class_proto = 0 # unsigned number + +# The number of newlines after a class constructor/destructor prototype, +# if not followed by another constructor/destructor prototype. +nl_after_func_class_proto_group = 0 # unsigned number + +# Whether one-line method definitions inside a class body should be treated +# as if they were prototypes for the purposes of adding newlines. +# +# Requires nl_class_leave_one_liners=true. Overrides nl_before_func_body_def +# and nl_before_func_class_def for one-liners. +nl_class_leave_one_liner_groups = false # true/false + +# The number of newlines after '}' of a multi-line function body. +# +# Overrides nl_min_after_func_body and nl_max_after_func_body. +nl_after_func_body = 0 # unsigned number + +# The minimum number of newlines after '}' of a multi-line function body. +# +# Only works when nl_after_func_body is 0. +nl_min_after_func_body = 0 # unsigned number + +# The maximum number of newlines after '}' of a multi-line function body. +# +# Only works when nl_after_func_body is 0. +# Takes precedence over nl_min_after_func_body. +nl_max_after_func_body = 0 # unsigned number + +# The number of newlines after '}' of a multi-line function body in a class +# declaration. Also affects class constructors/destructors. +# +# Overrides nl_after_func_body. +nl_after_func_body_class = 0 # unsigned number + +# The number of newlines after '}' of a single line function body. Also +# affects class constructors/destructors. +# +# Overrides nl_after_func_body and nl_after_func_body_class. +nl_after_func_body_one_liner = 0 # unsigned number + +# The number of newlines before a block of typedefs. If nl_after_access_spec +# is non-zero, that option takes precedence. +# +# 0: No change (default). +nl_typedef_blk_start = 0 # unsigned number + +# The number of newlines after a block of typedefs. +# +# 0: No change (default). +nl_typedef_blk_end = 0 # unsigned number + +# The maximum number of consecutive newlines within a block of typedefs. +# +# 0: No change (default). +nl_typedef_blk_in = 0 # unsigned number + +# The minimum number of blank lines after a block of variable definitions +# at the top of a function body. If any preprocessor directives appear +# between the opening brace of the function and the variable block, then +# it is considered as not at the top of the function.Newlines are added +# before trailing preprocessor directives, if any exist. +# +# 0: No change (default). +nl_var_def_blk_end_func_top = 0 # unsigned number + +# The minimum number of empty newlines before a block of variable definitions +# not at the top of a function body. If nl_after_access_spec is non-zero, +# that option takes precedence. Newlines are not added at the top of the +# file or just after an opening brace. Newlines are added above any +# preprocessor directives before the block. +# +# 0: No change (default). +nl_var_def_blk_start = 0 # unsigned number + +# The minimum number of empty newlines after a block of variable definitions +# not at the top of a function body. Newlines are not added if the block +# is at the bottom of the file or just before a preprocessor directive. +# +# 0: No change (default). +nl_var_def_blk_end = 0 # unsigned number + +# The maximum number of consecutive newlines within a block of variable +# definitions. +# +# 0: No change (default). +nl_var_def_blk_in = 0 # unsigned number + +# The minimum number of newlines before a multi-line comment. +# Doesn't apply if after a brace open or another multi-line comment. +nl_before_block_comment = 0 # unsigned number + +# The minimum number of newlines before a single-line C comment. +# Doesn't apply if after a brace open or other single-line C comments. +nl_before_c_comment = 0 # unsigned number + +# The minimum number of newlines before a CPP comment. +# Doesn't apply if after a brace open or other CPP comments. +nl_before_cpp_comment = 0 # unsigned number + +# Whether to force a newline after a multi-line comment. +nl_after_multiline_comment = false # true/false + +# Whether to force a newline after a label's colon. +nl_after_label_colon = false # true/false + +# The number of newlines before a struct definition. +nl_before_struct = 0 # unsigned number + +# The number of newlines after '}' or ';' of a struct/enum/union definition. +nl_after_struct = 0 # unsigned number + +# The number of newlines before a class definition. +nl_before_class = 0 # unsigned number + +# The number of newlines after '}' or ';' of a class definition. +nl_after_class = 0 # unsigned number + +# The number of newlines before a namespace. +nl_before_namespace = 0 # unsigned number + +# The number of newlines after '{' of a namespace. This also adds newlines +# before the matching '}'. +# +# 0: Apply eat_blanks_after_open_brace or eat_blanks_before_close_brace if +# applicable, otherwise no change. +# +# Overrides eat_blanks_after_open_brace and eat_blanks_before_close_brace. +nl_inside_namespace = 0 # unsigned number + +# The number of newlines after '}' of a namespace. +nl_after_namespace = 0 # unsigned number + +# The number of newlines before an access specifier label. This also includes +# the Qt-specific 'signals:' and 'slots:'. Will not change the newline count +# if after a brace open. +# +# 0: No change (default). +nl_before_access_spec = 0 # unsigned number + +# The number of newlines after an access specifier label. This also includes +# the Qt-specific 'signals:' and 'slots:'. Will not change the newline count +# if after a brace open. +# +# 0: No change (default). +# +# Overrides nl_typedef_blk_start and nl_var_def_blk_start. +nl_after_access_spec = 0 # unsigned number + +# The number of newlines between a function definition and the function +# comment, as in '// comment\n <here> void foo() {...}'. +# +# 0: No change (default). +nl_comment_func_def = 0 # unsigned number + +# The number of newlines after a try-catch-finally block that isn't followed +# by a brace close. +# +# 0: No change (default). +nl_after_try_catch_finally = 0 # unsigned number + +# (C#) The number of newlines before and after a property, indexer or event +# declaration. +# +# 0: No change (default). +nl_around_cs_property = 0 # unsigned number + +# (C#) The number of newlines between the get/set/add/remove handlers. +# +# 0: No change (default). +nl_between_get_set = 0 # unsigned number + +# (C#) Add or remove newline between property and the '{'. +nl_property_brace = ignore # ignore/add/remove/force/not_defined + +# Whether to remove blank lines after '{'. +eat_blanks_after_open_brace = false # true/false + +# Whether to remove blank lines before '}'. +eat_blanks_before_close_brace = false # true/false + +# How aggressively to remove extra newlines not in preprocessor. +# +# 0: No change (default) +# 1: Remove most newlines not handled by other config +# 2: Remove all newlines and reformat completely by config +nl_remove_extra_newlines = 0 # unsigned number + +# (Java) Add or remove newline after an annotation statement. Only affects +# annotations that are after a newline. +nl_after_annotation = ignore # ignore/add/remove/force/not_defined + +# (Java) Add or remove newline between two annotations. +nl_between_annotation = ignore # ignore/add/remove/force/not_defined + +# The number of newlines before a whole-file #ifdef. +# +# 0: No change (default). +nl_before_whole_file_ifdef = 0 # unsigned number + +# The number of newlines after a whole-file #ifdef. +# +# 0: No change (default). +nl_after_whole_file_ifdef = 0 # unsigned number + +# The number of newlines before a whole-file #endif. +# +# 0: No change (default). +nl_before_whole_file_endif = 0 # unsigned number + +# The number of newlines after a whole-file #endif. +# +# 0: No change (default). +nl_after_whole_file_endif = 0 # unsigned number + +# +# Positioning options +# + +# The position of arithmetic operators in wrapped expressions. +pos_arith = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of assignment in wrapped expressions. Do not affect '=' +# followed by '{'. +pos_assign = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of Boolean operators in wrapped expressions. +pos_bool = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of comparison operators in wrapped expressions. +pos_compare = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of conditional operators, as in the '?' and ':' of +# 'expr ? stmt : stmt', in wrapped expressions. +pos_conditional = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of the comma in wrapped expressions. +pos_comma = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of the comma in enum entries. +pos_enum_comma = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of the comma in the base class list if there is more than one +# line. Affects nl_class_init_args. +pos_class_comma = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of the comma in the constructor initialization list. +# Related to nl_constr_colon, nl_constr_init_args and pos_constr_colon. +pos_constr_comma = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of trailing/leading class colon, between class and base class +# list. Affects nl_class_colon. +pos_class_colon = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of colons between constructor and member initialization. +# Related to nl_constr_colon, nl_constr_init_args and pos_constr_comma. +pos_constr_colon = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# The position of shift operators in wrapped expressions. +pos_shift = ignore # ignore/break/force/lead/trail/join/lead_break/lead_force/trail_break/trail_force + +# +# Line splitting options +# + +# Try to limit code width to N columns. +code_width = 80 # unsigned number + +# Whether to fully split long 'for' statements at semi-colons. +ls_for_split_full = false # true/false + +# Whether to fully split long function prototypes/calls at commas. +# The option ls_code_width has priority over the option ls_func_split_full. +ls_func_split_full = false # true/false + +# Whether to split lines as close to code_width as possible and ignore some +# groupings. +# The option ls_code_width has priority over the option ls_func_split_full. +ls_code_width = false # true/false + +# +# Code alignment options (not left column spaces/tabs) +# + +# Whether to keep non-indenting tabs. +align_keep_tabs = false # true/false + +# Whether to use tabs for aligning. +align_with_tabs = false # true/false + +# Whether to bump out to the next tab when aligning. +align_on_tabstop = false # true/false + +# Whether to right-align numbers. +align_number_right = false # true/false + +# Whether to keep whitespace not required for alignment. +align_keep_extra_space = false # true/false + +# Whether to align variable definitions in prototypes and functions. +align_func_params = false # true/false + +# The span for aligning parameter definitions in function on parameter name. +# +# 0: Don't align (default). +align_func_params_span = 0 # unsigned number + +# The threshold for aligning function parameter definitions. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_func_params_thresh = 0 # number + +# The gap for aligning function parameter definitions. +align_func_params_gap = 0 # unsigned number + +# The span for aligning constructor value. +# +# 0: Don't align (default). +align_constr_value_span = 0 # unsigned number + +# The threshold for aligning constructor value. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_constr_value_thresh = 0 # number + +# The gap for aligning constructor value. +align_constr_value_gap = 0 # unsigned number + +# Whether to align parameters in single-line functions that have the same +# name. The function names must already be aligned with each other. +align_same_func_call_params = false # true/false + +# The span for aligning function-call parameters for single line functions. +# +# 0: Don't align (default). +align_same_func_call_params_span = 0 # unsigned number + +# The threshold for aligning function-call parameters for single line +# functions. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_same_func_call_params_thresh = 0 # number + +# The span for aligning variable definitions. +# +# 0: Don't align (default). +align_var_def_span = 0 # unsigned number + +# How to consider (or treat) the '*' in the alignment of variable definitions. +# +# 0: Part of the type 'void * foo;' (default) +# 1: Part of the variable 'void *foo;' +# 2: Dangling 'void *foo;' +# Dangling: the '*' will not be taken into account when aligning. +align_var_def_star_style = 0 # unsigned number + +# How to consider (or treat) the '&' in the alignment of variable definitions. +# +# 0: Part of the type 'long & foo;' (default) +# 1: Part of the variable 'long &foo;' +# 2: Dangling 'long &foo;' +# Dangling: the '&' will not be taken into account when aligning. +align_var_def_amp_style = 0 # unsigned number + +# The threshold for aligning variable definitions. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_var_def_thresh = 0 # number + +# The gap for aligning variable definitions. +align_var_def_gap = 0 # unsigned number + +# Whether to align the colon in struct bit fields. +align_var_def_colon = false # true/false + +# The gap for aligning the colon in struct bit fields. +align_var_def_colon_gap = 0 # unsigned number + +# Whether to align any attribute after the variable name. +align_var_def_attribute = false # true/false + +# Whether to align inline struct/enum/union variable definitions. +align_var_def_inline = false # true/false + +# The span for aligning on '=' in assignments. +# +# 0: Don't align (default). +align_assign_span = 0 # unsigned number + +# The span for aligning on '=' in function prototype modifier. +# +# 0: Don't align (default). +align_assign_func_proto_span = 0 # unsigned number + +# The threshold for aligning on '=' in assignments. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_assign_thresh = 0 # number + +# Whether to align on the left most assignment when multiple +# definitions are found on the same line. +# Depends on 'align_assign_span' and 'align_assign_thresh' settings. +align_assign_on_multi_var_defs = false # true/false + +# The span for aligning on '{' in braced init list. +# +# 0: Don't align (default). +align_braced_init_list_span = 0 # unsigned number + +# The threshold for aligning on '{' in braced init list. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_braced_init_list_thresh = 0 # number + +# How to apply align_assign_span to function declaration "assignments", i.e. +# 'virtual void foo() = 0' or '~foo() = {default|delete}'. +# +# 0: Align with other assignments (default) +# 1: Align with each other, ignoring regular assignments +# 2: Don't align +align_assign_decl_func = 0 # unsigned number + +# The span for aligning on '=' in enums. +# +# 0: Don't align (default). +align_enum_equ_span = 0 # unsigned number + +# The threshold for aligning on '=' in enums. +# Use a negative number for absolute thresholds. +# +# 0: no limit (default). +align_enum_equ_thresh = 0 # number + +# The span for aligning class member definitions. +# +# 0: Don't align (default). +align_var_class_span = 0 # unsigned number + +# The threshold for aligning class member definitions. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_var_class_thresh = 0 # number + +# The gap for aligning class member definitions. +align_var_class_gap = 0 # unsigned number + +# The span for aligning struct/union member definitions. +# +# 0: Don't align (default). +align_var_struct_span = 0 # unsigned number + +# The threshold for aligning struct/union member definitions. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_var_struct_thresh = 0 # number + +# The gap for aligning struct/union member definitions. +align_var_struct_gap = 0 # unsigned number + +# The span for aligning struct initializer values. +# +# 0: Don't align (default). +align_struct_init_span = 0 # unsigned number + +# The span for aligning single-line typedefs. +# +# 0: Don't align (default). +align_typedef_span = 0 # unsigned number + +# The minimum space between the type and the synonym of a typedef. +align_typedef_gap = 0 # unsigned number + +# How to align typedef'd functions with other typedefs. +# +# 0: Don't mix them at all (default) +# 1: Align the open parenthesis with the types +# 2: Align the function type name with the other type names +align_typedef_func = 0 # unsigned number + +# How to consider (or treat) the '*' in the alignment of typedefs. +# +# 0: Part of the typedef type, 'typedef int * pint;' (default) +# 1: Part of type name: 'typedef int *pint;' +# 2: Dangling: 'typedef int *pint;' +# Dangling: the '*' will not be taken into account when aligning. +align_typedef_star_style = 0 # unsigned number + +# How to consider (or treat) the '&' in the alignment of typedefs. +# +# 0: Part of the typedef type, 'typedef int & intref;' (default) +# 1: Part of type name: 'typedef int &intref;' +# 2: Dangling: 'typedef int &intref;' +# Dangling: the '&' will not be taken into account when aligning. +align_typedef_amp_style = 0 # unsigned number + +# The span for aligning comments that end lines. +# +# 0: Don't align (default). +align_right_cmt_span = 0 # unsigned number + +# Minimum number of columns between preceding text and a trailing comment in +# order for the comment to qualify for being aligned. Must be non-zero to have +# an effect. +align_right_cmt_gap = 0 # unsigned number + +# If aligning comments, whether to mix with comments after '}' and #endif with +# less than three spaces before the comment. +align_right_cmt_mix = false # true/false + +# Whether to only align trailing comments that are at the same brace level. +align_right_cmt_same_level = false # true/false + +# Minimum column at which to align trailing comments. Comments which are +# aligned beyond this column, but which can be aligned in a lesser column, +# may be "pulled in". +# +# 0: Ignore (default). +align_right_cmt_at_col = 0 # unsigned number + +# The span for aligning function prototypes. +# +# 0: Don't align (default). +align_func_proto_span = 0 # unsigned number + +# Whether to ignore continuation lines when evaluating the number of +# new lines for the function prototype alignment's span. +# +# false: continuation lines are part of the newlines count +# true: continuation lines are not counted +align_func_proto_span_ignore_cont_lines = false # true/false + +# How to consider (or treat) the '*' in the alignment of function prototypes. +# +# 0: Part of the type 'void * foo();' (default) +# 1: Part of the function 'void *foo();' +# 2: Dangling 'void *foo();' +# Dangling: the '*' will not be taken into account when aligning. +align_func_proto_star_style = 0 # unsigned number + +# How to consider (or treat) the '&' in the alignment of function prototypes. +# +# 0: Part of the type 'long & foo();' (default) +# 1: Part of the function 'long &foo();' +# 2: Dangling 'long &foo();' +# Dangling: the '&' will not be taken into account when aligning. +align_func_proto_amp_style = 0 # unsigned number + +# The threshold for aligning function prototypes. +# Use a negative number for absolute thresholds. +# +# 0: No limit (default). +align_func_proto_thresh = 0 # number + +# Minimum gap between the return type and the function name. +align_func_proto_gap = 0 # unsigned number + +# Whether to align function prototypes on the 'operator' keyword instead of +# what follows. +align_on_operator = false # true/false + +# Whether to mix aligning prototype and variable declarations. If true, +# align_var_def_XXX options are used instead of align_func_proto_XXX options. +align_mix_var_proto = false # true/false + +# Whether to align single-line functions with function prototypes. +# Uses align_func_proto_span. +align_single_line_func = false # true/false + +# Whether to align the open brace of single-line functions. +# Requires align_single_line_func=true. Uses align_func_proto_span. +align_single_line_brace = false # true/false + +# Gap for align_single_line_brace. +align_single_line_brace_gap = 0 # unsigned number + +# (OC) The span for aligning Objective-C message specifications. +# +# 0: Don't align (default). +align_oc_msg_spec_span = 0 # unsigned number + +# Whether and how to align backslashes that split a macro onto multiple lines. +# This will not work right if the macro contains a multi-line comment. +# +# 0: Do nothing (default) +# 1: Align the backslashes in the column at the end of the longest line +# 2: Align with the backslash that is farthest to the left, or, if that +# backslash is farther left than the end of the longest line, at the end of +# the longest line +# 3: Align with the backslash that is farthest to the right +align_nl_cont = 1 # unsigned number + +# The minimum number of spaces between the end of a line and its continuation +# backslash. Requires align_nl_cont. +# +# Default: 1 +align_nl_cont_spaces = 1 # unsigned number + +# Whether to align macro functions and variables together. +align_pp_define_together = false # true/false + +# The span for aligning on '#define' bodies. +# +# =0: Don't align (default) +# >0: Number of lines (including comments) between blocks +align_pp_define_span = 0 # unsigned number + +# The minimum space between label and value of a preprocessor define. +align_pp_define_gap = 0 # unsigned number + +# Whether to align lines that start with '<<' with previous '<<'. +# +# Default: true +align_left_shift = true # true/false + +# Whether to align comma-separated statements following '<<' (as used to +# initialize Eigen matrices). +align_eigen_comma_init = false # true/false + +# Whether to align text after 'asm volatile ()' colons. +align_asm_colon = false # true/false + +# (OC) Span for aligning parameters in an Objective-C message call +# on the ':'. +# +# 0: Don't align. +align_oc_msg_colon_span = 0 # unsigned number + +# (OC) Whether to always align with the first parameter, even if it is too +# short. +align_oc_msg_colon_first = false # true/false + +# (OC) Whether to align parameters in an Objective-C '+' or '-' declaration +# on the ':'. +align_oc_decl_colon = false # true/false + +# (OC) Whether to not align parameters in an Objectve-C message call if first +# colon is not on next line of the message call (the same way Xcode does +# alignment) +align_oc_msg_colon_xcode_like = false # true/false + +# +# Comment modification options +# + +# Try to wrap comments at N columns. +cmt_width = 0 # unsigned number + +# How to reflow comments. +# +# 0: No reflowing (apart from the line wrapping due to cmt_width) (default) +# 1: No touching at all +# 2: Full reflow (enable cmt_indent_multi for indent with line wrapping due to cmt_width) +cmt_reflow_mode = 0 # unsigned number + +# Path to a file that contains regular expressions describing patterns for +# which the end of one line and the beginning of the next will be folded into +# the same sentence or paragraph during full comment reflow. The regular +# expressions are described using ECMAScript syntax. The syntax for this +# specification is as follows, where "..." indicates the custom regular +# expression and "n" indicates the nth end_of_prev_line_regex and +# beg_of_next_line_regex regular expression pair: +# +# end_of_prev_line_regex[1] = "...$" +# beg_of_next_line_regex[1] = "^..." +# end_of_prev_line_regex[2] = "...$" +# beg_of_next_line_regex[2] = "^..." +# . +# . +# . +# end_of_prev_line_regex[n] = "...$" +# beg_of_next_line_regex[n] = "^..." +# +# Note that use of this option overrides the default reflow fold regular +# expressions, which are internally defined as follows: +# +# end_of_prev_line_regex[1] = "[\w,\]\)]$" +# beg_of_next_line_regex[1] = "^[\w,\[\(]" +# end_of_prev_line_regex[2] = "\.$" +# beg_of_next_line_regex[2] = "^[A-Z]" +cmt_reflow_fold_regex_file = "" # string + +# Whether to indent wrapped lines to the start of the encompassing paragraph +# during full comment reflow (cmt_reflow_mode = 2). Overrides the value +# specified by cmt_sp_after_star_cont. +# +# Note that cmt_align_doxygen_javadoc_tags overrides this option for +# paragraphs associated with javadoc tags +cmt_reflow_indent_to_paragraph_start = false # true/false + +# Whether to convert all tabs to spaces in comments. If false, tabs in +# comments are left alone, unless used for indenting. +cmt_convert_tab_to_spaces = false # true/false + +# Whether to apply changes to multi-line comments, including cmt_width, +# keyword substitution and leading chars. +# +# Default: true +cmt_indent_multi = true # true/false + +# Whether to align doxygen javadoc-style tags ('@param', '@return', etc.) +# and corresponding fields such that groups of consecutive block tags, +# parameter names, and descriptions align with one another. Overrides that +# which is specified by the cmt_sp_after_star_cont. If cmt_width > 0, it may +# be necessary to enable cmt_indent_multi and set cmt_reflow_mode = 2 +# in order to achieve the desired alignment for line-wrapping. +cmt_align_doxygen_javadoc_tags = false # true/false + +# The number of spaces to insert after the star and before doxygen +# javadoc-style tags (@param, @return, etc). Requires enabling +# cmt_align_doxygen_javadoc_tags. Overrides that which is specified by the +# cmt_sp_after_star_cont. +# +# Default: 1 +cmt_sp_before_doxygen_javadoc_tags = 1 # unsigned number + +# Whether to change trailing, single-line c-comments into cpp-comments. +cmt_trailing_single_line_c_to_cpp = false # true/false + +# Whether to group c-comments that look like they are in a block. +cmt_c_group = false # true/false + +# Whether to put an empty '/*' on the first line of the combined c-comment. +cmt_c_nl_start = false # true/false + +# Whether to add a newline before the closing '*/' of the combined c-comment. +cmt_c_nl_end = false # true/false + +# Whether to change cpp-comments into c-comments. +cmt_cpp_to_c = false # true/false + +# Whether to group cpp-comments that look like they are in a block. Only +# meaningful if cmt_cpp_to_c=true. +cmt_cpp_group = false # true/false + +# Whether to put an empty '/*' on the first line of the combined cpp-comment +# when converting to a c-comment. +# +# Requires cmt_cpp_to_c=true and cmt_cpp_group=true. +cmt_cpp_nl_start = false # true/false + +# Whether to add a newline before the closing '*/' of the combined cpp-comment +# when converting to a c-comment. +# +# Requires cmt_cpp_to_c=true and cmt_cpp_group=true. +cmt_cpp_nl_end = false # true/false + +# Whether to put a star on subsequent comment lines. +cmt_star_cont = false # true/false + +# The number of spaces to insert at the start of subsequent comment lines. +cmt_sp_before_star_cont = 0 # unsigned number + +# The number of spaces to insert after the star on subsequent comment lines. +cmt_sp_after_star_cont = 0 # unsigned number + +# For multi-line comments with a '*' lead, remove leading spaces if the first +# and last lines of the comment are the same length. +# +# Default: true +cmt_multi_check_last = true # true/false + +# For multi-line comments with a '*' lead, remove leading spaces if the first +# and last lines of the comment are the same length AND if the length is +# bigger as the first_len minimum. +# +# Default: 4 +cmt_multi_first_len_minimum = 4 # unsigned number + +# Path to a file that contains text to insert at the beginning of a file if +# the file doesn't start with a C/C++ comment. If the inserted text contains +# '$(filename)', that will be replaced with the current file's name. +cmt_insert_file_header = "" # string + +# Path to a file that contains text to insert at the end of a file if the +# file doesn't end with a C/C++ comment. If the inserted text contains +# '$(filename)', that will be replaced with the current file's name. +cmt_insert_file_footer = "" # string + +# Path to a file that contains text to insert before a function definition if +# the function isn't preceded by a C/C++ comment. If the inserted text +# contains '$(function)', '$(javaparam)' or '$(fclass)', these will be +# replaced with, respectively, the name of the function, the javadoc '@param' +# and '@return' stuff, or the name of the class to which the member function +# belongs. +cmt_insert_func_header = "" # string + +# Path to a file that contains text to insert before a class if the class +# isn't preceded by a C/C++ comment. If the inserted text contains '$(class)', +# that will be replaced with the class name. +cmt_insert_class_header = "" # string + +# Path to a file that contains text to insert before an Objective-C message +# specification, if the method isn't preceded by a C/C++ comment. If the +# inserted text contains '$(message)' or '$(javaparam)', these will be +# replaced with, respectively, the name of the function, or the javadoc +# '@param' and '@return' stuff. +cmt_insert_oc_msg_header = "" # string + +# Whether a comment should be inserted if a preprocessor is encountered when +# stepping backwards from a function name. +# +# Applies to cmt_insert_oc_msg_header, cmt_insert_func_header and +# cmt_insert_class_header. +cmt_insert_before_preproc = false # true/false + +# Whether a comment should be inserted if a function is declared inline to a +# class definition. +# +# Applies to cmt_insert_func_header. +# +# Default: true +cmt_insert_before_inlines = true # true/false + +# Whether a comment should be inserted if the function is a class constructor +# or destructor. +# +# Applies to cmt_insert_func_header. +cmt_insert_before_ctor_dtor = false # true/false + +# +# Code modifying options (non-whitespace) +# + +# Add or remove braces on a single-line 'do' statement. +mod_full_brace_do = ignore # ignore/add/remove/force/not_defined + +# Add or remove braces on a single-line 'for' statement. +mod_full_brace_for = ignore # ignore/add/remove/force/not_defined + +# (Pawn) Add or remove braces on a single-line function definition. +mod_full_brace_function = ignore # ignore/add/remove/force/not_defined + +# Add or remove braces on a single-line 'if' statement. Braces will not be +# removed if the braced statement contains an 'else'. +mod_full_brace_if = ignore # ignore/add/remove/force/not_defined + +# Whether to enforce that all blocks of an 'if'/'else if'/'else' chain either +# have, or do not have, braces. Overrides mod_full_brace_if. +# +# 0: Don't override mod_full_brace_if +# 1: Add braces to all blocks if any block needs braces and remove braces if +# they can be removed from all blocks +# 2: Add braces to all blocks if any block already has braces, regardless of +# whether it needs them +# 3: Add braces to all blocks if any block needs braces and remove braces if +# they can be removed from all blocks, except if all blocks have braces +# despite none needing them +mod_full_brace_if_chain = 0 # unsigned number + +# Whether to add braces to all blocks of an 'if'/'else if'/'else' chain. +# If true, mod_full_brace_if_chain will only remove braces from an 'if' that +# does not have an 'else if' or 'else'. +mod_full_brace_if_chain_only = false # true/false + +# Add or remove braces on single-line 'while' statement. +mod_full_brace_while = ignore # ignore/add/remove/force/not_defined + +# Add or remove braces on single-line 'using ()' statement. +mod_full_brace_using = ignore # ignore/add/remove/force/not_defined + +# Don't remove braces around statements that span N newlines +mod_full_brace_nl = 0 # unsigned number + +# Whether to prevent removal of braces from 'if'/'for'/'while'/etc. blocks +# which span multiple lines. +# +# Affects: +# mod_full_brace_for +# mod_full_brace_if +# mod_full_brace_if_chain +# mod_full_brace_if_chain_only +# mod_full_brace_while +# mod_full_brace_using +# +# Does not affect: +# mod_full_brace_do +# mod_full_brace_function +mod_full_brace_nl_block_rem_mlcond = false # true/false + +# Add or remove unnecessary parentheses on 'return' statement. +mod_paren_on_return = ignore # ignore/add/remove/force/not_defined + +# Add or remove unnecessary parentheses on 'throw' statement. +mod_paren_on_throw = ignore # ignore/add/remove/force/not_defined + +# (Pawn) Whether to change optional semicolons to real semicolons. +mod_pawn_semicolon = false # true/false + +# Whether to fully parenthesize Boolean expressions in 'while' and 'if' +# statement, as in 'if (a && b > c)' => 'if (a && (b > c))'. +mod_full_paren_if_bool = false # true/false + +# Whether to fully parenthesize Boolean expressions after '=' +# statement, as in 'x = a && b > c;' => 'x = (a && (b > c));'. +mod_full_paren_assign_bool = false # true/false + +# Whether to fully parenthesize Boolean expressions after '=' +# statement, as in 'return a && b > c;' => 'return (a && (b > c));'. +mod_full_paren_return_bool = false # true/false + +# Whether to remove superfluous semicolons. +mod_remove_extra_semicolon = false # true/false + +# Whether to remove duplicate include. +mod_remove_duplicate_include = false # true/false + +# the following options (mod_XX_closebrace_comment) use different comment, +# depending of the setting of the next option. +# false: Use the c comment (default) +# true : Use the cpp comment +mod_add_force_c_closebrace_comment = false # true/false + +# If a function body exceeds the specified number of newlines and doesn't have +# a comment after the close brace, a comment will be added. +mod_add_long_function_closebrace_comment = 0 # unsigned number + +# If a namespace body exceeds the specified number of newlines and doesn't +# have a comment after the close brace, a comment will be added. +mod_add_long_namespace_closebrace_comment = 0 # unsigned number + +# If a class body exceeds the specified number of newlines and doesn't have a +# comment after the close brace, a comment will be added. +mod_add_long_class_closebrace_comment = 0 # unsigned number + +# If a switch body exceeds the specified number of newlines and doesn't have a +# comment after the close brace, a comment will be added. +mod_add_long_switch_closebrace_comment = 0 # unsigned number + +# If an #ifdef body exceeds the specified number of newlines and doesn't have +# a comment after the #endif, a comment will be added. +mod_add_long_ifdef_endif_comment = 0 # unsigned number + +# If an #ifdef or #else body exceeds the specified number of newlines and +# doesn't have a comment after the #else, a comment will be added. +mod_add_long_ifdef_else_comment = 0 # unsigned number + +# Whether to take care of the case by the mod_sort_xx options. +mod_sort_case_sensitive = false # true/false + +# Whether to sort consecutive single-line 'import' statements. +mod_sort_import = false # true/false + +# (C#) Whether to sort consecutive single-line 'using' statements. +mod_sort_using = false # true/false + +# Whether to sort consecutive single-line '#include' statements (C/C++) and +# '#import' statements (Objective-C). Be aware that this has the potential to +# break your code if your includes/imports have ordering dependencies. +mod_sort_include = false # true/false + +# Whether to prioritize '#include' and '#import' statements that contain +# filename without extension when sorting is enabled. +mod_sort_incl_import_prioritize_filename = false # true/false + +# Whether to prioritize '#include' and '#import' statements that does not +# contain extensions when sorting is enabled. +mod_sort_incl_import_prioritize_extensionless = false # true/false + +# Whether to prioritize '#include' and '#import' statements that contain +# angle over quotes when sorting is enabled. +mod_sort_incl_import_prioritize_angle_over_quotes = false # true/false + +# Whether to ignore file extension in '#include' and '#import' statements +# for sorting comparison. +mod_sort_incl_import_ignore_extension = false # true/false + +# Whether to group '#include' and '#import' statements when sorting is enabled. +mod_sort_incl_import_grouping_enabled = false # true/false + +# Whether to move a 'break' that appears after a fully braced 'case' before +# the close brace, as in 'case X: { ... } break;' => 'case X: { ... break; }'. +mod_move_case_break = false # true/false + +# Whether to move a 'return' that appears after a fully braced 'case' before +# the close brace, as in 'case X: { ... } return;' => 'case X: { ... return; }'. +mod_move_case_return = false # true/false + +# Add or remove braces around a fully braced case statement. Will only remove +# braces if there are no variable declarations in the block. +mod_case_brace = ignore # ignore/add/remove/force/not_defined + +# Whether to remove a void 'return;' that appears as the last statement in a +# function. +mod_remove_empty_return = false # true/false + +# Add or remove the comma after the last value of an enumeration. +mod_enum_last_comma = ignore # ignore/add/remove/force/not_defined + +# Syntax to use for infinite loops. +# +# 0: Leave syntax alone (default) +# 1: Rewrite as `for(;;)` +# 2: Rewrite as `while(true)` +# 3: Rewrite as `do`...`while(true);` +# 4: Rewrite as `while(1)` +# 5: Rewrite as `do`...`while(1);` +# +# Infinite loops that do not already match one of these syntaxes are ignored. +# Other options that affect loop formatting will be applied after transforming +# the syntax. +mod_infinite_loop = 0 # unsigned number + +# Add or remove the 'int' keyword in 'int short'. +mod_int_short = ignore # ignore/add/remove/force/not_defined + +# Add or remove the 'int' keyword in 'short int'. +mod_short_int = ignore # ignore/add/remove/force/not_defined + +# Add or remove the 'int' keyword in 'int long'. +mod_int_long = ignore # ignore/add/remove/force/not_defined + +# Add or remove the 'int' keyword in 'long int'. +mod_long_int = ignore # ignore/add/remove/force/not_defined + +# Add or remove the 'int' keyword in 'int signed'. +mod_int_signed = ignore # ignore/add/remove/force/not_defined + +# Add or remove the 'int' keyword in 'signed int'. +mod_signed_int = ignore # ignore/add/remove/force/not_defined + +# Add or remove the 'int' keyword in 'int unsigned'. +mod_int_unsigned = ignore # ignore/add/remove/force/not_defined + +# Add or remove the 'int' keyword in 'unsigned int'. +mod_unsigned_int = ignore # ignore/add/remove/force/not_defined + +# If there is a situation where mod_int_* and mod_*_int would result in +# multiple int keywords, whether to keep the rightmost int (the default) or the +# leftmost int. +mod_int_prefer_int_on_left = false # true/false + +# (OC) Whether to organize the properties. If true, properties will be +# rearranged according to the mod_sort_oc_property_*_weight factors. +mod_sort_oc_properties = false # true/false + +# (OC) Weight of a class property modifier. +mod_sort_oc_property_class_weight = 0 # number + +# (OC) Weight of 'atomic' and 'nonatomic'. +mod_sort_oc_property_thread_safe_weight = 0 # number + +# (OC) Weight of 'readwrite' when organizing properties. +mod_sort_oc_property_readwrite_weight = 0 # number + +# (OC) Weight of a reference type specifier ('retain', 'copy', 'assign', +# 'weak', 'strong') when organizing properties. +mod_sort_oc_property_reference_weight = 0 # number + +# (OC) Weight of getter type ('getter=') when organizing properties. +mod_sort_oc_property_getter_weight = 0 # number + +# (OC) Weight of setter type ('setter=') when organizing properties. +mod_sort_oc_property_setter_weight = 0 # number + +# (OC) Weight of nullability type ('nullable', 'nonnull', 'null_unspecified', +# 'null_resettable') when organizing properties. +mod_sort_oc_property_nullability_weight = 0 # number + +# +# Preprocessor options +# + +# How to use tabs when indenting preprocessor code. +# +# -1: Use 'indent_with_tabs' setting (default) +# 0: Spaces only +# 1: Indent with tabs to brace level, align with spaces +# 2: Indent and align with tabs, using spaces when not on a tabstop +# +# Default: -1 +pp_indent_with_tabs = -1 # number + +# Add or remove indentation of preprocessor directives inside #if blocks +# at brace level 0 (file-level). +pp_indent = ignore # ignore/add/remove/force/not_defined + +# Whether to indent #if/#else/#endif at the brace level. If false, these are +# indented from column 1. +pp_indent_at_level = false # true/false + +# Whether to indent #if/#else/#endif at the parenthesis level if the brace +# level is 0. If false, these are indented from column 1. +pp_indent_at_level0 = false # true/false + +# Specifies the number of columns to indent preprocessors per level +# at brace level 0 (file-level). If pp_indent_at_level=false, also specifies +# the number of columns to indent preprocessors per level +# at brace level > 0 (function-level). +# +# Default: 1 +pp_indent_count = 1 # unsigned number + +# Add or remove space after # based on pp level of #if blocks. +pp_space_after = ignore # ignore/add/remove/force/not_defined + +# Sets the number of spaces per level added with pp_space_after. +pp_space_count = 0 # unsigned number + +# The indent for '#region' and '#endregion' in C# and '#pragma region' in +# C/C++. Negative values decrease indent down to the first column. +pp_indent_region = 0 # number + +# Whether to indent the code between #region and #endregion. +pp_region_indent_code = false # true/false + +# If pp_indent_at_level=true, sets the indent for #if, #else and #endif when +# not at file-level. Negative values decrease indent down to the first column. +# +# =0: Indent preprocessors using output_tab_size +# >0: Column at which all preprocessors will be indented +pp_indent_if = 0 # number + +# Whether to indent the code between #if, #else and #endif. +pp_if_indent_code = false # true/false + +# Whether to indent the body of an #if that encompasses all the code in the file. +pp_indent_in_guard = false # true/false + +# Whether to indent '#define' at the brace level. If false, these are +# indented from column 1. +pp_define_at_level = false # true/false + +# Whether to indent '#include' at the brace level. +pp_include_at_level = false # true/false + +# Whether to ignore the '#define' body while formatting. +pp_ignore_define_body = false # true/false + +# An offset value that controls the indentation of the body of a multiline #define. +# 'body' refers to all the lines of a multiline #define except the first line. +# Requires 'pp_ignore_define_body = false'. +# +# <0: Absolute column: the body indentation starts off at the specified column +# (ex. -3 ==> the body is indented starting from column 3) +# >=0: Relative to the column of the '#' of '#define' +# (ex. 3 ==> the body is indented starting 3 columns at the right of '#') +# +# Default: 8 +pp_multiline_define_body_indent = 8 # number + +# Whether to indent case statements between #if, #else, and #endif. +# Only applies to the indent of the preprocessor that the case statements +# directly inside of. +# +# Default: true +pp_indent_case = true # true/false + +# Whether to indent whole function definitions between #if, #else, and #endif. +# Only applies to the indent of the preprocessor that the function definition +# is directly inside of. +# +# Default: true +pp_indent_func_def = true # true/false + +# Whether to indent extern C blocks between #if, #else, and #endif. +# Only applies to the indent of the preprocessor that the extern block is +# directly inside of. +# +# Default: true +pp_indent_extern = true # true/false + +# How to indent braces directly inside #if, #else, and #endif. +# Requires pp_if_indent_code=true and only applies to the indent of the +# preprocessor that the braces are directly inside of. +# 0: No extra indent +# 1: Indent by one level +# -1: Preserve original indentation +# +# Default: 1 +pp_indent_brace = 1 # number + +# Whether to print warning messages for unbalanced #if and #else blocks. +# This will print a message in the following cases: +# - if an #ifdef block ends on a different indent level than +# where it started from. Example: +# +# #ifdef TEST +# int i; +# { +# int j; +# #endif +# +# - an #elif/#else block ends on a different indent level than +# the corresponding #ifdef block. Example: +# +# #ifdef TEST +# int i; +# #else +# } +# int j; +# #endif +pp_warn_unbalanced_if = false # true/false + +# +# Sort includes options +# + +# The regex for include category with priority 0. +include_category_0 = "" # string + +# The regex for include category with priority 1. +include_category_1 = "" # string + +# The regex for include category with priority 2. +include_category_2 = "" # string + +# +# Use or Do not Use options +# + +# true: indent_func_call_param will be used (default) +# false: indent_func_call_param will NOT be used +# +# Default: true +use_indent_func_call_param = true # true/false + +# The value of the indentation for a continuation line is calculated +# differently if the statement is: +# - a declaration: your case with QString fileName ... +# - an assignment: your case with pSettings = new QSettings( ... +# +# At the second case the indentation value might be used twice: +# - at the assignment +# - at the function call (if present) +# +# To prevent the double use of the indentation value, use this option with the +# value 'true'. +# +# true: indent_continue will be used only once +# false: indent_continue will be used every time (default) +# +# Requires indent_ignore_first_continue=false. +use_indent_continue_only_once = false # true/false + +# The indentation can be: +# - after the assignment, at the '[' character +# - at the beginning of the lambda body +# +# true: indentation will be at the beginning of the lambda body +# false: indentation will be after the assignment (default) +indent_cpp_lambda_only_once = false # true/false + +# Whether sp_after_angle takes precedence over sp_inside_fparen. This was the +# historic behavior, but is probably not the desired behavior, so this is off +# by default. +use_sp_after_angle_always = false # true/false + +# Whether to apply special formatting for Qt SIGNAL/SLOT macros. Essentially, +# this tries to format these so that they match Qt's normalized form (i.e. the +# result of QMetaObject::normalizedSignature), which can slightly improve the +# performance of the QObject::connect call, rather than how they would +# otherwise be formatted. +# +# See options_for_QT.cpp for details. +# +# Default: true +use_options_overriding_for_qt_macros = true # true/false + +# If true: the form feed character is removed from the list of whitespace +# characters. See https://en.cppreference.com/w/cpp/string/byte/isspace. +use_form_feed_no_more_as_whitespace_character = false # true/false + +# +# Warn levels - 1: error, 2: warning (default), 3: note +# + +# (C#) Warning is given if doing tab-to-\t replacement and we have found one +# in a C# verbatim string literal. +# +# Default: 2 +warn_level_tabs_found_in_verbatim_string_literals = 2 # unsigned number + +# Limit the number of loops. +# Used by uncrustify.cpp to exit from infinite loop. +# 0: no limit. +debug_max_number_of_loops = 0 # number + +# Set the number of the line to protocol; +# Used in the function prot_the_line if the 2. parameter is zero. +# 0: nothing protocol. +debug_line_number_to_protocol = 0 # number + +# Set the number of second(s) before terminating formatting the current file, +# 0: no timeout. +# only for linux +debug_timeout = 0 # number + +# Set the number of characters to be printed if the text is too long, +# 0: do not truncate. +debug_truncate = 0 # unsigned number + +# sort (or not) the tracking info. +# +# Default: true +debug_sort_the_tracks = true # true/false + +# decode (or not) the flags as a new line. +# only if the -p option is set. +debug_decode_the_flags = false # true/false + +# use (or not) the exit(EX_SOFTWARE) function. +# +# Default: true +debug_use_the_exit_function_pop = true # true/false + +# insert the number of the line at the beginning of each line +set_numbering_for_html_output = false # true/false + +# Meaning of the settings: +# Ignore - do not do any changes +# Add - makes sure there is 1 or more space/brace/newline/etc +# Force - makes sure there is exactly 1 space/brace/newline/etc, +# behaves like Add in some contexts +# Remove - removes space/brace/newline/etc +# +# +# - Token(s) can be treated as specific type(s) with the 'set' option: +# `set tokenType tokenString [tokenString...]` +# +# Example: +# `set BOOL __AND__ __OR__` +# +# tokenTypes are defined in src/token_enum.h, use them without the +# 'CT_' prefix: 'CT_BOOL' => 'BOOL' +# +# +# - Token(s) can be treated as type(s) with the 'type' option. +# `type tokenString [tokenString...]` +# +# Example: +# `type int c_uint_8 Rectangle` +# +# This can also be achieved with `set TYPE int c_uint_8 Rectangle` +# +# +# To embed whitespace in tokenStrings use the '\' escape character, or quote +# the tokenStrings. These quotes are supported: "'` +# +# +# - Support for the auto detection of languages through the file ending can be +# added using the 'file_ext' command. +# `file_ext langType langString [langString..]` +# +# Example: +# `file_ext CPP .ch .cxx .cpp.in` +# +# langTypes are defined in uncrusify_types.h in the lang_flag_e enum, use +# them without the 'LANG_' prefix: 'LANG_CPP' => 'CPP' +# +# +# - Custom macro-based indentation can be set up using 'macro-open', +# 'macro-else' and 'macro-close'. +# `(macro-open | macro-else | macro-close) tokenString` +# +# Example: +# `macro-open BEGIN_TEMPLATE_MESSAGE_MAP` +# `macro-open BEGIN_MESSAGE_MAP` +# `macro-close END_MESSAGE_MAP` +# +# +# option(s) with 'not default' value: 8 +# |