2019-11-07 02:58:00 -05:00
|
|
|
|
|
|
|
def collect_builtin iseq_ary, bs
|
|
|
|
code = iseq_ary[13]
|
|
|
|
|
|
|
|
code.each{|insn|
|
|
|
|
next unless Array === insn
|
|
|
|
case insn[0]
|
|
|
|
when :send
|
|
|
|
ci = insn[1]
|
|
|
|
if /\A__builtin_(.+)/ =~ ci[:mid]
|
|
|
|
func_name = $1
|
|
|
|
argc = ci[:orig_argc]
|
|
|
|
|
|
|
|
if bs[func_name] && bs[func_name] != argc
|
|
|
|
raise
|
|
|
|
end
|
|
|
|
bs[func_name] = argc
|
|
|
|
end
|
|
|
|
else
|
|
|
|
insn[1..-1].each{|op|
|
|
|
|
if op[0] == "YARVInstructionSequence/SimpleDataFormat"
|
|
|
|
collect_builtin op, bs
|
|
|
|
end
|
|
|
|
}
|
|
|
|
end
|
|
|
|
}
|
|
|
|
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-08 02:24:24 -05:00
|
|
|
ofile = "#{base}.rbinc"
|
2019-11-07 02:58:00 -05:00
|
|
|
|
|
|
|
collect_builtin(RubyVM::InstructionSequence.compile_file(file, false).to_a, bs = {})
|
|
|
|
|
|
|
|
open(ofile, 'w'){|f|
|
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
|
|
|
|
|
|
|
|
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}[] = {"
|
|
|
|
bs.each.with_index{|(func, argc), i|
|
|
|
|
f.puts " RB_BUILTIN_FUNCTION(#{i}, #{func}, #{argc}),"
|
|
|
|
}
|
|
|
|
f.puts " RB_BUILTIN_FUNCTION(-1, NULL, 0),"
|
|
|
|
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-11-07 02:58:00 -05:00
|
|
|
bs.each{|func, argc|
|
|
|
|
f.puts " if (0) rb_builtin_function_check_arity#{argc}(#{func});"
|
|
|
|
}
|
2019-11-08 01:13:24 -05:00
|
|
|
f.puts "COMPILER_WARNING_POP"
|
|
|
|
|
2019-11-07 02:58:00 -05:00
|
|
|
|
|
|
|
path = File.expand_path(file)
|
|
|
|
f.puts
|
|
|
|
f.puts " // load"
|
|
|
|
f.puts " rb_load_with_builtin_functions(\"#{base}\", \"#{file}\", #{table});"
|
|
|
|
|
|
|
|
f.puts "}"
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
ARGV.each{|file|
|
|
|
|
# feature.rb => load_feature.inc
|
|
|
|
path = File.expand_path(file)
|
|
|
|
mk_builtin_header path
|
|
|
|
}
|