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.is_a?(Array) && op[0] == "YARVInstructionSequence/SimpleDataFormat" collect_builtin op, bs end } end } end # ruby mk_builtin_loader.rb TARGET_FILE.rb # #=> generate TARGET_FILE.rbinc # def mk_builtin_header file base = File.basename(file, '.rb') ofile = "#{base}.rbinc" collect_builtin(RubyVM::InstructionSequence.compile_file(file, false).to_a, bs = {}) open(ofile, 'w'){|f| f.puts "// -*- c -*-" 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" 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" bs.each{|func, argc| f.puts " if (0) rb_builtin_function_check_arity#{argc}(#{func});" } f.puts "COMPILER_WARNING_POP" f.puts f.puts " // load" f.puts " rb_load_with_builtin_functions(#{base.dump}, #{table});" f.puts "}" } end ARGV.each{|file| # feature.rb => load_feature.inc mk_builtin_header file }