From 0b99f62c3b4cacb836892ccce146be700073a6bf Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Wed, 18 Oct 2023 15:35:29 -0500 Subject: [PATCH 01/13] simpler lifetime --- crates/libs/bindgen/src/metadata.rs | 2 +- crates/libs/bindgen/src/rdl/from_reader.rs | 28 +++++++++++----------- crates/libs/bindgen/src/rust/mod.rs | 8 +++---- crates/libs/bindgen/src/rust/writer.rs | 16 ++++++------- crates/libs/bindgen/src/tree.rs | 14 +++++------ 5 files changed, 34 insertions(+), 34 deletions(-) diff --git a/crates/libs/bindgen/src/metadata.rs b/crates/libs/bindgen/src/metadata.rs index 994cba7f7b..8db2636ef7 100644 --- a/crates/libs/bindgen/src/metadata.rs +++ b/crates/libs/bindgen/src/metadata.rs @@ -498,7 +498,7 @@ pub fn type_interfaces(ty: &Type) -> Vec { result } -fn type_name<'a>(ty: &Type) -> &'a str { +fn type_name(ty: &Type) -> &str { match ty { Type::TypeDef(row, _) => row.name(), _ => "", diff --git a/crates/libs/bindgen/src/rdl/from_reader.rs b/crates/libs/bindgen/src/rdl/from_reader.rs index 3842614fc2..7a909fb52d 100644 --- a/crates/libs/bindgen/src/rdl/from_reader.rs +++ b/crates/libs/bindgen/src/rdl/from_reader.rs @@ -3,7 +3,7 @@ use crate::tokens::{quote, to_ident, TokenStream}; use crate::{rdl, Error, Result, Tree}; use metadata::*; -pub fn from_reader(reader: &metadata::Reader, mut config: std::collections::BTreeMap<&str, &str>, output: &str) -> Result<()> { +pub fn from_reader(reader: &'static metadata::Reader, mut config: std::collections::BTreeMap<&str, &str>, output: &str) -> Result<()> { let dialect = match config.remove("type") { Some("winrt") => Dialect::WinRT, Some("win32") => Dialect::Win32, @@ -30,7 +30,7 @@ pub fn from_reader(reader: &metadata::Reader, mut config: std::collections::BTre fn gen_split(writer: &Writer) -> Result<()> { let tree = Tree::new(writer.reader); - let directory = crate::directory(writer.output); + let directory = crate::directory(&writer.output); // TODO: parallelize for tree in tree.flatten() { @@ -48,7 +48,7 @@ fn gen_split(writer: &Writer) -> Result<()> { fn gen_file(writer: &Writer) -> Result<()> { let tree = Tree::new(writer.reader); let tokens = writer.tree(&tree); - writer.write_to_file(writer.output, tokens) + writer.write_to_file(&writer.output, tokens) } #[derive(Debug, Copy, Clone, PartialEq)] @@ -57,21 +57,21 @@ enum Dialect { WinRT, } -struct Writer<'a> { - reader: &'a metadata::Reader, - namespace: &'a str, +struct Writer { + reader: &'static metadata::Reader, + namespace: &'static str, dialect: Dialect, split: bool, - output: &'a str, + output: String, } -impl<'a> Writer<'a> { - fn new(reader: &'a metadata::Reader, output: &'a str, dialect: Dialect) -> Self { - Self { reader, namespace: "", output, dialect, split: false } +impl Writer { + fn new(reader: &'static metadata::Reader, output: &str, dialect: Dialect) -> Self { + Self { reader, namespace: "", output: output.to_string(), dialect, split: false } } - fn with_namespace(&self, namespace: &'a str) -> Self { - Self { reader: self.reader, namespace, dialect: self.dialect, output: self.output, split: self.split } + fn with_namespace(&self, namespace: &'static str) -> Self { + Self { reader: self.reader, namespace, dialect: self.dialect, output: self.output.clone(), split: self.split } } fn write_to_file(&self, output: &str, tokens: TokenStream) -> Result<()> { @@ -90,7 +90,7 @@ impl<'a> Writer<'a> { //crate::write_to_file(output, tokens.into_string()) } - fn tree(&self, tree: &'a Tree) -> TokenStream { + fn tree(&self, tree: &Tree) -> TokenStream { let items = self.items(tree); if self.split { @@ -128,7 +128,7 @@ impl<'a> Writer<'a> { } } - fn items(&self, tree: &'a Tree) -> TokenStream { + fn items(&self, tree: &Tree) -> TokenStream { let mut functions = vec![]; let mut constants = vec![]; let mut types = vec![]; diff --git a/crates/libs/bindgen/src/rust/mod.rs b/crates/libs/bindgen/src/rust/mod.rs index fbc3a85c71..342a6879a9 100644 --- a/crates/libs/bindgen/src/rust/mod.rs +++ b/crates/libs/bindgen/src/rust/mod.rs @@ -21,7 +21,7 @@ use crate::{Error, Result, Tree}; use cfg::*; use rayon::prelude::*; -pub fn from_reader(reader: &metadata::Reader, mut config: std::collections::BTreeMap<&str, &str>, output: &str) -> Result<()> { +pub fn from_reader(reader: &'static metadata::Reader, mut config: std::collections::BTreeMap<&str, &str>, output: &str) -> Result<()> { let mut writer = Writer::new(reader, output); writer.package = config.remove("package").is_some(); writer.flatten = config.remove("flatten").is_some(); @@ -56,7 +56,7 @@ fn gen_file(writer: &Writer) -> Result<()> { if writer.flatten { let tokens = standalone::standalone_imp(writer); - crate::write_to_file(writer.output, try_format(writer, &tokens)) + crate::write_to_file(&writer.output, try_format(writer, &tokens)) } else { let mut tokens = String::new(); let root = Tree::new(writer.reader); @@ -65,12 +65,12 @@ fn gen_file(writer: &Writer) -> Result<()> { tokens.push_str(&namespace(writer, tree)); } - crate::write_to_file(writer.output, try_format(writer, &tokens)) + crate::write_to_file(&writer.output, try_format(writer, &tokens)) } } fn gen_package(writer: &Writer) -> Result<()> { - let directory = crate::directory(writer.output); + let directory = crate::directory(&writer.output); let root = Tree::new(writer.reader); let mut root_len = 0; diff --git a/crates/libs/bindgen/src/rust/writer.rs b/crates/libs/bindgen/src/rust/writer.rs index 5145817ceb..7e66c93457 100644 --- a/crates/libs/bindgen/src/rust/writer.rs +++ b/crates/libs/bindgen/src/rust/writer.rs @@ -1,10 +1,10 @@ use super::*; #[derive(Clone)] -pub struct Writer<'a> { - pub reader: &'a Reader, - pub output: &'a str, - pub namespace: &'a str, +pub struct Writer { + pub reader: &'static Reader, + pub output: String, + pub namespace: &'static str, pub implement: bool, // TODO: ideally we can use this to generate implementation traits on the fly and // and have a single interface definition macro for consumption that expands to include // impl traits when the `implement` cfg flag is set and then this writer option would be @@ -20,11 +20,11 @@ pub struct Writer<'a> { pub no_inner_attributes: bool, // skips the inner attributes at the start of the file } -impl<'a> Writer<'a> { - pub fn new(reader: &'a Reader, output: &'a str) -> Self { +impl Writer { + pub fn new(reader: &'static Reader, output: &str) -> Self { Self { reader, - output, + output: output.to_string(), namespace: "", implement: false, std: false, @@ -403,7 +403,7 @@ impl<'a> Writer<'a> { quote! { #arch #features } } - fn cfg_features_imp(&self, cfg: &'a Cfg, namespace: &'a str) -> Vec<&'a str> { + fn cfg_features_imp(&self, cfg: &Cfg, namespace: &str) -> Vec<&'static str> { let mut compact = Vec::<&'static str>::new(); if self.package { for feature in cfg.types.keys() { diff --git a/crates/libs/bindgen/src/tree.rs b/crates/libs/bindgen/src/tree.rs index 9fda6a302a..6a619940eb 100644 --- a/crates/libs/bindgen/src/tree.rs +++ b/crates/libs/bindgen/src/tree.rs @@ -1,13 +1,13 @@ use super::*; #[derive(Debug)] -pub struct Tree<'a> { - pub namespace: &'a str, - pub nested: std::collections::BTreeMap<&'a str, Tree<'a>>, +pub struct Tree { + pub namespace: &'static str, + pub nested: std::collections::BTreeMap<&'static str, Tree>, } -impl<'a> Tree<'a> { - pub fn new(reader: &'a metadata::Reader) -> Self { +impl Tree { + pub fn new(reader: &'static metadata::Reader) -> Self { let mut tree = Tree::from_namespace(""); for ns in reader.namespaces() { if reader.includes_namespace(ns) { @@ -17,10 +17,10 @@ impl<'a> Tree<'a> { tree } - fn from_namespace(namespace: &'a str) -> Self { + fn from_namespace(namespace: &'static str) -> Self { Self { namespace, nested: std::collections::BTreeMap::new() } } - fn insert_namespace(&mut self, namespace: &'a str, pos: usize) -> &mut Self { + fn insert_namespace(&mut self, namespace: &'static str, pos: usize) -> &mut Self { if let Some(next) = namespace[pos..].find('.') { let next = pos + next; self.nested.entry(&namespace[pos..next]).or_insert_with(|| Self::from_namespace(&namespace[..next])).insert_namespace(namespace, next + 1) From 53d281cd61ea9928d6fbc235bb62abdb56dac973 Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 11:36:38 -0500 Subject: [PATCH 02/13] TypeRef --- crates/libs/bindgen/src/rdl/from_reader.rs | 4 +--- crates/libs/bindgen/src/winmd/from_reader.rs | 1 - crates/libs/bindgen/src/winmd/verify.rs | 4 ++-- crates/libs/bindgen/src/winmd/writer/mod.rs | 5 ----- crates/libs/bindgen/src/winmd/writer/type.rs | 1 - crates/libs/metadata/src/lib.rs | 3 +-- crates/libs/metadata/src/reader.rs | 7 ++++--- crates/libs/metadata/src/tables.rs | 17 +++++++---------- crates/libs/metadata/src/type.rs | 5 +---- crates/libs/metadata/src/type_name.rs | 2 +- 10 files changed, 17 insertions(+), 32 deletions(-) diff --git a/crates/libs/bindgen/src/rdl/from_reader.rs b/crates/libs/bindgen/src/rdl/from_reader.rs index 7a909fb52d..02b4f55435 100644 --- a/crates/libs/bindgen/src/rdl/from_reader.rs +++ b/crates/libs/bindgen/src/rdl/from_reader.rs @@ -358,8 +358,7 @@ impl Writer { } } - metadata::Type::TypeRef(code) => { - let type_name = code.type_name(); + metadata::Type::TypeRef(type_name) => { let namespace = self.namespace(type_name.namespace); let name = to_ident(type_name.name); quote! { #namespace #name } @@ -379,7 +378,6 @@ impl Writer { metadata::Type::PCWSTR => quote! { PCWSTR }, metadata::Type::BSTR => quote! { BSTR }, metadata::Type::PrimitiveOrEnum(_, ty) => self.ty(ty), - rest => unimplemented!("{rest:?}"), } } diff --git a/crates/libs/bindgen/src/winmd/from_reader.rs b/crates/libs/bindgen/src/winmd/from_reader.rs index 51bc55b140..ab0fc8d64f 100644 --- a/crates/libs/bindgen/src/winmd/from_reader.rs +++ b/crates/libs/bindgen/src/winmd/from_reader.rs @@ -122,7 +122,6 @@ fn winmd_type(ty: &metadata::Type) -> winmd::Type { metadata::Type::PCSTR => winmd::Type::PCSTR, metadata::Type::PCWSTR => winmd::Type::PCWSTR, metadata::Type::BSTR => winmd::Type::BSTR, - metadata::Type::TypeName => winmd::Type::TypeName, metadata::Type::TypeDef(def, generics) => winmd::Type::TypeRef(winmd::TypeName { namespace: def.namespace().to_string(), name: def.name().to_string(), generics: generics.iter().map(winmd_type).collect() }), metadata::Type::GenericParam(generic) => winmd::Type::GenericParam(generic.number()), metadata::Type::ConstRef(ty) => winmd::Type::ConstRef(Box::new(winmd_type(ty))), diff --git a/crates/libs/bindgen/src/winmd/verify.rs b/crates/libs/bindgen/src/winmd/verify.rs index 6b36213fc3..7fcf385256 100644 --- a/crates/libs/bindgen/src/winmd/verify.rs +++ b/crates/libs/bindgen/src/winmd/verify.rs @@ -35,8 +35,8 @@ pub fn verify(reader: &metadata::Reader) -> crate::Result<()> { } fn not_type_ref(ty: &metadata::Type) -> crate::Result<()> { - if let metadata::Type::TypeRef(ty) = ty { - return Err(crate::Error::new(&format!("missing type definition `{}`", ty.type_name()))); + if let metadata::Type::TypeRef(type_name) = ty { + return Err(crate::Error::new(&format!("missing type definition `{}`", type_name))); } Ok(()) } diff --git a/crates/libs/bindgen/src/winmd/writer/mod.rs b/crates/libs/bindgen/src/winmd/writer/mod.rs index bd5f12f5bb..25170bb9e3 100644 --- a/crates/libs/bindgen/src/winmd/writer/mod.rs +++ b/crates/libs/bindgen/src/winmd/writer/mod.rs @@ -234,11 +234,6 @@ impl Writer { usize_blob(1, blob); // count usize_blob(*bounds, blob); } - Type::TypeName => { - let code = self.insert_type_ref("System", "Type"); - blob.push(metadata::ELEMENT_TYPE_CLASS); - usize_blob(code as usize, blob); - } Type::MutPtr(ty, pointers) | Type::ConstPtr(ty, pointers) => { for _ in 0..*pointers { usize_blob(metadata::ELEMENT_TYPE_PTR as usize, blob); diff --git a/crates/libs/bindgen/src/winmd/writer/type.rs b/crates/libs/bindgen/src/winmd/writer/type.rs index 3f0178654a..33482dc20b 100644 --- a/crates/libs/bindgen/src/winmd/writer/type.rs +++ b/crates/libs/bindgen/src/winmd/writer/type.rs @@ -34,7 +34,6 @@ pub enum Type { PCSTR, PCWSTR, BSTR, - TypeName, TypeRef(TypeName), GenericParam(u16), MutPtr(Box, usize), diff --git a/crates/libs/metadata/src/lib.rs b/crates/libs/metadata/src/lib.rs index 4606ca7ff4..0b71ca397c 100644 --- a/crates/libs/metadata/src/lib.rs +++ b/crates/libs/metadata/src/lib.rs @@ -19,7 +19,7 @@ pub use attributes::*; pub use bindings::*; pub use blob::*; pub use codes::*; -pub use column::*; +use column::*; pub use file::*; use filter::*; pub use r#type::*; @@ -96,7 +96,6 @@ pub enum Value { F64(f64), String(String), TypeName(String), - TypeRef(TypeDefOrRef), // TODO: needed? EnumDef(TypeDef, Box), } diff --git a/crates/libs/metadata/src/reader.rs b/crates/libs/metadata/src/reader.rs index 51b4056d98..b8dbeb314c 100644 --- a/crates/libs/metadata/src/reader.rs +++ b/crates/libs/metadata/src/reader.rs @@ -16,6 +16,8 @@ pub struct Reader { // TODO: riddle should just avoid nested structs nested: HashMap>, + // The reader needs to store the filter since standalone code generation needs more than just the filtered items + // in order to chase dependencies automatically. This is why `Reader::filter` can't just filter everything up front. filter: Filter, } @@ -170,11 +172,10 @@ impl Reader { if let Some(def) = self.get_type_def(full_name.namespace, full_name.name).next() { Type::TypeDef(def, Vec::new()) } else { - Type::TypeRef(code) + Type::TypeRef(full_name) } } - // TODO: this shouldn't be public pub fn type_from_blob(&self, blob: &mut Blob, enclosing: Option, generics: &[Type]) -> Type { // Used by WinRT to indicate that a struct input parameter is passed by reference rather than by value on the ABI. let is_const = blob.read_modifiers().iter().any(|def| def.type_name() == TypeName::IsConst); @@ -251,4 +252,4 @@ impl Reader { pub const REMAP_TYPES: [(TypeName, TypeName); 2] = [(TypeName::D2D_MATRIX_3X2_F, TypeName::Matrix3x2), (TypeName::D3DMATRIX, TypeName::Matrix4x4)]; // TODO: get rid of at least the second tuple if not the whole thing. -pub const CORE_TYPES: [(TypeName, Type); 11] = [(TypeName::GUID, Type::GUID), (TypeName::IUnknown, Type::IUnknown), (TypeName::HResult, Type::HRESULT), (TypeName::HRESULT, Type::HRESULT), (TypeName::HSTRING, Type::String), (TypeName::BSTR, Type::BSTR), (TypeName::IInspectable, Type::IInspectable), (TypeName::PSTR, Type::PSTR), (TypeName::PWSTR, Type::PWSTR), (TypeName::Type, Type::TypeName), (TypeName::CHAR, Type::U8)]; +pub const CORE_TYPES: [(TypeName, Type); 10] = [(TypeName::GUID, Type::GUID), (TypeName::IUnknown, Type::IUnknown), (TypeName::HResult, Type::HRESULT), (TypeName::HRESULT, Type::HRESULT), (TypeName::HSTRING, Type::String), (TypeName::BSTR, Type::BSTR), (TypeName::IInspectable, Type::IInspectable), (TypeName::PSTR, Type::PSTR), (TypeName::PWSTR, Type::PWSTR), (TypeName::CHAR, Type::U8)]; diff --git a/crates/libs/metadata/src/tables.rs b/crates/libs/metadata/src/tables.rs index dcebe18664..f7c340a4f9 100644 --- a/crates/libs/metadata/src/tables.rs +++ b/crates/libs/metadata/src/tables.rs @@ -48,16 +48,14 @@ impl Attribute { } pub fn name(&self) -> &'static str { - let AttributeType::MemberRef(member) = self.ty(); - assert_eq!(member.name(), ".ctor"); - let MemberRefParent::TypeRef(ty) = member.parent(); + let AttributeType::MemberRef(ctor) = self.ty(); + let MemberRefParent::TypeRef(ty) = ctor.parent(); ty.name() } pub fn type_name(&self) -> TypeName { - let AttributeType::MemberRef(member) = self.ty(); - assert_eq!(member.name(), ".ctor"); - let MemberRefParent::TypeRef(ty) = member.parent(); + let AttributeType::MemberRef(ctor) = self.ty(); + let MemberRefParent::TypeRef(ty) = ctor.parent(); ty.type_name() } @@ -84,7 +82,7 @@ impl Attribute { Type::I64 => Value::I64(values.read_i64()), Type::U64 => Value::U64(values.read_u64()), Type::String => Value::String(values.read_str().to_string()), - Type::TypeName => Value::TypeName(values.read_str().to_string()), + Type::TypeRef(type_name) if type_name == TypeName::Type => Value::TypeName(values.read_str().to_string()), Type::TypeDef(def, _) => Value::EnumDef(def, Box::new(values.read_integer(def.underlying_type()))), rest => unimplemented!("{rest:?}"), }; @@ -117,9 +115,8 @@ impl Attribute { args.push((name.to_string(), arg)); } - // TODO: can we debug assert these? - assert_eq!(sig.slice.len(), 0); - assert_eq!(values.slice.len(), 0); + debug_assert_eq!(sig.slice.len(), 0); + debug_assert_eq!(values.slice.len(), 0); args } diff --git a/crates/libs/metadata/src/type.rs b/crates/libs/metadata/src/type.rs index d0f60e4de2..0f5f9a4b9d 100644 --- a/crates/libs/metadata/src/type.rs +++ b/crates/libs/metadata/src/type.rs @@ -24,11 +24,8 @@ pub enum Type { String, // TODO: Win32 should use System.String when referring to an HSTRING IInspectable, // TODO: Win32 should use System.Object when referring to an IInspectable - // Meta-type indicating type name in attribute blob. - TypeName, - // Regular ECMA-335 types that map to metadata - TypeRef(TypeDefOrRef), // Note: this ought to be a TypeName but that would require Type to have a lifetime reference. + TypeRef(TypeName), GenericParam(GenericParam), TypeDef(TypeDef, Vec), diff --git a/crates/libs/metadata/src/type_name.rs b/crates/libs/metadata/src/type_name.rs index ec599d5920..e6e81ff360 100644 --- a/crates/libs/metadata/src/type_name.rs +++ b/crates/libs/metadata/src/type_name.rs @@ -2,7 +2,7 @@ use super::*; -#[derive(Copy, Clone, PartialEq, Eq)] +#[derive(Copy, Clone, Debug, Hash, PartialEq, Eq, Ord, PartialOrd)] pub struct TypeName { pub namespace: &'static str, pub name: &'static str, From 026b5a4d5e2b3bc2974f81455e8888eaffbc3b52 Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 12:54:48 -0500 Subject: [PATCH 03/13] args --- crates/libs/bindgen/src/metadata.rs | 128 +++++++++--------- crates/libs/bindgen/src/winmd/from_reader.rs | 2 +- crates/libs/metadata/src/blob.rs | 2 +- crates/libs/metadata/src/tables.rs | 12 +- crates/tests/metadata/tests/attribute_enum.rs | 2 +- 5 files changed, 73 insertions(+), 73 deletions(-) diff --git a/crates/libs/bindgen/src/metadata.rs b/crates/libs/bindgen/src/metadata.rs index 8db2636ef7..379b2c6174 100644 --- a/crates/libs/bindgen/src/metadata.rs +++ b/crates/libs/bindgen/src/metadata.rs @@ -437,67 +437,6 @@ pub fn type_def_has_callback(row: TypeDef) -> bool { } } -pub fn type_interfaces(ty: &Type) -> Vec { - // TODO: collect into btree map and then return collected vec - // This will both sort the results and should make finding dupes faster - fn walk(result: &mut Vec, parent: &Type, is_base: bool) { - if let Type::TypeDef(row, generics) = parent { - for imp in row.interface_impls() { - let mut child = Interface { ty: imp.ty(generics), kind: if imp.has_attribute("DefaultAttribute") { InterfaceKind::Default } else { InterfaceKind::None } }; - - child.kind = if !is_base && child.kind == InterfaceKind::Default { - InterfaceKind::Default - } else if child.kind == InterfaceKind::Overridable { - continue; - } else if is_base { - InterfaceKind::Base - } else { - InterfaceKind::None - }; - let mut found = false; - for existing in result.iter_mut() { - if existing.ty == child.ty { - found = true; - if child.kind == InterfaceKind::Default { - existing.kind = child.kind - } - } - } - if !found { - walk(result, &child.ty, is_base); - result.push(child); - } - } - } - } - let mut result = Vec::new(); - walk(&mut result, ty, false); - if let Type::TypeDef(row, _) = ty { - if row.kind() == TypeKind::Class { - for base in type_def_bases(*row) { - walk(&mut result, &Type::TypeDef(base, Vec::new()), true); - } - for attribute in row.attributes() { - match attribute.name() { - "StaticAttribute" | "ActivatableAttribute" => { - for (_, arg) in attribute.args() { - if let Value::TypeName(type_name) = arg { - let type_name = parse_type_name(&type_name); - let def = row.reader().get_type_def(type_name.0, type_name.1).next().expect("Type not found"); - result.push(Interface { ty: Type::TypeDef(def, Vec::new()), kind: InterfaceKind::Static }); - break; - } - } - } - _ => {} - } - } - } - } - result.sort_by(|a, b| type_name(&a.ty).cmp(type_name(&b.ty))); - result -} - fn type_name(ty: &Type) -> &str { match ty { Type::TypeDef(row, _) => row.name(), @@ -658,7 +597,7 @@ pub fn type_def_has_packing(row: TypeDef) -> bool { } pub fn type_def_default_interface(row: TypeDef) -> Option { - row.interface_impls().find_map(move |row| if row.has_attribute("DefaultAttribute") { Some(row.ty(&[])) } else { None }) + row.interface_impls().find_map(move |imp| if imp.has_attribute("DefaultAttribute") { Some(imp.ty(&[])) } else { None }) } fn type_signature(ty: &Type) -> String { @@ -792,7 +731,7 @@ pub fn type_def_vtables(row: TypeDef) -> Vec { } } else { let mut next = row; - while let Some(base) = type_def_interfaces(next, &[]).next() { + while let Some(base) = next.interface_impls().map(move |imp| imp.ty(&[])).next() { match base { Type::TypeDef(row, _) => { next = row; @@ -815,5 +754,66 @@ pub fn type_def_vtables(row: TypeDef) -> Vec { } pub fn type_def_interfaces(row: TypeDef, generics: &[Type]) -> impl Iterator + '_ { - row.interface_impls().map(move |row| row.ty(generics)) + row.interface_impls().map(move |imp| imp.ty(generics)) +} + +pub fn type_interfaces(ty: &Type) -> Vec { + // TODO: collect into btree map and then return collected vec + // This will both sort the results and should make finding dupes faster + fn walk(result: &mut Vec, parent: &Type, is_base: bool) { + if let Type::TypeDef(row, generics) = parent { + for imp in row.interface_impls() { + let mut child = Interface { ty: imp.ty(generics), kind: if imp.has_attribute("DefaultAttribute") { InterfaceKind::Default } else { InterfaceKind::None } }; + + child.kind = if !is_base && child.kind == InterfaceKind::Default { + InterfaceKind::Default + } else if child.kind == InterfaceKind::Overridable { + continue; + } else if is_base { + InterfaceKind::Base + } else { + InterfaceKind::None + }; + let mut found = false; + for existing in result.iter_mut() { + if existing.ty == child.ty { + found = true; + if child.kind == InterfaceKind::Default { + existing.kind = child.kind + } + } + } + if !found { + walk(result, &child.ty, is_base); + result.push(child); + } + } + } + } + let mut result = Vec::new(); + walk(&mut result, ty, false); + if let Type::TypeDef(row, _) = ty { + if row.kind() == TypeKind::Class { + for base in type_def_bases(*row) { + walk(&mut result, &Type::TypeDef(base, Vec::new()), true); + } + for attribute in row.attributes() { + match attribute.name() { + "StaticAttribute" | "ActivatableAttribute" => { + for (_, arg) in attribute.args() { + if let Value::TypeName(type_name) = arg { + let type_name = parse_type_name(&type_name); + let def = row.reader().get_type_def(type_name.0, type_name.1).next().expect("Type not found"); + result.push(Interface { ty: Type::TypeDef(def, Vec::new()), kind: InterfaceKind::Static }); + break; + } + } + } + _ => {} + } + } + } + } + result.sort_by(|a, b| type_name(&a.ty).cmp(type_name(&b.ty))); + result } diff --git a/crates/libs/bindgen/src/winmd/from_reader.rs b/crates/libs/bindgen/src/winmd/from_reader.rs index ab0fc8d64f..fa6843435d 100644 --- a/crates/libs/bindgen/src/winmd/from_reader.rs +++ b/crates/libs/bindgen/src/winmd/from_reader.rs @@ -36,7 +36,7 @@ pub fn from_reader(reader: &metadata::Reader, config: std::collections::BTreeMap for generic in def.generics() { writer.tables.GenericParam.push(writer::GenericParam { - Number: generic.number(), + Number: generic.number(), // TODO: isn't this just going to be incremental? Flags: 0, Owner: writer::TypeOrMethodDef::TypeDef(writer.tables.TypeDef.len() as u32 - 1).encode(), Name: writer.strings.insert(generic.name()), diff --git a/crates/libs/metadata/src/blob.rs b/crates/libs/metadata/src/blob.rs index 7d37bfefeb..398d5fb3b3 100644 --- a/crates/libs/metadata/src/blob.rs +++ b/crates/libs/metadata/src/blob.rs @@ -58,7 +58,7 @@ impl Blob { mods } - pub fn read_str(&mut self) -> &str { + pub fn read_str(&mut self) -> &'static str { let len = self.read_usize(); let value = unsafe { std::str::from_utf8_unchecked(&self.slice[..len]) }; self.offset(len); diff --git a/crates/libs/metadata/src/tables.rs b/crates/libs/metadata/src/tables.rs index f7c340a4f9..1f0f3b0c83 100644 --- a/crates/libs/metadata/src/tables.rs +++ b/crates/libs/metadata/src/tables.rs @@ -59,7 +59,7 @@ impl Attribute { ty.type_name() } - pub fn args(&self) -> Vec<(String, Value)> { + pub fn args(&self) -> Vec<(&'static str, Value)> { let AttributeType::MemberRef(member) = self.ty(); let mut sig = member.blob(2); let mut values = self.blob(2); @@ -67,7 +67,7 @@ impl Attribute { let _this_and_gen_param_count = sig.read_usize(); let fixed_arg_count = sig.read_usize(); let _ret_type = sig.read_usize(); - let mut args: Vec<(String, Value)> = Vec::with_capacity(fixed_arg_count); + let mut args = Vec::with_capacity(fixed_arg_count); let reader = self.reader(); for _ in 0..fixed_arg_count { @@ -87,7 +87,7 @@ impl Attribute { rest => unimplemented!("{rest:?}"), }; - args.push((String::new(), arg)); + args.push(("", arg)); } let named_arg_count = values.read_u16(); @@ -96,7 +96,7 @@ impl Attribute { for _ in 0..named_arg_count { let _id = values.read_u8(); let arg_type = values.read_u8(); - let mut name = values.read_str().to_string(); + let mut name = values.read_str(); let arg = match arg_type { ELEMENT_TYPE_BOOLEAN => Value::Bool(values.read_bool()), ELEMENT_TYPE_I2 => Value::I16(values.read_i16()), @@ -107,12 +107,12 @@ impl Attribute { 0x55 => { let type_name = parse_type_name(&name); let def = reader.get_type_def(type_name.0, type_name.1).next().expect("Type not found"); - name = values.read_str().to_string(); + name = values.read_str(); Value::EnumDef(def, Box::new(values.read_integer(def.underlying_type()))) } rest => unimplemented!("{rest:?}"), }; - args.push((name.to_string(), arg)); + args.push((name, arg)); } debug_assert_eq!(sig.slice.len(), 0); diff --git a/crates/tests/metadata/tests/attribute_enum.rs b/crates/tests/metadata/tests/attribute_enum.rs index 6944bbf45e..6692e3d028 100644 --- a/crates/tests/metadata/tests/attribute_enum.rs +++ b/crates/tests/metadata/tests/attribute_enum.rs @@ -37,7 +37,7 @@ fn check_attr_arg_enum(attr: Attribute, arg_name: &str, expected_type: &str, exp let (_, value) = attr .args() .drain(..) - .find(|(name, _)| name == arg_name) + .find(|(name, _)| *name == arg_name) .unwrap(); if let Value::EnumDef(ty, value) = value { From 9b1f1b52572fdde0b1ba43c7f9f180f7e264bdf4 Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 13:07:11 -0500 Subject: [PATCH 04/13] TypeName --- crates/libs/bindgen/src/metadata.rs | 5 ++--- crates/libs/metadata/src/lib.rs | 2 +- crates/libs/metadata/src/tables.rs | 4 ++-- crates/libs/metadata/src/type_name.rs | 5 +++++ 4 files changed, 10 insertions(+), 6 deletions(-) diff --git a/crates/libs/bindgen/src/metadata.rs b/crates/libs/bindgen/src/metadata.rs index 379b2c6174..3654725b21 100644 --- a/crates/libs/bindgen/src/metadata.rs +++ b/crates/libs/bindgen/src/metadata.rs @@ -80,7 +80,7 @@ pub enum AsyncKind { pub struct Guid(pub u32, pub u16, pub u16, pub u8, pub u8, pub u8, pub u8, pub u8, pub u8, pub u8, pub u8); impl Guid { - pub fn from_args(args: &[(String, Value)]) -> Self { + pub fn from_args(args: &[(&str, Value)]) -> Self { fn unwrap_u32(value: &Value) -> u32 { match value { Value::U32(value) => *value, @@ -802,8 +802,7 @@ pub fn type_interfaces(ty: &Type) -> Vec { "StaticAttribute" | "ActivatableAttribute" => { for (_, arg) in attribute.args() { if let Value::TypeName(type_name) = arg { - let type_name = parse_type_name(&type_name); - let def = row.reader().get_type_def(type_name.0, type_name.1).next().expect("Type not found"); + let def = row.reader().get_type_def(type_name.namespace, type_name.name).next().expect("Type not found"); result.push(Interface { ty: Type::TypeDef(def, Vec::new()), kind: InterfaceKind::Static }); break; } diff --git a/crates/libs/metadata/src/lib.rs b/crates/libs/metadata/src/lib.rs index 0b71ca397c..2484783510 100644 --- a/crates/libs/metadata/src/lib.rs +++ b/crates/libs/metadata/src/lib.rs @@ -95,7 +95,7 @@ pub enum Value { F32(f32), F64(f64), String(String), - TypeName(String), + TypeName(TypeName), EnumDef(TypeDef, Box), } diff --git a/crates/libs/metadata/src/tables.rs b/crates/libs/metadata/src/tables.rs index 1f0f3b0c83..89b1edad75 100644 --- a/crates/libs/metadata/src/tables.rs +++ b/crates/libs/metadata/src/tables.rs @@ -82,7 +82,7 @@ impl Attribute { Type::I64 => Value::I64(values.read_i64()), Type::U64 => Value::U64(values.read_u64()), Type::String => Value::String(values.read_str().to_string()), - Type::TypeRef(type_name) if type_name == TypeName::Type => Value::TypeName(values.read_str().to_string()), + Type::TypeRef(type_name) if type_name == TypeName::Type => Value::TypeName(TypeName::parse(values.read_str())), Type::TypeDef(def, _) => Value::EnumDef(def, Box::new(values.read_integer(def.underlying_type()))), rest => unimplemented!("{rest:?}"), }; @@ -103,7 +103,7 @@ impl Attribute { ELEMENT_TYPE_I4 => Value::I32(values.read_i32()), ELEMENT_TYPE_U4 => Value::U32(values.read_u32()), ELEMENT_TYPE_STRING => Value::String(values.read_str().to_string()), - 0x50 => Value::TypeName(values.read_str().to_string()), + 0x50 => Value::TypeName(TypeName::parse(values.read_str())), 0x55 => { let type_name = parse_type_name(&name); let def = reader.get_type_def(type_name.0, type_name.1).next().expect("Type not found"); diff --git a/crates/libs/metadata/src/type_name.rs b/crates/libs/metadata/src/type_name.rs index e6e81ff360..8b4459293f 100644 --- a/crates/libs/metadata/src/type_name.rs +++ b/crates/libs/metadata/src/type_name.rs @@ -56,6 +56,11 @@ impl TypeName { pub fn new(namespace: &'static str, name: &'static str) -> Self { Self { namespace, name: trim_tick(name) } } + + pub fn parse(full_name: &'static str) -> Self { + let index = full_name.rfind('.').expect("Expected full name separated with `.`"); + Self::new(&full_name[0..index], &full_name[index + 1..]) + } } impl std::fmt::Display for TypeName { From 6820565f9bf36a304d439ddb87473942ad3e0345 Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 13:08:54 -0500 Subject: [PATCH 05/13] parse --- crates/libs/metadata/src/filter.rs | 4 ++-- crates/libs/metadata/src/tables.rs | 4 ++-- crates/libs/metadata/src/type_name.rs | 5 ----- 3 files changed, 4 insertions(+), 9 deletions(-) diff --git a/crates/libs/metadata/src/filter.rs b/crates/libs/metadata/src/filter.rs index 057586a6c9..5405266b56 100644 --- a/crates/libs/metadata/src/filter.rs +++ b/crates/libs/metadata/src/filter.rs @@ -83,8 +83,8 @@ mod tests { use super::*; fn includes_type_name(filter: &Filter, full_name: &'static str) -> bool { - let type_name = crate::parse_type_name(full_name); - filter.includes_type_name(type_name.0, type_name.1) + let type_name = TypeName::parse(full_name); + filter.includes_type_name(type_name.namespace, type_name.name) } #[test] diff --git a/crates/libs/metadata/src/tables.rs b/crates/libs/metadata/src/tables.rs index 89b1edad75..eefc93f1af 100644 --- a/crates/libs/metadata/src/tables.rs +++ b/crates/libs/metadata/src/tables.rs @@ -105,8 +105,8 @@ impl Attribute { ELEMENT_TYPE_STRING => Value::String(values.read_str().to_string()), 0x50 => Value::TypeName(TypeName::parse(values.read_str())), 0x55 => { - let type_name = parse_type_name(&name); - let def = reader.get_type_def(type_name.0, type_name.1).next().expect("Type not found"); + let type_name = TypeName::parse(&name); + let def = reader.get_type_def(type_name.namespace, type_name.name).next().expect("Type not found"); name = values.read_str(); Value::EnumDef(def, Box::new(values.read_integer(def.underlying_type()))) } diff --git a/crates/libs/metadata/src/type_name.rs b/crates/libs/metadata/src/type_name.rs index 8b4459293f..2740b153d4 100644 --- a/crates/libs/metadata/src/type_name.rs +++ b/crates/libs/metadata/src/type_name.rs @@ -68,8 +68,3 @@ impl std::fmt::Display for TypeName { write!(fmt, "{}.{}", self.namespace, self.name) } } - -pub fn parse_type_name(full_name: &str) -> (&str, &str) { - let index = full_name.rfind('.').expect("Expected full name separated with `.`"); - (&full_name[0..index], &full_name[index + 1..]) -} From 0a9b37b26a98f25f24a3140372a1939522111acf Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 13:09:28 -0500 Subject: [PATCH 06/13] clippy --- crates/libs/metadata/src/tables.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crates/libs/metadata/src/tables.rs b/crates/libs/metadata/src/tables.rs index eefc93f1af..094e195e34 100644 --- a/crates/libs/metadata/src/tables.rs +++ b/crates/libs/metadata/src/tables.rs @@ -105,7 +105,7 @@ impl Attribute { ELEMENT_TYPE_STRING => Value::String(values.read_str().to_string()), 0x50 => Value::TypeName(TypeName::parse(values.read_str())), 0x55 => { - let type_name = TypeName::parse(&name); + let type_name = TypeName::parse(name); let def = reader.get_type_def(type_name.namespace, type_name.name).next().expect("Type not found"); name = values.read_str(); Value::EnumDef(def, Box::new(values.read_integer(def.underlying_type()))) From 4eb070c61dbb4dc93f11710c7d27a563766bc939 Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 14:12:03 -0500 Subject: [PATCH 07/13] type --- crates/libs/bindgen/src/rdl/from_reader.rs | 1 + crates/libs/bindgen/src/rust/cfg.rs | 2 ++ crates/libs/bindgen/src/winmd/from_reader.rs | 1 + crates/libs/bindgen/src/winmd/verify.rs | 4 ++-- crates/libs/bindgen/src/winmd/writer/mod.rs | 5 +++++ crates/libs/bindgen/src/winmd/writer/type.rs | 1 + crates/libs/metadata/src/reader.rs | 2 +- crates/libs/metadata/src/tables.rs | 2 +- crates/libs/metadata/src/type.rs | 1 + 9 files changed, 15 insertions(+), 4 deletions(-) diff --git a/crates/libs/bindgen/src/rdl/from_reader.rs b/crates/libs/bindgen/src/rdl/from_reader.rs index 02b4f55435..b5bc799604 100644 --- a/crates/libs/bindgen/src/rdl/from_reader.rs +++ b/crates/libs/bindgen/src/rdl/from_reader.rs @@ -378,6 +378,7 @@ impl Writer { metadata::Type::PCWSTR => quote! { PCWSTR }, metadata::Type::BSTR => quote! { BSTR }, metadata::Type::PrimitiveOrEnum(_, ty) => self.ty(ty), + rest => unimplemented!("{rest:?}"), } } diff --git a/crates/libs/bindgen/src/rust/cfg.rs b/crates/libs/bindgen/src/rust/cfg.rs index 59989f08a0..fdbfeb9e30 100644 --- a/crates/libs/bindgen/src/rust/cfg.rs +++ b/crates/libs/bindgen/src/rust/cfg.rs @@ -58,6 +58,8 @@ pub fn type_def_cfg_impl(def: TypeDef, generics: &[Type]) -> Cfg { combine(def, generics, &mut cfg); + // TODO: why do we need both type_def_interfaces for WinRT and type_def_vtables for Win32? + for def in type_def_vtables(def) { if let Type::TypeDef(def, generics) = def { combine(def, &generics, &mut cfg); diff --git a/crates/libs/bindgen/src/winmd/from_reader.rs b/crates/libs/bindgen/src/winmd/from_reader.rs index fa6843435d..bfe8aa3d09 100644 --- a/crates/libs/bindgen/src/winmd/from_reader.rs +++ b/crates/libs/bindgen/src/winmd/from_reader.rs @@ -122,6 +122,7 @@ fn winmd_type(ty: &metadata::Type) -> winmd::Type { metadata::Type::PCSTR => winmd::Type::PCSTR, metadata::Type::PCWSTR => winmd::Type::PCWSTR, metadata::Type::BSTR => winmd::Type::BSTR, + metadata::Type::Type => winmd::Type::Type, metadata::Type::TypeDef(def, generics) => winmd::Type::TypeRef(winmd::TypeName { namespace: def.namespace().to_string(), name: def.name().to_string(), generics: generics.iter().map(winmd_type).collect() }), metadata::Type::GenericParam(generic) => winmd::Type::GenericParam(generic.number()), metadata::Type::ConstRef(ty) => winmd::Type::ConstRef(Box::new(winmd_type(ty))), diff --git a/crates/libs/bindgen/src/winmd/verify.rs b/crates/libs/bindgen/src/winmd/verify.rs index 7fcf385256..c376d9d19d 100644 --- a/crates/libs/bindgen/src/winmd/verify.rs +++ b/crates/libs/bindgen/src/winmd/verify.rs @@ -35,8 +35,8 @@ pub fn verify(reader: &metadata::Reader) -> crate::Result<()> { } fn not_type_ref(ty: &metadata::Type) -> crate::Result<()> { - if let metadata::Type::TypeRef(type_name) = ty { - return Err(crate::Error::new(&format!("missing type definition `{}`", type_name))); + if let metadata::Type::TypeRef(ty) = ty { + return Err(crate::Error::new(&format!("missing type definition `{}`", ty))); } Ok(()) } diff --git a/crates/libs/bindgen/src/winmd/writer/mod.rs b/crates/libs/bindgen/src/winmd/writer/mod.rs index 25170bb9e3..7ac4a08d61 100644 --- a/crates/libs/bindgen/src/winmd/writer/mod.rs +++ b/crates/libs/bindgen/src/winmd/writer/mod.rs @@ -234,6 +234,11 @@ impl Writer { usize_blob(1, blob); // count usize_blob(*bounds, blob); } + Type::Type => { + let code = self.insert_type_ref("System", "Type"); + blob.push(metadata::ELEMENT_TYPE_CLASS); + usize_blob(code as usize, blob); + } Type::MutPtr(ty, pointers) | Type::ConstPtr(ty, pointers) => { for _ in 0..*pointers { usize_blob(metadata::ELEMENT_TYPE_PTR as usize, blob); diff --git a/crates/libs/bindgen/src/winmd/writer/type.rs b/crates/libs/bindgen/src/winmd/writer/type.rs index 33482dc20b..c0668c8ab9 100644 --- a/crates/libs/bindgen/src/winmd/writer/type.rs +++ b/crates/libs/bindgen/src/winmd/writer/type.rs @@ -34,6 +34,7 @@ pub enum Type { PCSTR, PCWSTR, BSTR, + Type, TypeRef(TypeName), GenericParam(u16), MutPtr(Box, usize), diff --git a/crates/libs/metadata/src/reader.rs b/crates/libs/metadata/src/reader.rs index b8dbeb314c..d60c98ba95 100644 --- a/crates/libs/metadata/src/reader.rs +++ b/crates/libs/metadata/src/reader.rs @@ -252,4 +252,4 @@ impl Reader { pub const REMAP_TYPES: [(TypeName, TypeName); 2] = [(TypeName::D2D_MATRIX_3X2_F, TypeName::Matrix3x2), (TypeName::D3DMATRIX, TypeName::Matrix4x4)]; // TODO: get rid of at least the second tuple if not the whole thing. -pub const CORE_TYPES: [(TypeName, Type); 10] = [(TypeName::GUID, Type::GUID), (TypeName::IUnknown, Type::IUnknown), (TypeName::HResult, Type::HRESULT), (TypeName::HRESULT, Type::HRESULT), (TypeName::HSTRING, Type::String), (TypeName::BSTR, Type::BSTR), (TypeName::IInspectable, Type::IInspectable), (TypeName::PSTR, Type::PSTR), (TypeName::PWSTR, Type::PWSTR), (TypeName::CHAR, Type::U8)]; +pub const CORE_TYPES: [(TypeName, Type); 11] = [(TypeName::GUID, Type::GUID), (TypeName::IUnknown, Type::IUnknown), (TypeName::HResult, Type::HRESULT), (TypeName::HRESULT, Type::HRESULT), (TypeName::HSTRING, Type::String), (TypeName::BSTR, Type::BSTR), (TypeName::IInspectable, Type::IInspectable), (TypeName::PSTR, Type::PSTR), (TypeName::PWSTR, Type::PWSTR), (TypeName::Type, Type::Type), (TypeName::CHAR, Type::U8)]; diff --git a/crates/libs/metadata/src/tables.rs b/crates/libs/metadata/src/tables.rs index 094e195e34..4efce73d81 100644 --- a/crates/libs/metadata/src/tables.rs +++ b/crates/libs/metadata/src/tables.rs @@ -82,7 +82,7 @@ impl Attribute { Type::I64 => Value::I64(values.read_i64()), Type::U64 => Value::U64(values.read_u64()), Type::String => Value::String(values.read_str().to_string()), - Type::TypeRef(type_name) if type_name == TypeName::Type => Value::TypeName(TypeName::parse(values.read_str())), + Type::Type => Value::TypeName(TypeName::parse(values.read_str())), Type::TypeDef(def, _) => Value::EnumDef(def, Box::new(values.read_integer(def.underlying_type()))), rest => unimplemented!("{rest:?}"), }; diff --git a/crates/libs/metadata/src/type.rs b/crates/libs/metadata/src/type.rs index 0f5f9a4b9d..5c2c24eb61 100644 --- a/crates/libs/metadata/src/type.rs +++ b/crates/libs/metadata/src/type.rs @@ -23,6 +23,7 @@ pub enum Type { GUID, // Both Win32 and WinRT agree that this is represented by System.Guid String, // TODO: Win32 should use System.String when referring to an HSTRING IInspectable, // TODO: Win32 should use System.Object when referring to an IInspectable + Type, // System.Type is needed since WinRT attribute use this as a parameter type. // Regular ECMA-335 types that map to metadata TypeRef(TypeName), From 022b7ce94a83d0f241c2135219d3e995e4e4dcdd Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 14:13:03 -0500 Subject: [PATCH 08/13] diff --- crates/libs/bindgen/src/metadata.rs | 120 ++++++++++++++-------------- 1 file changed, 60 insertions(+), 60 deletions(-) diff --git a/crates/libs/bindgen/src/metadata.rs b/crates/libs/bindgen/src/metadata.rs index 3654725b21..d650ecbc4f 100644 --- a/crates/libs/bindgen/src/metadata.rs +++ b/crates/libs/bindgen/src/metadata.rs @@ -437,6 +437,66 @@ pub fn type_def_has_callback(row: TypeDef) -> bool { } } +pub fn type_interfaces(ty: &Type) -> Vec { + // TODO: collect into btree map and then return collected vec + // This will both sort the results and should make finding dupes faster + fn walk(result: &mut Vec, parent: &Type, is_base: bool) { + if let Type::TypeDef(row, generics) = parent { + for imp in row.interface_impls() { + let mut child = Interface { ty: imp.ty(generics), kind: if imp.has_attribute("DefaultAttribute") { InterfaceKind::Default } else { InterfaceKind::None } }; + + child.kind = if !is_base && child.kind == InterfaceKind::Default { + InterfaceKind::Default + } else if child.kind == InterfaceKind::Overridable { + continue; + } else if is_base { + InterfaceKind::Base + } else { + InterfaceKind::None + }; + let mut found = false; + for existing in result.iter_mut() { + if existing.ty == child.ty { + found = true; + if child.kind == InterfaceKind::Default { + existing.kind = child.kind + } + } + } + if !found { + walk(result, &child.ty, is_base); + result.push(child); + } + } + } + } + let mut result = Vec::new(); + walk(&mut result, ty, false); + if let Type::TypeDef(row, _) = ty { + if row.kind() == TypeKind::Class { + for base in type_def_bases(*row) { + walk(&mut result, &Type::TypeDef(base, Vec::new()), true); + } + for attribute in row.attributes() { + match attribute.name() { + "StaticAttribute" | "ActivatableAttribute" => { + for (_, arg) in attribute.args() { + if let Value::TypeName(type_name) = arg { + let def = row.reader().get_type_def(type_name.namespace, type_name.name).next().expect("Type not found"); + result.push(Interface { ty: Type::TypeDef(def, Vec::new()), kind: InterfaceKind::Static }); + break; + } + } + } + _ => {} + } + } + } + } + result.sort_by(|a, b| type_name(&a.ty).cmp(type_name(&b.ty))); + result +} + fn type_name(ty: &Type) -> &str { match ty { Type::TypeDef(row, _) => row.name(), @@ -756,63 +816,3 @@ pub fn type_def_vtables(row: TypeDef) -> Vec { pub fn type_def_interfaces(row: TypeDef, generics: &[Type]) -> impl Iterator + '_ { row.interface_impls().map(move |imp| imp.ty(generics)) } - -pub fn type_interfaces(ty: &Type) -> Vec { - // TODO: collect into btree map and then return collected vec - // This will both sort the results and should make finding dupes faster - fn walk(result: &mut Vec, parent: &Type, is_base: bool) { - if let Type::TypeDef(row, generics) = parent { - for imp in row.interface_impls() { - let mut child = Interface { ty: imp.ty(generics), kind: if imp.has_attribute("DefaultAttribute") { InterfaceKind::Default } else { InterfaceKind::None } }; - - child.kind = if !is_base && child.kind == InterfaceKind::Default { - InterfaceKind::Default - } else if child.kind == InterfaceKind::Overridable { - continue; - } else if is_base { - InterfaceKind::Base - } else { - InterfaceKind::None - }; - let mut found = false; - for existing in result.iter_mut() { - if existing.ty == child.ty { - found = true; - if child.kind == InterfaceKind::Default { - existing.kind = child.kind - } - } - } - if !found { - walk(result, &child.ty, is_base); - result.push(child); - } - } - } - } - let mut result = Vec::new(); - walk(&mut result, ty, false); - if let Type::TypeDef(row, _) = ty { - if row.kind() == TypeKind::Class { - for base in type_def_bases(*row) { - walk(&mut result, &Type::TypeDef(base, Vec::new()), true); - } - for attribute in row.attributes() { - match attribute.name() { - "StaticAttribute" | "ActivatableAttribute" => { - for (_, arg) in attribute.args() { - if let Value::TypeName(type_name) = arg { - let def = row.reader().get_type_def(type_name.namespace, type_name.name).next().expect("Type not found"); - result.push(Interface { ty: Type::TypeDef(def, Vec::new()), kind: InterfaceKind::Static }); - break; - } - } - } - _ => {} - } - } - } - } - result.sort_by(|a, b| type_name(&a.ty).cmp(type_name(&b.ty))); - result -} From 417d8b9a59547714ed2b0f829b1b40613f803767 Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 14:36:34 -0500 Subject: [PATCH 09/13] cfg --- crates/libs/bindgen/src/rust/cfg.rs | 15 +++------------ 1 file changed, 3 insertions(+), 12 deletions(-) diff --git a/crates/libs/bindgen/src/rust/cfg.rs b/crates/libs/bindgen/src/rust/cfg.rs index fdbfeb9e30..c1ed18553f 100644 --- a/crates/libs/bindgen/src/rust/cfg.rs +++ b/crates/libs/bindgen/src/rust/cfg.rs @@ -58,22 +58,12 @@ pub fn type_def_cfg_impl(def: TypeDef, generics: &[Type]) -> Cfg { combine(def, generics, &mut cfg); - // TODO: why do we need both type_def_interfaces for WinRT and type_def_vtables for Win32? - - for def in type_def_vtables(def) { - if let Type::TypeDef(def, generics) = def { + for interface in type_interfaces(&Type::TypeDef(def, generics.to_vec())) { + if let Type::TypeDef(def, generics) = interface.ty { combine(def, &generics, &mut cfg); } } - if def.flags().contains(TypeAttributes::WindowsRuntime) { - for interface in type_def_interfaces(def, generics) { - if let Type::TypeDef(def, generics) = interface { - combine(def, &generics, &mut cfg); - } - } - } - cfg_add_attributes(&mut cfg, def); cfg } @@ -158,6 +148,7 @@ pub fn type_cfg(ty: &Type) -> Cfg { type_cfg_combine(ty, &mut cfg); cfg } + fn type_cfg_combine(ty: &Type, cfg: &mut Cfg) { match ty { Type::TypeDef(row, generics) => type_def_cfg_combine(*row, generics, cfg), From d35c2833cfb91028cfea0d3e4733db658a165908 Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 14:41:43 -0500 Subject: [PATCH 10/13] simpler --- crates/libs/bindgen/src/metadata.rs | 4 ---- crates/libs/bindgen/src/rust/writer.rs | 2 +- 2 files changed, 1 insertion(+), 5 deletions(-) diff --git a/crates/libs/bindgen/src/metadata.rs b/crates/libs/bindgen/src/metadata.rs index d650ecbc4f..9a60ddc868 100644 --- a/crates/libs/bindgen/src/metadata.rs +++ b/crates/libs/bindgen/src/metadata.rs @@ -812,7 +812,3 @@ pub fn type_def_vtables(row: TypeDef) -> Vec { } result } - -pub fn type_def_interfaces(row: TypeDef, generics: &[Type]) -> impl Iterator + '_ { - row.interface_impls().map(move |imp| imp.ty(generics)) -} diff --git a/crates/libs/bindgen/src/rust/writer.rs b/crates/libs/bindgen/src/rust/writer.rs index 7e66c93457..4911244bbb 100644 --- a/crates/libs/bindgen/src/rust/writer.rs +++ b/crates/libs/bindgen/src/rust/writer.rs @@ -568,7 +568,7 @@ impl Writer { let mut async_generics = generics.to_vec(); if kind == AsyncKind::None { - for interface in type_def_interfaces(def, generics) { + for interface in def.interface_impls().map(move |imp| imp.ty(generics)) { if let Type::TypeDef(interface_def, interface_generics) = &interface { kind = type_def_async_kind(*interface_def); if kind != AsyncKind::None { From 16420a59333cbf626c3d6472cc1d8c60c641f40a Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 19:40:01 -0500 Subject: [PATCH 11/13] class --- crates/libs/bindgen/src/rust/classes.rs | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/crates/libs/bindgen/src/rust/classes.rs b/crates/libs/bindgen/src/rust/classes.rs index 865b6734fb..d30cde31a2 100644 --- a/crates/libs/bindgen/src/rust/classes.rs +++ b/crates/libs/bindgen/src/rust/classes.rs @@ -2,7 +2,7 @@ use super::*; pub fn writer(writer: &Writer, def: TypeDef) -> TokenStream { if writer.sys { - if type_def_has_default_interface(def) { + if def.interface_impls().next().is_some() { let name = to_ident(def.name()); quote! { pub type #name = *mut ::core::ffi::c_void; @@ -64,7 +64,7 @@ fn gen_class(writer: &Writer, def: TypeDef) -> TokenStream { _ => None, }); - if type_def_has_default_interface(def) { + if def.interface_impls().next().is_some() { let new = if type_def_has_default_constructor(def) { quote! { pub fn new() -> ::windows_core::Result { @@ -172,10 +172,6 @@ fn type_def_has_default_constructor(row: TypeDef) -> bool { false } -fn type_def_has_default_interface(row: TypeDef) -> bool { - row.interface_impls().any(|imp| imp.has_attribute("DefaultAttribute")) -} - fn type_is_exclusive(ty: &Type) -> bool { match ty { Type::TypeDef(row, _) => type_def_is_exclusive(*row), From 9dd482e49b82ad0eae23ac750a41dadf8d2006cd Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 20:47:04 -0500 Subject: [PATCH 12/13] type_def_interfaces --- crates/libs/bindgen/src/metadata.rs | 13 +++++++++---- crates/libs/bindgen/src/rdl/from_reader.rs | 9 ++++----- crates/libs/bindgen/src/winmd/from_reader.rs | 5 ++--- 3 files changed, 15 insertions(+), 12 deletions(-) diff --git a/crates/libs/bindgen/src/metadata.rs b/crates/libs/bindgen/src/metadata.rs index 9a60ddc868..0a0dff5b07 100644 --- a/crates/libs/bindgen/src/metadata.rs +++ b/crates/libs/bindgen/src/metadata.rs @@ -442,9 +442,7 @@ pub fn type_interfaces(ty: &Type) -> Vec { // This will both sort the results and should make finding dupes faster fn walk(result: &mut Vec, parent: &Type, is_base: bool) { if let Type::TypeDef(row, generics) = parent { - for imp in row.interface_impls() { - let mut child = Interface { ty: imp.ty(generics), kind: if imp.has_attribute("DefaultAttribute") { InterfaceKind::Default } else { InterfaceKind::None } }; - + for mut child in type_def_interfaces(*row, generics) { child.kind = if !is_base && child.kind == InterfaceKind::Default { InterfaceKind::Default } else if child.kind == InterfaceKind::Overridable { @@ -656,8 +654,15 @@ pub fn type_def_has_packing(row: TypeDef) -> bool { } } +pub fn type_def_interfaces(def: TypeDef, generics: &[Type]) -> impl Iterator + '_ { + def.interface_impls().map(|imp| { + let kind = if imp.has_attribute("DefaultAttribute") { InterfaceKind::Default } else { InterfaceKind::None }; + Interface { kind, ty: imp.ty(generics) } + }) +} + pub fn type_def_default_interface(row: TypeDef) -> Option { - row.interface_impls().find_map(move |imp| if imp.has_attribute("DefaultAttribute") { Some(imp.ty(&[])) } else { None }) + type_def_interfaces(row, &[]).find_map(move |interface| if interface.kind == InterfaceKind::Default { Some(interface.ty) } else { None }) } fn type_signature(ty: &Type) -> String { diff --git a/crates/libs/bindgen/src/rdl/from_reader.rs b/crates/libs/bindgen/src/rdl/from_reader.rs index b5bc799604..856e8a131c 100644 --- a/crates/libs/bindgen/src/rdl/from_reader.rs +++ b/crates/libs/bindgen/src/rdl/from_reader.rs @@ -286,12 +286,11 @@ impl Writer { // TODO: then list default interface first // Then everything else - for imp in def.interface_impls() { - let ty = imp.ty(generics); - if imp.has_attribute("DefaultAttribute") { - types.insert(0, self.ty(&ty)); + for interface in type_def_interfaces(def, generics) { + if interface.kind == InterfaceKind::Default { + types.insert(0, self.ty(&interface.ty)); } else { - types.push(self.ty(&ty)); + types.push(self.ty(&interface.ty)); } } diff --git a/crates/libs/bindgen/src/winmd/from_reader.rs b/crates/libs/bindgen/src/winmd/from_reader.rs index bfe8aa3d09..ed5f4d1783 100644 --- a/crates/libs/bindgen/src/winmd/from_reader.rs +++ b/crates/libs/bindgen/src/winmd/from_reader.rs @@ -43,9 +43,8 @@ pub fn from_reader(reader: &metadata::Reader, config: std::collections::BTreeMap }); } - for imp in def.interface_impls() { - let ty = imp.ty(generics); - let ty = winmd_type(&ty); + for interface in metadata::type_def_interfaces(def, generics) { + let ty = winmd_type(&interface.ty); let reference = match &ty { winmd::Type::TypeRef(type_name) if type_name.generics.is_empty() => writer.insert_type_ref(&type_name.namespace, &type_name.name), From 891c4bbbac07d54dfe93f2533d732e68fd22a250 Mon Sep 17 00:00:00 2001 From: Kenny Kerr Date: Thu, 19 Oct 2023 21:49:47 -0500 Subject: [PATCH 13/13] test --- crates/libs/metadata/src/filter.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/crates/libs/metadata/src/filter.rs b/crates/libs/metadata/src/filter.rs index 5405266b56..f9efcfe7e1 100644 --- a/crates/libs/metadata/src/filter.rs +++ b/crates/libs/metadata/src/filter.rs @@ -83,7 +83,7 @@ mod tests { use super::*; fn includes_type_name(filter: &Filter, full_name: &'static str) -> bool { - let type_name = TypeName::parse(full_name); + let type_name = crate::TypeName::parse(full_name); filter.includes_type_name(type_name.namespace, type_name.name) }