guile-commits
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

[Guile-commits] 02/06: Allow for bind-optionals without alloc-frame


From: Andy Wingo
Subject: [Guile-commits] 02/06: Allow for bind-optionals without alloc-frame
Date: Fri, 7 Jun 2019 11:06:13 -0400 (EDT)

wingo pushed a commit to branch master
in repository guile.

commit c86758c298fdd06456939d16d604f9ff9a6c7b10
Author: Andy Wingo <address@hidden>
Date:   Thu Jun 6 17:26:59 2019 +0200

    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.
---
 libguile/gsubr.c    |  17 ++++----
 libguile/jit.c      | 113 +++++++++++++++++++++++++++++++---------------------
 libguile/programs.c |  12 ++++++
 3 files changed, 90 insertions(+), 52 deletions(-)

diff --git a/libguile/gsubr.c b/libguile/gsubr.c
index b99cc67..6442627 100644
--- a/libguile/gsubr.c
+++ b/libguile/gsubr.c
@@ -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 ();
diff --git a/libguile/jit.c b/libguile/jit.c
index 4e4a355..6e92730 100644
--- a/libguile/jit.c
+++ b/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;
 }
diff --git a/libguile/programs.c b/libguile/programs.c
index 6750357..81495a5 100644
--- a/libguile/programs.c
+++ b/libguile/programs.c
@@ -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;



reply via email to

[Prev in Thread] Current Thread [Next in Thread]