Skip to content

Commit 86c6e95

Browse files
snakefangoxBromeon
authored andcommitted
Allow virtual methods to be GdSelf in special cases
1 parent 80c24f6 commit 86c6e95

File tree

7 files changed

+40
-6
lines changed

7 files changed

+40
-6
lines changed

godot-codegen/src/generator/default_parameters.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -211,8 +211,8 @@ fn make_extender_receiver(sig: &dyn Function) -> ExtenderReceiver {
211211
let builder_mut = match sig.qualifier() {
212212
FnQualifier::Const | FnQualifier::Static => quote! {},
213213
FnQualifier::Mut => quote! { mut },
214-
FnQualifier::Global => {
215-
unreachable!("default parameters not supported for global methods; {sig}")
214+
FnQualifier::Global | FnQualifier::GdSelf => {
215+
unreachable!("default parameters not supported for global or gdself methods; {sig}")
216216
}
217217
};
218218

godot-codegen/src/generator/functions_common.rs

+4
Original file line numberDiff line numberDiff line change
@@ -308,6 +308,7 @@ pub fn make_receiver(qualifier: FnQualifier, ffi_arg_in: TokenStream) -> FnRecei
308308
let (param, param_lifetime_a) = match qualifier {
309309
FnQualifier::Const => (quote! { &self, }, quote! { &'a self, }),
310310
FnQualifier::Mut => (quote! { &mut self, }, quote! { &'a mut self, }),
311+
FnQualifier::GdSelf => (quote! { this: Gd<Self>, }, quote! { this: Gd<Self>, }),
311312
FnQualifier::Static => (quote! {}, quote! {}),
312313
FnQualifier::Global => (quote! {}, quote! {}),
313314
};
@@ -316,6 +317,9 @@ pub fn make_receiver(qualifier: FnQualifier, ffi_arg_in: TokenStream) -> FnRecei
316317
if matches!(qualifier, FnQualifier::Static) {
317318
ffi_arg = quote! { std::ptr::null_mut() };
318319
self_prefix = quote! { Self:: };
320+
} else if matches!(qualifier, FnQualifier::Static) {
321+
ffi_arg = ffi_arg_in;
322+
self_prefix = quote! { this. };
319323
} else {
320324
ffi_arg = ffi_arg_in;
321325
self_prefix = quote! { self. };

godot-codegen/src/generator/virtual_traits.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -145,7 +145,10 @@ fn make_virtual_method(method: &ClassMethod) -> Option<TokenStream> {
145145

146146
// Virtual methods are never static.
147147
let qualifier = method.qualifier();
148-
assert!(matches!(qualifier, FnQualifier::Mut | FnQualifier::Const));
148+
assert!(matches!(
149+
qualifier,
150+
FnQualifier::Mut | FnQualifier::Const | FnQualifier::GdSelf
151+
));
149152

150153
let definition = functions_common::make_function_definition(
151154
method,

godot-codegen/src/models/domain.rs

+1
Original file line numberDiff line numberDiff line change
@@ -462,6 +462,7 @@ pub enum FnDirection {
462462
pub enum FnQualifier {
463463
Mut, // &mut self
464464
Const, // &self
465+
GdSelf, // Gd<Self>
465466
Static, // Self
466467
Global, // (nothing) // TODO remove?
467468
}

godot-codegen/src/models/domain_mapping.rs

+6-1
Original file line numberDiff line numberDiff line change
@@ -458,6 +458,7 @@ impl ClassMethod {
458458
}
459459

460460
let is_private = special_cases::is_method_private(class_name, &method.name);
461+
let is_gdself = special_cases::is_virtual_method_gdself(class_name, rust_method_name);
461462

462463
let godot_method_name = method.name.clone();
463464

@@ -468,7 +469,11 @@ impl ClassMethod {
468469
is_actually_const = override_const;
469470
}
470471

471-
FnQualifier::from_const_static(is_actually_const, method.is_static)
472+
if is_gdself {
473+
FnQualifier::GdSelf
474+
} else {
475+
FnQualifier::from_const_static(is_actually_const, method.is_static)
476+
}
472477
};
473478

474479
// Since Godot 4.4, GDExtension advertises whether virtual methods have a default implementation or are required to be overridden.

godot-codegen/src/special_cases/special_cases.rs

+13
Original file line numberDiff line numberDiff line change
@@ -855,6 +855,19 @@ pub fn is_enum_exhaustive(class_name: Option<&TyName>, enum_name: &str) -> bool
855855
}
856856
}
857857

858+
/// Certain virtual methods can make it very difficult to avoid double bind issues
859+
/// if they take &self or &mut self.
860+
///
861+
/// These methods take Gd<Self> as a receiver so the implementer can decide when to bind
862+
/// and unbind them.
863+
pub fn is_virtual_method_gdself(class_name: &TyName, method: &str) -> bool {
864+
match (class_name.godot_ty.as_str(), method) {
865+
("MultiplayerAPIExtension", "poll") => true,
866+
867+
_ => false,
868+
}
869+
}
870+
858871
/// Whether an enum can be combined with another enum (return value) for bitmasking purposes.
859872
///
860873
/// If multiple masks are ever necessary, this can be extended to return a slice instead of Option.

godot-macros/src/class/data_models/interface_trait_impl.rs

+10-2
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ use crate::{util, ParseResult};
1111

1212
use proc_macro2::{Ident, TokenStream};
1313
use quote::quote;
14+
use venial::FnParam;
1415

1516
/// Codegen for `#[godot_api] impl ISomething for MyType`
1617
pub fn transform_trait_impl(original_impl: venial::Impl) -> ParseResult<TokenStream> {
@@ -211,7 +212,7 @@ pub fn transform_trait_impl(original_impl: venial::Impl) -> ParseResult<TokenStr
211212
method_name_str => {
212213
#[cfg(since_api = "4.4")]
213214
let method_name_ident = method.name.clone();
214-
let method = util::reduce_to_signature(method);
215+
let mut method = util::reduce_to_signature(method);
215216

216217
// Godot-facing name begins with underscore.
217218
//
@@ -223,7 +224,14 @@ pub fn transform_trait_impl(original_impl: venial::Impl) -> ParseResult<TokenStr
223224
format!("_{method_name_str}")
224225
};
225226

226-
let signature_info = into_signature_info(method, &class_name, false);
227+
// Some virtual methods are GdSelf, but none are static, so if the first param isn't a receiver (&self, etc) it's GdSelf
228+
let is_gd_self = matches!(method.params.first(), Some((FnParam::Typed(_), _)));
229+
if is_gd_self {
230+
// The GdSelf receiver is handled by `into_signature_info`
231+
method.params.inner.remove(0);
232+
}
233+
234+
let signature_info = into_signature_info(method, &class_name, is_gd_self);
227235

228236
// Overridden ready() methods additionally have an additional `__before_ready()` call (for OnReady inits).
229237
let before_kind = if method_name_str == "ready" {

0 commit comments

Comments
 (0)