aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/lj_arch.h12
-rw-r--r--src/lj_asm.c71
-rw-r--r--src/lj_dispatch.h4
-rw-r--r--src/lj_err.c247
-rw-r--r--src/lj_err.h19
-rw-r--r--src/lj_ffrecord.c2
-rw-r--r--src/lj_jit.h2
-rw-r--r--src/lj_mcode.c5
-rw-r--r--src/lj_opt_loop.c1
-rw-r--r--src/lj_record.c3
-rw-r--r--src/lj_snap.c1
-rw-r--r--src/lj_state.c1
-rw-r--r--src/lj_target_x86.h2
-rw-r--r--src/lj_trace.c55
-rw-r--r--src/lj_trace.h3
-rw-r--r--src/lj_vm.h3
-rw-r--r--src/vm_arm.dasc3
-rw-r--r--src/vm_arm64.dasc3
-rw-r--r--src/vm_mips.dasc9
-rw-r--r--src/vm_mips64.dasc9
-rw-r--r--src/vm_ppc.dasc3
-rw-r--r--src/vm_x64.dasc4
-rw-r--r--src/vm_x86.dasc4
23 files changed, 421 insertions, 45 deletions
diff --git a/src/lj_arch.h b/src/lj_arch.h
index ac3e3753..0a6e1b9f 100644
--- a/src/lj_arch.h
+++ b/src/lj_arch.h
@@ -173,6 +173,7 @@
173#define LJ_TARGET_X86 1 173#define LJ_TARGET_X86 1
174#define LJ_TARGET_X86ORX64 1 174#define LJ_TARGET_X86ORX64 1
175#define LJ_TARGET_EHRETREG 0 175#define LJ_TARGET_EHRETREG 0
176#define LJ_TARGET_EHRAREG 8
176#define LJ_TARGET_MASKSHIFT 1 177#define LJ_TARGET_MASKSHIFT 1
177#define LJ_TARGET_MASKROT 1 178#define LJ_TARGET_MASKROT 1
178#define LJ_TARGET_UNALIGNED 1 179#define LJ_TARGET_UNALIGNED 1
@@ -186,6 +187,7 @@
186#define LJ_TARGET_X64 1 187#define LJ_TARGET_X64 1
187#define LJ_TARGET_X86ORX64 1 188#define LJ_TARGET_X86ORX64 1
188#define LJ_TARGET_EHRETREG 0 189#define LJ_TARGET_EHRETREG 0
190#define LJ_TARGET_EHRAREG 16
189#define LJ_TARGET_JUMPRANGE 31 /* +-2^31 = +-2GB */ 191#define LJ_TARGET_JUMPRANGE 31 /* +-2^31 = +-2GB */
190#define LJ_TARGET_MASKSHIFT 1 192#define LJ_TARGET_MASKSHIFT 1
191#define LJ_TARGET_MASKROT 1 193#define LJ_TARGET_MASKROT 1
@@ -211,6 +213,7 @@
211#define LJ_ABI_EABI 1 213#define LJ_ABI_EABI 1
212#define LJ_TARGET_ARM 1 214#define LJ_TARGET_ARM 1
213#define LJ_TARGET_EHRETREG 0 215#define LJ_TARGET_EHRETREG 0
216#define LJ_TARGET_EHRAREG 14
214#define LJ_TARGET_JUMPRANGE 25 /* +-2^25 = +-32MB */ 217#define LJ_TARGET_JUMPRANGE 25 /* +-2^25 = +-32MB */
215#define LJ_TARGET_MASKSHIFT 0 218#define LJ_TARGET_MASKSHIFT 0
216#define LJ_TARGET_MASKROT 1 219#define LJ_TARGET_MASKROT 1
@@ -241,6 +244,7 @@
241#endif 244#endif
242#define LJ_TARGET_ARM64 1 245#define LJ_TARGET_ARM64 1
243#define LJ_TARGET_EHRETREG 0 246#define LJ_TARGET_EHRETREG 0
247#define LJ_TARGET_EHRAREG 30
244#define LJ_TARGET_JUMPRANGE 27 /* +-2^27 = +-128MB */ 248#define LJ_TARGET_JUMPRANGE 27 /* +-2^27 = +-128MB */
245#define LJ_TARGET_MASKSHIFT 1 249#define LJ_TARGET_MASKSHIFT 1
246#define LJ_TARGET_MASKROT 1 250#define LJ_TARGET_MASKROT 1
@@ -296,6 +300,7 @@
296 300
297#define LJ_TARGET_PPC 1 301#define LJ_TARGET_PPC 1
298#define LJ_TARGET_EHRETREG 3 302#define LJ_TARGET_EHRETREG 3
303#define LJ_TARGET_EHRAREG 65
299#define LJ_TARGET_JUMPRANGE 25 /* +-2^25 = +-32MB */ 304#define LJ_TARGET_JUMPRANGE 25 /* +-2^25 = +-32MB */
300#define LJ_TARGET_MASKSHIFT 0 305#define LJ_TARGET_MASKSHIFT 0
301#define LJ_TARGET_MASKROT 1 306#define LJ_TARGET_MASKROT 1
@@ -398,6 +403,7 @@
398#endif 403#endif
399#define LJ_TARGET_MIPS 1 404#define LJ_TARGET_MIPS 1
400#define LJ_TARGET_EHRETREG 4 405#define LJ_TARGET_EHRETREG 4
406#define LJ_TARGET_EHRAREG 31
401#define LJ_TARGET_JUMPRANGE 27 /* 2*2^27 = 256MB-aligned region */ 407#define LJ_TARGET_JUMPRANGE 27 /* 2*2^27 = 256MB-aligned region */
402#define LJ_TARGET_MASKSHIFT 1 408#define LJ_TARGET_MASKSHIFT 1
403#define LJ_TARGET_MASKROT 1 409#define LJ_TARGET_MASKROT 1
@@ -631,6 +637,12 @@ extern void *LJ_WIN_LOADLIBA(const char *path);
631#define LJ_UNWIND_EXT 0 637#define LJ_UNWIND_EXT 0
632#endif 638#endif
633 639
640#if LJ_UNWIND_EXT && LJ_HASJIT && !LJ_TARGET_ARM && !(LJ_ABI_WIN && LJ_TARGET_X86)
641#define LJ_UNWIND_JIT 1
642#else
643#define LJ_UNWIND_JIT 0
644#endif
645
634/* Compatibility with Lua 5.1 vs. 5.2. */ 646/* Compatibility with Lua 5.1 vs. 5.2. */
635#ifdef LUAJIT_ENABLE_LUA52COMPAT 647#ifdef LUAJIT_ENABLE_LUA52COMPAT
636#define LJ_52 1 648#define LJ_52 1
diff --git a/src/lj_asm.c b/src/lj_asm.c
index 7c9a4237..8a516eba 100644
--- a/src/lj_asm.c
+++ b/src/lj_asm.c
@@ -71,6 +71,7 @@ typedef struct ASMState {
71 IRRef snaprename; /* Rename highwater mark for snapshot check. */ 71 IRRef snaprename; /* Rename highwater mark for snapshot check. */
72 SnapNo snapno; /* Current snapshot number. */ 72 SnapNo snapno; /* Current snapshot number. */
73 SnapNo loopsnapno; /* Loop snapshot number. */ 73 SnapNo loopsnapno; /* Loop snapshot number. */
74 int snapalloc; /* Current snapshot needs allocation. */
74 75
75 IRRef fuseref; /* Fusion limit (loopref, 0 or FUSE_DISABLED). */ 76 IRRef fuseref; /* Fusion limit (loopref, 0 or FUSE_DISABLED). */
76 IRRef sectref; /* Section base reference (loopref or 0). */ 77 IRRef sectref; /* Section base reference (loopref or 0). */
@@ -84,6 +85,7 @@ typedef struct ASMState {
84 85
85 MCode *mcbot; /* Bottom of reserved MCode. */ 86 MCode *mcbot; /* Bottom of reserved MCode. */
86 MCode *mctop; /* Top of generated MCode. */ 87 MCode *mctop; /* Top of generated MCode. */
88 MCode *mctoporig; /* Original top of generated MCode. */
87 MCode *mcloop; /* Pointer to loop MCode (or NULL). */ 89 MCode *mcloop; /* Pointer to loop MCode (or NULL). */
88 MCode *invmcp; /* Points to invertible loop branch (or NULL). */ 90 MCode *invmcp; /* Points to invertible loop branch (or NULL). */
89 MCode *flagmcp; /* Pending opportunity to merge flag setting ins. */ 91 MCode *flagmcp; /* Pending opportunity to merge flag setting ins. */
@@ -947,9 +949,9 @@ static void asm_snap_alloc1(ASMState *as, IRRef ref)
947} 949}
948 950
949/* Allocate refs escaping to a snapshot. */ 951/* Allocate refs escaping to a snapshot. */
950static void asm_snap_alloc(ASMState *as) 952static void asm_snap_alloc(ASMState *as, int snapno)
951{ 953{
952 SnapShot *snap = &as->T->snap[as->snapno]; 954 SnapShot *snap = &as->T->snap[snapno];
953 SnapEntry *map = &as->T->snapmap[snap->mapofs]; 955 SnapEntry *map = &as->T->snapmap[snap->mapofs];
954 MSize n, nent = snap->nent; 956 MSize n, nent = snap->nent;
955 for (n = 0; n < nent; n++) { 957 for (n = 0; n < nent; n++) {
@@ -960,7 +962,7 @@ static void asm_snap_alloc(ASMState *as)
960 if (LJ_SOFTFP && (sn & SNAP_SOFTFPNUM)) { 962 if (LJ_SOFTFP && (sn & SNAP_SOFTFPNUM)) {
961 lj_assertA(irt_type(IR(ref+1)->t) == IRT_SOFTFP, 963 lj_assertA(irt_type(IR(ref+1)->t) == IRT_SOFTFP,
962 "snap %d[%d] points to bad SOFTFP IR %04d", 964 "snap %d[%d] points to bad SOFTFP IR %04d",
963 as->snapno, n, ref - REF_BIAS); 965 snapno, n, ref - REF_BIAS);
964 asm_snap_alloc1(as, ref+1); 966 asm_snap_alloc1(as, ref+1);
965 } 967 }
966 } 968 }
@@ -992,19 +994,16 @@ static int asm_snap_checkrename(ASMState *as, IRRef ren)
992 return 0; /* Not found. */ 994 return 0; /* Not found. */
993} 995}
994 996
995/* Prepare snapshot for next guard instruction. */ 997/* Prepare snapshot for next guard or throwing instruction. */
996static void asm_snap_prep(ASMState *as) 998static void asm_snap_prep(ASMState *as)
997{ 999{
998 if (as->curins < as->snapref) { 1000 if (as->snapalloc) {
999 do { 1001 /* Alloc on first invocation for each snapshot. */
1000 if (as->snapno == 0) return; /* Called by sunk stores before snap #0. */ 1002 as->snapalloc = 0;
1001 as->snapno--; 1003 asm_snap_alloc(as, as->snapno);
1002 as->snapref = as->T->snap[as->snapno].ref;
1003 } while (as->curins < as->snapref);
1004 asm_snap_alloc(as);
1005 as->snaprename = as->T->nins; 1004 as->snaprename = as->T->nins;
1006 } else { 1005 } else {
1007 /* Process any renames above the highwater mark. */ 1006 /* Check any renames above the highwater mark. */
1008 for (; as->snaprename < as->T->nins; as->snaprename++) { 1007 for (; as->snaprename < as->T->nins; as->snaprename++) {
1009 IRIns *ir = &as->T->ir[as->snaprename]; 1008 IRIns *ir = &as->T->ir[as->snaprename];
1010 if (asm_snap_checkrename(as, ir->op1)) 1009 if (asm_snap_checkrename(as, ir->op1))
@@ -1013,6 +1012,35 @@ static void asm_snap_prep(ASMState *as)
1013 } 1012 }
1014} 1013}
1015 1014
1015/* Move to previous snapshot when we cross the current snapshot ref. */
1016static void asm_snap_prev(ASMState *as)
1017{
1018 if (as->curins < as->snapref) {
1019 ptrdiff_t ofs = as->mctoporig - as->mcp;
1020 if (ofs >= 0x10000) lj_trace_err(as->J, LJ_TRERR_MCODEOV);
1021 do {
1022 if (as->snapno == 0) return;
1023 as->snapno--;
1024 as->snapref = as->T->snap[as->snapno].ref;
1025 as->T->snap[as->snapno].mcofs = ofs; /* Remember mcode offset. */
1026 } while (as->curins < as->snapref); /* May have no ins inbetween. */
1027 as->snapalloc = 1;
1028 }
1029}
1030
1031/* Fixup snapshot mcode offsetst. */
1032static void asm_snap_fixup_mcofs(ASMState *as)
1033{
1034 uint32_t sz = (uint32_t)(as->mctoporig - as->mcp);
1035 SnapShot *snap = as->T->snap;
1036 SnapNo i;
1037 for (i = as->T->nsnap-1; i > 0; i--) {
1038 /* Compute offset from mcode start and store in correct snapshot. */
1039 snap[i].mcofs = (uint16_t)(sz - snap[i-1].mcofs);
1040 }
1041 snap[0].mcofs = 0;
1042}
1043
1016/* -- Miscellaneous helpers ----------------------------------------------- */ 1044/* -- Miscellaneous helpers ----------------------------------------------- */
1017 1045
1018/* Calculate stack adjustment. */ 1046/* Calculate stack adjustment. */
@@ -1057,6 +1085,7 @@ static void asm_snew(ASMState *as, IRIns *ir)
1057{ 1085{
1058 const CCallInfo *ci = &lj_ir_callinfo[IRCALL_lj_str_new]; 1086 const CCallInfo *ci = &lj_ir_callinfo[IRCALL_lj_str_new];
1059 IRRef args[3]; 1087 IRRef args[3];
1088 asm_snap_prep(as);
1060 args[0] = ASMREF_L; /* lua_State *L */ 1089 args[0] = ASMREF_L; /* lua_State *L */
1061 args[1] = ir->op1; /* const char *str */ 1090 args[1] = ir->op1; /* const char *str */
1062 args[2] = ir->op2; /* size_t len */ 1091 args[2] = ir->op2; /* size_t len */
@@ -1069,6 +1098,7 @@ static void asm_tnew(ASMState *as, IRIns *ir)
1069{ 1098{
1070 const CCallInfo *ci = &lj_ir_callinfo[IRCALL_lj_tab_new1]; 1099 const CCallInfo *ci = &lj_ir_callinfo[IRCALL_lj_tab_new1];
1071 IRRef args[2]; 1100 IRRef args[2];
1101 asm_snap_prep(as);
1072 args[0] = ASMREF_L; /* lua_State *L */ 1102 args[0] = ASMREF_L; /* lua_State *L */
1073 args[1] = ASMREF_TMP1; /* uint32_t ahsize */ 1103 args[1] = ASMREF_TMP1; /* uint32_t ahsize */
1074 as->gcsteps++; 1104 as->gcsteps++;
@@ -1081,6 +1111,7 @@ static void asm_tdup(ASMState *as, IRIns *ir)
1081{ 1111{
1082 const CCallInfo *ci = &lj_ir_callinfo[IRCALL_lj_tab_dup]; 1112 const CCallInfo *ci = &lj_ir_callinfo[IRCALL_lj_tab_dup];
1083 IRRef args[2]; 1113 IRRef args[2];
1114 asm_snap_prep(as);
1084 args[0] = ASMREF_L; /* lua_State *L */ 1115 args[0] = ASMREF_L; /* lua_State *L */
1085 args[1] = ir->op1; /* const GCtab *kt */ 1116 args[1] = ir->op1; /* const GCtab *kt */
1086 as->gcsteps++; 1117 as->gcsteps++;
@@ -1201,6 +1232,7 @@ static void asm_tostr(ASMState *as, IRIns *ir)
1201{ 1232{
1202 const CCallInfo *ci; 1233 const CCallInfo *ci;
1203 IRRef args[2]; 1234 IRRef args[2];
1235 asm_snap_prep(as);
1204 args[0] = ASMREF_L; 1236 args[0] = ASMREF_L;
1205 as->gcsteps++; 1237 as->gcsteps++;
1206 if (ir->op2 == IRTOSTR_NUM) { 1238 if (ir->op2 == IRTOSTR_NUM) {
@@ -1257,6 +1289,7 @@ static void asm_newref(ASMState *as, IRIns *ir)
1257 IRRef args[3]; 1289 IRRef args[3];
1258 if (ir->r == RID_SINK) 1290 if (ir->r == RID_SINK)
1259 return; 1291 return;
1292 asm_snap_prep(as);
1260 args[0] = ASMREF_L; /* lua_State *L */ 1293 args[0] = ASMREF_L; /* lua_State *L */
1261 args[1] = ir->op1; /* GCtab *t */ 1294 args[1] = ir->op1; /* GCtab *t */
1262 args[2] = ASMREF_TMP1; /* cTValue *key */ 1295 args[2] = ASMREF_TMP1; /* cTValue *key */
@@ -1838,8 +1871,7 @@ static void asm_head_side(ASMState *as)
1838 1871
1839 if (as->snapno && as->topslot > as->parent->topslot) { 1872 if (as->snapno && as->topslot > as->parent->topslot) {
1840 /* Force snap #0 alloc to prevent register overwrite in stack check. */ 1873 /* Force snap #0 alloc to prevent register overwrite in stack check. */
1841 as->snapno = 0; 1874 asm_snap_alloc(as, 0);
1842 asm_snap_alloc(as);
1843 } 1875 }
1844 allow = asm_head_side_base(as, irp, allow); 1876 allow = asm_head_side_base(as, irp, allow);
1845 1877
@@ -2100,6 +2132,7 @@ static void asm_setup_regsp(ASMState *as)
2100 as->snaprename = nins; 2132 as->snaprename = nins;
2101 as->snapref = nins; 2133 as->snapref = nins;
2102 as->snapno = T->nsnap; 2134 as->snapno = T->nsnap;
2135 as->snapalloc = 0;
2103 2136
2104 as->stopins = REF_BASE; 2137 as->stopins = REF_BASE;
2105 as->orignins = nins; 2138 as->orignins = nins;
@@ -2327,7 +2360,6 @@ void lj_asm_trace(jit_State *J, GCtrace *T)
2327{ 2360{
2328 ASMState as_; 2361 ASMState as_;
2329 ASMState *as = &as_; 2362 ASMState *as = &as_;
2330 MCode *origtop;
2331 2363
2332 /* Remove nops/renames left over from ASM restart due to LJ_TRERR_MCODELM. */ 2364 /* Remove nops/renames left over from ASM restart due to LJ_TRERR_MCODELM. */
2333 { 2365 {
@@ -2355,7 +2387,7 @@ void lj_asm_trace(jit_State *J, GCtrace *T)
2355 as->parent = J->parent ? traceref(J, J->parent) : NULL; 2387 as->parent = J->parent ? traceref(J, J->parent) : NULL;
2356 2388
2357 /* Reserve MCode memory. */ 2389 /* Reserve MCode memory. */
2358 as->mctop = origtop = lj_mcode_reserve(J, &as->mcbot); 2390 as->mctop = as->mctoporig = lj_mcode_reserve(J, &as->mcbot);
2359 as->mcp = as->mctop; 2391 as->mcp = as->mctop;
2360 as->mclim = as->mcbot + MCLIM_REDZONE; 2392 as->mclim = as->mcbot + MCLIM_REDZONE;
2361 asm_setup_target(as); 2393 asm_setup_target(as);
@@ -2417,6 +2449,7 @@ void lj_asm_trace(jit_State *J, GCtrace *T)
2417 lj_assertA(!(LJ_32 && irt_isint64(ir->t)), 2449 lj_assertA(!(LJ_32 && irt_isint64(ir->t)),
2418 "IR %04d has unsplit 64 bit type", 2450 "IR %04d has unsplit 64 bit type",
2419 (int)(ir - as->ir) - REF_BIAS); 2451 (int)(ir - as->ir) - REF_BIAS);
2452 asm_snap_prev(as);
2420 if (!ra_used(ir) && !ir_sideeff(ir) && (as->flags & JIT_F_OPT_DCE)) 2453 if (!ra_used(ir) && !ir_sideeff(ir) && (as->flags & JIT_F_OPT_DCE))
2421 continue; /* Dead-code elimination can be soooo easy. */ 2454 continue; /* Dead-code elimination can be soooo easy. */
2422 if (irt_isguard(ir->t)) 2455 if (irt_isguard(ir->t))
@@ -2450,6 +2483,9 @@ void lj_asm_trace(jit_State *J, GCtrace *T)
2450 memcpy(J->curfinal->ir + as->orignins, T->ir + as->orignins, 2483 memcpy(J->curfinal->ir + as->orignins, T->ir + as->orignins,
2451 (T->nins - as->orignins) * sizeof(IRIns)); /* Copy RENAMEs. */ 2484 (T->nins - as->orignins) * sizeof(IRIns)); /* Copy RENAMEs. */
2452 T->nins = J->curfinal->nins; 2485 T->nins = J->curfinal->nins;
2486 /* Fill mcofs of any unprocessed snapshots. */
2487 as->curins = REF_FIRST;
2488 asm_snap_prev(as);
2453 break; /* Done. */ 2489 break; /* Done. */
2454 } 2490 }
2455 2491
@@ -2471,10 +2507,11 @@ void lj_asm_trace(jit_State *J, GCtrace *T)
2471 if (!as->loopref) 2507 if (!as->loopref)
2472 asm_tail_fixup(as, T->link); /* Note: this may change as->mctop! */ 2508 asm_tail_fixup(as, T->link); /* Note: this may change as->mctop! */
2473 T->szmcode = (MSize)((char *)as->mctop - (char *)as->mcp); 2509 T->szmcode = (MSize)((char *)as->mctop - (char *)as->mcp);
2510 asm_snap_fixup_mcofs(as);
2474#if LJ_TARGET_MCODE_FIXUP 2511#if LJ_TARGET_MCODE_FIXUP
2475 asm_mcode_fixup(T->mcode, T->szmcode); 2512 asm_mcode_fixup(T->mcode, T->szmcode);
2476#endif 2513#endif
2477 lj_mcode_sync(T->mcode, origtop); 2514 lj_mcode_sync(T->mcode, as->mctoporig);
2478} 2515}
2479 2516
2480#undef IR 2517#undef IR
diff --git a/src/lj_dispatch.h b/src/lj_dispatch.h
index be7c410a..2331bd42 100644
--- a/src/lj_dispatch.h
+++ b/src/lj_dispatch.h
@@ -31,7 +31,7 @@ extern double __divdf3(double a, double b);
31#define SFGOTDEF(_) 31#define SFGOTDEF(_)
32#endif 32#endif
33#if LJ_HASJIT 33#if LJ_HASJIT
34#define JITGOTDEF(_) _(lj_trace_exit) _(lj_trace_hot) 34#define JITGOTDEF(_) _(lj_err_trace) _(lj_trace_exit) _(lj_trace_hot)
35#else 35#else
36#define JITGOTDEF(_) 36#define JITGOTDEF(_)
37#endif 37#endif
@@ -46,7 +46,7 @@ extern double __divdf3(double a, double b);
46 _(asin) _(acos) _(atan) _(sinh) _(cosh) _(tanh) _(frexp) _(modf) _(atan2) \ 46 _(asin) _(acos) _(atan) _(sinh) _(cosh) _(tanh) _(frexp) _(modf) _(atan2) \
47 _(pow) _(fmod) _(ldexp) _(lj_vm_modi) \ 47 _(pow) _(fmod) _(ldexp) _(lj_vm_modi) \
48 _(lj_dispatch_call) _(lj_dispatch_ins) _(lj_dispatch_stitch) \ 48 _(lj_dispatch_call) _(lj_dispatch_ins) _(lj_dispatch_stitch) \
49 _(lj_dispatch_profile) _(lj_err_throw) _(lj_err_run) \ 49 _(lj_dispatch_profile) _(lj_err_throw) \
50 _(lj_ffh_coroutine_wrap_err) _(lj_func_closeuv) _(lj_func_newL_gc) \ 50 _(lj_ffh_coroutine_wrap_err) _(lj_func_closeuv) _(lj_func_newL_gc) \
51 _(lj_gc_barrieruv) _(lj_gc_step) _(lj_gc_step_fixtop) _(lj_meta_arith) \ 51 _(lj_gc_barrieruv) _(lj_gc_step) _(lj_gc_step_fixtop) _(lj_meta_arith) \
52 _(lj_meta_call) _(lj_meta_cat) _(lj_meta_comp) _(lj_meta_equal) \ 52 _(lj_meta_call) _(lj_meta_cat) _(lj_meta_comp) _(lj_meta_equal) \
diff --git a/src/lj_err.c b/src/lj_err.c
index ba0fac0a..9fc3adc7 100644
--- a/src/lj_err.c
+++ b/src/lj_err.c
@@ -52,6 +52,11 @@
52** the wrapper function feature. Lua errors thrown through C++ frames 52** the wrapper function feature. Lua errors thrown through C++ frames
53** cannot be caught by C++ code and C++ destructors are not run. 53** cannot be caught by C++ code and C++ destructors are not run.
54** 54**
55** - EXT can handle errors from internal helper functions that are called
56** from JIT-compiled code (except for Windows/x86 and 32 bit ARM).
57** INT has no choice but to call the panic handler, if this happens.
58** Note: this is mainly relevant for out-of-memory errors.
59**
55** EXT is the default on all systems where the toolchain produces unwind 60** EXT is the default on all systems where the toolchain produces unwind
56** tables by default (*). This is hard-coded and/or detected in src/Makefile. 61** tables by default (*). This is hard-coded and/or detected in src/Makefile.
57** You can thwart the detection with: TARGET_XCFLAGS=-DLUAJIT_UNWIND_INTERNAL 62** You can thwart the detection with: TARGET_XCFLAGS=-DLUAJIT_UNWIND_INTERNAL
@@ -305,12 +310,59 @@ LJ_FUNCA int lj_err_unwind_win(EXCEPTION_RECORD *rec,
305 return 1; /* ExceptionContinueSearch */ 310 return 1; /* ExceptionContinueSearch */
306} 311}
307 312
313#if LJ_UNWIND_JIT
314
315#if LJ_TARGET_X64
316#define CONTEXT_REG_PC Rip
317#elif LJ_TARGET_ARM64
318#define CONTEXT_REG_PC Pc
319#else
320#error "NYI: Windows arch-specific unwinder for JIT-compiled code"
321#endif
322
323/* Windows unwinder for JIT-compiled code. */
324static void err_unwind_win_jit(global_State *g, int errcode)
325{
326 CONTEXT ctx;
327 UNWIND_HISTORY_TABLE hist;
328
329 memset(&hist, 0, sizeof(hist));
330 RtlCaptureContext(&ctx);
331 while (1) {
332 uintptr_t frame, base, addr = ctx.CONTEXT_REG_PC;
333 void *hdata;
334 PRUNTIME_FUNCTION func = RtlLookupFunctionEntry(addr, &base, &hist);
335 if (!func) { /* Found frame without .pdata: must be JIT-compiled code. */
336 ExitNo exitno;
337 uintptr_t stub = lj_trace_unwind(G2J(g), addr - sizeof(MCode), &exitno);
338 if (stub) { /* Jump to side exit to unwind the trace. */
339 ctx.CONTEXT_REG_PC = stub;
340 G2J(g)->exitcode = errcode;
341 RtlRestoreContext(&ctx, NULL); /* Does not return. */
342 }
343 break;
344 }
345 RtlVirtualUnwind(UNW_FLAG_NHANDLER, base, addr, func,
346 &ctx, &hdata, &frame, NULL);
347 if (!addr) break;
348 }
349 /* Unwinding failed, if we end up here. */
350}
351#endif
352
308/* Raise Windows exception. */ 353/* Raise Windows exception. */
309static void err_raise_ext(global_State *g, int errcode) 354static void err_raise_ext(global_State *g, int errcode)
310{ 355{
311#if LJ_HASJIT 356#if LJ_UNWIND_JIT
357 if (tvref(g->jit_base)) {
358 err_unwind_win_jit(g, errcode);
359 return; /* Unwinding failed. */
360 }
361#elif LJ_HASJIT
362 /* Cannot catch on-trace errors for Windows/x86 SEH. Unwind to interpreter. */
312 setmref(g->jit_base, NULL); 363 setmref(g->jit_base, NULL);
313#endif 364#endif
365 UNUSED(g);
314 RaiseException(LJ_EXCODE_MAKE(errcode), 1 /* EH_NONCONTINUABLE */, 0, NULL); 366 RaiseException(LJ_EXCODE_MAKE(errcode), 1 /* EH_NONCONTINUABLE */, 0, NULL);
315} 367}
316 368
@@ -324,6 +376,7 @@ static void err_raise_ext(global_State *g, int errcode)
324typedef struct _Unwind_Context _Unwind_Context; 376typedef struct _Unwind_Context _Unwind_Context;
325 377
326#define _URC_OK 0 378#define _URC_OK 0
379#define _URC_FATAL_PHASE2_ERROR 2
327#define _URC_FATAL_PHASE1_ERROR 3 380#define _URC_FATAL_PHASE1_ERROR 3
328#define _URC_HANDLER_FOUND 6 381#define _URC_HANDLER_FOUND 6
329#define _URC_INSTALL_CONTEXT 7 382#define _URC_INSTALL_CONTEXT 7
@@ -343,9 +396,11 @@ typedef struct _Unwind_Exception
343 void (*excleanup)(int, struct _Unwind_Exception *); 396 void (*excleanup)(int, struct _Unwind_Exception *);
344 uintptr_t p1, p2; 397 uintptr_t p1, p2;
345} __attribute__((__aligned__)) _Unwind_Exception; 398} __attribute__((__aligned__)) _Unwind_Exception;
399#define UNWIND_EXCEPTION_TYPE _Unwind_Exception
346 400
347extern uintptr_t _Unwind_GetCFA(_Unwind_Context *); 401extern uintptr_t _Unwind_GetCFA(_Unwind_Context *);
348extern void _Unwind_SetGR(_Unwind_Context *, int, uintptr_t); 402extern void _Unwind_SetGR(_Unwind_Context *, int, uintptr_t);
403extern uintptr_t _Unwind_GetIP(_Unwind_Context *);
349extern void _Unwind_SetIP(_Unwind_Context *, uintptr_t); 404extern void _Unwind_SetIP(_Unwind_Context *, uintptr_t);
350extern void _Unwind_DeleteException(_Unwind_Exception *); 405extern void _Unwind_DeleteException(_Unwind_Exception *);
351extern int _Unwind_RaiseException(_Unwind_Exception *); 406extern int _Unwind_RaiseException(_Unwind_Exception *);
@@ -418,8 +473,130 @@ LJ_FUNCA int lj_err_unwind_dwarf(int version, int actions,
418 return _URC_CONTINUE_UNWIND; 473 return _URC_CONTINUE_UNWIND;
419} 474}
420 475
421#if LJ_UNWIND_EXT 476#if LJ_UNWIND_EXT && defined(LUA_USE_ASSERT)
422static __thread _Unwind_Exception static_uex; 477struct dwarf_eh_bases { void *tbase, *dbase, *func; };
478extern const void *_Unwind_Find_FDE(void *pc, struct dwarf_eh_bases *bases);
479
480/* Verify that external error handling actually has a chance to work. */
481void lj_err_verify(void)
482{
483 struct dwarf_eh_bases ehb;
484 lj_assertX(_Unwind_Find_FDE((void *)lj_err_throw, &ehb), "broken build: external frame unwinding enabled, but missing -funwind-tables");
485 lj_assertX(_Unwind_Find_FDE((void *)_Unwind_RaiseException, &ehb), "broken build: external frame unwinding enabled, but system libraries have no unwind tables");
486}
487#endif
488
489#if LJ_UNWIND_JIT
490/* DWARF2 personality handler for JIT-compiled code. */
491static int err_unwind_jit(int version, int actions,
492 uint64_t uexclass, _Unwind_Exception *uex, _Unwind_Context *ctx)
493{
494 /* NYI: FFI C++ exception interoperability. */
495 if (version != 1 || !LJ_UEXCLASS_CHECK(uexclass))
496 return _URC_FATAL_PHASE1_ERROR;
497 if ((actions & _UA_SEARCH_PHASE)) {
498 return _URC_HANDLER_FOUND;
499 }
500 if ((actions & _UA_CLEANUP_PHASE)) {
501 global_State *g = *(global_State **)(uex+1);
502 ExitNo exitno;
503 uintptr_t addr = _Unwind_GetIP(ctx); /* Return address _after_ call. */
504 uintptr_t stub = lj_trace_unwind(G2J(g), addr - sizeof(MCode), &exitno);
505 lj_assertG(tvref(g->jit_base), "unexpected throw across mcode frame");
506 if (stub) { /* Jump to side exit to unwind the trace. */
507 G2J(g)->exitcode = LJ_UEXCLASS_ERRCODE(uexclass);
508#ifdef LJ_TARGET_MIPS
509 _Unwind_SetGR(ctx, 4, stub);
510 _Unwind_SetGR(ctx, 5, exitno);
511 _Unwind_SetIP(ctx, (uintptr_t)(void *)lj_vm_unwind_stub);
512#else
513 _Unwind_SetIP(ctx, stub);
514#endif
515 return _URC_INSTALL_CONTEXT;
516 }
517 return _URC_FATAL_PHASE2_ERROR;
518 }
519 return _URC_FATAL_PHASE1_ERROR;
520}
521
522/* DWARF2 template frame info for JIT-compiled code.
523**
524** After copying the template to the start of the mcode segment,
525** the frame handler function and the code size is patched.
526** The frame handler always installs a new context to jump to the exit,
527** so don't bother to add any unwind opcodes.
528*/
529static const uint8_t err_frame_jit_template[] = {
530#if LJ_BE
531 0,0,0,
532#endif
533 LJ_64 ? 0x1c : 0x14, /* CIE length. */
534#if LJ_LE
535 0,0,0,
536#endif
537 0,0,0,0, 1, 'z','P','R',0, /* CIE mark, CIE version, augmentation. */
538 1, LJ_64 ? 0x78 : 0x7c, LJ_TARGET_EHRAREG, /* Code/data align, RA. */
539#if LJ_64
540 10, 0, 0,0,0,0,0,0,0,0, 0x1b, /* Aug. data ABS handler, PCREL|SDATA4 code. */
541 0,0,0,0,0, /* Alignment. */
542#else
543 6, 0, 0,0,0,0, 0x1b, /* Aug. data ABS handler, PCREL|SDATA4 code. */
544 0, /* Alignment. */
545#endif
546#if LJ_BE
547 0,0,0,
548#endif
549 LJ_64 ? 0x14 : 0x10, /* FDE length. */
550 0,0,0,
551 LJ_64 ? 0x24 : 0x1c, /* CIE offset. */
552 0,0,0,
553 LJ_64 ? 0x14 : 0x10, /* Code offset. After Final FDE. */
554#if LJ_LE
555 0,0,0,
556#endif
557 0,0,0,0, 0, 0,0,0, /* Code size, augmentation length, alignment. */
558#if LJ_64
559 0,0,0,0, /* Alignment. */
560#endif
561 0,0,0,0 /* Final FDE. */
562};
563
564#define ERR_FRAME_JIT_OFS_HANDLER 0x12
565#define ERR_FRAME_JIT_OFS_FDE (LJ_64 ? 0x20 : 0x18)
566#define ERR_FRAME_JIT_OFS_CODE_SIZE (LJ_64 ? 0x2c : 0x24)
567#if LJ_TARGET_OSX
568#define ERR_FRAME_JIT_OFS_REGISTER ERR_FRAME_JIT_OFS_FDE
569#else
570#define ERR_FRAME_JIT_OFS_REGISTER 0
571#endif
572
573extern void __register_frame(const void *);
574extern void __deregister_frame(const void *);
575
576uint8_t *lj_err_register_mcode(void *base, size_t sz, uint8_t *info)
577{
578 void **handler;
579 memcpy(info, err_frame_jit_template, sizeof(err_frame_jit_template));
580 handler = (void *)err_unwind_jit;
581 memcpy(info + ERR_FRAME_JIT_OFS_HANDLER, &handler, sizeof(handler));
582 *(uint32_t *)(info + ERR_FRAME_JIT_OFS_CODE_SIZE) =
583 (uint32_t)(sz - sizeof(err_frame_jit_template) - (info - (uint8_t *)base));
584 __register_frame(info + ERR_FRAME_JIT_OFS_REGISTER);
585#ifdef LUA_USE_ASSERT
586 {
587 struct dwarf_eh_bases ehb;
588 lj_assertX(_Unwind_Find_FDE(info + sizeof(err_frame_jit_template)+1, &ehb),
589 "bad JIT unwind table registration");
590 }
591#endif
592 return info + sizeof(err_frame_jit_template);
593}
594
595void lj_err_deregister_mcode(void *base, size_t sz, uint8_t *info)
596{
597 UNUSED(base); UNUSED(sz);
598 __deregister_frame(info + ERR_FRAME_JIT_OFS_REGISTER);
599}
423#endif 600#endif
424 601
425#else /* LJ_TARGET_ARM */ 602#else /* LJ_TARGET_ARM */
@@ -430,6 +607,7 @@ static __thread _Unwind_Exception static_uex;
430#define _US_FORCE_UNWIND 8 607#define _US_FORCE_UNWIND 8
431 608
432typedef struct _Unwind_Control_Block _Unwind_Control_Block; 609typedef struct _Unwind_Control_Block _Unwind_Control_Block;
610#define UNWIND_EXCEPTION_TYPE _Unwind_Control_Block
433 611
434struct _Unwind_Control_Block { 612struct _Unwind_Control_Block {
435 uint64_t exclass; 613 uint64_t exclass;
@@ -488,25 +666,62 @@ LJ_FUNCA int lj_err_unwind_arm(int state, _Unwind_Control_Block *ucb,
488 } 666 }
489 if (__gnu_unwind_frame(ucb, ctx) != _URC_OK) 667 if (__gnu_unwind_frame(ucb, ctx) != _URC_OK)
490 return _URC_FAILURE; 668 return _URC_FAILURE;
669#ifdef LUA_USE_ASSERT
670 /* We should never get here unless this is a forced unwind aka backtrace. */
671 if (_Unwind_GetGR(ctx, 0) == 0xff33aa77) {
672 _Unwind_SetGR(ctx, 0, 0xff33aa88);
673 }
674#endif
491 return _URC_CONTINUE_UNWIND; 675 return _URC_CONTINUE_UNWIND;
492} 676}
493 677
494#if LJ_UNWIND_EXT 678#if LJ_UNWIND_EXT && defined(LUA_USE_ASSERT)
495static __thread _Unwind_Control_Block static_uex; 679typedef int (*_Unwind_Trace_Fn)(_Unwind_Context *, void *);
680extern int _Unwind_Backtrace(_Unwind_Trace_Fn, void *);
681
682static int err_verify_bt(_Unwind_Context *ctx, int *got)
683{
684 if (_Unwind_GetGR(ctx, 0) == 0xff33aa88) { *got = 2; }
685 else if (*got == 0) { *got = 1; _Unwind_SetGR(ctx, 0, 0xff33aa77); }
686 return _URC_OK;
687}
688
689/* Verify that external error handling actually has a chance to work. */
690void lj_err_verify(void)
691{
692 int got = 0;
693 _Unwind_Backtrace((_Unwind_Trace_Fn)err_verify_bt, &got);
694 lj_assertX(got == 2, "broken build: external frame unwinding enabled, but missing -funwind-tables");
695}
496#endif 696#endif
697
698/*
699** Note: LJ_UNWIND_JIT is not implemented for 32 bit ARM.
700**
701** The quirky ARM unwind API doesn't have __register_frame().
702** A potential workaround might involve _Unwind_Backtrace.
703** But most 32 bit ARM targets don't qualify for LJ_UNWIND_EXT, anyway,
704** since they are built without unwind tables by default.
705*/
706
497#endif /* LJ_TARGET_ARM */ 707#endif /* LJ_TARGET_ARM */
498 708
709
499#if LJ_UNWIND_EXT 710#if LJ_UNWIND_EXT
711static __thread struct {
712 UNWIND_EXCEPTION_TYPE ex;
713 global_State *g;
714} static_uex;
715
500/* Raise external exception. */ 716/* Raise external exception. */
501static void err_raise_ext(global_State *g, int errcode) 717static void err_raise_ext(global_State *g, int errcode)
502{ 718{
503#if LJ_HASJIT
504 setmref(g->jit_base, NULL);
505#endif
506 memset(&static_uex, 0, sizeof(static_uex)); 719 memset(&static_uex, 0, sizeof(static_uex));
507 static_uex.exclass = LJ_UEXCLASS_MAKE(errcode); 720 static_uex.ex.exclass = LJ_UEXCLASS_MAKE(errcode);
508 _Unwind_RaiseException(&static_uex); 721 static_uex.g = g;
722 _Unwind_RaiseException(&static_uex.ex);
509} 723}
724
510#endif 725#endif
511 726
512#endif 727#endif
@@ -615,7 +830,7 @@ static ptrdiff_t finderrfunc(lua_State *L)
615/* Runtime error. */ 830/* Runtime error. */
616LJ_NOINLINE void LJ_FASTCALL lj_err_run(lua_State *L) 831LJ_NOINLINE void LJ_FASTCALL lj_err_run(lua_State *L)
617{ 832{
618 ptrdiff_t ef = finderrfunc(L); 833 ptrdiff_t ef = (LJ_HASJIT && tvref(G(L)->jit_base)) ? 0 : finderrfunc(L);
619 if (ef) { 834 if (ef) {
620 TValue *errfunc = restorestack(L, ef); 835 TValue *errfunc = restorestack(L, ef);
621 TValue *top = L->top; 836 TValue *top = L->top;
@@ -634,6 +849,16 @@ LJ_NOINLINE void LJ_FASTCALL lj_err_run(lua_State *L)
634 lj_err_throw(L, LUA_ERRRUN); 849 lj_err_throw(L, LUA_ERRRUN);
635} 850}
636 851
852#if LJ_HASJIT
853LJ_NOINLINE void LJ_FASTCALL lj_err_trace(lua_State *L, int errcode)
854{
855 if (errcode == LUA_ERRRUN)
856 lj_err_run(L);
857 else
858 lj_err_throw(L, errcode);
859}
860#endif
861
637/* Formatted runtime error message. */ 862/* Formatted runtime error message. */
638LJ_NORET LJ_NOINLINE static void err_msgv(lua_State *L, ErrMsg em, ...) 863LJ_NORET LJ_NOINLINE static void err_msgv(lua_State *L, ErrMsg em, ...)
639{ 864{
diff --git a/src/lj_err.h b/src/lj_err.h
index 770553fc..2e8a251f 100644
--- a/src/lj_err.h
+++ b/src/lj_err.h
@@ -23,7 +23,10 @@ LJ_DATA const char *lj_err_allmsg;
23LJ_FUNC GCstr *lj_err_str(lua_State *L, ErrMsg em); 23LJ_FUNC GCstr *lj_err_str(lua_State *L, ErrMsg em);
24LJ_FUNCA_NORET void LJ_FASTCALL lj_err_throw(lua_State *L, int errcode); 24LJ_FUNCA_NORET void LJ_FASTCALL lj_err_throw(lua_State *L, int errcode);
25LJ_FUNC_NORET void lj_err_mem(lua_State *L); 25LJ_FUNC_NORET void lj_err_mem(lua_State *L);
26LJ_FUNCA_NORET void LJ_FASTCALL lj_err_run(lua_State *L); 26LJ_FUNC_NORET void LJ_FASTCALL lj_err_run(lua_State *L);
27#if LJ_HASJIT
28LJ_FUNCA_NORET void LJ_FASTCALL lj_err_trace(lua_State *L, int errcode);
29#endif
27LJ_FUNC_NORET void lj_err_msg(lua_State *L, ErrMsg em); 30LJ_FUNC_NORET void lj_err_msg(lua_State *L, ErrMsg em);
28LJ_FUNC_NORET void lj_err_lex(lua_State *L, GCstr *src, const char *tok, 31LJ_FUNC_NORET void lj_err_lex(lua_State *L, GCstr *src, const char *tok,
29 BCLine line, ErrMsg em, va_list argp); 32 BCLine line, ErrMsg em, va_list argp);
@@ -38,4 +41,18 @@ LJ_FUNC_NORET void lj_err_argv(lua_State *L, int narg, ErrMsg em, ...);
38LJ_FUNC_NORET void lj_err_argtype(lua_State *L, int narg, const char *xname); 41LJ_FUNC_NORET void lj_err_argtype(lua_State *L, int narg, const char *xname);
39LJ_FUNC_NORET void lj_err_argt(lua_State *L, int narg, int tt); 42LJ_FUNC_NORET void lj_err_argt(lua_State *L, int narg, int tt);
40 43
44#if LJ_UNWIND_JIT && !LJ_ABI_WIN
45LJ_FUNC uint8_t *lj_err_register_mcode(void *base, size_t sz, uint8_t *info);
46LJ_FUNC void lj_err_deregister_mcode(void *base, size_t sz, uint8_t *info);
47#else
48#define lj_err_register_mcode(base, sz, info) (info)
49#define lj_err_deregister_mcode(base, sz, info) UNUSED(base)
50#endif
51
52#if LJ_UNWIND_EXT && !LJ_ABI_WIN && defined(LUA_USE_ASSERT)
53LJ_FUNC void lj_err_verify(void);
54#else
55#define lj_err_verify() ((void)0)
56#endif
57
41#endif 58#endif
diff --git a/src/lj_ffrecord.c b/src/lj_ffrecord.c
index 6c7eb2a0..844fc497 100644
--- a/src/lj_ffrecord.c
+++ b/src/lj_ffrecord.c
@@ -455,6 +455,7 @@ static void LJ_FASTCALL recff_pcall(jit_State *J, RecordFFData *rd)
455#endif 455#endif
456 lj_record_call(J, 0, J->maxslot - 1); 456 lj_record_call(J, 0, J->maxslot - 1);
457 rd->nres = -1; /* Pending call. */ 457 rd->nres = -1; /* Pending call. */
458 J->needsnap = 1; /* Start catching on-trace errors. */
458 } /* else: Interpreter will throw. */ 459 } /* else: Interpreter will throw. */
459} 460}
460 461
@@ -490,6 +491,7 @@ static void LJ_FASTCALL recff_xpcall(jit_State *J, RecordFFData *rd)
490 if (errcode) 491 if (errcode)
491 lj_err_throw(J->L, errcode); /* Propagate errors. */ 492 lj_err_throw(J->L, errcode); /* Propagate errors. */
492 rd->nres = -1; /* Pending call. */ 493 rd->nres = -1; /* Pending call. */
494 J->needsnap = 1; /* Start catching on-trace errors. */
493 } /* else: Interpreter will throw. */ 495 } /* else: Interpreter will throw. */
494} 496}
495 497
diff --git a/src/lj_jit.h b/src/lj_jit.h
index 03ed3ea0..34ddf907 100644
--- a/src/lj_jit.h
+++ b/src/lj_jit.h
@@ -184,6 +184,7 @@ typedef struct MCLink {
184typedef struct SnapShot { 184typedef struct SnapShot {
185 uint32_t mapofs; /* Offset into snapshot map. */ 185 uint32_t mapofs; /* Offset into snapshot map. */
186 IRRef1 ref; /* First IR ref for this snapshot. */ 186 IRRef1 ref; /* First IR ref for this snapshot. */
187 uint16_t mcofs; /* Offset into machine code in MCode units. */
187 uint8_t nslots; /* Number of valid slots. */ 188 uint8_t nslots; /* Number of valid slots. */
188 uint8_t topslot; /* Maximum frame extent. */ 189 uint8_t topslot; /* Maximum frame extent. */
189 uint8_t nent; /* Number of compressed entries. */ 190 uint8_t nent; /* Number of compressed entries. */
@@ -485,6 +486,7 @@ typedef struct jit_State {
485 const BCIns *startpc; /* Bytecode PC of starting instruction. */ 486 const BCIns *startpc; /* Bytecode PC of starting instruction. */
486 TraceNo parent; /* Parent of current side trace (0 for root traces). */ 487 TraceNo parent; /* Parent of current side trace (0 for root traces). */
487 ExitNo exitno; /* Exit number in parent of current side trace. */ 488 ExitNo exitno; /* Exit number in parent of current side trace. */
489 int exitcode; /* Exit code from unwound trace. */
488 490
489 BCIns *patchpc; /* PC for pending re-patch. */ 491 BCIns *patchpc; /* PC for pending re-patch. */
490 BCIns patchins; /* Instruction for pending re-patch. */ 492 BCIns patchins; /* Instruction for pending re-patch. */
diff --git a/src/lj_mcode.c b/src/lj_mcode.c
index fc67eaee..b34d7c85 100644
--- a/src/lj_mcode.c
+++ b/src/lj_mcode.c
@@ -269,6 +269,7 @@ static void mcode_allocarea(jit_State *J)
269 ((MCLink *)J->mcarea)->next = oldarea; 269 ((MCLink *)J->mcarea)->next = oldarea;
270 ((MCLink *)J->mcarea)->size = sz; 270 ((MCLink *)J->mcarea)->size = sz;
271 J->szallmcarea += sz; 271 J->szallmcarea += sz;
272 J->mcbot = (MCode *)lj_err_register_mcode(J->mcarea, sz, (uint8_t *)J->mcbot);
272} 273}
273 274
274/* Free all MCode areas. */ 275/* Free all MCode areas. */
@@ -279,7 +280,9 @@ void lj_mcode_free(jit_State *J)
279 J->szallmcarea = 0; 280 J->szallmcarea = 0;
280 while (mc) { 281 while (mc) {
281 MCode *next = ((MCLink *)mc)->next; 282 MCode *next = ((MCLink *)mc)->next;
282 mcode_free(J, mc, ((MCLink *)mc)->size); 283 size_t sz = ((MCLink *)mc)->size;
284 lj_err_deregister_mcode(mc, sz, (uint8_t *)mc + sizeof(MCLink));
285 mcode_free(J, mc, sz);
283 mc = next; 286 mc = next;
284 } 287 }
285} 288}
diff --git a/src/lj_opt_loop.c b/src/lj_opt_loop.c
index 4c709c63..df5811a9 100644
--- a/src/lj_opt_loop.c
+++ b/src/lj_opt_loop.c
@@ -225,6 +225,7 @@ static void loop_subst_snap(jit_State *J, SnapShot *osnap,
225 /* Setup new snapshot. */ 225 /* Setup new snapshot. */
226 snap->mapofs = (uint32_t)nmapofs; 226 snap->mapofs = (uint32_t)nmapofs;
227 snap->ref = (IRRef1)J->cur.nins; 227 snap->ref = (IRRef1)J->cur.nins;
228 snap->mcofs = 0;
228 snap->nslots = nslots; 229 snap->nslots = nslots;
229 snap->topslot = osnap->topslot; 230 snap->topslot = osnap->topslot;
230 snap->count = 0; 231 snap->count = 0;
diff --git a/src/lj_record.c b/src/lj_record.c
index a4779933..3ca2fb4f 100644
--- a/src/lj_record.c
+++ b/src/lj_record.c
@@ -832,6 +832,7 @@ void lj_record_ret(jit_State *J, BCReg rbase, ptrdiff_t gotresults)
832 J->base -= cbase; 832 J->base -= cbase;
833 J->base[--rbase] = TREF_TRUE; /* Prepend true to results. */ 833 J->base[--rbase] = TREF_TRUE; /* Prepend true to results. */
834 frame = frame_prevd(frame); 834 frame = frame_prevd(frame);
835 J->needsnap = 1; /* Stop catching on-trace errors. */
835 } 836 }
836 /* Return to lower frame via interpreter for unhandled cases. */ 837 /* Return to lower frame via interpreter for unhandled cases. */
837 if (J->framedepth == 0 && J->pt && bc_isret(bc_op(*J->pc)) && 838 if (J->framedepth == 0 && J->pt && bc_isret(bc_op(*J->pc)) &&
@@ -2050,7 +2051,7 @@ void lj_record_ins(jit_State *J)
2050 /* Need snapshot before recording next bytecode (e.g. after a store). */ 2051 /* Need snapshot before recording next bytecode (e.g. after a store). */
2051 if (J->needsnap) { 2052 if (J->needsnap) {
2052 J->needsnap = 0; 2053 J->needsnap = 0;
2053 lj_snap_purge(J); 2054 if (J->pt) lj_snap_purge(J);
2054 lj_snap_add(J); 2055 lj_snap_add(J);
2055 J->mergesnap = 1; 2056 J->mergesnap = 1;
2056 } 2057 }
diff --git a/src/lj_snap.c b/src/lj_snap.c
index 21f27e1f..fee68ba5 100644
--- a/src/lj_snap.c
+++ b/src/lj_snap.c
@@ -171,6 +171,7 @@ static void snapshot_stack(jit_State *J, SnapShot *snap, MSize nsnapmap)
171 nent += snapshot_framelinks(J, p + nent, &snap->topslot); 171 nent += snapshot_framelinks(J, p + nent, &snap->topslot);
172 snap->mapofs = (uint32_t)nsnapmap; 172 snap->mapofs = (uint32_t)nsnapmap;
173 snap->ref = (IRRef1)J->cur.nins; 173 snap->ref = (IRRef1)J->cur.nins;
174 snap->mcofs = 0;
174 snap->nslots = (uint8_t)nslots; 175 snap->nslots = (uint8_t)nslots;
175 snap->count = 0; 176 snap->count = 0;
176 J->cur.nsnapmap = (uint32_t)(nsnapmap + nent); 177 J->cur.nsnapmap = (uint32_t)(nsnapmap + nent);
diff --git a/src/lj_state.c b/src/lj_state.c
index 9585c18a..e87b945a 100644
--- a/src/lj_state.c
+++ b/src/lj_state.c
@@ -156,6 +156,7 @@ static TValue *cpluaopen(lua_State *L, lua_CFunction dummy, void *ud)
156 fixstring(lj_err_str(L, LJ_ERR_ERRMEM)); /* Preallocate memory error msg. */ 156 fixstring(lj_err_str(L, LJ_ERR_ERRMEM)); /* Preallocate memory error msg. */
157 g->gc.threshold = 4*g->gc.total; 157 g->gc.threshold = 4*g->gc.total;
158 lj_trace_initstate(g); 158 lj_trace_initstate(g);
159 lj_err_verify();
159 return NULL; 160 return NULL;
160} 161}
161 162
diff --git a/src/lj_target_x86.h b/src/lj_target_x86.h
index 4be5f04c..a403f820 100644
--- a/src/lj_target_x86.h
+++ b/src/lj_target_x86.h
@@ -165,6 +165,8 @@ typedef struct {
165#define EXITSTUB_SPACING (2+2) 165#define EXITSTUB_SPACING (2+2)
166#define EXITSTUBS_PER_GROUP 32 166#define EXITSTUBS_PER_GROUP 32
167 167
168#define EXITTRACE_VMSTATE 1 /* g->vmstate has traceno on exit. */
169
168/* -- x86 ModRM operand encoding ------------------------------------------ */ 170/* -- x86 ModRM operand encoding ------------------------------------------ */
169 171
170typedef enum { 172typedef enum {
diff --git a/src/lj_trace.c b/src/lj_trace.c
index 4608f81f..a0ff8864 100644
--- a/src/lj_trace.c
+++ b/src/lj_trace.c
@@ -821,7 +821,7 @@ static void trace_exit_regs(lua_State *L, ExitState *ex)
821} 821}
822#endif 822#endif
823 823
824#ifdef EXITSTATE_PCREG 824#if defined(EXITSTATE_PCREG) || (LJ_UNWIND_JIT && !EXITTRACE_VMSTATE)
825/* Determine trace number from pc of exit instruction. */ 825/* Determine trace number from pc of exit instruction. */
826static TraceNo trace_exit_find(jit_State *J, MCode *pc) 826static TraceNo trace_exit_find(jit_State *J, MCode *pc)
827{ 827{
@@ -843,10 +843,18 @@ int LJ_FASTCALL lj_trace_exit(jit_State *J, void *exptr)
843 lua_State *L = J->L; 843 lua_State *L = J->L;
844 ExitState *ex = (ExitState *)exptr; 844 ExitState *ex = (ExitState *)exptr;
845 ExitDataCP exd; 845 ExitDataCP exd;
846 int errcode; 846 int errcode, exitcode = J->exitcode;
847 TValue exiterr;
847 const BCIns *pc; 848 const BCIns *pc;
848 void *cf; 849 void *cf;
849 GCtrace *T; 850 GCtrace *T;
851
852 setnilV(&exiterr);
853 if (exitcode) { /* Trace unwound with error code. */
854 J->exitcode = 0;
855 copyTV(L, &exiterr, L->top-1);
856 }
857
850#ifdef EXITSTATE_PCREG 858#ifdef EXITSTATE_PCREG
851 J->parent = trace_exit_find(J, (MCode *)(intptr_t)ex->gpr[EXITSTATE_PCREG]); 859 J->parent = trace_exit_find(J, (MCode *)(intptr_t)ex->gpr[EXITSTATE_PCREG]);
852#endif 860#endif
@@ -866,6 +874,8 @@ int LJ_FASTCALL lj_trace_exit(jit_State *J, void *exptr)
866 if (errcode) 874 if (errcode)
867 return -errcode; /* Return negated error code. */ 875 return -errcode; /* Return negated error code. */
868 876
877 if (exitcode) copyTV(L, L->top++, &exiterr); /* Anchor the error object. */
878
869 if (!(LJ_HASPROFILE && (G(L)->hookmask & HOOK_PROFILE))) 879 if (!(LJ_HASPROFILE && (G(L)->hookmask & HOOK_PROFILE)))
870 lj_vmevent_send(L, TEXIT, 880 lj_vmevent_send(L, TEXIT,
871 lj_state_checkstack(L, 4+RID_NUM_GPR+RID_NUM_FPR+LUA_MINSTACK); 881 lj_state_checkstack(L, 4+RID_NUM_GPR+RID_NUM_FPR+LUA_MINSTACK);
@@ -877,7 +887,9 @@ int LJ_FASTCALL lj_trace_exit(jit_State *J, void *exptr)
877 pc = exd.pc; 887 pc = exd.pc;
878 cf = cframe_raw(L->cframe); 888 cf = cframe_raw(L->cframe);
879 setcframe_pc(cf, pc); 889 setcframe_pc(cf, pc);
880 if (LJ_HASPROFILE && (G(L)->hookmask & HOOK_PROFILE)) { 890 if (exitcode) {
891 return -exitcode;
892 } else if (LJ_HASPROFILE && (G(L)->hookmask & HOOK_PROFILE)) {
881 /* Just exit to interpreter. */ 893 /* Just exit to interpreter. */
882 } else if (G(L)->gc.state == GCSatomic || G(L)->gc.state == GCSfinalize) { 894 } else if (G(L)->gc.state == GCSatomic || G(L)->gc.state == GCSfinalize) {
883 if (!(G(L)->hookmask & HOOK_GC)) 895 if (!(G(L)->hookmask & HOOK_GC))
@@ -915,4 +927,41 @@ int LJ_FASTCALL lj_trace_exit(jit_State *J, void *exptr)
915 } 927 }
916} 928}
917 929
930#if LJ_UNWIND_JIT
931/* Given an mcode address determine trace exit address for unwinding. */
932uintptr_t LJ_FASTCALL lj_trace_unwind(jit_State *J, uintptr_t addr, ExitNo *ep)
933{
934#if EXITTRACE_VMSTATE
935 TraceNo traceno = J2G(J)->vmstate;
936#else
937 TraceNo traceno = trace_exit_find(J, (MCode *)addr);
938#endif
939 GCtrace *T = traceref(J, traceno);
940 if (T
941#if EXITTRACE_VMSTATE
942 && addr >= (uintptr_t)T->mcode && addr < (uintptr_t)T->mcode + T->szmcode
943#endif
944 ) {
945 SnapShot *snap = T->snap;
946 SnapNo lo = 0, exitno = T->nsnap;
947 uintptr_t ofs = (uintptr_t)((MCode *)addr - T->mcode); /* MCode units! */
948 /* Rightmost binary search for mcode offset to determine exit number. */
949 do {
950 SnapNo mid = (lo+exitno) >> 1;
951 if (ofs < snap[mid].mcofs) exitno = mid; else lo = mid + 1;
952 } while (lo < exitno);
953 exitno--;
954 *ep = exitno;
955#ifdef EXITSTUBS_PER_GROUP
956 return (uintptr_t)exitstub_addr(J, exitno);
957#else
958 return (uintptr_t)exitstub_trace_addr(T, exitno);
959#endif
960 }
961 /* Cannot correlate addr with trace/exit. This will be fatal. */
962 lj_assertJ(0, "bad exit pc");
963 return 0;
964}
965#endif
966
918#endif 967#endif
diff --git a/src/lj_trace.h b/src/lj_trace.h
index 7e86d963..e4cf2dc4 100644
--- a/src/lj_trace.h
+++ b/src/lj_trace.h
@@ -37,6 +37,9 @@ LJ_FUNC void lj_trace_ins(jit_State *J, const BCIns *pc);
37LJ_FUNCA void LJ_FASTCALL lj_trace_hot(jit_State *J, const BCIns *pc); 37LJ_FUNCA void LJ_FASTCALL lj_trace_hot(jit_State *J, const BCIns *pc);
38LJ_FUNCA void LJ_FASTCALL lj_trace_stitch(jit_State *J, const BCIns *pc); 38LJ_FUNCA void LJ_FASTCALL lj_trace_stitch(jit_State *J, const BCIns *pc);
39LJ_FUNCA int LJ_FASTCALL lj_trace_exit(jit_State *J, void *exptr); 39LJ_FUNCA int LJ_FASTCALL lj_trace_exit(jit_State *J, void *exptr);
40#if LJ_UNWIND_EXT
41LJ_FUNC uintptr_t LJ_FASTCALL lj_trace_unwind(jit_State *J, uintptr_t addr, ExitNo *ep);
42#endif
40 43
41/* Signal asynchronous abort of trace or end of trace. */ 44/* Signal asynchronous abort of trace or end of trace. */
42#define lj_trace_abort(g) (G2J(g)->state &= ~LJ_TRACE_ACTIVE) 45#define lj_trace_abort(g) (G2J(g)->state &= ~LJ_TRACE_ACTIVE)
diff --git a/src/lj_vm.h b/src/lj_vm.h
index 6358dc58..84348e7a 100644
--- a/src/lj_vm.h
+++ b/src/lj_vm.h
@@ -26,6 +26,9 @@ LJ_ASMF void lj_vm_unwind_ff_eh(void);
26#if LJ_TARGET_X86ORX64 26#if LJ_TARGET_X86ORX64
27LJ_ASMF void lj_vm_unwind_rethrow(void); 27LJ_ASMF void lj_vm_unwind_rethrow(void);
28#endif 28#endif
29#if LJ_TARGET_MIPS
30LJ_ASMF void lj_vm_unwind_stub(void);
31#endif
29 32
30/* Miscellaneous functions. */ 33/* Miscellaneous functions. */
31#if LJ_TARGET_X86ORX64 34#if LJ_TARGET_X86ORX64
diff --git a/src/vm_arm.dasc b/src/vm_arm.dasc
index 5efda227..c0934ac8 100644
--- a/src/vm_arm.dasc
+++ b/src/vm_arm.dasc
@@ -2245,8 +2245,9 @@ static void build_subroutines(BuildCtx *ctx)
2245 | b <2 2245 | b <2
2246 | 2246 |
2247 |9: // Rethrow error from the right C frame. 2247 |9: // Rethrow error from the right C frame.
2248 | rsb CARG2, CARG1, #0
2248 | mov CARG1, L 2249 | mov CARG1, L
2249 | bl extern lj_err_run // (lua_State *L) 2250 | bl extern lj_err_trace // (lua_State *L, int errcode)
2250 |.endif 2251 |.endif
2251 | 2252 |
2252 |//----------------------------------------------------------------------- 2253 |//-----------------------------------------------------------------------
diff --git a/src/vm_arm64.dasc b/src/vm_arm64.dasc
index 755d9a64..3a63d23b 100644
--- a/src/vm_arm64.dasc
+++ b/src/vm_arm64.dasc
@@ -2035,8 +2035,9 @@ static void build_subroutines(BuildCtx *ctx)
2035 | b <2 2035 | b <2
2036 | 2036 |
2037 |9: // Rethrow error from the right C frame. 2037 |9: // Rethrow error from the right C frame.
2038 | neg CARG2w, CARG1w
2038 | mov CARG1, L 2039 | mov CARG1, L
2039 | bl extern lj_err_run // (lua_State *L) 2040 | bl extern lj_err_trace // (lua_State *L, int errcode)
2040 |.endif 2041 |.endif
2041 | 2042 |
2042 |//----------------------------------------------------------------------- 2043 |//-----------------------------------------------------------------------
diff --git a/src/vm_mips.dasc b/src/vm_mips.dasc
index 7a6a83be..e16066d7 100644
--- a/src/vm_mips.dasc
+++ b/src/vm_mips.dasc
@@ -501,6 +501,10 @@ static void build_subroutines(BuildCtx *ctx)
501 | b ->vm_returnc 501 | b ->vm_returnc
502 |. li RD, 16 // 2 results: false + error message. 502 |. li RD, 16 // 2 results: false + error message.
503 | 503 |
504 |->vm_unwind_stub: // Jump to exit stub from unwinder.
505 | jr CARG1
506 |. move ra, CARG2
507 |
504 |//----------------------------------------------------------------------- 508 |//-----------------------------------------------------------------------
505 |//-- Grow stack for calls ----------------------------------------------- 509 |//-- Grow stack for calls -----------------------------------------------
506 |//----------------------------------------------------------------------- 510 |//-----------------------------------------------------------------------
@@ -2520,8 +2524,9 @@ static void build_subroutines(BuildCtx *ctx)
2520 |. addu RA, RA, BASE 2524 |. addu RA, RA, BASE
2521 | 2525 |
2522 |9: // Rethrow error from the right C frame. 2526 |9: // Rethrow error from the right C frame.
2523 | load_got lj_err_run 2527 | load_got lj_err_trace
2524 | call_intern lj_err_run // (lua_State *L) 2528 | sub CARG2, r0, CRET1
2529 | call_intern lj_err_trace // (lua_State *L, int errcode)
2525 |. move CARG1, L 2530 |. move CARG1, L
2526 |.endif 2531 |.endif
2527 | 2532 |
diff --git a/src/vm_mips64.dasc b/src/vm_mips64.dasc
index 0b642095..877d9885 100644
--- a/src/vm_mips64.dasc
+++ b/src/vm_mips64.dasc
@@ -556,6 +556,10 @@ static void build_subroutines(BuildCtx *ctx)
556 | b ->vm_returnc 556 | b ->vm_returnc
557 |. li RD, 16 // 2 results: false + error message. 557 |. li RD, 16 // 2 results: false + error message.
558 | 558 |
559 |->vm_unwind_stub: // Jump to exit stub from unwinder.
560 | jr CARG1
561 |. move ra, CARG2
562 |
559 |//----------------------------------------------------------------------- 563 |//-----------------------------------------------------------------------
560 |//-- Grow stack for calls ----------------------------------------------- 564 |//-- Grow stack for calls -----------------------------------------------
561 |//----------------------------------------------------------------------- 565 |//-----------------------------------------------------------------------
@@ -2622,8 +2626,9 @@ static void build_subroutines(BuildCtx *ctx)
2622 |. daddu RA, RA, BASE 2626 |. daddu RA, RA, BASE
2623 | 2627 |
2624 |9: // Rethrow error from the right C frame. 2628 |9: // Rethrow error from the right C frame.
2625 | load_got lj_err_run 2629 | load_got lj_err_trace
2626 | call_intern lj_err_run // (lua_State *L) 2630 | sub CARG2, r0, CRET1
2631 | call_intern lj_err_trace // (lua_State *L, int errcode)
2627 |. move CARG1, L 2632 |. move CARG1, L
2628 |.endif 2633 |.endif
2629 | 2634 |
diff --git a/src/vm_ppc.dasc b/src/vm_ppc.dasc
index f4ebb9db..1cd8eb15 100644
--- a/src/vm_ppc.dasc
+++ b/src/vm_ppc.dasc
@@ -3090,8 +3090,9 @@ static void build_subroutines(BuildCtx *ctx)
3090 | bctr 3090 | bctr
3091 | 3091 |
3092 |9: // Rethrow error from the right C frame. 3092 |9: // Rethrow error from the right C frame.
3093 | neg CARG2, CARG1
3093 | mr CARG1, L 3094 | mr CARG1, L
3094 | bl extern lj_err_run // (lua_State *L) 3095 | bl extern lj_err_trace // (lua_State *L, int errcode)
3095 |.endif 3096 |.endif
3096 | 3097 |
3097 |//----------------------------------------------------------------------- 3098 |//-----------------------------------------------------------------------
diff --git a/src/vm_x64.dasc b/src/vm_x64.dasc
index 8ade0a6a..05bfa649 100644
--- a/src/vm_x64.dasc
+++ b/src/vm_x64.dasc
@@ -2509,8 +2509,10 @@ static void build_subroutines(BuildCtx *ctx)
2509 | jmp <2 2509 | jmp <2
2510 | 2510 |
2511 |9: // Rethrow error from the right C frame. 2511 |9: // Rethrow error from the right C frame.
2512 | mov CARG2d, RDd
2512 | mov CARG1, L:RB 2513 | mov CARG1, L:RB
2513 | call extern lj_err_run // (lua_State *L) 2514 | neg CARG2d
2515 | call extern lj_err_trace // (lua_State *L, int errcode)
2514 |.endif 2516 |.endif
2515 | 2517 |
2516 |//----------------------------------------------------------------------- 2518 |//-----------------------------------------------------------------------
diff --git a/src/vm_x86.dasc b/src/vm_x86.dasc
index 1fee4a2a..1e376e45 100644
--- a/src/vm_x86.dasc
+++ b/src/vm_x86.dasc
@@ -2964,8 +2964,10 @@ static void build_subroutines(BuildCtx *ctx)
2964 | jmp <2 2964 | jmp <2
2965 | 2965 |
2966 |9: // Rethrow error from the right C frame. 2966 |9: // Rethrow error from the right C frame.
2967 | mov FCARG2, RD
2967 | mov FCARG1, L:RB 2968 | mov FCARG1, L:RB
2968 | call extern lj_err_run@4 // (lua_State *L) 2969 | neg FCARG2
2970 | call extern lj_err_trace@8 // (lua_State *L, int errcode)
2969 |.endif 2971 |.endif
2970 | 2972 |
2971 |//----------------------------------------------------------------------- 2973 |//-----------------------------------------------------------------------