From bb046d23f435770b222d8dd5f9935447969fdd58 Mon Sep 17 00:00:00 2001 From: Gordon Williams Date: Fri, 20 Oct 2023 09:50:57 +0100 Subject: [PATCH] JIT: Ensure ternary block skips 'named' variables (and references them correctly) --- ChangeLog | 1 + src/jsjit.c | 14 +++++++++----- 2 files changed, 10 insertions(+), 5 deletions(-) diff --git a/ChangeLog b/ChangeLog index ec95a66203..93ffce8944 100644 --- a/ChangeLog +++ b/ChangeLog @@ -15,6 +15,7 @@ Remove createChild argument from jsvFindChildFromString, add jsvFindOrAddChildFromString instead Add ES9 optional catch binding (`try {} catch {}`) Bangle.js: Fix terminal's repeated call to '.flip' that broke double buffering on Bangle.js 1 + JIT: Ensure ternary block skips 'named' variables (and references them correctly) 2v19 : Fix Object.values/entries for numeric keys after 2v18 regression (fix #2375) nRF52: for SD>5 use static buffers for advertising and scan response data (#2367) diff --git a/src/jsjit.c b/src/jsjit.c index 2022bf8573..ec1a259376 100644 --- a/src/jsjit.c +++ b/src/jsjit.c @@ -745,21 +745,25 @@ void jsjConditionalExpression() { jsjBinaryExpression(); if (lex->tk=='?') { JSP_ASSERT_MATCH('?'); + if (jit.phase == JSJP_EMIT) { + /* we handle the condition here because it means the stack level is + then the same when we capture the true/false blocks as when we emit them */ + DEBUG_JIT("; ternary condition\n"); + jsjPopAsBool(0); + jsjcCompareImm(0, 0); + } DEBUG_JIT_EMIT("; capture ternary true block\n"); JsVar *oldBlock = jsjcStartBlock(); jsjAssignmentExpression(); - if (jit.phase == JSJP_EMIT) jsjPopAsVar(0); // we pop to r0 here so we can push after and avoid confusing the stack size checker + if (jit.phase == JSJP_EMIT) jsjPopNoName(0); // we pop to r0 here so we can push after and avoid confusing the stack size checker JsVar *trueBlock = jsjcStopBlock(oldBlock); JSP_MATCH(':'); DEBUG_JIT_EMIT("; capture ternary false block\n"); oldBlock = jsjcStartBlock(); jsjAssignmentExpression(); - if (jit.phase == JSJP_EMIT) jsjPopAsVar(0); // we pop to r0 here so we can push after and avoid confusing the stack size checker + if (jit.phase == JSJP_EMIT) jsjPopNoName(0); // we pop to r0 here so we can push after and avoid confusing the stack size checker JsVar *falseBlock = jsjcStopBlock(oldBlock); if (jit.phase == JSJP_EMIT) { - DEBUG_JIT("; ternary condition\n"); - jsjPopAsBool(0); - jsjcCompareImm(0, 0); DEBUG_JIT("; ternary jump after condition\n"); // if false, jump after true block (if an 'else' we need to jump over the jsjcBranchRelative jsjcBranchConditionalRelative(JSJAC_EQ, jsvGetStringLength(trueBlock) + 2);