summaryrefslogtreecommitdiff
path: root/zjit/bindgen/src/main.rs
diff options
context:
space:
mode:
Diffstat (limited to 'zjit/bindgen/src/main.rs')
-rw-r--r--zjit/bindgen/src/main.rs470
1 files changed, 470 insertions, 0 deletions
diff --git a/zjit/bindgen/src/main.rs b/zjit/bindgen/src/main.rs
new file mode 100644
index 0000000000..d015f975de
--- /dev/null
+++ b/zjit/bindgen/src/main.rs
@@ -0,0 +1,470 @@
+//! See https://docs.rs/bindgen/0.59.2/bindgen/struct.Builder.html
+//! This is the binding generation tool that the ZJIT cruby module talks about.
+//! More docs later once we have more experience with this, for now, check
+//! the output to make sure it looks reasonable and allowlist things you want
+//! to use in Rust.
+
+use std::env;
+use std::path::PathBuf;
+
+const SRC_ROOT_ENV: &str = "ZJIT_SRC_ROOT_PATH";
+const JIT_NAME: &str = "BINDGEN_JIT_NAME";
+
+fn main() {
+ // Path to repo is a required input for supporting running `configure`
+ // in a directory away from the code.
+ let src_root = env::var(SRC_ROOT_ENV).expect(
+ format!(
+ r#"The "{}" env var must be a path to the root of the Ruby repo"#,
+ SRC_ROOT_ENV
+ )
+ .as_ref(),
+ );
+ let src_root = PathBuf::from(src_root);
+
+ let jit_name = env::var(JIT_NAME).expect(JIT_NAME);
+ let c_file = format!("{}.c", jit_name);
+
+ assert!(
+ src_root.is_dir(),
+ "{} must be set to a path to a directory",
+ SRC_ROOT_ENV
+ );
+
+ // We want Bindgen warnings printed to console
+ env_logger::init();
+
+ // Remove this flag so rust-bindgen generates bindings
+ // that are internal functions not public in libruby
+ let filtered_clang_args = env::args().filter(|arg| arg != "-fvisibility=hidden");
+
+ let bindings = bindgen::builder()
+ .clang_args(filtered_clang_args)
+ .header("encindex.h")
+ .header("internal.h")
+ .header("internal/object.h")
+ .header("internal/re.h")
+ .header("include/ruby/ruby.h")
+ .header("shape.h")
+ .header("vm_core.h")
+ .header("vm_callinfo.h")
+
+ // Our C file for glue code
+ .header(src_root.join(c_file).to_str().unwrap())
+ .header(src_root.join("jit.c").to_str().unwrap())
+
+ // Don't want to copy over C comment
+ .generate_comments(false)
+
+ // Makes the output more compact
+ .merge_extern_blocks(true)
+
+ // Don't want layout tests as they are platform dependent
+ .layout_tests(false)
+
+ // Block for stability since output is different on Darwin and Linux
+ .blocklist_type("size_t")
+ .blocklist_type("fpos_t")
+
+ // Import YARV bytecode instruction constants
+ .allowlist_type("ruby_vminsn_type")
+
+ // JITFrame struct defined in zjit.h, imported into Rust via bindgen
+ .allowlist_type("zjit_jit_frame")
+
+ .allowlist_type("ruby_special_consts")
+ .allowlist_function("rb_utf8_str_new")
+ .allowlist_function("rb_str_buf_append")
+ .allowlist_function("rb_str_dup")
+ .allowlist_function("rb_str_getbyte")
+ .allowlist_type("ruby_preserved_encindex")
+ .allowlist_function("rb_class2name")
+
+ // This struct is public to Ruby C extensions
+ .allowlist_type("RBasic")
+
+ .allowlist_type("ruby_rstring_flags")
+ .allowlist_type("rbimpl_typeddata_flags")
+
+ // This function prints info about a value and is useful for debugging
+ .allowlist_function("rb_raw_obj_info")
+
+ .allowlist_function("ruby_init")
+ .allowlist_function("ruby_init_stack")
+ .allowlist_function("ruby_options")
+ .allowlist_function("ruby_executable_node")
+ .allowlist_function("rb_funcallv")
+ .allowlist_function("rb_protect")
+ .allowlist_function("rb_zjit_profile_disable")
+
+ // For crashing
+ .allowlist_function("rb_bug")
+
+ .allowlist_function("rb_obj_shape_id")
+ .allowlist_function("rb_shape_id_offset")
+ .allowlist_function("rb_shape_get_iv_index")
+ .allowlist_function("rb_shape_transition_add_ivar_no_warnings")
+ .allowlist_function("rb_jit_shape_capacity")
+ .allowlist_var("rb_invalid_shape_id")
+ .allowlist_type("shape_id_fl_type")
+ .allowlist_var("VM_KW_SPECIFIED_BITS_MAX")
+ .allowlist_var("SHAPE_ID_NUM_BITS")
+ .allowlist_function("rb_obj_is_kind_of")
+ .allowlist_function("rb_obj_frozen_p")
+ .allowlist_function("rb_class_inherited_p")
+ .allowlist_function("rb_class_real")
+ .allowlist_type("ruby_encoding_consts")
+ .allowlist_function("rb_hash_new")
+ .allowlist_function("rb_hash_new_with_size")
+ .allowlist_function("rb_hash_resurrect")
+ .allowlist_function("rb_hash_stlike_foreach")
+ .allowlist_function("rb_to_hash_type")
+ .allowlist_type("st_retval")
+ .allowlist_function("rb_hash_aset")
+ .allowlist_function("rb_hash_aref")
+ .allowlist_function("rb_hash_bulk_insert")
+ .allowlist_function("rb_hash_new_with_bulk_insert")
+ .allowlist_function("rb_hash_stlike_lookup")
+ .allowlist_function("rb_ary_new_capa")
+ .allowlist_function("rb_ary_store")
+ .allowlist_function("rb_ary_resurrect")
+ .allowlist_function("rb_ary_cat")
+ .allowlist_function("rb_ary_concat")
+ .allowlist_function("rb_ary_clear")
+ .allowlist_function("rb_ary_dup")
+ .allowlist_function("rb_ary_push")
+ .allowlist_function("rb_ary_pop")
+ .allowlist_function("rb_ary_unshift_m")
+ .allowlist_function("rb_ec_ary_new_from_values")
+ .allowlist_function("rb_ary_tmp_new_from_values")
+ .allowlist_function("rb_ary_entry")
+ .allowlist_function("rb_class_attached_object")
+ .allowlist_function("rb_singleton_class")
+ .allowlist_function("rb_define_class")
+ .allowlist_function("rb_class_get_superclass")
+ .allowlist_function("rb_gc_disable")
+ .allowlist_function("rb_gc_enable")
+ .allowlist_function("rb_gc_mark")
+ .allowlist_function("rb_gc_mark_movable")
+ .allowlist_function("rb_gc_location")
+ .allowlist_function("rb_gc_writebarrier")
+ .allowlist_function("rb_gc_writebarrier_remember")
+ .allowlist_function("rb_gc_register_mark_object")
+
+ // VALUE variables for Ruby class objects
+ .allowlist_var("rb_cBasicObject")
+ .allowlist_var("rb_cObject")
+ .allowlist_var("rb_cModule")
+ .allowlist_var("rb_cNilClass")
+ .allowlist_var("rb_cTrueClass")
+ .allowlist_var("rb_cFalseClass")
+ .allowlist_var("rb_cInteger")
+ .allowlist_var("rb_cIO")
+ .allowlist_var("rb_cSymbol")
+ .allowlist_var("rb_cFloat")
+ .allowlist_var("rb_cNumeric")
+ .allowlist_var("rb_cRange")
+ .allowlist_var("rb_cString")
+ .allowlist_var("rb_cThread")
+ .allowlist_var("rb_cArray")
+ .allowlist_var("rb_cHash")
+ .allowlist_var("rb_cSet")
+ .allowlist_var("rb_cClass")
+ .allowlist_var("rb_cRegexp")
+ .allowlist_var("rb_cISeq")
+ .allowlist_var("rb_cRubyVM")
+ .allowlist_function("rb_const_get")
+
+ .allowlist_type("ruby_fl_type")
+ .allowlist_type("ruby_fl_ushift")
+ .allowlist_type("ruby_robject_flags")
+ .allowlist_type("ruby_rarray_flags")
+ .allowlist_type("ruby_rarray_consts")
+ .allowlist_type("ruby_rmodule_flags")
+ .allowlist_var("rb_mKernel")
+ .allowlist_type("vm_call_flag_bits")
+ .allowlist_type("rb_call_data")
+ .blocklist_type("rb_callcache.*") // Not used yet - opaque to make it easy to import rb_call_data
+ .opaque_type("rb_callcache.*")
+ .allowlist_type("rb_callinfo")
+ .allowlist_var("VM_ENV_DATA_INDEX_ME_CREF")
+ .allowlist_var("rb_block_param_proxy")
+ .allowlist_function("rb_range_new")
+ .allowlist_function("rb_intern")
+ .allowlist_function("rb_intern2")
+ .allowlist_function("rb_id2sym")
+ .allowlist_function("rb_sym2id")
+ .allowlist_function("rb_str_intern")
+ .allowlist_function("rb_id2str")
+ .allowlist_function("rb_sym2str")
+ .allowlist_function("rb_fix_aref")
+ .allowlist_function("rb_float_plus")
+ .allowlist_function("rb_float_minus")
+ .allowlist_function("rb_float_mul")
+ .allowlist_function("rb_float_div")
+ .allowlist_function("rb_flo_to_i")
+ .allowlist_type("ruby_rstring_private_flags")
+ .allowlist_function("rb_ec_str_resurrect")
+ .allowlist_function("rb_str_concat_literals")
+ .allowlist_function("rb_obj_as_string_result")
+ .allowlist_function("rb_str_byte_substr")
+ .allowlist_function("rb_str_substr_two_fixnums")
+ .allowlist_function("rb_backref_get")
+ .allowlist_function("rb_reg_last_match")
+ .allowlist_function("rb_reg_match_pre")
+ .allowlist_function("rb_reg_match_post")
+ .allowlist_function("rb_reg_match_last")
+ .allowlist_function("rb_reg_nth_match")
+ .allowlist_function("rb_reg_new_from_values")
+ .allowlist_var("ARG_ENCODING_FIXED")
+ .allowlist_var("ARG_ENCODING_NONE")
+ .allowlist_var("ONIG_OPTION_IGNORECASE")
+ .allowlist_var("ONIG_OPTION_EXTEND")
+ .allowlist_var("ONIG_OPTION_MULTILINE")
+
+ // `ruby_value_type` is a C enum and this stops it from
+ // prefixing all the members with the name of the type
+ .prepend_enum_name(false)
+ .translate_enum_integer_types(true) // so we get fixed width Rust types for members
+ .allowlist_type("ruby_value_type") // really old C extension API
+
+ .allowlist_type("ruby_rhash_flags") // really old C extension API
+ .allowlist_type("rb_method_visibility_t")
+ .allowlist_type("rb_method_type_t")
+ .allowlist_type("method_optimized_type")
+ .allowlist_type("rb_callable_method_entry_t")
+ .allowlist_type("rb_callable_method_entry_struct")
+ .allowlist_function("rb_method_entry_at")
+ .allowlist_type("rb_method_entry_t")
+ .blocklist_type("rb_method_cfunc_t")
+ .blocklist_type("rb_method_definition_.*") // Large struct with a bitfield and union of many types - don't import (yet?)
+ .opaque_type("rb_method_definition_.*")
+ .allowlist_function("rb_float_new")
+ .allowlist_var("rb_mRubyVMFrozenCore")
+ .allowlist_var("VM_BLOCK_HANDLER_NONE")
+ .allowlist_type("vm_frame_env_flags")
+ .allowlist_type("rb_seq_param_keyword_struct")
+ .allowlist_type("rb_callinfo_kwarg")
+ .allowlist_type("ruby_basic_operators")
+ .allowlist_var(".*_REDEFINED_OP_FLAG")
+ .allowlist_type("rb_num_t")
+ .allowlist_function("rb_callable_method_entry")
+ .allowlist_function("rb_callable_method_entry_or_negative")
+ .allowlist_function("rb_vm_frame_method_entry")
+ .allowlist_type("IVC") // pointer to iseq_inline_iv_cache_entry
+ .allowlist_type("IC") // pointer to iseq_inline_constant_cache
+ .allowlist_type("iseq_inline_constant_cache_entry")
+ .blocklist_type("rb_cref_t") // don't need this directly, opaqued to allow IC import
+ .opaque_type("rb_cref_t")
+ .allowlist_type("iseq_inline_iv_cache_entry")
+ .allowlist_type("ICVARC") // pointer to iseq_inline_cvar_cache_entry
+ .allowlist_type("iseq_inline_cvar_cache_entry")
+ .blocklist_type("rb_execution_context_.*") // Large struct with various-type fields and an ifdef, so we don't import
+ .opaque_type("rb_execution_context_.*")
+ .allowlist_type("rb_control_frame_struct")
+ .allowlist_function("rb_vm_bh_to_procval")
+ .allowlist_function("rb_vm_env_write")
+ .allowlist_function("rb_vm_ep_local_ep")
+ .allowlist_type("vm_special_object_type")
+ .allowlist_var("VM_ENV_DATA_INDEX_SPECVAL")
+ .allowlist_var("VM_ENV_DATA_INDEX_FLAGS")
+ .allowlist_var("VM_ENV_DATA_SIZE")
+ .allowlist_function("rb_iseq_path")
+ .allowlist_type("rb_builtin_attr")
+ .allowlist_type("ruby_tag_type")
+ .allowlist_type("ruby_vm_throw_flags")
+ .allowlist_type("vm_check_match_type")
+ .allowlist_type("vm_opt_newarray_send_type")
+ .allowlist_type("rb_iseq_type")
+ .allowlist_type("rb_event_flag_t")
+ .allowlist_function("rb_object_shape_count")
+ .allowlist_function("rb_iseq_(get|set)_zjit_payload")
+ .allowlist_function("rb_iseq_pc_at_idx")
+ .allowlist_function("rb_iseq_opcode_at_pc")
+ .allowlist_function("rb_iseq_bare_opcode_at_pc")
+ .allowlist_function("rb_jit_reserve_addr_space")
+ .allowlist_function("rb_jit_mark_writable")
+ .allowlist_function("rb_jit_mark_executable")
+ .allowlist_function("rb_jit_mark_unused")
+ .allowlist_function("rb_jit_get_page_size")
+ .allowlist_function("rb_jit_array_len")
+ .allowlist_function("rb_jit_fix_div_fix")
+ .allowlist_function("rb_jit_iseq_builtin_attrs")
+ .allowlist_function("rb_jit_str_concat_codepoint")
+ .allowlist_function("rb_zjit_iseq_inspect")
+ .allowlist_function("rb_zjit_iseq_insn_set")
+ .allowlist_function("rb_zjit_local_id")
+ .allowlist_function("rb_set_cfp_(pc|sp)")
+ .allowlist_function("rb_c_method_tracing_currently_enabled")
+ .allowlist_function("rb_zjit_method_tracing_currently_enabled")
+ .allowlist_function("rb_zjit_iseq_tracing_currently_enabled")
+ .allowlist_function("rb_full_cfunc_return")
+ .allowlist_function("rb_assert_(iseq|cme)_handle")
+ .allowlist_function("rb_IMEMO_TYPE_P")
+ .allowlist_function("rb_RSTRING_PTR")
+ .allowlist_function("rb_RSTRING_LEN")
+ .allowlist_function("rb_ENCODING_GET")
+ .allowlist_function("rb_profile_frame_full_label")
+ .allowlist_function("rb_profile_frame_absolute_path")
+ .allowlist_function("rb_profile_frame_path")
+ .allowlist_function("rb_optimized_call")
+ .allowlist_function("rb_jit_icache_invalidate")
+ .allowlist_function("rb_zjit_print_exception")
+ .allowlist_function("rb_zjit_singleton_class_p")
+ .allowlist_function("rb_zjit_defined_ivar")
+ .allowlist_function("rb_zjit_insn_leaf")
+ .allowlist_type("jit_bindgen_constants")
+ .allowlist_type("zjit_struct_offsets")
+ .allowlist_var("ZJIT_JIT_RETURN_C_FRAME")
+ .allowlist_function("rb_assert_holding_vm_lock")
+ .allowlist_function("rb_jit_shape_complex_p")
+ .allowlist_function("rb_jit_multi_ractor_p")
+ .allowlist_function("rb_jit_class_fields_embedded_p")
+ .allowlist_function("rb_jit_data_fields_embedded_p")
+ .allowlist_function("rb_jit_vm_lock_then_barrier")
+ .allowlist_function("rb_jit_vm_unlock")
+ .allowlist_function("rb_jit_for_each_iseq")
+ .allowlist_function("rb_iseq_reset_jit_func")
+ .allowlist_function("rb_vm_barrier")
+
+ // Not sure why it's picking these up, but don't.
+ .blocklist_type("FILE")
+ .blocklist_type("_IO_.*")
+
+ .allowlist_function("rb_vm_insn_decode")
+ .allowlist_function("rb_jit_cont_each_iseq")
+ .allowlist_function("rb_vm_insn_addr2opcode")
+ .allowlist_function("rb_iseqw_to_iseq")
+ .allowlist_function("rb_iseq_label")
+ .allowlist_function("rb_iseq_line_no")
+ .allowlist_function("rb_iseq_defined_string")
+ .allowlist_type("defined_type")
+ .allowlist_type("rb_builtin_function.*")
+ .allowlist_function("rb_gvar_(get|set)")
+ .allowlist_function("rb_ensure_iv_list_size")
+ .allowlist_function("rb_attr_get")
+ .allowlist_function("rb_ivar_defined")
+ .allowlist_function("rb_ivar_get")
+ .allowlist_function("rb_ivar_get_at_no_ractor_check")
+ .allowlist_function("rb_ivar_set")
+ .allowlist_function("rb_mod_name")
+ .allowlist_var("rb_vm_insn_count")
+ .allowlist_function("rb_get_alloc_func")
+ .allowlist_function("rb_class_allocate_instance")
+ .allowlist_function("rb_obj_equal")
+ .allowlist_function("rb_class_new_instance_pass_kw")
+ .allowlist_function("rb_obj_alloc")
+ .allowlist_function("rb_obj_info")
+ .allowlist_function("rb_obj_frozen_p")
+ // From include/ruby/debug.h
+ .allowlist_function("rb_profile_frames")
+ .allowlist_function("ruby_xfree")
+ .allowlist_function("rb_profile_frames")
+
+ // Functions used for code generation
+ .allowlist_function("rb_insn_name")
+ .allowlist_function("rb_insn_len")
+ .allowlist_function("rb_yarv_class_of")
+ .allowlist_function("rb_zjit_class_initialized_p")
+ .allowlist_function("rb_zjit_class_has_default_allocator")
+ .allowlist_function("rb_zjit_class_get_alloc_func")
+ .allowlist_function("rb_get_ec_cfp")
+ .allowlist_function("rb_get_cfp_iseq")
+ .allowlist_function("rb_get_cfp_pc")
+ .allowlist_function("rb_get_cfp_sp")
+ .allowlist_function("rb_get_cfp_self")
+ .allowlist_function("rb_get_cfp_ep")
+ .allowlist_function("rb_get_cfp_ep_level")
+ .allowlist_function("rb_get_cme_def_type")
+ .allowlist_function("rb_zjit_constcache_shareable")
+ .allowlist_function("rb_zjit_vm_search_method")
+ .allowlist_function("rb_zjit_cme_is_cfunc")
+ .allowlist_function("rb_get_cme_def_body_attr_id")
+ .allowlist_function("rb_get_symbol_id")
+ .allowlist_function("rb_get_cme_def_body_optimized_type")
+ .allowlist_function("rb_get_cme_def_body_optimized_index")
+ .allowlist_function("rb_get_cme_def_body_cfunc")
+ .allowlist_function("rb_get_def_method_serial")
+ .allowlist_function("rb_get_def_original_id")
+ .allowlist_function("rb_get_mct_argc")
+ .allowlist_function("rb_get_mct_func")
+ .allowlist_function("rb_get_def_iseq_ptr")
+ .allowlist_function("rb_get_def_bmethod_proc")
+ .allowlist_function("rb_jit_get_proc_ptr")
+ .allowlist_function("rb_iseq_encoded_size")
+ .allowlist_function("rb_get_iseq_body_total_calls")
+ .allowlist_function("rb_get_iseq_body_local_iseq")
+ .allowlist_function("rb_get_iseq_body_parent_iseq")
+ .allowlist_function("rb_get_iseq_body_iseq_encoded")
+ .allowlist_function("rb_get_iseq_body_stack_max")
+ .allowlist_function("rb_get_iseq_body_type")
+ .allowlist_function("rb_get_iseq_flags_has_lead")
+ .allowlist_function("rb_get_iseq_flags_has_opt")
+ .allowlist_function("rb_get_iseq_flags_has_kw")
+ .allowlist_function("rb_get_iseq_flags_has_rest")
+ .allowlist_function("rb_get_iseq_flags_has_post")
+ .allowlist_function("rb_get_iseq_flags_has_kwrest")
+ .allowlist_function("rb_get_iseq_flags_anon_kwrest")
+ .allowlist_function("rb_get_iseq_flags_has_block")
+ .allowlist_function("rb_get_iseq_flags_ambiguous_param0")
+ .allowlist_function("rb_get_iseq_flags_accepts_no_kwarg")
+ .allowlist_function("rb_get_iseq_flags_ruby2_keywords")
+ .allowlist_function("rb_get_iseq_flags_forwardable")
+ .allowlist_function("rb_get_iseq_body_local_table_size")
+ .allowlist_function("rb_get_iseq_body_param_keyword")
+ .allowlist_function("rb_get_iseq_body_param_size")
+ .allowlist_function("rb_get_iseq_body_param_lead_num")
+ .allowlist_function("rb_get_iseq_body_param_opt_num")
+ .allowlist_function("rb_get_iseq_body_param_opt_table")
+ .allowlist_function("rb_get_cikw_keyword_len")
+ .allowlist_function("rb_get_cikw_keywords_idx")
+ .allowlist_function("rb_get_call_data_ci")
+ .allowlist_function("rb_yarv_str_eql_internal")
+ .allowlist_function("rb_str_neq_internal")
+ .allowlist_function("rb_yarv_ary_entry_internal")
+ .allowlist_function("rb_vm_get_untagged_block_handler")
+ .allowlist_function("rb_vm_untag_block_handler")
+ .allowlist_function("rb_FL_TEST")
+ .allowlist_function("rb_FL_TEST_RAW")
+ .allowlist_function("rb_RB_TYPE_P")
+ .allowlist_function("rb_BASIC_OP_UNREDEFINED_P")
+ .allowlist_function("rb_vm_ci_argc")
+ .allowlist_function("rb_vm_ci_mid")
+ .allowlist_function("rb_vm_ci_flag")
+ .allowlist_function("rb_vm_ci_kwarg")
+ .allowlist_function("rb_METHOD_ENTRY_VISI")
+ .allowlist_function("rb_RCLASS_ORIGIN")
+ .allowlist_function("rb_method_basic_definition_p")
+ .allowlist_function("rb_obj_class")
+ .allowlist_function("rb_obj_is_proc")
+ .allowlist_function("rb_vm_base_ptr")
+ .allowlist_function("rb_ec_stack_check")
+ .allowlist_function("rb_vm_top_self")
+ .allowlist_function("rb_const_lookup")
+
+ // We define these manually, don't import them
+ .blocklist_type("VALUE")
+ .blocklist_type("ID")
+ .blocklist_type("rb_iseq_constant_body")
+
+ // Avoid binding to stuff we don't use
+ .blocklist_item("rb_thread_struct.*")
+ .opaque_type("rb_thread_struct.*")
+ .blocklist_item("iseq_inline_storage_entry_.*")
+ .opaque_type("iseq_inline_storage_entry")
+ .opaque_type("iseq_compile_data")
+
+ // Finish the builder and generate the bindings.
+ .generate()
+ // Unwrap the Result and panic on failure.
+ .expect("Unable to generate bindings");
+
+ let mut out_path: PathBuf = src_root;
+ out_path.push(jit_name);
+ out_path.push("src");
+ out_path.push("cruby_bindings.inc.rs");
+
+ bindings
+ .write_to_file(out_path)
+ .expect("Couldn't write bindings!");
+}