Skip to content

Commit

Permalink
Required virtual methods should be required at compile-time
Browse files Browse the repository at this point in the history
  • Loading branch information
TitanNano committed Aug 9, 2024
1 parent 128c7ce commit 9d39621
Show file tree
Hide file tree
Showing 17 changed files with 511 additions and 95 deletions.
1 change: 1 addition & 0 deletions godot-codegen/src/generator/builtins.rs
Original file line number Diff line number Diff line change
Expand Up @@ -250,6 +250,7 @@ fn make_builtin_method_definition(
receiver,
varcall_invocation,
ptrcall_invocation,
is_virtual_required: false,
},
safety_doc,
&TokenStream::new(),
Expand Down
1 change: 1 addition & 0 deletions godot-codegen/src/generator/classes.rs
Original file line number Diff line number Diff line change
Expand Up @@ -535,6 +535,7 @@ fn make_class_method_definition(
receiver,
varcall_invocation,
ptrcall_invocation,
is_virtual_required: false,
},
None,
cfg_attributes,
Expand Down
10 changes: 7 additions & 3 deletions godot-codegen/src/generator/functions_common.rs
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ pub struct FnCode {
pub receiver: FnReceiver,
pub varcall_invocation: TokenStream,
pub ptrcall_invocation: TokenStream,
pub is_virtual_required: bool,
}

pub struct FnDefinition {
Expand Down Expand Up @@ -150,6 +151,11 @@ pub fn make_function_definition(
};

let return_decl = &sig.return_value().decl;
let fn_body = if code.is_virtual_required {
quote! { ; }
} else {
quote! { { unimplemented!() } }
};

let receiver_param = &code.receiver.param;
let primary_function = if sig.is_virtual() {
Expand All @@ -160,9 +166,7 @@ pub fn make_function_definition(
#maybe_unsafe fn #primary_fn_name(
#receiver_param
#( #params, )*
) #return_decl {
unimplemented!()
}
) #return_decl #fn_body
}
} else if sig.is_vararg() {
// Varargs (usually varcall, but not necessarily -- utilities use ptrcall)
Expand Down
1 change: 1 addition & 0 deletions godot-codegen/src/generator/utility_functions.rs
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,7 @@ pub(crate) fn make_utility_function_definition(function: &UtilityFunction) -> To
receiver: FnReceiver::global_function(),
varcall_invocation,
ptrcall_invocation,
is_virtual_required: false,
},
None,
&TokenStream::new(),
Expand Down
1 change: 1 addition & 0 deletions godot-codegen/src/generator/virtual_traits.rs
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,7 @@ fn make_virtual_method(method: &ClassMethod) -> Option<TokenStream> {
// make_return() requests following args, but they are not used for virtual methods. We can provide empty streams.
varcall_invocation: TokenStream::new(),
ptrcall_invocation: TokenStream::new(),
is_virtual_required: method.is_virtual_required(),
},
None,
&TokenStream::new(),
Expand Down
5 changes: 5 additions & 0 deletions godot-codegen/src/models/domain.rs
Original file line number Diff line number Diff line change
Expand Up @@ -280,6 +280,7 @@ pub struct FunctionCommon {
pub return_value: FnReturn,
pub is_vararg: bool,
pub is_private: bool,
pub is_virtual_required: bool,
pub direction: FnDirection,
}

Expand Down Expand Up @@ -314,6 +315,10 @@ pub trait Function: fmt::Display {
fn direction(&self) -> FnDirection {
self.common().direction
}

fn is_virtual_required(&self) -> bool {
self.common().is_virtual_required
}
}

// ----------------------------------------------------------------------------------------------------------------------------------------------
Expand Down
6 changes: 6 additions & 0 deletions godot-codegen/src/models/domain_mapping.rs
Original file line number Diff line number Diff line change
Expand Up @@ -358,6 +358,7 @@ impl BuiltinMethod {
return_value: FnReturn::new(&return_value, ctx),
is_vararg: method.is_vararg,
is_private: special_cases::is_method_private(builtin_name, &method.name),
is_virtual_required: false,
direction: FnDirection::Outbound {
hash: method.hash.expect("hash absent for builtin method"),
},
Expand Down Expand Up @@ -463,6 +464,10 @@ impl ClassMethod {
return_value: FnReturn::new(&method.return_value, ctx),
is_vararg: method.is_vararg,
is_private,
is_virtual_required: special_cases::is_virtual_method_required(
&class_name.rust_ty.to_string(),
rust_method_name,
),
direction,
},
qualifier,
Expand Down Expand Up @@ -511,6 +516,7 @@ impl UtilityFunction {
return_value: FnReturn::new(&return_value, ctx),
is_vararg: function.is_vararg,
is_private: false,
is_virtual_required: false,
direction: FnDirection::Outbound {
hash: function.hash,
},
Expand Down
184 changes: 184 additions & 0 deletions godot-codegen/src/special_cases/special_cases.rs
Original file line number Diff line number Diff line change
Expand Up @@ -336,6 +336,190 @@ pub fn get_interface_extra_docs(trait_name: &str) -> Option<&'static str> {
}
}

