summaryrefslogtreecommitdiffstats
path: root/third_party/rust/uniffi_meta
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/rust/uniffi_meta')
-rw-r--r--third_party/rust/uniffi_meta/.cargo-checksum.json1
-rw-r--r--third_party/rust/uniffi_meta/Cargo.toml35
-rw-r--r--third_party/rust/uniffi_meta/src/ffi_names.rs66
-rw-r--r--third_party/rust/uniffi_meta/src/group.rs273
-rw-r--r--third_party/rust/uniffi_meta/src/lib.rs544
-rw-r--r--third_party/rust/uniffi_meta/src/metadata.rs87
-rw-r--r--third_party/rust/uniffi_meta/src/reader.rs465
-rw-r--r--third_party/rust/uniffi_meta/src/types.rs172
8 files changed, 1643 insertions, 0 deletions
diff --git a/third_party/rust/uniffi_meta/.cargo-checksum.json b/third_party/rust/uniffi_meta/.cargo-checksum.json
new file mode 100644
index 0000000000..cb02cde83f
--- /dev/null
+++ b/third_party/rust/uniffi_meta/.cargo-checksum.json
@@ -0,0 +1 @@
+{"files":{"Cargo.toml":"cb9f8aad563572bd4f12ee234ede6773f189a79ba5bd3bfd7622d3c0ec49d6a3","src/ffi_names.rs":"422bbe9d49d5476de752a9f9b2330f59b37a79e67f19a828caceb64d1bdabff8","src/group.rs":"ae996e6b9f83d459af04eb392e36487d0fe19c7328a395823186cce76a0955ff","src/lib.rs":"a442e2271a0eb538ec1d4fc7573a3acc7e5f366e2b2ac8d0e659fd998fd7d995","src/metadata.rs":"4ae425a8eab7b8c19a6b96c914f2c02c5bee00358888fd55b936fd1fd175a93c","src/reader.rs":"57fb771584491b8e90b01c68f9d53bac7cfa3135888e11e24e14b59312185ff9","src/types.rs":"8c155ed1301e11a365863989e29c2271149048092fb7052ec145f58c948482d5"},"package":"71dc8573a7b1ac4b71643d6da34888273ebfc03440c525121f1b3634ad3417a2"} \ No newline at end of file
diff --git a/third_party/rust/uniffi_meta/Cargo.toml b/third_party/rust/uniffi_meta/Cargo.toml
new file mode 100644
index 0000000000..34999eee18
--- /dev/null
+++ b/third_party/rust/uniffi_meta/Cargo.toml
@@ -0,0 +1,35 @@
+# THIS FILE IS AUTOMATICALLY GENERATED BY CARGO
+#
+# When uploading crates to the registry Cargo will automatically
+# "normalize" Cargo.toml files for maximal compatibility
+# with all versions of Cargo and also rewrite `path` dependencies
+# to registry (e.g., crates.io) dependencies.
+#
+# If you are reading this file be aware that the original Cargo.toml
+# will likely look very different (and much more reasonable).
+# See Cargo.toml.orig for the original contents.
+
+[package]
+edition = "2021"
+name = "uniffi_meta"
+version = "0.25.3"
+description = "uniffi_meta"
+homepage = "https://mozilla.github.io/uniffi-rs"
+keywords = [
+ "ffi",
+ "bindgen",
+]
+license = "MPL-2.0"
+repository = "https://github.com/mozilla/uniffi-rs"
+
+[dependencies.anyhow]
+version = "1"
+
+[dependencies.bytes]
+version = "1.3"
+
+[dependencies.siphasher]
+version = "0.3"
+
+[dependencies.uniffi_checksum_derive]
+version = "0.25.3"
diff --git a/third_party/rust/uniffi_meta/src/ffi_names.rs b/third_party/rust/uniffi_meta/src/ffi_names.rs
new file mode 100644
index 0000000000..44a5bc3e63
--- /dev/null
+++ b/third_party/rust/uniffi_meta/src/ffi_names.rs
@@ -0,0 +1,66 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+//! Functions to calculate names for FFI symbols
+//!
+//! All of these functions input a `namespace` parameter which is:
+//! - The UDL namespace for UDL-based generation
+//! - The "module path" of the item for proc-macro based generation. Right now this is actually just the
+//! crate name, but we eventually hope to make this the full module path.
+//!
+//! This could cause collisions in the case where you combine UDL and proc-macro generation and you
+//! set the UDL namespace to the name of another crate. This seems so pathological that it's not
+//! worth the code complexity to prevent it.
+
+/// FFI symbol name for a top-level function
+pub fn fn_symbol_name(namespace: &str, name: &str) -> String {
+ let name = name.to_ascii_lowercase();
+ format!("uniffi_{namespace}_fn_func_{name}")
+}
+
+/// FFI symbol name for an object constructor
+pub fn constructor_symbol_name(namespace: &str, object_name: &str, name: &str) -> String {
+ let object_name = object_name.to_ascii_lowercase();
+ let name = name.to_ascii_lowercase();
+ format!("uniffi_{namespace}_fn_constructor_{object_name}_{name}")
+}
+
+/// FFI symbol name for an object method
+pub fn method_symbol_name(namespace: &str, object_name: &str, name: &str) -> String {
+ let object_name = object_name.to_ascii_lowercase();
+ let name = name.to_ascii_lowercase();
+ format!("uniffi_{namespace}_fn_method_{object_name}_{name}")
+}
+
+/// FFI symbol name for the `free` function for an object.
+pub fn free_fn_symbol_name(namespace: &str, object_name: &str) -> String {
+ let object_name = object_name.to_ascii_lowercase();
+ format!("uniffi_{namespace}_fn_free_{object_name}")
+}
+
+/// FFI symbol name for the `init_callback` function for a callback interface
+pub fn init_callback_fn_symbol_name(namespace: &str, callback_interface_name: &str) -> String {
+ let callback_interface_name = callback_interface_name.to_ascii_lowercase();
+ format!("uniffi_{namespace}_fn_init_callback_{callback_interface_name}")
+}
+
+/// FFI checksum symbol name for a top-level function
+pub fn fn_checksum_symbol_name(namespace: &str, name: &str) -> String {
+ let name = name.to_ascii_lowercase();
+ format!("uniffi_{namespace}_checksum_func_{name}")
+}
+
+/// FFI checksum symbol name for an object constructor
+pub fn constructor_checksum_symbol_name(namespace: &str, object_name: &str, name: &str) -> String {
+ let object_name = object_name.to_ascii_lowercase();
+ let name = name.to_ascii_lowercase();
+ format!("uniffi_{namespace}_checksum_constructor_{object_name}_{name}")
+}
+
+/// FFI checksum symbol name for an object method
+pub fn method_checksum_symbol_name(namespace: &str, object_name: &str, name: &str) -> String {
+ let object_name = object_name.to_ascii_lowercase();
+ let name = name.to_ascii_lowercase();
+ format!("uniffi_{namespace}_checksum_method_{object_name}_{name}")
+}
diff --git a/third_party/rust/uniffi_meta/src/group.rs b/third_party/rust/uniffi_meta/src/group.rs
new file mode 100644
index 0000000000..f0be2e5a98
--- /dev/null
+++ b/third_party/rust/uniffi_meta/src/group.rs
@@ -0,0 +1,273 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+use std::collections::{BTreeSet, HashMap};
+
+use crate::*;
+use anyhow::{bail, Result};
+
+type MetadataGroupMap = HashMap<String, MetadataGroup>;
+
+// Create empty metadata groups based on the metadata items.
+pub fn create_metadata_groups(items: &[Metadata]) -> MetadataGroupMap {
+ // Map crate names to MetadataGroup instances
+ items
+ .iter()
+ .filter_map(|i| match i {
+ Metadata::Namespace(namespace) => {
+ let group = MetadataGroup {
+ namespace: namespace.clone(),
+ items: BTreeSet::new(),
+ };
+ Some((namespace.crate_name.clone(), group))
+ }
+ Metadata::UdlFile(udl) => {
+ let namespace = NamespaceMetadata {
+ crate_name: udl.module_path.clone(),
+ name: udl.namespace.clone(),
+ };
+ let group = MetadataGroup {
+ namespace,
+ items: BTreeSet::new(),
+ };
+ Some((udl.module_path.clone(), group))
+ }
+ _ => None,
+ })
+ .collect::<HashMap<_, _>>()
+}
+
+/// Consume the items into the previously created metadata groups.
+pub fn group_metadata(group_map: &mut MetadataGroupMap, items: Vec<Metadata>) -> Result<()> {
+ for item in items {
+ if matches!(&item, Metadata::Namespace(_)) {
+ continue;
+ }
+
+ let crate_name = calc_crate_name(item.module_path()).to_owned(); // XXX - kill clone?
+
+ let item = fixup_external_type(item, group_map);
+ let group = match group_map.get_mut(&crate_name) {
+ Some(ns) => ns,
+ None => bail!("Unknown namespace for {item:?} ({crate_name})"),
+ };
+ if group.items.contains(&item) {
+ bail!("Duplicate metadata item: {item:?}");
+ }
+ group.add_item(item);
+ }
+ Ok(())
+}
+
+#[derive(Debug)]
+pub struct MetadataGroup {
+ pub namespace: NamespaceMetadata,
+ pub items: BTreeSet<Metadata>,
+}
+
+impl MetadataGroup {
+ pub fn add_item(&mut self, item: Metadata) {
+ self.items.insert(item);
+ }
+}
+
+pub fn fixup_external_type(item: Metadata, group_map: &MetadataGroupMap) -> Metadata {
+ let crate_name = calc_crate_name(item.module_path()).to_owned();
+ let converter = ExternalTypeConverter {
+ crate_name: &crate_name,
+ crate_to_namespace: group_map,
+ };
+ converter.convert_item(item)
+}
+
+/// Convert metadata items by replacing types from external crates with Type::External
+struct ExternalTypeConverter<'a> {
+ crate_name: &'a str,
+ crate_to_namespace: &'a MetadataGroupMap,
+}
+
+impl<'a> ExternalTypeConverter<'a> {
+ fn crate_to_namespace(&self, crate_name: &str) -> String {
+ self.crate_to_namespace
+ .get(crate_name)
+ .unwrap_or_else(|| panic!("Can't find namespace for module {crate_name}"))
+ .namespace
+ .name
+ .clone()
+ }
+
+ fn convert_item(&self, item: Metadata) -> Metadata {
+ match item {
+ Metadata::Func(meta) => Metadata::Func(FnMetadata {
+ inputs: self.convert_params(meta.inputs),
+ return_type: self.convert_optional(meta.return_type),
+ throws: self.convert_optional(meta.throws),
+ ..meta
+ }),
+ Metadata::Method(meta) => Metadata::Method(MethodMetadata {
+ inputs: self.convert_params(meta.inputs),
+ return_type: self.convert_optional(meta.return_type),
+ throws: self.convert_optional(meta.throws),
+ ..meta
+ }),
+ Metadata::TraitMethod(meta) => Metadata::TraitMethod(TraitMethodMetadata {
+ inputs: self.convert_params(meta.inputs),
+ return_type: self.convert_optional(meta.return_type),
+ throws: self.convert_optional(meta.throws),
+ ..meta
+ }),
+ Metadata::Constructor(meta) => Metadata::Constructor(ConstructorMetadata {
+ inputs: self.convert_params(meta.inputs),
+ throws: self.convert_optional(meta.throws),
+ ..meta
+ }),
+ Metadata::Record(meta) => Metadata::Record(RecordMetadata {
+ fields: self.convert_fields(meta.fields),
+ ..meta
+ }),
+ Metadata::Enum(meta) => Metadata::Enum(self.convert_enum(meta)),
+ Metadata::Error(meta) => Metadata::Error(match meta {
+ ErrorMetadata::Enum { enum_, is_flat } => ErrorMetadata::Enum {
+ enum_: self.convert_enum(enum_),
+ is_flat,
+ },
+ }),
+ _ => item,
+ }
+ }
+
+ fn convert_params(&self, params: Vec<FnParamMetadata>) -> Vec<FnParamMetadata> {
+ params
+ .into_iter()
+ .map(|param| FnParamMetadata {
+ ty: self.convert_type(param.ty),
+ ..param
+ })
+ .collect()
+ }
+
+ fn convert_fields(&self, fields: Vec<FieldMetadata>) -> Vec<FieldMetadata> {
+ fields
+ .into_iter()
+ .map(|field| FieldMetadata {
+ ty: self.convert_type(field.ty),
+ ..field
+ })
+ .collect()
+ }
+
+ fn convert_enum(&self, enum_: EnumMetadata) -> EnumMetadata {
+ EnumMetadata {
+ variants: enum_
+ .variants
+ .into_iter()
+ .map(|variant| VariantMetadata {
+ fields: self.convert_fields(variant.fields),
+ ..variant
+ })
+ .collect(),
+ ..enum_
+ }
+ }
+
+ fn convert_optional(&self, ty: Option<Type>) -> Option<Type> {
+ ty.map(|ty| self.convert_type(ty))
+ }
+
+ fn convert_type(&self, ty: Type) -> Type {
+ match ty {
+ // Convert `ty` if it's external
+ Type::Enum { module_path, name } | Type::Record { module_path, name }
+ if self.is_module_path_external(&module_path) =>
+ {
+ Type::External {
+ namespace: self.crate_to_namespace(&module_path),
+ module_path,
+ name,
+ kind: ExternalKind::DataClass,
+ tagged: false,
+ }
+ }
+ Type::Custom {
+ module_path, name, ..
+ } if self.is_module_path_external(&module_path) => {
+ // For now, it's safe to assume that all custom types are data classes.
+ // There's no reason to use a custom type with an interface.
+ Type::External {
+ namespace: self.crate_to_namespace(&module_path),
+ module_path,
+ name,
+ kind: ExternalKind::DataClass,
+ tagged: false,
+ }
+ }
+ Type::Object {
+ module_path, name, ..
+ } if self.is_module_path_external(&module_path) => Type::External {
+ namespace: self.crate_to_namespace(&module_path),
+ module_path,
+ name,
+ kind: ExternalKind::Interface,
+ tagged: false,
+ },
+ Type::CallbackInterface { module_path, name }
+ if self.is_module_path_external(&module_path) =>
+ {
+ panic!("External callback interfaces not supported ({name})")
+ }
+ // Convert child types
+ Type::Custom {
+ module_path,
+ name,
+ builtin,
+ ..
+ } => Type::Custom {
+ module_path,
+ name,
+ builtin: Box::new(self.convert_type(*builtin)),
+ },
+ Type::Optional { inner_type } => Type::Optional {
+ inner_type: Box::new(self.convert_type(*inner_type)),
+ },
+ Type::Sequence { inner_type } => Type::Sequence {
+ inner_type: Box::new(self.convert_type(*inner_type)),
+ },
+ Type::Map {
+ key_type,
+ value_type,
+ } => Type::Map {
+ key_type: Box::new(self.convert_type(*key_type)),
+ value_type: Box::new(self.convert_type(*value_type)),
+ },
+ // Existing External types probably need namespace fixed.
+ Type::External {
+ namespace,
+ module_path,
+ name,
+ kind,
+ tagged,
+ } => {
+ assert!(namespace.is_empty());
+ Type::External {
+ namespace: self.crate_to_namespace(&module_path),
+ module_path,
+ name,
+ kind,
+ tagged,
+ }
+ }
+
+ // Otherwise, just return the type unchanged
+ _ => ty,
+ }
+ }
+
+ fn is_module_path_external(&self, module_path: &str) -> bool {
+ calc_crate_name(module_path) != self.crate_name
+ }
+}
+
+fn calc_crate_name(module_path: &str) -> &str {
+ module_path.split("::").next().unwrap()
+}
diff --git a/third_party/rust/uniffi_meta/src/lib.rs b/third_party/rust/uniffi_meta/src/lib.rs
new file mode 100644
index 0000000000..e486d84d89
--- /dev/null
+++ b/third_party/rust/uniffi_meta/src/lib.rs
@@ -0,0 +1,544 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+use std::{collections::BTreeMap, hash::Hasher};
+pub use uniffi_checksum_derive::Checksum;
+
+mod ffi_names;
+pub use ffi_names::*;
+
+mod group;
+pub use group::{create_metadata_groups, fixup_external_type, group_metadata, MetadataGroup};
+
+mod reader;
+pub use reader::{read_metadata, read_metadata_type};
+
+mod types;
+pub use types::{AsType, ExternalKind, ObjectImpl, Type, TypeIterator};
+
+mod metadata;
+
+// This needs to match the minor version of the `uniffi` crate. See
+// `docs/uniffi-versioning.md` for details.
+//
+// Once we get to 1.0, then we'll need to update the scheme to something like 100 + major_version
+pub const UNIFFI_CONTRACT_VERSION: u32 = 24;
+
+/// Similar to std::hash::Hash.
+///
+/// Implementations of this trait are expected to update the hasher state in
+/// the same way across platforms. #[derive(Checksum)] will do the right thing.
+pub trait Checksum {
+ fn checksum<H: Hasher>(&self, state: &mut H);
+}
+
+impl Checksum for bool {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ state.write_u8(*self as u8);
+ }
+}
+
+impl Checksum for u64 {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ state.write(&self.to_le_bytes());
+ }
+}
+
+impl Checksum for i64 {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ state.write(&self.to_le_bytes());
+ }
+}
+
+impl<T: Checksum> Checksum for Box<T> {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ (**self).checksum(state)
+ }
+}
+
+impl<T: Checksum> Checksum for [T] {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ state.write(&(self.len() as u64).to_le_bytes());
+ for item in self {
+ Checksum::checksum(item, state);
+ }
+ }
+}
+
+impl<T: Checksum> Checksum for Vec<T> {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ Checksum::checksum(&**self, state);
+ }
+}
+
+impl<K: Checksum, V: Checksum> Checksum for BTreeMap<K, V> {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ state.write(&(self.len() as u64).to_le_bytes());
+ for (key, value) in self {
+ Checksum::checksum(key, state);
+ Checksum::checksum(value, state);
+ }
+ }
+}
+
+impl<T: Checksum> Checksum for Option<T> {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ match self {
+ None => state.write(&0u64.to_le_bytes()),
+ Some(value) => {
+ state.write(&1u64.to_le_bytes());
+ Checksum::checksum(value, state)
+ }
+ }
+ }
+}
+
+impl Checksum for str {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ state.write(self.as_bytes());
+ state.write_u8(0xff);
+ }
+}
+
+impl Checksum for String {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ (**self).checksum(state)
+ }
+}
+
+impl Checksum for &str {
+ fn checksum<H: Hasher>(&self, state: &mut H) {
+ (**self).checksum(state)
+ }
+}
+
+// The namespace of a Component interface.
+//
+// This is used to match up the macro metadata with the UDL items.
+#[derive(Clone, Debug, Default, PartialEq, Eq, PartialOrd, Ord)]
+pub struct NamespaceMetadata {
+ pub crate_name: String,
+ pub name: String,
+}
+
+// UDL file included with `include_scaffolding!()`
+//
+// This is to find the UDL files in library mode generation
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct UdlFile {
+ // The module path specified when the UDL file was parsed.
+ pub module_path: String,
+ pub namespace: String,
+ // the base filename of the udl file - no path, no extension.
+ pub file_stub: String,
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct FnMetadata {
+ pub module_path: String,
+ pub name: String,
+ pub is_async: bool,
+ pub inputs: Vec<FnParamMetadata>,
+ pub return_type: Option<Type>,
+ pub throws: Option<Type>,
+ pub checksum: Option<u16>,
+}
+
+impl FnMetadata {
+ pub fn ffi_symbol_name(&self) -> String {
+ fn_symbol_name(&self.module_path, &self.name)
+ }
+
+ pub fn checksum_symbol_name(&self) -> String {
+ fn_checksum_symbol_name(&self.module_path, &self.name)
+ }
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct ConstructorMetadata {
+ pub module_path: String,
+ pub self_name: String,
+ pub name: String,
+ pub inputs: Vec<FnParamMetadata>,
+ pub throws: Option<Type>,
+ pub checksum: Option<u16>,
+}
+
+impl ConstructorMetadata {
+ pub fn ffi_symbol_name(&self) -> String {
+ constructor_symbol_name(&self.module_path, &self.self_name, &self.name)
+ }
+
+ pub fn checksum_symbol_name(&self) -> String {
+ constructor_checksum_symbol_name(&self.module_path, &self.self_name, &self.name)
+ }
+
+ pub fn is_primary(&self) -> bool {
+ self.name == "new"
+ }
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct MethodMetadata {
+ pub module_path: String,
+ pub self_name: String,
+ pub name: String,
+ pub is_async: bool,
+ pub inputs: Vec<FnParamMetadata>,
+ pub return_type: Option<Type>,
+ pub throws: Option<Type>,
+ pub takes_self_by_arc: bool, // unused except by rust udl bindgen.
+ pub checksum: Option<u16>,
+}
+
+impl MethodMetadata {
+ pub fn ffi_symbol_name(&self) -> String {
+ method_symbol_name(&self.module_path, &self.self_name, &self.name)
+ }
+
+ pub fn checksum_symbol_name(&self) -> String {
+ method_checksum_symbol_name(&self.module_path, &self.self_name, &self.name)
+ }
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct TraitMethodMetadata {
+ pub module_path: String,
+ pub trait_name: String,
+ // Note: the position of `index` is important since it causes callback interface methods to be
+ // ordered correctly in MetadataGroup.items
+ pub index: u32,
+ pub name: String,
+ pub is_async: bool,
+ pub inputs: Vec<FnParamMetadata>,
+ pub return_type: Option<Type>,
+ pub throws: Option<Type>,
+ pub takes_self_by_arc: bool, // unused except by rust udl bindgen.
+ pub checksum: Option<u16>,
+}
+
+impl TraitMethodMetadata {
+ pub fn ffi_symbol_name(&self) -> String {
+ method_symbol_name(&self.module_path, &self.trait_name, &self.name)
+ }
+
+ pub fn checksum_symbol_name(&self) -> String {
+ method_checksum_symbol_name(&self.module_path, &self.trait_name, &self.name)
+ }
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct FnParamMetadata {
+ pub name: String,
+ pub ty: Type,
+ pub by_ref: bool,
+ pub optional: bool,
+ pub default: Option<LiteralMetadata>,
+}
+
+impl FnParamMetadata {
+ pub fn simple(name: &str, ty: Type) -> Self {
+ Self {
+ name: name.to_string(),
+ ty,
+ by_ref: false,
+ optional: false,
+ default: None,
+ }
+ }
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord, Checksum)]
+pub enum LiteralMetadata {
+ Boolean(bool),
+ String(String),
+ // Integers are represented as the widest representation we can.
+ // Number formatting vary with language and radix, so we avoid a lot of parsing and
+ // formatting duplication by using only signed and unsigned variants.
+ UInt(u64, Radix, Type),
+ Int(i64, Radix, Type),
+ // Pass the string representation through as typed in the UDL.
+ // This avoids a lot of uncertainty around precision and accuracy,
+ // though bindings for languages less sophisticated number parsing than WebIDL
+ // will have to do extra work.
+ Float(String, Type),
+ Enum(String, Type),
+ EmptySequence,
+ EmptyMap,
+ Null,
+}
+
+// Represent the radix of integer literal values.
+// We preserve the radix into the generated bindings for readability reasons.
+#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Checksum)]
+pub enum Radix {
+ Decimal = 10,
+ Octal = 8,
+ Hexadecimal = 16,
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct RecordMetadata {
+ pub module_path: String,
+ pub name: String,
+ pub fields: Vec<FieldMetadata>,
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct FieldMetadata {
+ pub name: String,
+ pub ty: Type,
+ pub default: Option<LiteralMetadata>,
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct EnumMetadata {
+ pub module_path: String,
+ pub name: String,
+ pub variants: Vec<VariantMetadata>,
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct VariantMetadata {
+ pub name: String,
+ pub fields: Vec<FieldMetadata>,
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct ObjectMetadata {
+ pub module_path: String,
+ pub name: String,
+ pub imp: types::ObjectImpl,
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct CallbackInterfaceMetadata {
+ pub module_path: String,
+ pub name: String,
+}
+
+impl ObjectMetadata {
+ /// FFI symbol name for the `free` function for this object.
+ ///
+ /// This function is used to free the memory used by this object.
+ pub fn free_ffi_symbol_name(&self) -> String {
+ free_fn_symbol_name(&self.module_path, &self.name)
+ }
+}
+
+/// The list of traits we support generating helper methods for.
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub enum UniffiTraitMetadata {
+ Debug {
+ fmt: MethodMetadata,
+ },
+ Display {
+ fmt: MethodMetadata,
+ },
+ Eq {
+ eq: MethodMetadata,
+ ne: MethodMetadata,
+ },
+ Hash {
+ hash: MethodMetadata,
+ },
+}
+
+impl UniffiTraitMetadata {
+ fn module_path(&self) -> &String {
+ &match self {
+ UniffiTraitMetadata::Debug { fmt } => fmt,
+ UniffiTraitMetadata::Display { fmt } => fmt,
+ UniffiTraitMetadata::Eq { eq, .. } => eq,
+ UniffiTraitMetadata::Hash { hash } => hash,
+ }
+ .module_path
+ }
+
+ pub fn self_name(&self) -> &String {
+ &match self {
+ UniffiTraitMetadata::Debug { fmt } => fmt,
+ UniffiTraitMetadata::Display { fmt } => fmt,
+ UniffiTraitMetadata::Eq { eq, .. } => eq,
+ UniffiTraitMetadata::Hash { hash } => hash,
+ }
+ .self_name
+ }
+}
+
+#[repr(u8)]
+pub enum UniffiTraitDiscriminants {
+ Debug,
+ Display,
+ Eq,
+ Hash,
+}
+
+impl UniffiTraitDiscriminants {
+ pub fn from(v: u8) -> anyhow::Result<Self> {
+ Ok(match v {
+ 0 => UniffiTraitDiscriminants::Debug,
+ 1 => UniffiTraitDiscriminants::Display,
+ 2 => UniffiTraitDiscriminants::Eq,
+ 3 => UniffiTraitDiscriminants::Hash,
+ _ => anyhow::bail!("invalid trait discriminant {v}"),
+ })
+ }
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub enum ErrorMetadata {
+ Enum { enum_: EnumMetadata, is_flat: bool },
+}
+
+impl ErrorMetadata {
+ pub fn name(&self) -> &String {
+ match self {
+ Self::Enum { enum_, .. } => &enum_.name,
+ }
+ }
+
+ pub fn module_path(&self) -> &String {
+ match self {
+ Self::Enum { enum_, .. } => &enum_.module_path,
+ }
+ }
+}
+
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub struct CustomTypeMetadata {
+ pub module_path: String,
+ pub name: String,
+ pub builtin: Type,
+}
+
+/// Returns the last 16 bits of the value's hash as computed with [`SipHasher13`].
+///
+/// This is used as a safeguard against different UniFFI versions being used for scaffolding and
+/// bindings generation.
+pub fn checksum<T: Checksum>(val: &T) -> u16 {
+ let mut hasher = siphasher::sip::SipHasher13::new();
+ val.checksum(&mut hasher);
+ (hasher.finish() & 0x000000000000FFFF) as u16
+}
+
+/// Enum covering all the possible metadata types
+#[derive(Clone, Debug, PartialEq, Eq, PartialOrd, Ord)]
+pub enum Metadata {
+ Namespace(NamespaceMetadata),
+ UdlFile(UdlFile),
+ Func(FnMetadata),
+ Object(ObjectMetadata),
+ CallbackInterface(CallbackInterfaceMetadata),
+ Record(RecordMetadata),
+ Enum(EnumMetadata),
+ Error(ErrorMetadata),
+ Constructor(ConstructorMetadata),
+ Method(MethodMetadata),
+ TraitMethod(TraitMethodMetadata),
+ CustomType(CustomTypeMetadata),
+ UniffiTrait(UniffiTraitMetadata),
+}
+
+impl Metadata {
+ pub fn read(data: &[u8]) -> anyhow::Result<Self> {
+ read_metadata(data)
+ }
+
+ pub(crate) fn module_path(&self) -> &String {
+ match self {
+ Metadata::Namespace(meta) => &meta.crate_name,
+ Metadata::UdlFile(meta) => &meta.module_path,
+ Metadata::Func(meta) => &meta.module_path,
+ Metadata::Constructor(meta) => &meta.module_path,
+ Metadata::Method(meta) => &meta.module_path,
+ Metadata::Record(meta) => &meta.module_path,
+ Metadata::Enum(meta) => &meta.module_path,
+ Metadata::Object(meta) => &meta.module_path,
+ Metadata::CallbackInterface(meta) => &meta.module_path,
+ Metadata::TraitMethod(meta) => &meta.module_path,
+ Metadata::Error(meta) => meta.module_path(),
+ Metadata::CustomType(meta) => &meta.module_path,
+ Metadata::UniffiTrait(meta) => meta.module_path(),
+ }
+ }
+}
+
+impl From<NamespaceMetadata> for Metadata {
+ fn from(value: NamespaceMetadata) -> Metadata {
+ Self::Namespace(value)
+ }
+}
+
+impl From<UdlFile> for Metadata {
+ fn from(value: UdlFile) -> Metadata {
+ Self::UdlFile(value)
+ }
+}
+
+impl From<FnMetadata> for Metadata {
+ fn from(value: FnMetadata) -> Metadata {
+ Self::Func(value)
+ }
+}
+
+impl From<ConstructorMetadata> for Metadata {
+ fn from(c: ConstructorMetadata) -> Self {
+ Self::Constructor(c)
+ }
+}
+
+impl From<MethodMetadata> for Metadata {
+ fn from(m: MethodMetadata) -> Self {
+ Self::Method(m)
+ }
+}
+
+impl From<RecordMetadata> for Metadata {
+ fn from(r: RecordMetadata) -> Self {
+ Self::Record(r)
+ }
+}
+
+impl From<EnumMetadata> for Metadata {
+ fn from(e: EnumMetadata) -> Self {
+ Self::Enum(e)
+ }
+}
+
+impl From<ErrorMetadata> for Metadata {
+ fn from(e: ErrorMetadata) -> Self {
+ Self::Error(e)
+ }
+}
+
+impl From<ObjectMetadata> for Metadata {
+ fn from(v: ObjectMetadata) -> Self {
+ Self::Object(v)
+ }
+}
+
+impl From<CallbackInterfaceMetadata> for Metadata {
+ fn from(v: CallbackInterfaceMetadata) -> Self {
+ Self::CallbackInterface(v)
+ }
+}
+
+impl From<TraitMethodMetadata> for Metadata {
+ fn from(v: TraitMethodMetadata) -> Self {
+ Self::TraitMethod(v)
+ }
+}
+
+impl From<CustomTypeMetadata> for Metadata {
+ fn from(v: CustomTypeMetadata) -> Self {
+ Self::CustomType(v)
+ }
+}
+
+impl From<UniffiTraitMetadata> for Metadata {
+ fn from(v: UniffiTraitMetadata) -> Self {
+ Self::UniffiTrait(v)
+ }
+}
diff --git a/third_party/rust/uniffi_meta/src/metadata.rs b/third_party/rust/uniffi_meta/src/metadata.rs
new file mode 100644
index 0000000000..6e490a4866
--- /dev/null
+++ b/third_party/rust/uniffi_meta/src/metadata.rs
@@ -0,0 +1,87 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+// Copied from uniffi_core/src/metadata.rs
+// Due to a [Rust bug](https://github.com/rust-lang/rust/issues/113104) we don't want to pull in
+// `uniffi_core`.
+// This is the easy way out of that issue and is a temporary hacky solution.
+
+/// Metadata constants, make sure to keep this in sync with copy in `uniffi_meta::reader`
+pub mod codes {
+ // Top-level metadata item codes
+ pub const FUNC: u8 = 0;
+ pub const METHOD: u8 = 1;
+ pub const RECORD: u8 = 2;
+ pub const ENUM: u8 = 3;
+ pub const INTERFACE: u8 = 4;
+ pub const ERROR: u8 = 5;
+ pub const NAMESPACE: u8 = 6;
+ pub const CONSTRUCTOR: u8 = 7;
+ pub const UDL_FILE: u8 = 8;
+ pub const CALLBACK_INTERFACE: u8 = 9;
+ pub const TRAIT_METHOD: u8 = 10;
+ pub const UNIFFI_TRAIT: u8 = 11;
+ //pub const UNKNOWN: u8 = 255;
+
+ // Type codes
+ pub const TYPE_U8: u8 = 0;
+ pub const TYPE_U16: u8 = 1;
+ pub const TYPE_U32: u8 = 2;
+ pub const TYPE_U64: u8 = 3;
+ pub const TYPE_I8: u8 = 4;
+ pub const TYPE_I16: u8 = 5;
+ pub const TYPE_I32: u8 = 6;
+ pub const TYPE_I64: u8 = 7;
+ pub const TYPE_F32: u8 = 8;
+ pub const TYPE_F64: u8 = 9;
+ pub const TYPE_BOOL: u8 = 10;
+ pub const TYPE_STRING: u8 = 11;
+ pub const TYPE_OPTION: u8 = 12;
+ pub const TYPE_RECORD: u8 = 13;
+ pub const TYPE_ENUM: u8 = 14;
+ // 15 no longer used.
+ pub const TYPE_INTERFACE: u8 = 16;
+ pub const TYPE_VEC: u8 = 17;
+ pub const TYPE_HASH_MAP: u8 = 18;
+ pub const TYPE_SYSTEM_TIME: u8 = 19;
+ pub const TYPE_DURATION: u8 = 20;
+ pub const TYPE_CALLBACK_INTERFACE: u8 = 21;
+ pub const TYPE_CUSTOM: u8 = 22;
+ pub const TYPE_RESULT: u8 = 23;
+ //pub const TYPE_FUTURE: u8 = 24;
+ pub const TYPE_FOREIGN_EXECUTOR: u8 = 25;
+ pub const TYPE_UNIT: u8 = 255;
+
+ // Literal codes
+ pub const LIT_STR: u8 = 0;
+ pub const LIT_INT: u8 = 1;
+ pub const LIT_FLOAT: u8 = 2;
+ pub const LIT_BOOL: u8 = 3;
+ pub const LIT_NULL: u8 = 4;
+}
+
+// Create a checksum for a MetadataBuffer
+//
+// This is used by the bindings code to verify that the library they link to is the same one
+// that the bindings were generated from.
+pub const fn checksum_metadata(buf: &[u8]) -> u16 {
+ calc_checksum(buf, buf.len())
+}
+
+const fn calc_checksum(bytes: &[u8], size: usize) -> u16 {
+ // Taken from the fnv_hash() function from the FNV crate (https://github.com/servo/rust-fnv/blob/master/lib.rs).
+ // fnv_hash() hasn't been released in a version yet.
+ const INITIAL_STATE: u64 = 0xcbf29ce484222325;
+ const PRIME: u64 = 0x100000001b3;
+
+ let mut hash = INITIAL_STATE;
+ let mut i = 0;
+ while i < size {
+ hash ^= bytes[i] as u64;
+ hash = hash.wrapping_mul(PRIME);
+ i += 1;
+ }
+ // Convert the 64-bit hash to a 16-bit hash by XORing everything together
+ (hash ^ (hash >> 16) ^ (hash >> 32) ^ (hash >> 48)) as u16
+}
diff --git a/third_party/rust/uniffi_meta/src/reader.rs b/third_party/rust/uniffi_meta/src/reader.rs
new file mode 100644
index 0000000000..bf6525f2b5
--- /dev/null
+++ b/third_party/rust/uniffi_meta/src/reader.rs
@@ -0,0 +1,465 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+use crate::metadata::{checksum_metadata, codes};
+use crate::*;
+use anyhow::{bail, ensure, Context, Result};
+
+pub fn read_metadata(data: &[u8]) -> Result<Metadata> {
+ MetadataReader::new(data).read_metadata()
+}
+
+// Read a metadata type, this is pub so that we can test it in the metadata fixture
+pub fn read_metadata_type(data: &[u8]) -> Result<Type> {
+ MetadataReader::new(data).read_type()
+}
+
+/// Helper struct for read_metadata()
+struct MetadataReader<'a> {
+ // This points to the initial data we were passed in
+ initial_data: &'a [u8],
+ // This points to the remaining data to be read
+ buf: &'a [u8],
+}
+
+impl<'a> MetadataReader<'a> {
+ fn new(data: &'a [u8]) -> Self {
+ Self {
+ initial_data: data,
+ buf: data,
+ }
+ }
+
+ // Read a top-level metadata item
+ //
+ // This consumes self because MetadataReader is only intended to read a single item.
+ fn read_metadata(mut self) -> Result<Metadata> {
+ let value = self.read_u8()?;
+ Ok(match value {
+ codes::NAMESPACE => NamespaceMetadata {
+ crate_name: self.read_string()?,
+ name: self.read_string()?,
+ }
+ .into(),
+ codes::UDL_FILE => UdlFile {
+ module_path: self.read_string()?,
+ namespace: self.read_string()?,
+ file_stub: self.read_string()?,
+ }
+ .into(),
+ codes::FUNC => self.read_func()?.into(),
+ codes::CONSTRUCTOR => self.read_constructor()?.into(),
+ codes::METHOD => self.read_method()?.into(),
+ codes::RECORD => self.read_record()?.into(),
+ codes::ENUM => self.read_enum(false)?.into(),
+ codes::ERROR => self.read_error()?.into(),
+ codes::INTERFACE => self.read_object()?.into(),
+ codes::CALLBACK_INTERFACE => self.read_callback_interface()?.into(),
+ codes::TRAIT_METHOD => self.read_trait_method()?.into(),
+ codes::UNIFFI_TRAIT => self.read_uniffi_trait()?.into(),
+ _ => bail!("Unexpected metadata code: {value:?}"),
+ })
+ }
+
+ fn read_u8(&mut self) -> Result<u8> {
+ if !self.buf.is_empty() {
+ let value = self.buf[0];
+ self.buf = &self.buf[1..];
+ Ok(value)
+ } else {
+ bail!("Buffer is empty")
+ }
+ }
+
+ fn peek_u8(&mut self) -> Result<u8> {
+ if !self.buf.is_empty() {
+ Ok(self.buf[0])
+ } else {
+ bail!("Buffer is empty")
+ }
+ }
+
+ fn read_u32(&mut self) -> Result<u32> {
+ if self.buf.len() >= 4 {
+ // read the value as little-endian
+ let value = self.buf[0] as u32
+ + ((self.buf[1] as u32) << 8)
+ + ((self.buf[2] as u32) << 16)
+ + ((self.buf[3] as u32) << 24);
+ self.buf = &self.buf[4..];
+ Ok(value)
+ } else {
+ bail!("Not enough data left in buffer to read a u32 value");
+ }
+ }
+
+ fn read_bool(&mut self) -> Result<bool> {
+ Ok(self.read_u8()? == 1)
+ }
+
+ fn read_string(&mut self) -> Result<String> {
+ let size = self.read_u8()? as usize;
+ let slice;
+ (slice, self.buf) = self.buf.split_at(size);
+ String::from_utf8(slice.into()).context("Invalid string data")
+ }
+
+ fn read_type(&mut self) -> Result<Type> {
+ let value = self.read_u8()?;
+ Ok(match value {
+ codes::TYPE_U8 => Type::UInt8,
+ codes::TYPE_I8 => Type::Int8,
+ codes::TYPE_U16 => Type::UInt16,
+ codes::TYPE_I16 => Type::Int16,
+ codes::TYPE_U32 => Type::UInt32,
+ codes::TYPE_I32 => Type::Int32,
+ codes::TYPE_U64 => Type::UInt64,
+ codes::TYPE_I64 => Type::Int64,
+ codes::TYPE_F32 => Type::Float32,
+ codes::TYPE_F64 => Type::Float64,
+ codes::TYPE_BOOL => Type::Boolean,
+ codes::TYPE_STRING => Type::String,
+ codes::TYPE_DURATION => Type::Duration,
+ codes::TYPE_SYSTEM_TIME => Type::Timestamp,
+ codes::TYPE_FOREIGN_EXECUTOR => Type::ForeignExecutor,
+ codes::TYPE_RECORD => Type::Record {
+ module_path: self.read_string()?,
+ name: self.read_string()?,
+ },
+ codes::TYPE_ENUM => Type::Enum {
+ module_path: self.read_string()?,
+ name: self.read_string()?,
+ },
+ codes::TYPE_INTERFACE => Type::Object {
+ module_path: self.read_string()?,
+ name: self.read_string()?,
+ imp: ObjectImpl::from_is_trait(self.read_bool()?),
+ },
+ codes::TYPE_CALLBACK_INTERFACE => Type::CallbackInterface {
+ module_path: self.read_string()?,
+ name: self.read_string()?,
+ },
+ codes::TYPE_CUSTOM => Type::Custom {
+ module_path: self.read_string()?,
+ name: self.read_string()?,
+ builtin: Box::new(self.read_type()?),
+ },
+ codes::TYPE_OPTION => Type::Optional {
+ inner_type: Box::new(self.read_type()?),
+ },
+ codes::TYPE_VEC => {
+ let inner_type = self.read_type()?;
+ if inner_type == Type::UInt8 {
+ Type::Bytes
+ } else {
+ Type::Sequence {
+ inner_type: Box::new(inner_type),
+ }
+ }
+ }
+ codes::TYPE_HASH_MAP => Type::Map {
+ key_type: Box::new(self.read_type()?),
+ value_type: Box::new(self.read_type()?),
+ },
+ codes::TYPE_UNIT => bail!("Unexpected TYPE_UNIT"),
+ codes::TYPE_RESULT => bail!("Unexpected TYPE_RESULT"),
+ _ => bail!("Unexpected metadata type code: {value:?}"),
+ })
+ }
+
+ fn read_optional_type(&mut self) -> Result<Option<Type>> {
+ Ok(match self.peek_u8()? {
+ codes::TYPE_UNIT => {
+ _ = self.read_u8();
+ None
+ }
+ _ => Some(self.read_type()?),
+ })
+ }
+
+ fn read_return_type(&mut self) -> Result<(Option<Type>, Option<Type>)> {
+ Ok(match self.peek_u8()? {
+ codes::TYPE_UNIT => {
+ _ = self.read_u8();
+ (None, None)
+ }
+ codes::TYPE_RESULT => {
+ _ = self.read_u8();
+ (self.read_optional_type()?, self.read_optional_type()?)
+ }
+ _ => (Some(self.read_type()?), None),
+ })
+ }
+
+ fn read_func(&mut self) -> Result<FnMetadata> {
+ let module_path = self.read_string()?;
+ let name = self.read_string()?;
+ let is_async = self.read_bool()?;
+ let inputs = self.read_inputs()?;
+ let (return_type, throws) = self.read_return_type()?;
+ Ok(FnMetadata {
+ module_path,
+ name,
+ is_async,
+ inputs,
+ return_type,
+ throws,
+ checksum: self.calc_checksum(),
+ })
+ }
+
+ fn read_constructor(&mut self) -> Result<ConstructorMetadata> {
+ let module_path = self.read_string()?;
+ let self_name = self.read_string()?;
+ let name = self.read_string()?;
+ let inputs = self.read_inputs()?;
+ let (return_type, throws) = self.read_return_type()?;
+
+ return_type
+ .filter(|t| {
+ matches!(
+ t,
+ Type::Object { name, imp: ObjectImpl::Struct, .. } if name == &self_name
+ )
+ })
+ .context("Constructor return type must be Arc<Self>")?;
+
+ Ok(ConstructorMetadata {
+ module_path,
+ self_name,
+ name,
+ inputs,
+ throws,
+ checksum: self.calc_checksum(),
+ })
+ }
+
+ fn read_method(&mut self) -> Result<MethodMetadata> {
+ let module_path = self.read_string()?;
+ let self_name = self.read_string()?;
+ let name = self.read_string()?;
+ let is_async = self.read_bool()?;
+ let inputs = self.read_inputs()?;
+ let (return_type, throws) = self.read_return_type()?;
+ Ok(MethodMetadata {
+ module_path,
+ self_name,
+ name,
+ is_async,
+ inputs,
+ return_type,
+ throws,
+ takes_self_by_arc: false, // not emitted by macros
+ checksum: self.calc_checksum(),
+ })
+ }
+
+ fn read_record(&mut self) -> Result<RecordMetadata> {
+ Ok(RecordMetadata {
+ module_path: self.read_string()?,
+ name: self.read_string()?,
+ fields: self.read_fields()?,
+ })
+ }
+
+ fn read_enum(&mut self, is_flat_error: bool) -> Result<EnumMetadata> {
+ let module_path = self.read_string()?;
+ let name = self.read_string()?;
+ let variants = if is_flat_error {
+ self.read_flat_variants()?
+ } else {
+ self.read_variants()?
+ };
+
+ Ok(EnumMetadata {
+ module_path,
+ name,
+ variants,
+ })
+ }
+
+ fn read_error(&mut self) -> Result<ErrorMetadata> {
+ let is_flat = self.read_bool()?;
+ let enum_ = self.read_enum(is_flat)?;
+ Ok(ErrorMetadata::Enum { enum_, is_flat })
+ }
+
+ fn read_object(&mut self) -> Result<ObjectMetadata> {
+ Ok(ObjectMetadata {
+ module_path: self.read_string()?,
+ name: self.read_string()?,
+ imp: ObjectImpl::from_is_trait(self.read_bool()?),
+ })
+ }
+
+ fn read_uniffi_trait(&mut self) -> Result<UniffiTraitMetadata> {
+ let code = self.read_u8()?;
+ let mut read_metadata_method = || -> Result<MethodMetadata> {
+ let code = self.read_u8()?;
+ ensure!(code == codes::METHOD, "expected METHOD but read {code}");
+ self.read_method()
+ };
+
+ Ok(match UniffiTraitDiscriminants::from(code)? {
+ UniffiTraitDiscriminants::Debug => UniffiTraitMetadata::Debug {
+ fmt: read_metadata_method()?,
+ },
+ UniffiTraitDiscriminants::Display => UniffiTraitMetadata::Display {
+ fmt: read_metadata_method()?,
+ },
+ UniffiTraitDiscriminants::Eq => UniffiTraitMetadata::Eq {
+ eq: read_metadata_method()?,
+ ne: read_metadata_method()?,
+ },
+ UniffiTraitDiscriminants::Hash => UniffiTraitMetadata::Hash {
+ hash: read_metadata_method()?,
+ },
+ })
+ }
+
+ fn read_callback_interface(&mut self) -> Result<CallbackInterfaceMetadata> {
+ Ok(CallbackInterfaceMetadata {
+ module_path: self.read_string()?,
+ name: self.read_string()?,
+ })
+ }
+
+ fn read_trait_method(&mut self) -> Result<TraitMethodMetadata> {
+ let module_path = self.read_string()?;
+ let trait_name = self.read_string()?;
+ let index = self.read_u32()?;
+ let name = self.read_string()?;
+ let is_async = self.read_bool()?;
+ let inputs = self.read_inputs()?;
+ let (return_type, throws) = self.read_return_type()?;
+ Ok(TraitMethodMetadata {
+ module_path,
+ trait_name,
+ index,
+ name,
+ is_async,
+ inputs,
+ return_type,
+ throws,
+ takes_self_by_arc: false, // not emitted by macros
+ checksum: self.calc_checksum(),
+ })
+ }
+
+ fn read_fields(&mut self) -> Result<Vec<FieldMetadata>> {
+ let len = self.read_u8()?;
+ (0..len)
+ .map(|_| {
+ let name = self.read_string()?;
+ let ty = self.read_type()?;
+ let default = self.read_default(&name, &ty)?;
+ Ok(FieldMetadata { name, ty, default })
+ })
+ .collect()
+ }
+
+ fn read_variants(&mut self) -> Result<Vec<VariantMetadata>> {
+ let len = self.read_u8()?;
+ (0..len)
+ .map(|_| {
+ Ok(VariantMetadata {
+ name: self.read_string()?,
+ fields: self.read_fields()?,
+ })
+ })
+ .collect()
+ }
+
+ fn read_flat_variants(&mut self) -> Result<Vec<VariantMetadata>> {
+ let len = self.read_u8()?;
+ (0..len)
+ .map(|_| {
+ Ok(VariantMetadata {
+ name: self.read_string()?,
+ fields: vec![],
+ })
+ })
+ .collect()
+ }
+
+ fn read_inputs(&mut self) -> Result<Vec<FnParamMetadata>> {
+ let len = self.read_u8()?;
+ (0..len)
+ .map(|_| {
+ Ok(FnParamMetadata {
+ name: self.read_string()?,
+ ty: self.read_type()?,
+ // not emitted by macros
+ by_ref: false,
+ optional: false,
+ default: None,
+ })
+ })
+ .collect()
+ }
+
+ fn calc_checksum(&self) -> Option<u16> {
+ let bytes_read = self.initial_data.len() - self.buf.len();
+ let metadata_buf = &self.initial_data[..bytes_read];
+ Some(checksum_metadata(metadata_buf))
+ }
+
+ fn read_default(&mut self, name: &str, ty: &Type) -> Result<Option<LiteralMetadata>> {
+ let has_default = self.read_bool()?;
+ if !has_default {
+ return Ok(None);
+ }
+
+ let literal_kind = self.read_u8()?;
+ Ok(Some(match literal_kind {
+ codes::LIT_STR => {
+ ensure!(
+ matches!(ty, Type::String),
+ "field {name} of type {ty:?} can't have a default value of type string"
+ );
+ LiteralMetadata::String(self.read_string()?)
+ }
+ codes::LIT_INT => {
+ let base10_digits = self.read_string()?;
+ macro_rules! parse_int {
+ ($ty:ident, $variant:ident) => {
+ LiteralMetadata::$variant(
+ base10_digits
+ .parse::<$ty>()
+ .with_context(|| format!("parsing default for field {name}"))?
+ .into(),
+ Radix::Decimal,
+ ty.to_owned(),
+ )
+ };
+ }
+
+ match ty {
+ Type::UInt8 => parse_int!(u8, UInt),
+ Type::Int8 => parse_int!(i8, Int),
+ Type::UInt16 => parse_int!(u16, UInt),
+ Type::Int16 => parse_int!(i16, Int),
+ Type::UInt32 => parse_int!(u32, UInt),
+ Type::Int32 => parse_int!(i32, Int),
+ Type::UInt64 => parse_int!(u64, UInt),
+ Type::Int64 => parse_int!(i64, Int),
+ _ => {
+ bail!("field {name} of type {ty:?} can't have a default value of type integer");
+ }
+ }
+ }
+ codes::LIT_FLOAT => match ty {
+ Type::Float32 | Type::Float64 => {
+ LiteralMetadata::Float(self.read_string()?, ty.to_owned())
+ }
+ _ => {
+ bail!("field {name} of type {ty:?} can't have a default value of type float");
+ }
+ },
+ codes::LIT_BOOL => LiteralMetadata::Boolean(self.read_bool()?),
+ codes::LIT_NULL => LiteralMetadata::Null,
+ _ => bail!("Unexpected literal kind code: {literal_kind:?}"),
+ }))
+ }
+}
diff --git a/third_party/rust/uniffi_meta/src/types.rs b/third_party/rust/uniffi_meta/src/types.rs
new file mode 100644
index 0000000000..24f8a6f2a8
--- /dev/null
+++ b/third_party/rust/uniffi_meta/src/types.rs
@@ -0,0 +1,172 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+//! # Basic typesystem for defining a component interface.
+//!
+//! This module provides the "API-level" typesystem of a UniFFI Rust Component, that is,
+//! the types provided by the Rust implementation and consumed callers of the foreign language
+//! bindings. Think "objects" and "enums" and "records".
+//!
+//! The [`Type`] enum represents high-level types that would appear in the public API of
+//! a component, such as enums and records as well as primitives like ints and strings.
+//! The Rust code that implements a component, and the foreign language bindings that consume it,
+//! will both typically deal with such types as their core concern.
+//!
+//! As a developer working on UniFFI itself, you're likely to spend a fair bit of time thinking
+//! about how these API-level types map into the lower-level types of the FFI layer as represented
+//! by the [`ffi::FfiType`](super::ffi::FfiType) enum, but that's a detail that is invisible to end users.
+
+use crate::Checksum;
+
+#[derive(Debug, Copy, Clone, Eq, PartialEq, Checksum, Ord, PartialOrd)]
+pub enum ObjectImpl {
+ Struct,
+ Trait,
+}
+
+impl ObjectImpl {
+ /// Return the fully qualified name which should be used by Rust code for
+ /// an object with the given name.
+ /// Includes `r#`, traits get a leading `dyn`. If we ever supported associated types, then
+ /// this would also include them.
+ pub fn rust_name_for(&self, name: &str) -> String {
+ if self == &ObjectImpl::Trait {
+ format!("dyn r#{name}")
+ } else {
+ format!("r#{name}")
+ }
+ }
+
+ // uniffi_meta and procmacro support tend to carry around `is_trait` bools. This makes that
+ // mildly less painful
+ pub fn from_is_trait(is_trait: bool) -> Self {
+ if is_trait {
+ ObjectImpl::Trait
+ } else {
+ ObjectImpl::Struct
+ }
+ }
+}
+
+#[derive(Debug, Clone, Copy, Eq, PartialEq, Checksum, Ord, PartialOrd)]
+pub enum ExternalKind {
+ Interface,
+ // Either a record or enum
+ DataClass,
+}
+
+/// Represents all the different high-level types that can be used in a component interface.
+/// At this level we identify user-defined types by name, without knowing any details
+/// of their internal structure apart from what type of thing they are (record, enum, etc).
+#[derive(Debug, Clone, Eq, PartialEq, Checksum, Ord, PartialOrd)]
+pub enum Type {
+ // Primitive types.
+ UInt8,
+ Int8,
+ UInt16,
+ Int16,
+ UInt32,
+ Int32,
+ UInt64,
+ Int64,
+ Float32,
+ Float64,
+ Boolean,
+ String,
+ Bytes,
+ Timestamp,
+ Duration,
+ Object {
+ // The module path to the object
+ module_path: String,
+ // The name in the "type universe"
+ name: String,
+ // How the object is implemented.
+ imp: ObjectImpl,
+ },
+ ForeignExecutor,
+ // Types defined in the component API, each of which has a string name.
+ Record {
+ module_path: String,
+ name: String,
+ },
+ Enum {
+ module_path: String,
+ name: String,
+ },
+ CallbackInterface {
+ module_path: String,
+ name: String,
+ },
+ // Structurally recursive types.
+ Optional {
+ inner_type: Box<Type>,
+ },
+ Sequence {
+ inner_type: Box<Type>,
+ },
+ Map {
+ key_type: Box<Type>,
+ value_type: Box<Type>,
+ },
+ // An FfiConverter we `use` from an external crate
+ External {
+ module_path: String,
+ name: String,
+ #[checksum_ignore] // The namespace is not known generating scaffolding.
+ namespace: String,
+ kind: ExternalKind,
+ tagged: bool, // does its FfiConverter use <UniFFITag>?
+ },
+ // Custom type on the scaffolding side
+ Custom {
+ module_path: String,
+ name: String,
+ builtin: Box<Type>,
+ },
+}
+
+impl Type {
+ pub fn iter_types(&self) -> TypeIterator<'_> {
+ let nested_types = match self {
+ Type::Optional { inner_type } | Type::Sequence { inner_type } => {
+ inner_type.iter_types()
+ }
+ Type::Map {
+ key_type,
+ value_type,
+ } => Box::new(key_type.iter_types().chain(value_type.iter_types())),
+ _ => Box::new(std::iter::empty()),
+ };
+ Box::new(std::iter::once(self).chain(nested_types))
+ }
+}
+
+// A trait so various things can turn into a type.
+pub trait AsType: core::fmt::Debug {
+ fn as_type(&self) -> Type;
+}
+
+impl AsType for Type {
+ fn as_type(&self) -> Type {
+ self.clone()
+ }
+}
+
+// Needed to handle &&Type and &&&Type values, which we sometimes end up with in the template code
+impl<T, C> AsType for T
+where
+ T: std::ops::Deref<Target = C> + std::fmt::Debug,
+ C: AsType,
+{
+ fn as_type(&self) -> Type {
+ self.deref().as_type()
+ }
+}
+
+/// An abstract type for an iterator over &Type references.
+///
+/// Ideally we would not need to name this type explicitly, and could just
+/// use an `impl Iterator<Item = &Type>` on any method that yields types.
+pub type TypeIterator<'a> = Box<dyn Iterator<Item = &'a Type> + 'a>;