2019-11-07 02:58:00 -05:00
|
|
|
|
2019-11-25 22:20:53 -05:00
|
|
|
def inline_text argc, prev_insn
|
|
|
|
raise "argc (#{argc}) of inline! should be 1" unless argc == 1
|
|
|
|
raise "1st argument should be string literal" unless prev_insn[0] == :putstring
|
|
|
|
prev_insn[1].rstrip
|
|
|
|
end
|
|
|
|
|
2019-12-13 03:26:12 -05:00
|
|
|
def make_cfunc_name inlines, name, lineno
|
|
|
|
case name
|
|
|
|
when /\[\]/
|
|
|
|
name = '_GETTER'
|
|
|
|
when /\[\]=/
|
|
|
|
name = '_SETTER'
|
|
|
|
else
|
|
|
|
name = name.tr('!?', 'EP')
|
|
|
|
end
|
|
|
|
|
|
|
|
base = "builtin_inline_#{name}_#{lineno}"
|
|
|
|
if inlines[base]
|
|
|
|
1000.times{|i|
|
|
|
|
name = "#{base}_#{i}"
|
|
|
|
return name unless inlines[name]
|
|
|
|
}
|
|
|
|
raise "too many functions in same line..."
|
|
|
|
else
|
|
|
|
base
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def collect_builtin base, iseq_ary, name, bs, inlines
|
|
|
|
case type = iseq_ary[9]
|
|
|
|
when :method
|
|
|
|
name = iseq_ary[5]
|
|
|
|
when :class
|
|
|
|
name = 'class'
|
|
|
|
else
|
|
|
|
end
|
|
|
|
|
2019-11-07 02:58:00 -05:00
|
|
|
code = iseq_ary[13]
|
2019-11-11 02:38:46 -05:00
|
|
|
params = iseq_ary[10]
|
|
|
|
prev_insn = nil
|
|
|
|
lineno = nil
|
2019-11-07 02:58:00 -05:00
|
|
|
|
|
|
|
code.each{|insn|
|
2019-11-11 02:38:46 -05:00
|
|
|
case insn
|
|
|
|
when Array
|
|
|
|
# ok
|
|
|
|
when Integer
|
|
|
|
lineno = insn
|
|
|
|
next
|
|
|
|
else
|
|
|
|
next
|
|
|
|
end
|
|
|
|
|
2019-11-07 02:58:00 -05:00
|
|
|
next unless Array === insn
|
|
|
|
case insn[0]
|
|
|
|
when :send
|
|
|
|
ci = insn[1]
|
|
|
|
if /\A__builtin_(.+)/ =~ ci[:mid]
|
2019-12-13 03:26:12 -05:00
|
|
|
cfunc_name = func_name = $1
|
2019-11-07 02:58:00 -05:00
|
|
|
argc = ci[:orig_argc]
|
|
|
|
|
2019-11-25 22:20:53 -05:00
|
|
|
if /(.+)\!\z/ =~ func_name
|
|
|
|
case $1
|
|
|
|
when 'cstmt'
|
|
|
|
text = inline_text argc, prev_insn
|
|
|
|
|
2019-12-13 03:26:12 -05:00
|
|
|
func_name = "_bi#{inlines.size}"
|
|
|
|
cfunc_name = make_cfunc_name(inlines, name, lineno)
|
|
|
|
inlines[cfunc_name] = [lineno, text, params, func_name]
|
2019-11-25 22:20:53 -05:00
|
|
|
argc -= 1
|
|
|
|
when 'cexpr', 'cconst'
|
|
|
|
text = inline_text argc, prev_insn
|
|
|
|
code = "return #{text};"
|
|
|
|
|
2019-12-13 03:26:12 -05:00
|
|
|
func_name = "_bi#{inlines.size}"
|
|
|
|
cfunc_name = make_cfunc_name(inlines, name, lineno)
|
|
|
|
|
2019-11-25 22:20:53 -05:00
|
|
|
params = [] if $1 == 'cconst'
|
2019-12-13 03:26:12 -05:00
|
|
|
inlines[cfunc_name] = [lineno, code, params, func_name]
|
2019-11-25 22:20:53 -05:00
|
|
|
argc -= 1
|
|
|
|
when 'cinit'
|
|
|
|
text = inline_text argc, prev_insn
|
|
|
|
func_name = nil
|
2019-12-13 03:26:12 -05:00
|
|
|
inlines[inlines.size] = [nil, [lineno, text, nil, nil]]
|
2019-11-25 22:20:53 -05:00
|
|
|
argc -= 1
|
|
|
|
end
|
2019-11-07 02:58:00 -05:00
|
|
|
end
|
2019-11-11 02:38:46 -05:00
|
|
|
|
|
|
|
if bs[func_name] &&
|
2019-12-13 03:26:12 -05:00
|
|
|
bs[func_name] != [argc, cfunc_name]
|
2019-11-11 02:38:46 -05:00
|
|
|
raise "same builtin function \"#{func_name}\", but different arity (was #{bs[func_name]} but #{argc})"
|
|
|
|
end
|
|
|
|
|
2019-12-13 03:26:12 -05:00
|
|
|
bs[func_name] = [argc, cfunc_name] if func_name
|
2019-11-07 02:58:00 -05:00
|
|
|
end
|
|
|
|
else
|
|
|
|
insn[1..-1].each{|op|
|
2019-11-07 22:58:17 -05:00
|
|
|
if op.is_a?(Array) && op[0] == "YARVInstructionSequence/SimpleDataFormat"
|
2019-12-13 03:26:12 -05:00
|
|
|
collect_builtin base, op, name, bs, inlines
|
2019-11-07 02:58:00 -05:00
|
|
|
end
|
|
|
|
}
|
|
|
|
end
|
2019-11-11 02:38:46 -05:00
|
|
|
prev_insn = insn
|
2019-11-07 02:58:00 -05:00
|
|
|
}
|
|
|
|
end
|
|
|
|
# ruby mk_builtin_loader.rb TARGET_FILE.rb
|
2019-11-08 02:24:24 -05:00
|
|
|
# #=> generate TARGET_FILE.rbinc
|
2019-11-07 02:58:00 -05:00
|
|
|
#
|
|
|
|
|
|
|
|
def mk_builtin_header file
|
|
|
|
base = File.basename(file, '.rb')
|
2019-11-26 05:08:56 -05:00
|
|
|
ofile = "#{file}inc"
|
2019-11-07 02:58:00 -05:00
|
|
|
|
2019-11-11 02:38:46 -05:00
|
|
|
# bs = { func_name => argc }
|
2019-12-13 03:26:12 -05:00
|
|
|
collect_builtin(base, RubyVM::InstructionSequence.compile_file(file, false).to_a, 'top', bs = {}, inlines = {})
|
2019-11-07 02:58:00 -05:00
|
|
|
|
2019-11-26 05:08:56 -05:00
|
|
|
begin
|
|
|
|
f = open(ofile, 'w')
|
|
|
|
rescue Errno::EACCESS
|
|
|
|
# Fall back to the current directory
|
|
|
|
f = open(File.basename(ofile), 'w')
|
|
|
|
end
|
|
|
|
begin
|
2019-11-08 02:37:42 -05:00
|
|
|
f.puts "// -*- c -*-"
|
2019-11-07 02:58:00 -05:00
|
|
|
f.puts "// DO NOT MODIFY THIS FILE DIRECTLY."
|
|
|
|
f.puts "// auto-generated file"
|
|
|
|
f.puts "// by #{__FILE__}"
|
|
|
|
f.puts "// with #{file}"
|
|
|
|
f.puts
|
2019-11-11 02:38:46 -05:00
|
|
|
lineno = 6
|
2019-11-25 22:20:53 -05:00
|
|
|
line_file = file.gsub('\\', '/')
|
2019-11-11 02:38:46 -05:00
|
|
|
|
2019-12-13 03:26:12 -05:00
|
|
|
inlines.each{|cfunc_name, (body_lineno, text, params, func_name)|
|
|
|
|
if String === cfunc_name
|
|
|
|
f.puts "static VALUE #{cfunc_name}(rb_execution_context_t *ec, const VALUE self) {"
|
2019-11-25 22:20:53 -05:00
|
|
|
lineno += 1
|
2019-11-11 02:38:46 -05:00
|
|
|
|
2019-11-25 22:20:53 -05:00
|
|
|
params.reverse_each.with_index{|param, i|
|
|
|
|
next unless Symbol === param
|
|
|
|
f.puts "MAYBE_UNUSED(const VALUE) #{param} = rb_vm_lvar(ec, #{-3 - i});"
|
|
|
|
lineno += 1
|
|
|
|
}
|
|
|
|
f.puts "#line #{body_lineno} \"#{line_file}\""
|
2019-11-11 02:38:46 -05:00
|
|
|
lineno += 1
|
|
|
|
|
2019-11-25 22:20:53 -05:00
|
|
|
f.puts text
|
|
|
|
lineno += text.count("\n") + 1
|
2019-11-11 02:38:46 -05:00
|
|
|
|
2019-11-25 22:20:53 -05:00
|
|
|
f.puts "#line #{lineno + 2} \"#{ofile}\"" # TODO: restore line number.
|
|
|
|
f.puts "}"
|
|
|
|
lineno += 2
|
|
|
|
else
|
|
|
|
# cinit!
|
|
|
|
f.puts "#line #{body_lineno} \"#{line_file}\""
|
|
|
|
lineno += 1
|
|
|
|
f.puts text
|
|
|
|
lineno += text.count("\n") + 1
|
|
|
|
f.puts "#line #{lineno + 2} \"#{ofile}\"" # TODO: restore line number.
|
|
|
|
lineno += 1
|
|
|
|
end
|
2019-11-11 02:38:46 -05:00
|
|
|
}
|
2019-11-07 02:58:00 -05:00
|
|
|
|
|
|
|
f.puts "static void load_#{base}(void)"
|
|
|
|
f.puts "{"
|
|
|
|
|
|
|
|
table = "#{base}_table"
|
|
|
|
f.puts " // table definition"
|
|
|
|
f.puts " static const struct rb_builtin_function #{table}[] = {"
|
2019-12-13 03:26:12 -05:00
|
|
|
bs.each.with_index{|(func, (argc, cfunc_name)), i|
|
|
|
|
f.puts " RB_BUILTIN_FUNCTION(#{i}, #{func}, #{cfunc_name}, #{argc}),"
|
2019-11-07 02:58:00 -05:00
|
|
|
}
|
2019-12-13 03:26:12 -05:00
|
|
|
f.puts " RB_BUILTIN_FUNCTION(-1, NULL, NULL, 0),"
|
2019-11-07 02:58:00 -05:00
|
|
|
f.puts " };"
|
|
|
|
|
|
|
|
f.puts
|
|
|
|
f.puts " // arity_check"
|
2019-11-08 01:13:24 -05:00
|
|
|
f.puts "COMPILER_WARNING_PUSH"
|
|
|
|
f.puts "#if GCC_VERSION_SINCE(5, 1, 0) || __clang__"
|
|
|
|
f.puts "COMPILER_WARNING_ERROR(-Wincompatible-pointer-types)"
|
|
|
|
f.puts "#endif"
|
2019-12-13 03:26:12 -05:00
|
|
|
bs.each{|func, (argc, cfunc_name)|
|
|
|
|
f.puts " if (0) rb_builtin_function_check_arity#{argc}(#{cfunc_name});"
|
2019-11-07 02:58:00 -05:00
|
|
|
}
|
2019-11-08 01:13:24 -05:00
|
|
|
f.puts "COMPILER_WARNING_POP"
|
|
|
|
|
2019-11-07 02:58:00 -05:00
|
|
|
|
|
|
|
f.puts
|
|
|
|
f.puts " // load"
|
2019-11-09 05:43:14 -05:00
|
|
|
f.puts " rb_load_with_builtin_functions(#{base.dump}, #{table});"
|
2019-11-07 02:58:00 -05:00
|
|
|
|
|
|
|
f.puts "}"
|
2019-11-26 05:08:56 -05:00
|
|
|
ensure
|
|
|
|
f.close
|
|
|
|
end
|
2019-11-07 02:58:00 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
ARGV.each{|file|
|
|
|
|
# feature.rb => load_feature.inc
|
2019-11-09 05:43:14 -05:00
|
|
|
mk_builtin_header file
|
2019-11-07 02:58:00 -05:00
|
|
|
}
|