mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-04-29 19:30:36 +02:00
Allow for bind-optionals without alloc-frame
This reduces subr trampoline instruction count for subrs with optional args. * libguile/gsubr.c (get_subr_stub_code): Insert bind-optionals instructions. (primitive_call_ip): Handle bind-optionals. * libguile/programs.c (try_parse_arity): Handle bind-optionals. * libguile/jit.c (struct scm_jit_state) (compile_call, compile_call_label, compile_tail_call) (compile_tail_call_label, compile_receive), compile_receive_values) (compile_shuffle_down, compile_return_values, compile_subr_call) (compile_foreign_call, compile_continuation_call) (compile_compose_continuation, compile_abort, compile_assert_nargs_ee) (compile_assert_nargs_ge, compile_assert_nargs_le) (compile_alloc_frame, compile_reset_frame, compile_push) (compile_pop, compile_drop, compile_expand_apply_argument) (compile_bind_kwargs, compile_bind_rest, compile_bind_optionals) (compile, compute_mcode): Separately track min and max frame sizes.
This commit is contained in:
parent
9fd978ed7e
commit
c86758c298
3 changed files with 90 additions and 52 deletions
|
@ -1,4 +1,4 @@
|
|||
/* Copyright 1995-2001,2006,2008-2011,2013,2015,2018
|
||||
/* Copyright 1995-2001,2006,2008-2011,2013,2015,2018-2019
|
||||
Free Software Foundation, Inc.
|
||||
|
||||
This file is part of Guile.
|
||||
|
@ -292,7 +292,7 @@ get_subr_stub_code (uint32_t subr_idx,
|
|||
case OPT:
|
||||
{
|
||||
uint32_t code[2] = { SCM_PACK_OP_24 (assert_nargs_le, nopt + 1),
|
||||
SCM_PACK_OP_24 (alloc_frame, nopt + 1) };
|
||||
SCM_PACK_OP_24 (bind_optionals, nopt + 1) };
|
||||
return alloc_subr_code (subr_idx, code, 2);
|
||||
}
|
||||
case REST:
|
||||
|
@ -304,7 +304,7 @@ get_subr_stub_code (uint32_t subr_idx,
|
|||
{
|
||||
uint32_t code[3] = { SCM_PACK_OP_24 (assert_nargs_ge, nreq + 1),
|
||||
SCM_PACK_OP_24 (assert_nargs_le, nreq + nopt + 1),
|
||||
SCM_PACK_OP_24 (alloc_frame, nreq + nopt + 1) };
|
||||
SCM_PACK_OP_24 (bind_optionals, nreq + nopt + 1) };
|
||||
return alloc_subr_code (subr_idx, code, 3);
|
||||
}
|
||||
case REQ_REST:
|
||||
|
@ -315,14 +315,16 @@ get_subr_stub_code (uint32_t subr_idx,
|
|||
}
|
||||
case OPT_REST:
|
||||
{
|
||||
uint32_t code[1] = { SCM_PACK_OP_24 (bind_rest, nopt + 1) };
|
||||
return alloc_subr_code (subr_idx, code, 1);
|
||||
uint32_t code[2] = { SCM_PACK_OP_24 (bind_optionals, nopt + 1),
|
||||
SCM_PACK_OP_24 (bind_rest, nopt + 1) };
|
||||
return alloc_subr_code (subr_idx, code, 2);
|
||||
}
|
||||
case REQ_OPT_REST:
|
||||
{
|
||||
uint32_t code[2] = { SCM_PACK_OP_24 (assert_nargs_ge, nreq + 1),
|
||||
uint32_t code[3] = { SCM_PACK_OP_24 (assert_nargs_ge, nreq + 1),
|
||||
SCM_PACK_OP_24 (bind_optionals, nreq + nopt + 1),
|
||||
SCM_PACK_OP_24 (bind_rest, nreq + nopt + 1) };
|
||||
return alloc_subr_code (subr_idx, code, 2);
|
||||
return alloc_subr_code (subr_idx, code, 3);
|
||||
}
|
||||
default:
|
||||
abort ();
|
||||
|
@ -380,6 +382,7 @@ primitive_call_ip (const uint32_t *code)
|
|||
case scm_op_assert_nargs_ee:
|
||||
case scm_op_assert_nargs_le:
|
||||
case scm_op_assert_nargs_ge:
|
||||
case scm_op_bind_optionals:
|
||||
case scm_op_bind_rest:
|
||||
case scm_op_alloc_frame:
|
||||
if (direction < 0) abort ();
|
||||
|
|
113
libguile/jit.c
113
libguile/jit.c
|
@ -182,7 +182,8 @@ struct scm_jit_state {
|
|||
size_t reloc_idx;
|
||||
size_t reloc_count;
|
||||
void **labels;
|
||||
int32_t frame_size;
|
||||
int32_t frame_size_min;
|
||||
int32_t frame_size_max;
|
||||
uint32_t register_state;
|
||||
jit_gpr_t sp_cache_gpr;
|
||||
jit_fpr_t sp_cache_fpr;
|
||||
|
@ -1462,7 +1463,8 @@ compile_call (scm_jit_state *j, uint32_t proc, uint32_t nlocals)
|
|||
jit_patch_here (j->jit, mcont);
|
||||
|
||||
reset_register_state (j, FP_IN_REGISTER | SP_IN_REGISTER);
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = proc;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1476,7 +1478,8 @@ compile_call_label (scm_jit_state *j, uint32_t proc, uint32_t nlocals, const uin
|
|||
jit_patch_here (j->jit, mcont);
|
||||
|
||||
reset_register_state (j, FP_IN_REGISTER | SP_IN_REGISTER);
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = proc;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1487,7 +1490,8 @@ compile_tail_call (scm_jit_state *j)
|
|||
|
||||
emit_indirect_tail_call (j);
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1498,7 +1502,8 @@ compile_tail_call_label (scm_jit_state *j, const uint32_t *vcode)
|
|||
|
||||
emit_direct_tail_call (j, vcode);
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1528,7 +1533,7 @@ compile_receive (scm_jit_state *j, uint16_t dst, uint16_t proc, uint32_t nlocals
|
|||
emit_fp_set_scm (j, dst, t);
|
||||
emit_reset_frame (j, nlocals);
|
||||
|
||||
j->frame_size = nlocals;
|
||||
j->frame_size_min = j->frame_size_max = nlocals;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1556,10 +1561,10 @@ compile_receive_values (scm_jit_state *j, uint32_t proc, uint8_t allow_extra,
|
|||
jit_operand_imm (JIT_OPERAND_ABI_UINT32, nvalues));
|
||||
j->register_state = saved_state;
|
||||
jit_patch_here (j->jit, k);
|
||||
|
||||
j->frame_size = proc + nvalues;
|
||||
}
|
||||
|
||||
j->frame_size_min = proc + nvalues;
|
||||
j->frame_size_max = allow_extra ? INT32_MAX : j->frame_size_min;
|
||||
clear_register_state (j, SP_CACHE_GPR | SP_CACHE_FPR);
|
||||
}
|
||||
|
||||
|
@ -1584,8 +1589,9 @@ compile_shuffle_down (scm_jit_state *j, uint16_t from, uint16_t to)
|
|||
|
||||
clear_register_state (j, SP_CACHE_GPR | SP_CACHE_FPR);
|
||||
|
||||
if (j->frame_size >= 0)
|
||||
j->frame_size -= (from - to);
|
||||
j->frame_size_min -= (from - to);
|
||||
if (j->frame_size_max != INT32_MAX)
|
||||
j->frame_size_max -= (from - to);
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1605,7 +1611,8 @@ compile_return_values (scm_jit_state *j)
|
|||
emit_store_ip (j, ra);
|
||||
emit_exit (j);
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1616,14 +1623,14 @@ compile_subr_call (scm_jit_state *j, uint32_t idx)
|
|||
jit_reloc_t immediate, not_values, k;
|
||||
jit_operand_t args[10];
|
||||
|
||||
ASSERT (j->frame_size > 0);
|
||||
size_t argc = j->frame_size - 1;
|
||||
ASSERT (j->frame_size_min == j->frame_size_max);
|
||||
size_t argc = j->frame_size_max - 1;
|
||||
ASSERT (argc <= 10);
|
||||
|
||||
subr = scm_subr_function_by_index (idx);
|
||||
emit_store_current_ip (j, t);
|
||||
for (size_t i = 2; i <= j->frame_size; i++)
|
||||
args[i - 2] = sp_scm_operand (j, (j->frame_size - i));
|
||||
for (size_t i = 2; i <= j->frame_size_max; i++)
|
||||
args[i - 2] = sp_scm_operand (j, (j->frame_size_max - i));
|
||||
jit_calli (j->jit, subr, argc, args);
|
||||
clear_scratch_register_state (j);
|
||||
jit_retval (j->jit, ret);
|
||||
|
@ -1647,7 +1654,8 @@ compile_subr_call (scm_jit_state *j, uint32_t idx)
|
|||
|
||||
clear_register_state (j, SP_CACHE_GPR | SP_CACHE_FPR);
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1655,10 +1663,10 @@ compile_foreign_call (scm_jit_state *j, uint16_t cif_idx, uint16_t ptr_idx)
|
|||
{
|
||||
uint32_t saved_state;
|
||||
|
||||
ASSERT (j->frame_size >= 0);
|
||||
ASSERT (j->frame_size_min == j->frame_size_max);
|
||||
|
||||
emit_store_current_ip (j, T0);
|
||||
emit_sp_ref_scm (j, T0, j->frame_size - 1);
|
||||
emit_sp_ref_scm (j, T0, j->frame_size_min - 1);
|
||||
|
||||
/* FIXME: Inline the foreign call. */
|
||||
saved_state = save_reloadable_register_state (j);
|
||||
|
@ -1667,7 +1675,7 @@ compile_foreign_call (scm_jit_state *j, uint16_t cif_idx, uint16_t ptr_idx)
|
|||
free_variable_operand (j, T0, ptr_idx));
|
||||
restore_reloadable_register_state (j, saved_state);
|
||||
|
||||
j->frame_size = 2; /* Return value and errno. */
|
||||
j->frame_size_min = j->frame_size_max = 2; /* Return value and errno. */
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1680,7 +1688,8 @@ compile_continuation_call (scm_jit_state *j, uint32_t contregs_idx)
|
|||
thread_operand (), free_variable_operand (j, T0, contregs_idx));
|
||||
/* Does not fall through. */
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1703,7 +1712,8 @@ compile_compose_continuation (scm_jit_state *j, uint32_t cont_idx)
|
|||
jit_patch_here (j->jit, interp);
|
||||
emit_exit (j);
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1739,7 +1749,8 @@ compile_abort (scm_jit_state *j)
|
|||
|
||||
jit_patch_here (j->jit, k);
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1797,7 +1808,7 @@ compile_assert_nargs_ee (scm_jit_state *j, uint32_t nlocals)
|
|||
jit_patch_here (j->jit, k);
|
||||
|
||||
j->register_state = saved_state;
|
||||
j->frame_size = nlocals;
|
||||
j->frame_size_min = j->frame_size_max = nlocals;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1816,6 +1827,8 @@ compile_assert_nargs_ge (scm_jit_state *j, uint32_t nlocals)
|
|||
jit_patch_here (j->jit, k);
|
||||
j->register_state = saved_state;
|
||||
}
|
||||
|
||||
j->frame_size_min = nlocals;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1832,6 +1845,7 @@ compile_assert_nargs_le (scm_jit_state *j, uint32_t nlocals)
|
|||
jit_patch_here (j->jit, k);
|
||||
|
||||
j->register_state = saved_state;
|
||||
j->frame_size_max = nlocals;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1839,15 +1853,15 @@ compile_alloc_frame (scm_jit_state *j, uint32_t nlocals)
|
|||
{
|
||||
jit_gpr_t t = T0, saved_frame_size = T1_PRESERVED;
|
||||
|
||||
if (j->frame_size < 0)
|
||||
if (j->frame_size_min != j->frame_size_max)
|
||||
jit_subr (j->jit, saved_frame_size, FP, SP);
|
||||
|
||||
/* This will clear the regalloc, so no need to track clobbers. */
|
||||
emit_alloc_frame (j, t, nlocals);
|
||||
|
||||
if (j->frame_size >= 0)
|
||||
if (j->frame_size_min == j->frame_size_max)
|
||||
{
|
||||
int32_t slots = nlocals - j->frame_size;
|
||||
int32_t slots = nlocals - j->frame_size_min;
|
||||
|
||||
if (slots > 0)
|
||||
{
|
||||
|
@ -1870,7 +1884,7 @@ compile_alloc_frame (scm_jit_state *j, uint32_t nlocals)
|
|||
jit_patch_here (j->jit, k);
|
||||
}
|
||||
|
||||
j->frame_size = nlocals;
|
||||
j->frame_size_min = j->frame_size_max = nlocals;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1879,7 +1893,7 @@ compile_reset_frame (scm_jit_state *j, uint32_t nlocals)
|
|||
restore_reloadable_register_state (j, FP_IN_REGISTER);
|
||||
emit_reset_frame (j, nlocals);
|
||||
|
||||
j->frame_size = nlocals;
|
||||
j->frame_size_min = j->frame_size_max = nlocals;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1892,8 +1906,9 @@ compile_push (scm_jit_state *j, uint32_t src)
|
|||
|
||||
clear_register_state (j, SP_CACHE_GPR | SP_CACHE_FPR);
|
||||
|
||||
if (j->frame_size >= 0)
|
||||
j->frame_size++;
|
||||
j->frame_size_min++;
|
||||
if (j->frame_size_max != INT32_MAX)
|
||||
j->frame_size_max++;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1905,8 +1920,9 @@ compile_pop (scm_jit_state *j, uint32_t dst)
|
|||
|
||||
clear_register_state (j, SP_CACHE_GPR | SP_CACHE_FPR);
|
||||
|
||||
if (j->frame_size >= 0)
|
||||
j->frame_size--;
|
||||
j->frame_size_min--;
|
||||
if (j->frame_size_max != INT32_MAX)
|
||||
j->frame_size_max--;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1917,8 +1933,9 @@ compile_drop (scm_jit_state *j, uint32_t nvalues)
|
|||
|
||||
clear_register_state (j, SP_CACHE_GPR | SP_CACHE_FPR);
|
||||
|
||||
if (j->frame_size >= 0)
|
||||
j->frame_size -= nvalues;
|
||||
j->frame_size_min -= nvalues;
|
||||
if (j->frame_size_max != INT32_MAX)
|
||||
j->frame_size_max -= nvalues;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1938,7 +1955,8 @@ compile_expand_apply_argument (scm_jit_state *j)
|
|||
emit_reload_sp (j);
|
||||
emit_reload_fp (j);
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min--;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -1978,7 +1996,7 @@ compile_bind_kwargs (scm_jit_state *j, uint32_t nreq, uint8_t flags,
|
|||
emit_reload_fp (j);
|
||||
|
||||
emit_reset_frame (j, ntotal);
|
||||
j->frame_size = ntotal;
|
||||
j->frame_size_min = j->frame_size_max = ntotal;
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -2003,21 +2021,25 @@ compile_bind_rest (scm_jit_state *j, uint32_t dst)
|
|||
emit_sp_set_scm (j, 0, t);
|
||||
|
||||
jit_patch_here (j->jit, k);
|
||||
|
||||
j->frame_size_min = dst + 1;
|
||||
}
|
||||
|
||||
static void
|
||||
compile_bind_optionals (scm_jit_state *j, uint32_t dst)
|
||||
compile_bind_optionals (scm_jit_state *j, uint32_t nlocals)
|
||||
{
|
||||
ASSERT_HAS_REGISTER_STATE (FP_IN_REGISTER | SP_IN_REGISTER);
|
||||
ASSERT(j->frame_size == -1);
|
||||
ASSERT(j->frame_size_min < nlocals);
|
||||
ASSERT(j->frame_size_min < j->frame_size_max);
|
||||
|
||||
jit_gpr_t saved_frame_size = T1_PRESERVED;
|
||||
jit_subr (j->jit, saved_frame_size, FP, SP);
|
||||
|
||||
jit_reloc_t no_optionals = jit_bgei
|
||||
(j->jit, saved_frame_size, dst * sizeof (union scm_vm_stack_element));
|
||||
(j->jit, saved_frame_size, nlocals * sizeof (union scm_vm_stack_element));
|
||||
|
||||
emit_alloc_frame (j, T0, dst);
|
||||
emit_alloc_frame (j, T0, nlocals);
|
||||
j->frame_size_min = nlocals;
|
||||
|
||||
jit_gpr_t walk = saved_frame_size;
|
||||
jit_subr (j->jit, walk, FP, saved_frame_size);
|
||||
|
@ -2032,8 +2054,6 @@ compile_bind_optionals (scm_jit_state *j, uint32_t dst)
|
|||
|
||||
jit_patch_here (j->jit, done);
|
||||
jit_patch_here (j->jit, no_optionals);
|
||||
|
||||
ASSERT(j->frame_size == -1);
|
||||
}
|
||||
|
||||
static void
|
||||
|
@ -4616,7 +4636,8 @@ compile (scm_jit_state *j)
|
|||
{
|
||||
j->ip = (uint32_t *) j->start;
|
||||
set_register_state (j, SP_IN_REGISTER | FP_IN_REGISTER);
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
|
||||
for (ptrdiff_t offset = 0; j->ip + offset < j->end; offset++)
|
||||
j->labels[offset] = NULL;
|
||||
|
@ -4740,7 +4761,8 @@ compute_mcode (scm_thread *thread, uint32_t *entry_ip,
|
|||
j->labels = calloc ((j->end - j->start), sizeof (*j->labels));
|
||||
ASSERT (j->labels);
|
||||
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
|
||||
INFO ("vcode: start=%p,+%zu entry=+%zu\n", j->start, j->end - j->start,
|
||||
j->entry - j->start);
|
||||
|
@ -4763,7 +4785,8 @@ compute_mcode (scm_thread *thread, uint32_t *entry_ip,
|
|||
j->reloc_count = 0;
|
||||
|
||||
j->start = j->end = j->ip = j->entry = NULL;
|
||||
j->frame_size = -1;
|
||||
j->frame_size_min = 0;
|
||||
j->frame_size_max = INT32_MAX;
|
||||
|
||||
return entry_mcode;
|
||||
}
|
||||
|
|
|
@ -295,6 +295,12 @@ try_parse_arity (SCM program, int *req, int *opt, int *rest)
|
|||
*opt = slots - 1;
|
||||
*rest = 0;
|
||||
return 1;
|
||||
case scm_op_bind_optionals:
|
||||
slots = code[0] >> 8;
|
||||
*req = 0;
|
||||
*opt = slots - 1;
|
||||
*rest = ((code[1] & 0xff) == scm_op_bind_rest);
|
||||
return 1;
|
||||
case scm_op_bind_rest:
|
||||
slots = code[0] >> 8;
|
||||
*req = 0;
|
||||
|
@ -310,6 +316,12 @@ try_parse_arity (SCM program, int *req, int *opt, int *rest)
|
|||
*opt = slots - 1 - *req;
|
||||
*rest = 0;
|
||||
return 1;
|
||||
case scm_op_bind_optionals:
|
||||
slots = code[1] >> 8;
|
||||
*req = min - 1;
|
||||
*opt = slots - 1 - *req;
|
||||
*rest = ((code[2] & 0xff) == scm_op_bind_rest);
|
||||
return 1;
|
||||
case scm_op_bind_rest:
|
||||
slots = code[1] >> 8;
|
||||
*req = min - 1;
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue