|
| 1 | +use bindgen::callbacks::TypeKind; |
1 | 2 | use bindgen::{
|
2 | 3 | builder, AliasVariation, Builder, CodegenConfig, EnumVariation,
|
3 |
| - MacroTypeVariation, NonCopyUnionStyle, RustTarget, |
| 4 | + MacroTypeVariation, NonCopyUnionStyle, RegexSet, RustTarget, |
4 | 5 | DEFAULT_ANON_FIELDS_PREFIX, RUST_TARGET_STRINGS,
|
5 | 6 | };
|
6 | 7 | use clap::Parser;
|
@@ -340,6 +341,18 @@ struct BindgenCommand {
|
340 | 341 | /// Wrap unsafe operations in unsafe blocks.
|
341 | 342 | #[arg(long)]
|
342 | 343 | wrap_unsafe_ops: bool,
|
| 344 | + /// Derive custom traits on any kind of type. The <CUSTOM> value must be of the shape <REGEX>=<DERIVE> where <DERIVE> is a coma-separated list of derive macros. |
| 345 | + #[arg(long, value_name = "CUSTOM")] |
| 346 | + with_derive_custom: Vec<String>, |
| 347 | + /// Derive custom traits on a `struct`. The <CUSTOM> value must be of the shape <REGEX>=<DERIVE> where <DERIVE> is a coma-separated list of derive macros. |
| 348 | + #[arg(long, value_name = "CUSTOM")] |
| 349 | + with_derive_custom_struct: Vec<String>, |
| 350 | + /// Derive custom traits on an `enum. The <CUSTOM> value must be of the shape <REGEX>=<DERIVE> where <DERIVE> is a coma-separated list of derive macros. |
| 351 | + #[arg(long, value_name = "CUSTOM")] |
| 352 | + with_derive_custom_enum: Vec<String>, |
| 353 | + /// Derive custom traits on a `union`. The <CUSTOM> value must be of the shape <REGEX>=<DERIVE> where <DERIVE> is a coma-separated list of derive macros. |
| 354 | + #[arg(long, value_name = "CUSTOM")] |
| 355 | + with_derive_custom_union: Vec<String>, |
343 | 356 | /// Prints the version, and exits
|
344 | 357 | #[arg(short = 'V', long)]
|
345 | 358 | version: bool,
|
@@ -456,6 +469,10 @@ where
|
456 | 469 | merge_extern_blocks,
|
457 | 470 | override_abi,
|
458 | 471 | wrap_unsafe_ops,
|
| 472 | + with_derive_custom, |
| 473 | + with_derive_custom_struct, |
| 474 | + with_derive_custom_enum, |
| 475 | + with_derive_custom_union, |
459 | 476 | version,
|
460 | 477 | clang_args,
|
461 | 478 | } = command;
|
@@ -894,5 +911,72 @@ where
|
894 | 911 | builder = builder.wrap_unsafe_ops(true);
|
895 | 912 | }
|
896 | 913 |
|
| 914 | + #[derive(Debug)] |
| 915 | + struct CustomDeriveCallback { |
| 916 | + derives: Vec<String>, |
| 917 | + kind: Option<TypeKind>, |
| 918 | + regex_set: bindgen::RegexSet, |
| 919 | + } |
| 920 | + |
| 921 | + impl bindgen::callbacks::ParseCallbacks for CustomDeriveCallback { |
| 922 | + fn cli_args(&self) -> Vec<String> { |
| 923 | + let mut args = vec![]; |
| 924 | + |
| 925 | + let flag = match &self.kind { |
| 926 | + None => "--with-derive-custom", |
| 927 | + Some(TypeKind::Struct) => "--with-derive-custom-struct", |
| 928 | + Some(TypeKind::Enum) => "--with-derive-custom-enum", |
| 929 | + Some(TypeKind::Union) => "--with-derive-custom-union", |
| 930 | + }; |
| 931 | + |
| 932 | + let derives = self.derives.join(","); |
| 933 | + |
| 934 | + for item in self.regex_set.get_items() { |
| 935 | + args.extend_from_slice(&[ |
| 936 | + flag.to_owned(), |
| 937 | + format!("{}={}", item, derives), |
| 938 | + ]); |
| 939 | + } |
| 940 | + |
| 941 | + args |
| 942 | + } |
| 943 | + |
| 944 | + fn add_derives( |
| 945 | + &self, |
| 946 | + info: &bindgen::callbacks::DeriveInfo<'_>, |
| 947 | + ) -> Vec<String> { |
| 948 | + if self.kind.map(|kind| kind == info.kind).unwrap_or(true) && |
| 949 | + self.regex_set.matches(info.name) |
| 950 | + { |
| 951 | + return self.derives.clone(); |
| 952 | + } |
| 953 | + vec![] |
| 954 | + } |
| 955 | + } |
| 956 | + |
| 957 | + for (custom_derives, kind) in [ |
| 958 | + (with_derive_custom, None), |
| 959 | + (with_derive_custom_struct, Some(TypeKind::Struct)), |
| 960 | + (with_derive_custom_enum, Some(TypeKind::Enum)), |
| 961 | + (with_derive_custom_union, Some(TypeKind::Union)), |
| 962 | + ] { |
| 963 | + for custom_derive in custom_derives { |
| 964 | + let (regex, derives) = custom_derive |
| 965 | + .rsplit_once('=') |
| 966 | + .expect("Invalid custom derive argument: Missing `=`"); |
| 967 | + let derives = derives.split(',').map(|s| s.to_owned()).collect(); |
| 968 | + |
| 969 | + let mut regex_set = RegexSet::new(); |
| 970 | + regex_set.insert(regex); |
| 971 | + regex_set.build(false); |
| 972 | + |
| 973 | + builder = builder.parse_callbacks(Box::new(CustomDeriveCallback { |
| 974 | + derives, |
| 975 | + kind, |
| 976 | + regex_set, |
| 977 | + })); |
| 978 | + } |
| 979 | + } |
| 980 | + |
897 | 981 | Ok((builder, output, verbose))
|
898 | 982 | }
|
0 commit comments