jscript: Use global accumulator for storing function call results.

Signed-off-by: Jacek Caban <jacek@codeweavers.com>
Signed-off-by: Alexandre Julliard <julliard@winehq.org>
This commit is contained in:
Jacek Caban 2018-06-12 14:04:11 +02:00 committed by Alexandre Julliard
parent 4b72060ff0
commit 5a90acf59c
6 changed files with 41 additions and 20 deletions

View File

@ -609,7 +609,7 @@ static HRESULT compile_new_expression(compiler_ctx_t *ctx, call_expression_t *ex
if(FAILED(hres)) if(FAILED(hres))
return hres; return hres;
return push_instr(ctx, OP_push_ret) ? S_OK : E_OUTOFMEMORY; return push_instr(ctx, OP_push_acc) ? S_OK : E_OUTOFMEMORY;
} }
static HRESULT compile_call_expression(compiler_ctx_t *ctx, call_expression_t *expr, BOOL emit_ret) static HRESULT compile_call_expression(compiler_ctx_t *ctx, call_expression_t *expr, BOOL emit_ret)
@ -651,7 +651,7 @@ static HRESULT compile_call_expression(compiler_ctx_t *ctx, call_expression_t *e
if(FAILED(hres)) if(FAILED(hres))
return hres; return hres;
return !emit_ret || push_instr(ctx, OP_push_ret) ? S_OK : E_OUTOFMEMORY; return !emit_ret || push_instr(ctx, OP_push_acc) ? S_OK : E_OUTOFMEMORY;
} }
static HRESULT compile_delete_expression(compiler_ctx_t *ctx, unary_expression_t *expr) static HRESULT compile_delete_expression(compiler_ctx_t *ctx, unary_expression_t *expr)

View File

@ -385,9 +385,10 @@ static inline jsval_t steal_ret(call_frame_t *frame)
return r; return r;
} }
static inline void clear_ret(call_frame_t *frame) static inline void clear_acc(script_ctx_t *ctx)
{ {
jsval_release(steal_ret(frame)); jsval_release(ctx->acc);
ctx->acc = jsval_undefined();
} }
static HRESULT scope_push(scope_chain_t *scope, jsdisp_t *jsobj, IDispatch *obj, scope_chain_t **ret) static HRESULT scope_push(scope_chain_t *scope, jsdisp_t *jsobj, IDispatch *obj, scope_chain_t **ret)
@ -1155,7 +1156,6 @@ static HRESULT interp_refval(script_ctx_t *ctx)
static HRESULT interp_new(script_ctx_t *ctx) static HRESULT interp_new(script_ctx_t *ctx)
{ {
const unsigned argc = get_op_uint(ctx, 0); const unsigned argc = get_op_uint(ctx, 0);
call_frame_t *frame = ctx->call_ctx;
jsval_t constr; jsval_t constr;
TRACE("%d\n", argc); TRACE("%d\n", argc);
@ -1171,9 +1171,9 @@ static HRESULT interp_new(script_ctx_t *ctx)
else if(!get_object(constr)) else if(!get_object(constr))
return throw_type_error(ctx, JS_E_INVALID_PROPERTY, NULL); return throw_type_error(ctx, JS_E_INVALID_PROPERTY, NULL);
clear_ret(frame); clear_acc(ctx);
return disp_call_value(ctx, get_object(constr), NULL, DISPATCH_CONSTRUCT | DISPATCH_JSCRIPT_CALLEREXECSSOURCE, return disp_call_value(ctx, get_object(constr), NULL, DISPATCH_CONSTRUCT | DISPATCH_JSCRIPT_CALLEREXECSSOURCE,
argc, stack_args(ctx, argc), &frame->ret); argc, stack_args(ctx, argc), &ctx->acc);
} }
/* ECMA-262 3rd Edition 11.2.3 */ /* ECMA-262 3rd Edition 11.2.3 */
@ -1181,7 +1181,6 @@ static HRESULT interp_call(script_ctx_t *ctx)
{ {
const unsigned argn = get_op_uint(ctx, 0); const unsigned argn = get_op_uint(ctx, 0);
const int do_ret = get_op_int(ctx, 1); const int do_ret = get_op_int(ctx, 1);
call_frame_t *frame = ctx->call_ctx;
jsval_t obj; jsval_t obj;
TRACE("%d %d\n", argn, do_ret); TRACE("%d %d\n", argn, do_ret);
@ -1190,9 +1189,9 @@ static HRESULT interp_call(script_ctx_t *ctx)
if(!is_object_instance(obj)) if(!is_object_instance(obj))
return throw_type_error(ctx, JS_E_INVALID_PROPERTY, NULL); return throw_type_error(ctx, JS_E_INVALID_PROPERTY, NULL);
clear_ret(frame); clear_acc(ctx);
return disp_call_value(ctx, get_object(obj), NULL, DISPATCH_METHOD | DISPATCH_JSCRIPT_CALLEREXECSSOURCE, return disp_call_value(ctx, get_object(obj), NULL, DISPATCH_METHOD | DISPATCH_JSCRIPT_CALLEREXECSSOURCE,
argn, stack_args(ctx, argn), do_ret ? &frame->ret : NULL); argn, stack_args(ctx, argn), do_ret ? &ctx->acc : NULL);
} }
/* ECMA-262 3rd Edition 11.2.3 */ /* ECMA-262 3rd Edition 11.2.3 */
@ -1200,7 +1199,6 @@ static HRESULT interp_call_member(script_ctx_t *ctx)
{ {
const unsigned argn = get_op_uint(ctx, 0); const unsigned argn = get_op_uint(ctx, 0);
const int do_ret = get_op_int(ctx, 1); const int do_ret = get_op_int(ctx, 1);
call_frame_t *frame = ctx->call_ctx;
exprval_t ref; exprval_t ref;
TRACE("%d %d\n", argn, do_ret); TRACE("%d %d\n", argn, do_ret);
@ -1208,9 +1206,9 @@ static HRESULT interp_call_member(script_ctx_t *ctx)
if(!stack_topn_exprval(ctx, argn, &ref)) if(!stack_topn_exprval(ctx, argn, &ref))
return throw_type_error(ctx, ref.u.hres, NULL); return throw_type_error(ctx, ref.u.hres, NULL);
clear_ret(frame); clear_acc(ctx);
return exprval_call(ctx, &ref, DISPATCH_METHOD | DISPATCH_JSCRIPT_CALLEREXECSSOURCE, return exprval_call(ctx, &ref, DISPATCH_METHOD | DISPATCH_JSCRIPT_CALLEREXECSSOURCE,
argn, stack_args(ctx, argn), do_ret ? &frame->ret : NULL); argn, stack_args(ctx, argn), do_ret ? &ctx->acc : NULL);
} }
/* ECMA-262 3rd Edition 11.1.1 */ /* ECMA-262 3rd Edition 11.1.1 */
@ -2600,16 +2598,15 @@ static HRESULT interp_setret(script_ctx_t *ctx)
return S_OK; return S_OK;
} }
static HRESULT interp_push_ret(script_ctx_t *ctx) static HRESULT interp_push_acc(script_ctx_t *ctx)
{ {
call_frame_t *frame = ctx->call_ctx;
HRESULT hres; HRESULT hres;
TRACE("\n"); TRACE("\n");
hres = stack_push(ctx, frame->ret); hres = stack_push(ctx, ctx->acc);
if(SUCCEEDED(hres)) if(SUCCEEDED(hres))
frame->ret = jsval_undefined(); ctx->acc = jsval_undefined();
return hres; return hres;
} }
@ -2796,8 +2793,8 @@ static HRESULT enter_bytecode(script_ctx_t *ctx, jsval_t *r)
assert(frame->scope == frame->base_scope); assert(frame->scope == frame->base_scope);
if(return_to_interp) { if(return_to_interp) {
clear_ret(frame->prev_frame); jsval_release(ctx->acc);
frame->prev_frame->ret = steal_ret(frame); ctx->acc = steal_ret(frame);
}else if(r) { }else if(r) {
*r = steal_ret(frame); *r = steal_ret(frame);
} }

