summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--common.mk1
-rw-r--r--test/ruby/test_yjit.rb4
-rw-r--r--yjit_codegen.c43
3 files changed, 48 insertions, 0 deletions
diff --git a/common.mk b/common.mk
index 76d62915f6..5ccc9a389d 100644
--- a/common.mk
+++ b/common.mk
@@ -16721,6 +16721,7 @@ yjit_codegen.$(OBJEXT): $(top_srcdir)/internal/compilers.h
yjit_codegen.$(OBJEXT): $(top_srcdir)/internal/gc.h
yjit_codegen.$(OBJEXT): $(top_srcdir)/internal/imemo.h
yjit_codegen.$(OBJEXT): $(top_srcdir)/internal/object.h
+yjit_codegen.$(OBJEXT): $(top_srcdir)/internal/re.h
yjit_codegen.$(OBJEXT): $(top_srcdir)/internal/serial.h
yjit_codegen.$(OBJEXT): $(top_srcdir)/internal/static_assert.h
yjit_codegen.$(OBJEXT): $(top_srcdir)/internal/string.h
diff --git a/test/ruby/test_yjit.rb b/test/ruby/test_yjit.rb
index 8160042cb9..1f5f99f9a8 100644
--- a/test/ruby/test_yjit.rb
+++ b/test/ruby/test_yjit.rb
@@ -67,6 +67,10 @@ class TestYJIT < Test::Unit::TestCase
assert_no_exits('"i am a string #{true}"')
end
+ def test_compile_regexp
+ assert_no_exits('/#{true}/')
+ end
+
def test_getlocal_with_level
assert_compiles(<<~RUBY, insns: %i[getlocal opt_plus], result: [[7]])
def foo(foo, bar)
diff --git a/yjit_codegen.c b/yjit_codegen.c
index fc4d46a329..3f9169a74b 100644
--- a/yjit_codegen.c
+++ b/yjit_codegen.c
@@ -10,6 +10,7 @@
#include "internal/object.h"
#include "internal/string.h"
#include "internal/variable.h"
+#include "internal/re.h"
#include "insns_info.inc"
#include "yjit.h"
#include "yjit_iface.h"
@@ -3521,6 +3522,47 @@ gen_tostring(jitstate_t* jit, ctx_t* ctx)
}
static codegen_status_t
+gen_toregexp(jitstate_t* jit, ctx_t* ctx)
+{
+ rb_num_t opt = jit_get_arg(jit, 0);
+ rb_num_t cnt = jit_get_arg(jit, 1);
+
+ // Save the PC and SP because this allocates an object and could
+ // raise an exception.
+ jit_save_pc(jit, REG0);
+ jit_save_sp(jit, ctx);
+
+ x86opnd_t values_ptr = ctx_sp_opnd(ctx, -(sizeof(VALUE) * (uint32_t)cnt));
+ ctx_stack_pop(ctx, cnt);
+
+ mov(cb, C_ARG_REGS[0], imm_opnd(0));
+ mov(cb, C_ARG_REGS[1], imm_opnd(cnt));
+ lea(cb, C_ARG_REGS[2], values_ptr);
+ call_ptr(cb, REG0, (void *)&rb_ary_tmp_new_from_values);
+
+ // Save the array so we can clear it later
+ push(cb, RAX);
+ push(cb, RAX); // Alignment
+ mov(cb, C_ARG_REGS[0], RAX);
+ mov(cb, C_ARG_REGS[1], imm_opnd(opt));
+ call_ptr(cb, REG0, (void *)&rb_reg_new_ary);
+
+ // The actual regex is in RAX now. Pop the temp array from
+ // rb_ary_tmp_new_from_values into C arg regs so we can clear it
+ pop(cb, REG1); // Alignment
+ pop(cb, C_ARG_REGS[0]);
+
+ // The value we want to push on the stack is in RAX right now
+ x86opnd_t stack_ret = ctx_stack_push(ctx, TYPE_UNKNOWN);
+ mov(cb, stack_ret, RAX);
+
+ // Clear the temp array.
+ call_ptr(cb, REG0, (void *)&rb_ary_clear);
+
+ return YJIT_KEEP_COMPILING;
+}
+
+static codegen_status_t
gen_opt_getinlinecache(jitstate_t *jit, ctx_t *ctx)
{
VALUE jump_offset = jit_get_arg(jit, 0);
@@ -3758,6 +3800,7 @@ yjit_init_codegen(void)
yjit_reg_op(BIN(getglobal), gen_getglobal);
yjit_reg_op(BIN(setglobal), gen_setglobal);
yjit_reg_op(BIN(tostring), gen_tostring);
+ yjit_reg_op(BIN(toregexp), gen_toregexp);
yjit_method_codegen_table = st_init_numtable();