pub fn is_virtual_method_required(class_name: &str, method: &str) -> bool {
match (class_name, method) {
("ScriptLanguageExtension", _) => method != "get_doc_comment_delimiters",

("ScriptExtension", "editor_can_reload_from_file")
| ("ScriptExtension", "can_instantiate")
| ("ScriptExtension", "get_base_script")
| ("ScriptExtension", "get_global_name")
| ("ScriptExtension", "inherits_script")
| ("ScriptExtension", "get_instance_base_type")
| ("ScriptExtension", "instance_create")
| ("ScriptExtension", "placeholder_instance_create")
| ("ScriptExtension", "instance_has")
| ("ScriptExtension", "has_source_code")
| ("ScriptExtension", "get_source_code")
| ("ScriptExtension", "set_source_code")
| ("ScriptExtension", "reload")
| ("ScriptExtension", "get_documentation")
| ("ScriptExtension", "has_method")
| ("ScriptExtension", "has_static_method")
| ("ScriptExtension", "get_method_info")
| ("ScriptExtension", "is_tool")
| ("ScriptExtension", "is_valid")
| ("ScriptExtension", "get_language")
| ("ScriptExtension", "has_script_signal")
| ("ScriptExtension", "get_script_signal_list")
| ("ScriptExtension", "has_property_default_value")
| ("ScriptExtension", "get_property_default_value")
| ("ScriptExtension", "update_exports")
| ("ScriptExtension", "get_script_method_list")
| ("ScriptExtension", "get_script_property_list")
| ("ScriptExtension", "get_member_line")
| ("ScriptExtension", "get_constants")
| ("ScriptExtension", "get_members")
| ("ScriptExtension", "is_placeholder_fallback_enabled")
| ("ScriptExtension", "get_rpc_config")
| ("EditorExportPlugin", "customize_resource")
| ("EditorExportPlugin", "customize_scene")
| ("EditorExportPlugin", "get_customization_configuration_hash")
| ("EditorExportPlugin", "get_name")
| ("EditorVcsInterface", _)
| ("MovieWriter", _)
| ("TextServerExtension", "has_feature")
| ("TextServerExtension", "get_name")
| ("TextServerExtension", "get_features")
| ("TextServerExtension", "free_rid")
| ("TextServerExtension", "has")
| ("TextServerExtension", "create_font")
| ("TextServerExtension", "font_set_fixed_size")
| ("TextServerExtension", "font_get_fixed_size")
| ("TextServerExtension", "font_set_fixed_size_scale_mode")
| ("TextServerExtension", "font_get_fixed_size_scale_mode")
| ("TextServerExtension", "font_get_size_cache_list")
| ("TextServerExtension", "font_clear_size_cache")
| ("TextServerExtension", "font_remove_size_cache")
| ("TextServerExtension", "font_set_ascent")
| ("TextServerExtension", "font_get_ascent")
| ("TextServerExtension", "font_set_descent")
| ("TextServerExtension", "font_get_descent")
| ("TextServerExtension", "font_set_underline_position")
| ("TextServerExtension", "font_get_underline_position")
| ("TextServerExtension", "font_set_underline_thickness")
| ("TextServerExtension", "font_get_underline_thickness")
| ("TextServerExtension", "font_set_scale")
| ("TextServerExtension", "font_get_scale")
| ("TextServerExtension", "font_get_texture_count")
| ("TextServerExtension", "font_clear_textures")
| ("TextServerExtension", "font_remove_texture")
| ("TextServerExtension", "font_set_texture_image")
| ("TextServerExtension", "font_get_texture_image")
| ("TextServerExtension", "font_get_glyph_list")
| ("TextServerExtension", "font_clear_glyphs")
| ("TextServerExtension", "font_remove_glyph")
| ("TextServerExtension", "font_get_glyph_advance")
| ("TextServerExtension", "font_set_glyph_advance")
| ("TextServerExtension", "font_get_glyph_offset")
| ("TextServerExtension", "font_set_glyph_offset")
| ("TextServerExtension", "font_get_glyph_size")
| ("TextServerExtension", "font_set_glyph_size")
| ("TextServerExtension", "font_get_glyph_uv_rect")
| ("TextServerExtension", "font_set_glyph_uv_rect")
| ("TextServerExtension", "font_get_glyph_texture_idx")
| ("TextServerExtension", "font_set_glyph_texture_idx")
| ("TextServerExtension", "font_get_glyph_texture_rid")
| ("TextServerExtension", "font_get_glyph_texture_size")
| ("TextServerExtension", "font_get_glyph_index")
| ("TextServerExtension", "font_get_char_from_glyph_index")
| ("TextServerExtension", "font_has_char")
| ("TextServerExtension", "font_get_supported_chars")
| ("TextServerExtension", "font_draw_glyph")
| ("TextServerExtension", "font_draw_glyph_outline")
| ("TextServerExtension", "create_shaped_text")
| ("TextServerExtension", "shaped_text_clear")
| ("TextServerExtension", "shaped_text_add_string")
| ("TextServerExtension", "shaped_text_add_object")
| ("TextServerExtension", "shaped_text_resize_object")
| ("TextServerExtension", "shaped_get_span_count")
| ("TextServerExtension", "shaped_get_span_meta")
| ("TextServerExtension", "shaped_set_span_update_font")
| ("TextServerExtension", "shaped_text_substr")
| ("TextServerExtension", "shaped_text_get_parent")
| ("TextServerExtension", "shaped_text_shape")
| ("TextServerExtension", "shaped_text_is_ready")
| ("TextServerExtension", "shaped_text_get_glyphs")
| ("TextServerExtension", "shaped_text_sort_logical")
| ("TextServerExtension", "shaped_text_get_glyph_count")
| ("TextServerExtension", "shaped_text_get_range")
| ("TextServerExtension", "shaped_text_get_trim_pos")
| ("TextServerExtension", "shaped_text_get_ellipsis_pos")
| ("TextServerExtension", "shaped_text_get_ellipsis_glyphs")
| ("TextServerExtension", "shaped_text_get_ellipsis_glyph_count")
| ("TextServerExtension", "shaped_text_get_objects")
| ("TextServerExtension", "shaped_text_get_object_rect")
| ("TextServerExtension", "shaped_text_get_object_range")
| ("TextServerExtension", "shaped_text_get_object_glyph")
| ("TextServerExtension", "shaped_text_get_size")
| ("TextServerExtension", "shaped_text_get_ascent")
| ("TextServerExtension", "shaped_text_get_descent")
| ("TextServerExtension", "shaped_text_get_width")
| ("TextServerExtension", "shaped_text_get_underline_position")
| ("TextServerExtension", "shaped_text_get_underline_thickness")
| ("AudioStreamPlayback", "mix")
| ("AudioStreamPlaybackResampled", "mix_resampled")
| ("AudioStreamPlaybackResampled", "get_stream_sampling_rate")
| ("AudioEffectInstance", "process")
| ("AudioEffect", "instantiate")
| ("PhysicsDirectBodyState2DExtension", _)
| ("PhysicsDirectBodyState3DExtension", _)
| ("PhysicsDirectSpaceState2DExtension", _)
| ("PhysicsDirectSpaceState3DExtension", _)
| ("PhysicsServer3DExtension", _)
| ("PhysicsServer2DExtension", _)
| ("EditorScript", "run")
| ("VideoStreamPlayback", "update")
| ("EditorFileSystemImportFormatSupportQuery", _)
| ("Mesh", _)
| ("Texture2D", "get_width")
| ("Texture2D", "get_height")
| ("Texture3D", _)
| ("TextureLayered", _)
| ("StyleBox", "draw")
| ("Material", "get_shader_rid")
| ("Material", "get_shader_mode")
| ("PacketPeerExtension", "get_available_packet_count")
| ("PacketPeerExtension", "get_max_packet_size")
| ("StreamPeerExtension", "get_available_bytes")
| ("WebRtcDataChannelExtension", "poll")
| ("WebRtcDataChannelExtension", "close")
| ("WebRtcDataChannelExtension", "set_write_mode")
| ("WebRtcDataChannelExtension", "get_write_mode")
| ("WebRtcDataChannelExtension", "was_string_packet")
| ("WebRtcDataChannelExtension", "get_ready_state")
| ("WebRtcDataChannelExtension", "get_label")
| ("WebRtcDataChannelExtension", "is_ordered")
| ("WebRtcDataChannelExtension", "get_id")
| ("WebRtcDataChannelExtension", "get_max_packet_life_time")
| ("WebRtcDataChannelExtension", "get_max_retransmits")
| ("WebRtcDataChannelExtension", "get_protocol")
| ("WebRtcDataChannelExtension", "is_negotiated")
| ("WebRtcDataChannelExtension", "get_buffered_amount")
| ("WebRtcDataChannelExtension", "get_available_packet_count")
| ("WebRtcDataChannelExtension", "get_max_packet_size")
| ("WebRtcPeerConnectionExtension", _)
| ("MultiplayerPeerExtension", "get_available_packet_count")
| ("MultiplayerPeerExtension", "get_max_packet_size")
| ("MultiplayerPeerExtension", "set_transfer_channel")
| ("MultiplayerPeerExtension", "get_transfer_channel")
| ("MultiplayerPeerExtension", "set_transfer_mode")
| ("MultiplayerPeerExtension", "get_transfer_mode")
| ("MultiplayerPeerExtension", "set_target_peer")
| ("MultiplayerPeerExtension", "get_packet_peer")
| ("MultiplayerPeerExtension", "get_packet_mode")
| ("MultiplayerPeerExtension", "get_packet_channel")
| ("MultiplayerPeerExtension", "is_server")
| ("MultiplayerPeerExtension", "poll")
| ("MultiplayerPeerExtension", "close")
| ("MultiplayerPeerExtension", "disconnect_peer")
| ("MultiplayerPeerExtension", "get_unique_id")
| ("MultiplayerPeerExtension", "get_connection_status") => true,

(_, _) => false,
}
}