View File

@ -73,8 +73,8 @@
X(pop_scope, 1, 0,0) \ X(pop_scope, 1, 0,0) \
X(postinc, 1, ARG_INT, 0) \ X(postinc, 1, ARG_INT, 0) \
X(preinc, 1, ARG_INT, 0) \ X(preinc, 1, ARG_INT, 0) \
X(push_acc, 1, 0,0) \
X(push_except,1, ARG_ADDR, ARG_UINT) \ X(push_except,1, ARG_ADDR, ARG_UINT) \
X(push_ret, 1, 0,0) \
X(push_scope, 1, 0,0) \ X(push_scope, 1, 0,0) \
X(regexp, 1, ARG_STR, ARG_UINT) \ X(regexp, 1, ARG_STR, ARG_UINT) \
X(rshift, 1, 0,0) \ X(rshift, 1, 0,0) \

View File

@ -69,6 +69,7 @@ void script_release(script_ctx_t *ctx)
if(--ctx->ref) if(--ctx->ref)
return; return;
jsval_release(ctx->acc);
clear_ei(ctx); clear_ei(ctx);
if(ctx->cc) if(ctx->cc)
release_cc(ctx->cc); release_cc(ctx->cc);
@ -715,6 +716,7 @@ static HRESULT WINAPI JScriptParse_InitNew(IActiveScriptParse *iface)
ctx->version = This->version; ctx->version = This->version;
ctx->html_mode = This->html_mode; ctx->html_mode = This->html_mode;
ctx->ei.val = jsval_undefined(); ctx->ei.val = jsval_undefined();
ctx->acc = jsval_undefined();
heap_pool_init(&ctx->tmp_heap); heap_pool_init(&ctx->tmp_heap);
hres = create_jscaller(ctx); hres = create_jscaller(ctx);

View File

@ -424,6 +424,7 @@ struct _script_ctx_t {
jsval_t *stack; jsval_t *stack;
unsigned stack_size; unsigned stack_size;
unsigned stack_top; unsigned stack_top;
jsval_t acc;
jsstr_t *last_match; jsstr_t *last_match;
match_result_t match_parens[9]; match_result_t match_parens[9];

View File

@ -1146,6 +1146,27 @@ case 3:
})(); })();
expect(ret, "ret"); expect(ret, "ret");
expect(x, "try,try2,finally2,finally,ret"); expect(x, "try,try2,finally2,finally,ret");
ret = (function() {
try {
return "try";
unreachable();
}catch(e) {
unreachable();
}finally {
new Object();
var tmp = (function() {
var s = new String();
try {
s.length;
}finally {
return 1;
}
})();
}
unreachable();
})();
expect(ret, "try");
})(); })();
tmp = eval("1"); tmp = eval("1");