From 6bbef8dec6fd377a43982d256d74b02a5fb4aaad Mon Sep 17 00:00:00 2001 From: Andrew Hayzen Date: Wed, 26 Jul 2023 16:35:04 +0100 Subject: [PATCH] cxx-qt-gen: store module ident in parser and simplify populating mapping This also simplifies extern "C++Qt" support later. Related to #577 --- crates/cxx-qt-gen/src/parser/cxxqtdata.rs | 88 ++++++++++------------- crates/cxx-qt-gen/src/parser/mod.rs | 22 +++--- 2 files changed, 45 insertions(+), 65 deletions(-) diff --git a/crates/cxx-qt-gen/src/parser/cxxqtdata.rs b/crates/cxx-qt-gen/src/parser/cxxqtdata.rs index 3af4bbc03..c72bb1019 100644 --- a/crates/cxx-qt-gen/src/parser/cxxqtdata.rs +++ b/crates/cxx-qt-gen/src/parser/cxxqtdata.rs @@ -49,7 +49,6 @@ impl ParsedCxxMappings { } } -#[derive(Default)] pub struct ParsedCxxQtData { /// Mappings for CXX types when used in C++ pub cxx_mappings: ParsedCxxMappings, @@ -62,9 +61,22 @@ pub struct ParsedCxxQtData { pub qobjects: BTreeMap, /// The namespace of the CXX-Qt module pub namespace: String, + /// The ident of the module, used for mappings + pub module_ident: Ident, } impl ParsedCxxQtData { + /// Create a ParsedCxxQtData from a given module and namespace + pub fn new(module_ident: Ident, namespace: String) -> Self { + Self { + cxx_mappings: ParsedCxxMappings::default(), + qualified_mappings: BTreeMap::::default(), + qobjects: BTreeMap::::default(), + module_ident, + namespace, + } + } + /// Find the QObjects within the module and add into the qobjects BTreeMap pub fn find_qobject_types(&mut self, items: &[Item]) -> Result<()> { for item in items { @@ -107,19 +119,16 @@ impl ParsedCxxQtData { /// /// We need to know this as it affects the type name used in the C++ generation /// And it is used to create the qualified Rust name - pub fn populate_mappings_from_item( - &mut self, - item: &Item, - bridge_namespace: &str, - module_ident: &Ident, - ) -> Result<()> { + pub fn populate_mappings_from_item(&mut self, item: &Item) -> Result<()> { + let bridge_namespace = self.namespace.clone(); + // Consider if shared types have mappings match item { Item::Enum(item) => { - self.populate_mappings(&item.ident, &item.attrs, bridge_namespace, module_ident)?; + self.populate_mappings(&item.ident, &item.attrs, &bridge_namespace)?; } Item::Struct(item) => { - self.populate_mappings(&item.ident, &item.attrs, bridge_namespace, module_ident)?; + self.populate_mappings(&item.ident, &item.attrs, &bridge_namespace)?; } _others => {} } @@ -136,12 +145,7 @@ impl ParsedCxxQtData { // Read each of the types in the mod (type A;) for foreign_type in foreign_mod_to_foreign_item_types(foreign_mod)? { - self.populate_mappings( - &foreign_type.ident, - &foreign_type.attrs, - &block_namespace, - module_ident, - )?; + self.populate_mappings(&foreign_type.ident, &foreign_type.attrs, &block_namespace)?; } } @@ -154,7 +158,6 @@ impl ParsedCxxQtData { ident: &Ident, attrs: &[Attribute], parent_namespace: &str, - module_ident: &Ident, ) -> Result<()> { // Retrieve the namespace for the type itself if there is one let namespace = if let Some(index) = attribute_find_path(attrs, &["namespace"]) { @@ -188,6 +191,7 @@ impl ParsedCxxQtData { } else { ident.clone() }; + let module_ident = &self.module_ident; self.qualified_mappings .insert(ident.clone(), parse_quote! { #module_ident::#rust_ident }); @@ -318,7 +322,7 @@ mod tests { /// Creates a ParsedCxxQtData with a QObject definition already found fn create_parsed_cxx_qt_data() -> ParsedCxxQtData { - let mut cxx_qt_data = ParsedCxxQtData::default(); + let mut cxx_qt_data = ParsedCxxQtData::new(format_ident!("ffi"), "".to_string()); cxx_qt_data .qobjects .insert(qobject_ident(), create_parsed_qobject()); @@ -327,7 +331,7 @@ mod tests { #[test] fn test_find_qobjects_one_qobject() { - let mut cxx_qt_data = ParsedCxxQtData::default(); + let mut cxx_qt_data = ParsedCxxQtData::new(format_ident!("ffi"), "".to_string()); let module: ItemMod = parse_quote! { mod module { @@ -347,7 +351,7 @@ mod tests { #[test] fn test_find_qobjects_multiple_qobject() { - let mut cxx_qt_data = ParsedCxxQtData::default(); + let mut cxx_qt_data = ParsedCxxQtData::new(format_ident!("ffi"), "".to_string()); let module: ItemMod = parse_quote! { mod module { @@ -372,10 +376,8 @@ mod tests { #[test] fn test_find_qobjects_namespace() { - let mut cxx_qt_data = ParsedCxxQtData { - namespace: "bridge_namespace".to_owned(), - ..Default::default() - }; + let mut cxx_qt_data = + ParsedCxxQtData::new(format_ident!("ffi"), "bridge_namespace".to_string()); let module: ItemMod = parse_quote! { mod module { @@ -412,7 +414,7 @@ mod tests { #[test] fn test_find_qobjects_no_macro() { - let mut cxx_qt_data = ParsedCxxQtData::default(); + let mut cxx_qt_data = ParsedCxxQtData::new(format_ident!("ffi"), "".to_string()); let module: ItemMod = parse_quote! { mod module { @@ -551,9 +553,7 @@ mod tests { type A; } }; - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert!(cxx_qt_data.cxx_mappings.cxx_names.is_empty()); assert_eq!(cxx_qt_data.qualified_mappings.len(), 1); @@ -576,9 +576,7 @@ mod tests { type A; } }; - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.len(), 1); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.get("A").unwrap(), "B"); @@ -602,9 +600,7 @@ mod tests { type A = C; } }; - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.len(), 1); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.get("A").unwrap(), "B"); @@ -621,6 +617,7 @@ mod tests { #[test] fn test_cxx_mappings_cxx_name_namespace_bridge() { let mut cxx_qt_data = create_parsed_cxx_qt_data(); + cxx_qt_data.namespace = "bridge_namespace".to_owned(); let item: Item = parse_quote! { extern "C++" { @@ -630,9 +627,7 @@ mod tests { type B; } }; - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "bridge_namespace", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.len(), 1); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.get("B").unwrap(), "C"); @@ -666,6 +661,7 @@ mod tests { #[test] fn test_cxx_mappings_cxx_name_namespace_items() { let mut cxx_qt_data = create_parsed_cxx_qt_data(); + cxx_qt_data.namespace = "namespace".to_owned(); let item: Item = parse_quote! { #[namespace = "extern_namespace"] @@ -677,9 +673,7 @@ mod tests { } }; // Also ensure item namespace is chosen instead of bridge namespace - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "namespace", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.len(), 0); assert_eq!(cxx_qt_data.cxx_mappings.namespaces.len(), 2); @@ -724,9 +718,7 @@ mod tests { type C; } }; - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.len(), 2); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.get("A").unwrap(), "B"); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.get("C").unwrap(), "D"); @@ -770,9 +762,7 @@ mod tests { } }; - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.len(), 1); assert_eq!( cxx_qt_data.cxx_mappings.cxx_names.get("EnumA").unwrap(), @@ -807,9 +797,7 @@ mod tests { } }; - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert_eq!(cxx_qt_data.cxx_mappings.cxx_names.len(), 1); assert_eq!( cxx_qt_data.cxx_mappings.cxx_names.get("StructA").unwrap(), @@ -842,9 +830,7 @@ mod tests { type A; } }; - assert!(cxx_qt_data - .populate_mappings_from_item(&item, "", &format_ident!("ffi")) - .is_ok()); + assert!(cxx_qt_data.populate_mappings_from_item(&item).is_ok()); assert!(cxx_qt_data.cxx_mappings.cxx_names.is_empty()); assert_eq!(cxx_qt_data.qualified_mappings.len(), 1); diff --git a/crates/cxx-qt-gen/src/parser/mod.rs b/crates/cxx-qt-gen/src/parser/mod.rs index 928ca2dcb..7a140796a 100644 --- a/crates/cxx-qt-gen/src/parser/mod.rs +++ b/crates/cxx-qt-gen/src/parser/mod.rs @@ -34,7 +34,7 @@ pub struct Parser { impl Parser { /// Constructs a Parser object from a given [syn::ItemMod] block pub fn from(mut module: ItemMod) -> Result { - let mut cxx_qt_data = ParsedCxxQtData::default(); + let mut namespace = "".to_owned(); let mut others = vec![]; let mut cxx_file_stem = module.ident.to_string(); @@ -44,13 +44,11 @@ impl Parser { &module.attrs[index], AttributeDefault::None, )?; + // Parse any namespace in the cxx_qt::bridge macro - cxx_qt_data.namespace = - if let Some(lit_str) = attr_map.get("e::format_ident!("namespace")) { - lit_str.value() - } else { - "".to_owned() - }; + if let Some(lit_str) = attr_map.get("e::format_ident!("namespace")) { + namespace = lit_str.value(); + } // Parse any custom file stem if let Some(stem) = attr_map.get("e::format_ident!("cxx_file_stem")) { @@ -65,10 +63,10 @@ impl Parser { )); } + let mut cxx_qt_data = ParsedCxxQtData::new(module.ident.clone(), namespace); + // Check that there are items in the module if let Some(mut items) = module.content { - let bridge_namespace = cxx_qt_data.namespace.clone(); - // Find any QObject structs cxx_qt_data.find_qobject_types(&items.1)?; @@ -78,11 +76,7 @@ impl Parser { // qobject. Otherwise return them to be added to other if let Some(other) = cxx_qt_data.parse_cxx_qt_item(item)? { // Load any CXX name mappings - cxx_qt_data.populate_mappings_from_item( - &other, - &bridge_namespace, - &module.ident, - )?; + cxx_qt_data.populate_mappings_from_item(&other)?; // Unknown item so add to the other list others.push(other);