#[rustfmt::skip]
pub fn is_class_level_server(class_name: &str) -> bool {
// Unclear on if some of these classes should be registered earlier than `Scene`:
Expand Down
12 changes: 7 additions & 5 deletions godot-core/src/meta/signature.rs
Original file line number Diff line number Diff line change
Expand Up @@ -654,9 +654,9 @@ impl<'a> fmt::Display for CallContext<'a> {
// Trace diagnostics for integration tests
#[cfg(feature = "trace")]
pub mod trace {
use crate::meta::CallContext;
use std::cell::Cell;

use super::sys::Global;
use crate::meta::CallContext;

/// Stores information about the current call for diagnostic purposes.
pub struct CallReport {
Expand All @@ -667,7 +667,7 @@ pub mod trace {
}

pub fn pop() -> CallReport {
let lock = TRACE.lock().take();
let lock = TRACE.take();
// let th = std::thread::current().id();
// println!("trace::pop [{th:?}]...");

Expand All @@ -688,8 +688,10 @@ pub mod trace {
is_ptrcall: ptrcall,
};

*TRACE.lock() = Some(report);
TRACE.set(Some(report));
}

static TRACE: Global<Option<CallReport>> = Global::default();
thread_local! {
static TRACE: Cell<Option<CallReport>> = Cell::default();
}
}
16 changes: 4 additions & 12 deletions godot-core/src/obj/script.rs
Original file line number Diff line number Diff line change
Expand Up @@ -45,12 +45,12 @@ use self::bounded_ptr_list::BoundedPtrList;
/// # // Trick 17 to avoid listing all the methods. Needs also a method.
/// # mod godot {
/// # pub use ::godot::*;
/// # pub mod extras { pub trait ScriptInstance {} }
/// # pub mod extras { pub trait ScriptInstance {} pub trait IScriptExtension {} }
/// # }
/// # fn create_script_instance(_: MyInstance) -> *mut std::ffi::c_void { std::ptr::null_mut() }
/// use godot::prelude::*;
/// use godot::classes::{IScriptExtension, Script, ScriptExtension};
/// use godot::extras::ScriptInstance;
/// use godot::classes::{Script, ScriptExtension};
/// use godot::extras::{IScriptExtension, ScriptInstance};
///
/// // 1) Define the script.
/// #[derive(GodotClass)]
Expand All @@ -75,15 +75,7 @@ use self::bounded_ptr_list::BoundedPtrList;
/// // 3) Implement the script's virtual interface to wire up 1) and 2).
/// #[godot_api]
/// impl IScriptExtension for MyScript {
/// unsafe fn instance_create(&self, _for_object: Gd<Object>) -> *mut std::ffi::c_void {
/// // Upcast Gd<ScriptExtension> to Gd<Script>.
/// let script = self.to_gd().upcast();
/// let script_instance = MyInstance::from_gd(script);
///
/// // Note on safety: the returned pointer must be obtained
/// // through create_script_instance().
/// create_script_instance(script_instance)
/// }
/// // Implement all the methods...
/// }
/// ```
pub trait ScriptInstance: Sized {
Expand Down
5 changes: 3 additions & 2 deletions itest/rust/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -10,8 +10,9 @@ publish = false
crate-type = ["cdylib"]

[features]
default = []
codegen-full-experimental = ["godot/__codegen-full", "godot/experimental-godot-api"]
default = ["codegen-full"]
codegen-full = ["godot/__codegen-full"]
codegen-full-experimental = ["codegen-full", "godot/experimental-godot-api"]
experimental-threads = ["godot/experimental-threads"]
register-docs = ["godot/register-docs"] # TODO remove as soon as constant_test.rs checks bitfields with #[constant] proc-macro
serde = ["dep:serde", "dep:serde_json", "godot/serde"]
Expand Down
Loading

0 comments on commit 9d39621

Please sign in to comment.