Skip to content

Commit

Permalink
YJIT: Implement concatarray in yjit (#405)
Browse files Browse the repository at this point in the history
* Create code generation func

* Make rb_vm_concat_array available to use in Rust

* Map opcode to code gen func

* Implement code gen for concatarray

* Add test for concatarray

* Use new asm backend

* Add comment to C func wrapper
  • Loading branch information
wildmaples authored and noahgibbs committed Aug 26, 2022
1 parent d8edb0a commit 0bcdbbf
Show file tree
Hide file tree
Showing 4 changed files with 46 additions and 0 deletions.
10 changes: 10 additions & 0 deletions bootstraptest/test_yjit.rb
Expand Up @@ -3245,3 +3245,13 @@ def foo(&block)
end
foo { "foo" }.call
}

assert_equal '[1, 2]', %q{
def foo
x = [2]
[1, *x]
end
foo
foo
}
10 changes: 10 additions & 0 deletions vm_insnhelper.c
Expand Up @@ -4380,6 +4380,14 @@ vm_concat_array(VALUE ary1, VALUE ary2st)
return rb_ary_concat(tmp1, tmp2);
}

// YJIT implementation is using the C function
// and needs to call a non-static function
VALUE
rb_vm_concat_array(VALUE ary1, VALUE ary2st)
{
return vm_concat_array(ary1, ary2st);
}

static VALUE
vm_splat_array(VALUE flag, VALUE ary)
{
Expand All @@ -4395,6 +4403,8 @@ vm_splat_array(VALUE flag, VALUE ary)
}
}

// YJIT implementation is using the C function
// and needs to call a non-static function
VALUE
rb_vm_splat_array(VALUE flag, VALUE ary)
{
Expand Down
25 changes: 25 additions & 0 deletions yjit/src/codegen.rs
Expand Up @@ -1256,6 +1256,30 @@ fn gen_splatarray(
KeepCompiling
}

// concat two arrays
fn gen_concatarray(
jit: &mut JITState,
ctx: &mut Context,
asm: &mut Assembler,
_ocb: &mut OutlinedCb,
) -> CodegenStatus {
// Save the PC and SP because the callee may allocate
// Note that this modifies REG_SP, which is why we do it first
jit_prepare_routine_call(jit, ctx, asm);

// Get the operands from the stack
let ary2st_opnd = ctx.stack_pop(1);
let ary1_opnd = ctx.stack_pop(1);

// Call rb_vm_concat_array(ary1, ary2st)
let ary = asm.ccall(rb_vm_concat_array as *const u8, vec![ary1_opnd, ary2st_opnd]);

let stack_ret = ctx.stack_push(Type::Array);
asm.mov(stack_ret, ary);

KeepCompiling
}

// new range initialized from top 2 values
fn gen_newrange(
jit: &mut JITState,
Expand Down Expand Up @@ -5862,6 +5886,7 @@ fn get_gen_fn(opcode: VALUE) -> Option<InsnGenFn> {
YARVINSN_opt_str_freeze => Some(gen_opt_str_freeze),
YARVINSN_opt_str_uminus => Some(gen_opt_str_uminus),
YARVINSN_splatarray => Some(gen_splatarray),
YARVINSN_concatarray => Some(gen_concatarray),
YARVINSN_newrange => Some(gen_newrange),
YARVINSN_putstring => Some(gen_putstring),
YARVINSN_expandarray => Some(gen_expandarray),
Expand Down
1 change: 1 addition & 0 deletions yjit/src/cruby.rs
Expand Up @@ -112,6 +112,7 @@ pub use autogened::*;
#[cfg_attr(test, allow(unused))] // We don't link against C code when testing
extern "C" {
pub fn rb_vm_splat_array(flag: VALUE, ary: VALUE) -> VALUE;
pub fn rb_vm_concat_array(ary1: VALUE, ary2st: VALUE) -> VALUE;
pub fn rb_vm_defined(
ec: EcPtr,
reg_cfp: CfpPtr,
Expand Down

0 comments on commit 0bcdbbf

Please sign in to comment.