blob: 33b3fbccf64075ca8fecda0137550a967c066874 [file] [log] [blame]
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07001// Part of the Crubit project, under the Apache License v2.0 with LLVM
2// Exceptions. See /LICENSE for license information.
3// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
4
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005use anyhow::{anyhow, bail, ensure, Context, Result};
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08006use code_gen_utils::{
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07007 escape_non_identifier_chars, format_cc_ident, format_cc_includes,
8 format_namespace_bound_cc_tokens, make_rs_ident, CcInclude, NamespaceQualifier,
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08009};
Lukasz Anforowicz903fc632022-10-25 08:55:33 -070010use itertools::Itertools;
Lukasz Anforowicza691cf52023-03-08 12:24:33 -080011use proc_macro2::{Ident, Literal, TokenStream};
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -070012use quote::{format_ident, quote, ToTokens};
Lukasz Anforowicz9924c432023-04-04 12:51:22 -070013use rustc_hir::{
14 AssocItemKind, Impl, ImplItemKind, ImplicitSelfKind, Item, ItemKind, Node, Unsafety,
15};
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -070016use rustc_middle::dep_graph::DepContext;
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -080017use rustc_middle::mir::Mutability;
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -070018use rustc_middle::ty::{self, Ty, TyCtxt}; // See <internal link>/ty.html#import-conventions
Lukasz Anforowicz903fc632022-10-25 08:55:33 -070019use rustc_span::def_id::{DefId, LocalDefId, LOCAL_CRATE};
Lukasz Anforowicz9924c432023-04-04 12:51:22 -070020use rustc_span::symbol::{sym, Symbol};
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -070021use rustc_target::abi::{Abi, FieldsShape, Integer, Layout, Primitive, Scalar};
Lukasz Anforowicz24160d52022-10-19 06:45:45 -070022use rustc_target::spec::PanicStrategy;
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -080023use std::collections::{BTreeSet, HashMap, HashSet};
Googler47fd9572023-01-20 09:57:32 -080024use std::iter::once;
Lukasz Anforowicz7f31f802022-12-16 08:24:13 -080025use std::ops::AddAssign;
Lukasz Anforowicz8e5042c2023-01-03 11:19:07 -080026use std::rc::Rc;
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -070027
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080028pub struct Input<'tcx> {
29 /// Compilation context for the crate that the bindings should be generated
30 /// for.
31 pub tcx: TyCtxt<'tcx>,
32
Lukasz Anforowicza782bda2023-01-17 14:04:50 -080033 /// Path to a the `crubit/support` directory in a format that should be used
34 /// in the `#include` directives inside the generated C++ files.
35 /// Example: "crubit/support".
36 pub crubit_support_path: Rc<str>,
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080037
Lukasz Anforowiczcfe84552023-04-20 13:38:54 -070038 /// A map from a crate name to the include path with the corresponding C++
39 /// bindings. This is used when formatting a type exported from another
40 /// crate.
41 // TODO(b/271857814): A crate name might not be globally unique - the key needs to also cover
42 // a "hash" of the crate version and compilation flags.
43 pub crate_name_to_include_path: HashMap<Rc<str>, CcInclude>,
44
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080045 // TODO(b/262878759): Provide a set of enabled/disabled Crubit features.
46 pub _features: (),
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080047}
48
Lukasz Anforowicza3b7db02023-03-09 17:34:05 -080049impl<'tcx> Input<'tcx> {
50 // TODO(b/259724276): This function's results should be memoized. It may be
51 // easier if separate functions are provided for each support header - e.g.
52 // `rs_char()`, `return_value_slot()`, etc.
53 fn support_header(&self, suffix: &str) -> CcInclude {
54 let support_path = &*self.crubit_support_path;
55 let full_path = format!("{support_path}/{suffix}");
56 CcInclude::user_header(full_path.into())
57 }
58}
59
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080060pub struct Output {
Lukasz Anforowicz2b38d272022-09-23 08:08:18 -070061 pub h_body: TokenStream,
Lukasz Anforowicze7a25002022-11-10 06:21:42 -080062 pub rs_body: TokenStream,
Lukasz Anforowicz581fd752022-09-21 11:30:15 -070063}
64
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080065pub fn generate_bindings(input: &Input) -> Result<Output> {
66 match input.tcx.sess().panic_strategy() {
67 PanicStrategy::Unwind => bail!("No support for panic=unwind strategy (b/254049425)"),
68 PanicStrategy::Abort => (),
69 };
Lukasz Anforowicz24160d52022-10-19 06:45:45 -070070
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080071 let top_comment = {
72 let crate_name = input.tcx.crate_name(LOCAL_CRATE);
73 let txt = format!(
74 "Automatically @generated C++ bindings for the following Rust crate:\n\
75 {crate_name}"
76 );
77 quote! { __COMMENT__ #txt __NEWLINE__ }
78 };
Lukasz Anforowiczd9ff4ab2022-09-23 08:11:18 -070079
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080080 let Output { h_body, rs_body } = format_crate(input).unwrap_or_else(|err| {
81 let txt = format!("Failed to generate bindings for the crate: {err}");
82 let src = quote! { __COMMENT__ #txt };
83 Output { h_body: src.clone(), rs_body: src }
84 });
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -080085
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080086 let h_body = quote! {
87 #top_comment
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -080088
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080089 // TODO(b/251445877): Replace `#pragma once` with include guards.
90 __HASH_TOKEN__ pragma once __NEWLINE__
91 __NEWLINE__
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -080092
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080093 #h_body
94 };
Lukasz Anforowicz581fd752022-09-21 11:30:15 -070095
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -080096 let rs_body = quote! {
97 #top_comment
Lukasz Anforowiczd16b6bf2022-11-22 18:35:08 -080098
Lukasz Anforowicz843d1612023-03-02 16:52:53 -080099 // `rust_builtin_type_abi_assumptions.md` documents why the generated
100 // bindings need to relax the `improper_ctypes_definitions` warning
101 // for `char` (and possibly for other built-in types in the future).
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -0800102 #![allow(improper_ctypes_definitions)] __NEWLINE__
103 __NEWLINE__
Lukasz Anforowiczd16b6bf2022-11-22 18:35:08 -0800104
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -0800105 #rs_body
106 };
Lukasz Anforowicze7a25002022-11-10 06:21:42 -0800107
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -0800108 Ok(Output { h_body, rs_body })
Lukasz Anforowicz581fd752022-09-21 11:30:15 -0700109}
110
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -0800111#[derive(Clone, Debug, Default)]
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800112struct CcPrerequisites {
113 /// Set of `#include`s that a `CcSnippet` depends on. For example if
114 /// `CcSnippet::tokens` expands to `std::int32_t`, then `includes`
115 /// need to cover the `#include <cstdint>`.
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700116 includes: BTreeSet<CcInclude>,
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -0800117
118 /// Set of local definitions that a `CcSnippet` depends on. For example if
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700119 /// `CcSnippet::tokens` expands to `void foo(S s) { ... }` then the
120 /// definition of `S` should have appeared earlier - in this case `defs`
121 /// will include the `LocalDefId` corresponding to `S`. Note that the
122 /// definition of `S` is covered by `ApiSnippets::main_api` (i.e. the
123 /// predecessor of a toposort edge is `ApiSnippets::main_api` - it is not
124 /// possible to depend on `ApiSnippets::cc_details`).
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -0800125 defs: HashSet<LocalDefId>,
126
127 /// Set of forward declarations that a `CcSnippet` depends on. For example
128 /// if `CcSnippet::tokens` expands to `void foo(S* s)` then a forward
129 /// declaration of `S` should have appeared earlier - in this case
130 /// `fwd_decls` will include the `LocalDefId` corresponding to `S`.
131 /// Note that in this particular example the *definition* of `S` does
132 /// *not* need to appear earlier (and therefore `defs` will *not*
133 /// contain `LocalDefId` corresponding to `S`).
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -0800134 fwd_decls: HashSet<LocalDefId>,
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800135}
136
137impl CcPrerequisites {
138 #[cfg(test)]
139 fn is_empty(&self) -> bool {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -0800140 let &Self { ref includes, ref defs, ref fwd_decls } = self;
141 includes.is_empty() && defs.is_empty() && fwd_decls.is_empty()
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800142 }
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -0800143
144 /// Weakens all dependencies to only require a forward declaration. Example
145 /// usage scenarios:
146 /// - Computing prerequisites of pointer types (the pointee type can just be
147 /// forward-declared),
148 /// - Computing prerequisites of function declarations (parameter types and
149 /// return type can just be forward-declared).
150 fn move_defs_to_fwd_decls(&mut self) {
151 self.fwd_decls.extend(std::mem::take(&mut self.defs))
152 }
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800153}
154
155impl AddAssign for CcPrerequisites {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -0800156 fn add_assign(&mut self, rhs: Self) {
157 let Self { mut includes, defs, fwd_decls } = rhs;
158
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -0800159 // `BTreeSet::append` is used because it _seems_ to be more efficient than
160 // calling `extend`. This is because `extend` takes an iterator
161 // (processing each `rhs` include one-at-a-time) while `append` steals
162 // the whole backing data store from `rhs.includes`. OTOH, this is a bit
163 // speculative, since the (expected / guessed) performance difference is
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800164 // not documented at
165 // https://doc.rust-lang.org/std/collections/struct.BTreeSet.html#method.append
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -0800166 self.includes.append(&mut includes);
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -0800167
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -0800168 self.defs.extend(defs);
169 self.fwd_decls.extend(fwd_decls);
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800170 }
171}
172
173#[derive(Debug, Default)]
174struct CcSnippet {
175 tokens: TokenStream,
176 prereqs: CcPrerequisites,
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700177}
178
179impl CcSnippet {
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800180 /// Consumes `self` and returns its `tokens`, while preserving
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800181 /// its `prereqs` into `prereqs_accumulator`.
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800182 fn into_tokens(self, prereqs_accumulator: &mut CcPrerequisites) -> TokenStream {
183 let Self { tokens, prereqs } = self;
184 *prereqs_accumulator += prereqs;
185 tokens
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700186 }
Lukasz Anforowicza2f1cae2022-12-15 10:42:25 -0800187
188 /// Creates a new CcSnippet (with no `CcPrerequisites`).
189 fn new(tokens: TokenStream) -> Self {
190 Self { tokens, ..Default::default() }
191 }
192
193 /// Creates a CcSnippet that depends on a single `CcInclude`.
194 fn with_include(tokens: TokenStream, include: CcInclude) -> Self {
195 let mut prereqs = CcPrerequisites::default();
196 prereqs.includes.insert(include);
197 Self { tokens, prereqs }
198 }
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700199}
200
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700201impl AddAssign for CcSnippet {
202 fn add_assign(&mut self, rhs: Self) {
203 self.tokens.extend(rhs.into_tokens(&mut self.prereqs));
204 }
205}
206
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800207/// Represents the fully qualified name of a Rust item (e.g. of a `struct` or a
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800208/// function).
209struct FullyQualifiedName {
210 /// Name of the crate that defines the item.
211 /// For example, this would be `std` for `std::cmp::Ordering`.
212 krate: Symbol,
213
214 /// Path to the module where the item is located.
215 /// For example, this would be `cmp` for `std::cmp::Ordering`.
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800216 /// The path may contain multiple modules - e.g. `foo::bar::baz`.
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800217 mod_path: NamespaceQualifier,
218
219 /// Name of the item.
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -0800220 /// For example, this would be:
221 /// * `Some("Ordering")` for `std::cmp::Ordering`.
222 /// * `None` for `ItemKind::Use` - e.g.: `use submodule::*`
223 name: Option<Symbol>,
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800224}
225
226impl FullyQualifiedName {
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -0800227 /// Computes a `FullyQualifiedName` for `def_id`.
228 ///
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -0800229 /// May panic if `def_id` is an invalid id.
Lukasz Anforowicz1ab5e872022-12-05 10:07:00 -0800230 // TODO(b/259724276): This function's results should be memoized.
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800231 fn new(tcx: TyCtxt, def_id: DefId) -> Self {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800232 let krate = tcx.crate_name(def_id.krate);
233
234 let mut full_path = tcx.def_path(def_id).data; // mod_path + name
235 let name = full_path.pop().expect("At least the item's name should be present");
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -0800236 let name = name.data.get_opt_name();
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800237
Lukasz Anforowicz8e5042c2023-01-03 11:19:07 -0800238 let mod_path = NamespaceQualifier::new(
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -0800239 full_path
240 .into_iter()
241 .filter_map(|p| p.data.get_opt_name())
242 .map(|s| Rc::<str>::from(s.as_str())),
Lukasz Anforowicz8e5042c2023-01-03 11:19:07 -0800243 );
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800244
245 Self { krate, mod_path, name }
246 }
247
248 fn format_for_cc(&self) -> Result<TokenStream> {
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -0800249 let name =
250 self.name.as_ref().expect("`format_for_cc` can't be called on name-less item kinds");
251
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800252 let top_level_ns = format_cc_ident(self.krate.as_str())?;
253 let ns_path = self.mod_path.format_for_cc()?;
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -0800254 let name = format_cc_ident(name.as_str())?;
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800255 Ok(quote! { :: #top_level_ns :: #ns_path #name })
256 }
257
258 fn format_for_rs(&self) -> TokenStream {
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -0800259 let name =
260 self.name.as_ref().expect("`format_for_cc` can't be called on name-less item kinds");
261
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800262 let krate = make_rs_ident(self.krate.as_str());
263 let mod_path = self.mod_path.format_for_rs();
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -0800264 let name = make_rs_ident(name.as_str());
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800265 quote! { :: #krate :: #mod_path #name }
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800266 }
267}
268
Lukasz Anforowicza3b7db02023-03-09 17:34:05 -0800269/// Whether functions using `extern "C"` ABI can safely handle values of type
270/// `ty` (e.g. when passing by value arguments or return values of such type).
271fn is_c_abi_compatible_by_value(ty: Ty) -> bool {
272 match ty.kind() {
273 // `improper_ctypes_definitions` warning doesn't complain about the following types:
274 ty::TyKind::Bool |
275 ty::TyKind::Float{..} |
276 ty::TyKind::Int{..} |
277 ty::TyKind::Uint{..} |
278 ty::TyKind::Never |
279 ty::TyKind::RawPtr{..} |
280 ty::TyKind::FnPtr{..} => true,
281 ty::TyKind::Tuple(types) if types.len() == 0 => true,
282
283 // Crubit assumes that `char` is compatible with a certain `extern "C"` ABI.
284 // See `rust_builtin_type_abi_assumptions.md` for more details.
285 ty::TyKind::Char => true,
286
287 // Crubit's C++ bindings for tuples, structs, and other ADTs may not preserve
288 // their ABI (even if they *do* preserve their memory layout). For example:
289 // - In System V ABI replacing a field with a fixed-length array of bytes may affect
290 // whether the whole struct is classified as an integer and passed in general purpose
291 // registers VS classified as SSE2 and passed in floating-point registers like xmm0).
292 // See also b/270454629.
293 // - To replicate field offsets, Crubit may insert explicit padding fields. These
294 // extra fields may also impact the ABI of the generated bindings.
295 //
296 // TODO(lukasza): In the future, some additional performance gains may be realized by
297 // returning `true` in a few limited cases (this may require additional complexity to
298 // ensure that `format_adt` never injects explicit padding into such structs):
299 // - `#[repr(C)]` structs and unions,
300 // - `#[repr(transparent)]` struct that wraps an ABI-safe type,
301 // - Discriminant-only enums (b/259984090).
302 ty::TyKind::Tuple{..} | // An empty tuple (`()` - the unit type) is handled above.
303 ty::TyKind::Adt{..} => false,
304
305 // These kinds of reference-related types are not implemented yet - `is_c_abi_compatible_by_value`
306 // should never need to handle them, because `format_ty_for_cc` fails for such types.
307 //
308 // TODO(b/258235219): When implementing support for references we should
309 // consider returning `true` for `TyKind::Ref` and document the rationale
310 // for such decision - maybe something like this will be sufficient:
311 // - In general `TyKind::Ref` should have the same ABI as `TyKind::RawPtr`
312 // - References to slices (`&[T]`) or strings (`&str`) rely on assumptions
313 // spelled out in `rust_builtin_type_abi_assumptions.md`..
314 ty::TyKind::Ref{..} |
315 ty::TyKind::Str |
316 ty::TyKind::Array{..} |
317 ty::TyKind::Slice{..} =>
318 unimplemented!(),
319
320 // `format_ty_for_cc` is expected to fail for other kinds of types
321 // and therefore `is_c_abi_compatible_by_value` should never be called for
322 // these other types
323 _ => unimplemented!(),
324 }
325}
326
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700327/// Location where a type is used.
328enum TypeLocation {
329 /// The top-level return type.
330 ///
331 /// The "top-level" part can be explained by looking at an example of `fn
332 /// foo() -> *const T`:
333 /// - The top-level return type `*const T` is in the `FnReturn` location
334 /// - The nested pointee type `T` is in the `Other` location
335 FnReturn,
336
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700337 /// The top-level parameter type.
338 ///
339 /// The "top-level" part can be explained by looking at an example of:
340 /// `fn foo(param: *const T)`:
341 /// - The top-level parameter type `*const T` is in the `FnParam` location
342 /// - The nested pointee type `T` is in the `Other` location
343 // TODO(b/278141494, b/278141418): Once `const` and `static` items are supported,
344 // we may want to apply parameter-like formatting to their types (e.g. have
345 // `format_ty_for_cc` emit `T&` rather than `T*`).
346 FnParam,
347
348 /// Other location (e.g. pointee type, field type, etc.).
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700349 Other,
350}
351
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700352/// Formats `ty` into a `CcSnippet` that represents how the type should be
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800353/// spelled in a C++ declaration of a function parameter or field.
Lukasz Anforowicz1ab5e872022-12-05 10:07:00 -0800354//
355// TODO(b/259724276): This function's results should be memoized.
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700356fn format_ty_for_cc<'tcx>(
357 input: &Input<'tcx>,
358 ty: Ty<'tcx>,
359 location: TypeLocation,
360) -> Result<CcSnippet> {
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800361 fn cstdint(tokens: TokenStream) -> CcSnippet {
Lukasz Anforowicza2f1cae2022-12-15 10:42:25 -0800362 CcSnippet::with_include(tokens, CcInclude::cstdint())
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700363 }
Lukasz Anforowicz3744e502022-12-02 08:40:38 -0800364 fn keyword(tokens: TokenStream) -> CcSnippet {
Lukasz Anforowicza2f1cae2022-12-15 10:42:25 -0800365 CcSnippet::new(tokens)
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700366 }
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700367 Ok(match ty.kind() {
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700368 ty::TyKind::Never => match location {
369 TypeLocation::FnReturn => keyword(quote! { void }),
Googler7a77a802023-04-21 08:32:50 -0700370 _ => {
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700371 // TODO(b/254507801): Maybe translate into `crubit::Never`?
372 bail!("The never type `!` is only supported as a return type (b/254507801)");
Googler7a77a802023-04-21 08:32:50 -0700373 }
374 },
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700375 ty::TyKind::Tuple(types) => {
376 if types.len() == 0 {
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700377 match location {
378 TypeLocation::FnReturn => keyword(quote! { void }),
Googler7a77a802023-04-21 08:32:50 -0700379 _ => {
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700380 // TODO(b/254507801): Maybe translate into `crubit::Unit`?
381 bail!("`()` / `void` is only supported as a return type (b/254507801)");
Googler7a77a802023-04-21 08:32:50 -0700382 }
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700383 }
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700384 } else {
Lukasz Anforowicz13794df2022-10-21 07:56:34 -0700385 // TODO(b/254099023): Add support for tuples.
386 bail!("Tuples are not supported yet: {} (b/254099023)", ty);
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700387 }
388 }
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700389
Lukasz Anforowicze3c39472023-01-12 15:07:38 -0800390 // https://rust-lang.github.io/unsafe-code-guidelines/layout/scalars.html#bool documents
391 // that "Rust's bool has the same layout as C17's _Bool". The details (e.g. size, valid
392 // bit patterns) are implementation-defined, but this is okay, because `bool` in the
393 // `extern "C"` functions in the generated `..._cc_api.h` will also be the C17's _Bool.
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700394 ty::TyKind::Bool => keyword(quote! { bool }),
395
396 // https://rust-lang.github.io/unsafe-code-guidelines/layout/scalars.html#fixed-width-floating-point-types
397 // documents that "When the platforms' "math.h" header defines the __STDC_IEC_559__ macro,
398 // Rust's floating-point types are safe to use directly in C FFI where the appropriate C
399 // types are expected (f32 for float, f64 for double)."
400 //
401 // TODO(b/255768062): Generated bindings should explicitly check `__STDC_IEC_559__`
402 ty::TyKind::Float(ty::FloatTy::F32) => keyword(quote! { float }),
403 ty::TyKind::Float(ty::FloatTy::F64) => keyword(quote! { double }),
404
Lukasz Anforowicza782bda2023-01-17 14:04:50 -0800405 // ABI compatibility and other details are described in the doc comments in
Lukasz Anforowiczec0b64e2023-02-17 14:31:12 -0800406 // `crubit/support/rs_std/rs_char.h` and `crubit/support/rs_std/char_test.cc` (search for
Lukasz Anforowiczd71686c2023-02-17 14:29:55 -0800407 // "Layout tests").
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700408 ty::TyKind::Char => {
Lukasz Anforowiczb06e0812023-03-02 15:54:32 -0800409 // Asserting that the target architecture meets the assumption from Crubit's
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700410 // `rust_builtin_type_abi_assumptions.md` - we assume that Rust's `char` has the
411 // same ABI as `u32`.
Lukasz Anforowiczb06e0812023-03-02 15:54:32 -0800412 let param_env = ty::ParamEnv::empty();
413 let layout = input
414 .tcx
415 .layout_of(param_env.and(ty))
416 .expect("`layout_of` is expected to succeed for the builtin `char` type")
417 .layout;
418 assert_eq!(4, layout.align().abi.bytes());
419 assert_eq!(4, layout.size().bytes());
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700420 assert!(matches!(
421 layout.abi(),
422 Abi::Scalar(Scalar::Initialized {
423 value: Primitive::Int(Integer::I32, /* signedness = */ false),
424 ..
425 })
426 ));
Lukasz Anforowiczb06e0812023-03-02 15:54:32 -0800427
Lukasz Anforowicza782bda2023-01-17 14:04:50 -0800428 CcSnippet::with_include(
Lukasz Anforowiczec0b64e2023-02-17 14:31:12 -0800429 quote! { rs_std::rs_char },
Lukasz Anforowicza3b7db02023-03-09 17:34:05 -0800430 input.support_header("rs_std/rs_char.h"),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -0800431 )
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700432 }
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700433
434 // https://rust-lang.github.io/unsafe-code-guidelines/layout/scalars.html#isize-and-usize
435 // documents that "Rust's signed and unsigned fixed-width integer types {i,u}{8,16,32,64}
436 // have the same layout the C fixed-width integer types from the <stdint.h> header
437 // {u,}int{8,16,32,64}_t. These fixed-width integer types are therefore safe to use
438 // directly in C FFI where the corresponding C fixed-width integer types are expected.
439 //
440 // https://rust-lang.github.io/unsafe-code-guidelines/layout/scalars.html#layout-compatibility-with-c-native-integer-types
441 // documents that "Rust does not support C platforms on which the C native integer type are
442 // not compatible with any of Rust's fixed-width integer type (e.g. because of
443 // padding-bits, lack of 2's complement, etc.)."
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700444 ty::TyKind::Int(ty::IntTy::I8) => cstdint(quote! { std::int8_t }),
445 ty::TyKind::Int(ty::IntTy::I16) => cstdint(quote! { std::int16_t }),
446 ty::TyKind::Int(ty::IntTy::I32) => cstdint(quote! { std::int32_t }),
447 ty::TyKind::Int(ty::IntTy::I64) => cstdint(quote! { std::int64_t }),
448 ty::TyKind::Uint(ty::UintTy::U8) => cstdint(quote! { std::uint8_t }),
449 ty::TyKind::Uint(ty::UintTy::U16) => cstdint(quote! { std::uint16_t }),
450 ty::TyKind::Uint(ty::UintTy::U32) => cstdint(quote! { std::uint32_t }),
451 ty::TyKind::Uint(ty::UintTy::U64) => cstdint(quote! { std::uint64_t }),
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700452
453 // https://rust-lang.github.io/unsafe-code-guidelines/layout/scalars.html#isize-and-usize
454 // documents that "The isize and usize types are [...] layout compatible with C's uintptr_t
455 // and intptr_t types.".
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700456 ty::TyKind::Int(ty::IntTy::Isize) => cstdint(quote! { std::intptr_t }),
457 ty::TyKind::Uint(ty::UintTy::Usize) => cstdint(quote! { std::uintptr_t }),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700458
459 ty::TyKind::Int(ty::IntTy::I128) | ty::TyKind::Uint(ty::UintTy::U128) => {
Lukasz Anforowiczed17d052022-11-02 12:07:28 -0700460 // Note that "the alignment of Rust's {i,u}128 is unspecified and allowed to
461 // change" according to
462 // https://rust-lang.github.io/unsafe-code-guidelines/layout/scalars.html#fixed-width-integer-types
463 //
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700464 // TODO(b/254094650): Consider mapping this to Clang's (and GCC's) `__int128`
465 // or to `absl::in128`.
466 bail!("C++ doesn't have a standard equivalent of `{ty}` (b/254094650)");
467 }
468
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800469 ty::TyKind::Adt(adt, substs) => {
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800470 ensure!(substs.len() == 0, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -0800471 ensure!(
472 is_directly_public(input.tcx, adt.did()),
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700473 "Not directly public type (re-exports are not supported yet - b/262052635)"
474 );
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800475
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800476 let def_id = adt.did();
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -0800477 let mut prereqs = CcPrerequisites::default();
478 if def_id.krate == LOCAL_CRATE {
479 prereqs.defs.insert(def_id.expect_local());
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800480 } else {
Lukasz Anforowiczcfe84552023-04-20 13:38:54 -0700481 let other_crate_name = input.tcx.crate_name(def_id.krate);
482 let include = input
483 .crate_name_to_include_path
484 .get(other_crate_name.as_str())
Googler7a77a802023-04-21 08:32:50 -0700485 .ok_or_else(|| {
486 anyhow!(
Lukasz Anforowiczcfe84552023-04-20 13:38:54 -0700487 "Type `{ty}` comes from the `{other_crate_name}` crate, \
Googler7a77a802023-04-21 08:32:50 -0700488 but no `--other-crate-bindings` were specified for this crate"
489 )
490 })?;
Lukasz Anforowiczcfe84552023-04-20 13:38:54 -0700491 prereqs.includes.insert(include.clone());
492 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800493
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -0800494 // Verify if definition of `ty` can be succesfully imported and bail otherwise.
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700495 format_adt_core(input.tcx, def_id).with_context(|| {
496 format!("Failed to generate bindings for the definition of `{ty}`")
497 })?;
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -0800498
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800499 CcSnippet {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -0800500 tokens: FullyQualifiedName::new(input.tcx, def_id).format_for_cc()?,
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800501 prereqs,
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800502 }
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700503 }
504
505 ty::TyKind::RawPtr(ty::TypeAndMut { ty, mutbl }) => {
506 let const_qualifier = match mutbl {
507 Mutability::Mut => quote! {},
508 Mutability::Not => quote! { const },
509 };
510 let CcSnippet { tokens, mut prereqs } =
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700511 format_ty_for_cc(input, *ty, TypeLocation::Other).with_context(|| {
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700512 format!("Failed to format the pointee of the pointer type `{ty}`")
513 })?;
514 prereqs.move_defs_to_fwd_decls();
515 CcSnippet { prereqs, tokens: quote! { #const_qualifier #tokens * } }
516 }
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800517
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700518 ty::TyKind::FnPtr(sig) => {
519 let sig = match sig.no_bound_vars() {
520 None => bail!("Generic functions are not supported yet (b/259749023)"),
521 Some(sig) => sig,
522 };
523 check_fn_sig(&sig)?;
524 is_thunk_required(&sig).context("Function pointers can't have a thunk")?;
525
526 // `is_thunk_required` check above implies `extern "C"` (or `"C-unwind"`).
527 // This assertion reinforces that the generated C++ code doesn't need
528 // to use calling convention attributes like `_stdcall`, etc.
529 assert!(matches!(sig.abi, rustc_target::spec::abi::Abi::C { .. }));
530
Googler7a77a802023-04-21 08:32:50 -0700531 // C++ references are not rebindable and therefore can't be used to replicate
532 // semantics of Rust field types (or, say, element types of Rust
533 // arrays). Because of this, C++ references are only used for
534 // top-level return types and parameter types (and pointers are used
535 // in other locations).
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700536 let ptr_or_ref_sigil = match location {
Googler7a77a802023-04-21 08:32:50 -0700537 TypeLocation::FnReturn | TypeLocation::FnParam => quote! { & },
538 TypeLocation::Other => quote! { * },
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700539 };
540
541 let mut prereqs = CcPrerequisites::default();
542 prereqs.includes.insert(input.support_header("internal/cxx20_backports.h"));
543 let ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut prereqs);
544 let param_types = format_param_types_for_cc(input, &sig)?
545 .into_iter()
546 .map(|snippet| snippet.into_tokens(&mut prereqs));
547 let tokens = quote! {
548 crubit::type_identity_t<
549 #ret_type( #( #param_types ),* )
550 > #ptr_or_ref_sigil
551 };
552
553 CcSnippet { tokens, prereqs }
Googler7a77a802023-04-21 08:32:50 -0700554 }
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700555
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800556 // TODO(b/260268230, b/260729464): When recursively processing nested types (e.g. an
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -0800557 // element type of an Array, a referent of a Ref, a parameter type of an FnPtr, etc), one
558 // should also 1) propagate `CcPrerequisites::defs`, 2) cover `CcPrerequisites::defs` in
559 // `test_format_ty_for_cc...`. For ptr/ref it might be possible to use
560 // `CcPrerequisites::move_defs_to_fwd_decls`.
Lukasz Anforowicz087dff72023-02-17 12:13:32 -0800561 _ => bail!("The following Rust type is not supported yet: {ty}"),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700562 })
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -0700563}
564
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700565fn format_ret_ty_for_cc<'tcx>(input: &Input<'tcx>, sig: &ty::FnSig<'tcx>) -> Result<CcSnippet> {
566 format_ty_for_cc(input, sig.output(), TypeLocation::FnReturn)
567 .context("Error formatting function return type")
568}
569
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700570fn format_param_types_for_cc<'tcx>(
571 input: &Input<'tcx>,
572 sig: &ty::FnSig<'tcx>,
573) -> Result<Vec<CcSnippet>> {
574 sig.inputs()
575 .iter()
576 .enumerate()
577 .map(|(i, &ty)| {
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700578 Ok(format_ty_for_cc(input, ty, TypeLocation::FnParam)
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700579 .with_context(|| format!("Error handling parameter #{i}"))?)
580 })
581 .collect()
582}
583
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800584/// Formats `ty` for Rust - to be used in `..._cc_api_impl.rs` (e.g. as a type
585/// of a parameter in a Rust thunk). Because `..._cc_api_impl.rs` is a
586/// distinct, separate crate, the returned `TokenStream` uses crate-qualified
587/// names whenever necessary - for example: `target_crate::SomeStruct` rather
588/// than just `SomeStruct`.
Lukasz Anforowicz1ab5e872022-12-05 10:07:00 -0800589//
590// TODO(b/259724276): This function's results should be memoized.
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800591fn format_ty_for_rs(tcx: TyCtxt, ty: Ty) -> Result<TokenStream> {
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -0800592 Ok(match ty.kind() {
593 ty::TyKind::Bool
594 | ty::TyKind::Float(_)
595 | ty::TyKind::Char
596 | ty::TyKind::Int(_)
597 | ty::TyKind::Uint(_)
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700598 | ty::TyKind::FnPtr(_)
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -0800599 | ty::TyKind::Never => ty
600 .to_string()
601 .parse()
602 .expect("rustc_middle::ty::Ty::to_string() should produce no parsing errors"),
603 ty::TyKind::Tuple(types) => {
604 if types.len() == 0 {
605 quote! { () }
606 } else {
607 // TODO(b/254099023): Add support for tuples.
608 bail!("Tuples are not supported yet: {} (b/254099023)", ty);
609 }
610 }
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800611 ty::TyKind::Adt(adt, substs) => {
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800612 ensure!(substs.len() == 0, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800613 FullyQualifiedName::new(tcx, adt.did()).format_for_rs()
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700614 }
615 ty::TyKind::RawPtr(ty::TypeAndMut { ty, mutbl }) => {
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800616 let qualifier = match mutbl {
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700617 Mutability::Mut => quote! { mut },
618 Mutability::Not => quote! { const },
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800619 };
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700620 let ty = format_ty_for_rs(tcx, *ty).with_context(|| {
621 format!("Failed to format the pointee of the pointer type `{ty}`")
622 })?;
623 quote! { * #qualifier #ty }
624 }
Lukasz Anforowicz087dff72023-02-17 12:13:32 -0800625 _ => bail!("The following Rust type is not supported yet: {ty}"),
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -0800626 })
627}
628
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700629#[derive(Debug, Default)]
630struct ApiSnippets {
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800631 /// Main API - for example:
632 /// - A C++ declaration of a function (with a doc comment),
633 /// - A C++ definition of a struct (with a doc comment).
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700634 main_api: CcSnippet,
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800635
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700636 /// C++ implementation details - for example:
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800637 /// - A C++ declaration of an `extern "C"` thunk,
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700638 /// - C++ `static_assert`s about struct size, aligment, and field offsets.
639 cc_details: CcSnippet,
640
641 /// Rust implementation details - for exmaple:
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800642 /// - A Rust implementation of an `extern "C"` thunk,
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700643 /// - Rust `assert!`s about struct size, aligment, and field offsets.
644 rs_details: TokenStream,
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800645}
646
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700647impl FromIterator<ApiSnippets> for ApiSnippets {
648 fn from_iter<I: IntoIterator<Item = ApiSnippets>>(iter: I) -> Self {
649 let mut result = ApiSnippets::default();
650 for ApiSnippets { main_api, cc_details, rs_details } in iter.into_iter() {
651 result.main_api += main_api;
652 result.cc_details += cc_details;
653 result.rs_details.extend(rs_details);
654 }
655 result
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -0800656 }
657}
658
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -0700659fn get_fn_sig<'tcx>(tcx: TyCtxt<'tcx>, fn_def_id: LocalDefId) -> Result<ty::FnSig<'tcx>> {
660 match tcx.fn_sig(fn_def_id).subst_identity().no_bound_vars() {
661 None => bail!("Generic functions are not supported yet (b/259749023)"),
662 Some(sig) => Ok(sig),
663 }
664}
665
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700666/// Formats a C++ function declaration of a thunk that wraps a Rust function
667/// identified by `fn_def_id`. `format_thunk_impl` may panic if `fn_def_id`
668/// doesn't identify a function.
Lukasz Anforowicz9924c432023-04-04 12:51:22 -0700669fn format_thunk_decl(
670 input: &Input,
671 fn_def_id: LocalDefId,
672 thunk_name: &TokenStream,
673) -> Result<CcSnippet> {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700674 let tcx = input.tcx;
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700675
676 let mut prereqs = CcPrerequisites::default();
677 let sig = get_fn_sig(tcx, fn_def_id)?;
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700678 let main_api_ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut prereqs);
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700679
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700680 let mut thunk_params = {
681 let cc_types = format_param_types_for_cc(input, &sig)?;
682 sig.inputs()
683 .iter()
684 .zip(cc_types.into_iter())
685 .map(|(&ty, cc_type)| -> Result<TokenStream> {
686 let cc_type = cc_type.into_tokens(&mut prereqs);
687 if is_c_abi_compatible_by_value(ty) {
688 Ok(quote! { #cc_type })
689 } else {
690 // Rust thunk will move a value via memcpy - we need to `ensure` that
691 // invoking the C++ destructor (on the moved-away value) is safe.
692 // TODO(b/259749095): Support generic structs (with non-empty ParamEnv).
693 ensure!(
694 !ty.needs_drop(tcx, ty::ParamEnv::empty()),
695 "Only trivially-movable and trivially-destructible types \
696 may be passed by value over the FFI boundary"
697 );
698 Ok(quote! { #cc_type* })
699 }
700 })
701 .collect::<Result<Vec<_>>>()?
702 };
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700703
704 let thunk_ret_type: TokenStream;
705 if is_c_abi_compatible_by_value(sig.output()) {
706 thunk_ret_type = main_api_ret_type.clone();
707 } else {
708 thunk_ret_type = quote! { void };
709 thunk_params.push(quote! { #main_api_ret_type* __ret_ptr });
710 prereqs.includes.insert(CcInclude::utility());
711 prereqs.includes.insert(input.support_header("internal/return_value_slot.h"));
712 };
713 Ok(CcSnippet {
714 prereqs,
715 tokens: quote! {
716 namespace __crubit_internal {
717 extern "C" #thunk_ret_type #thunk_name ( #( #thunk_params ),* );
718 }
719 },
720 })
721}
722
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -0700723/// Formats a thunk implementation in Rust that provides an `extern "C"` ABI for
724/// calling a Rust function identified by `fn_def_id`. `format_thunk_impl` may
725/// panic if `fn_def_id` doesn't identify a function.
726///
727/// `fully_qualified_fn_name` specifies how the thunk can identify the function
728/// to call. Examples of valid arguments:
729/// - `::crate_name::some_module::free_function`
730/// - `::crate_name::some_module::SomeStruct::method`
731/// - `<::create_name::some_module::SomeStruct as
732/// ::core::default::Default>::default`
733fn format_thunk_impl(
734 tcx: TyCtxt,
735 fn_def_id: LocalDefId,
736 thunk_name: &str,
737 fully_qualified_fn_name: TokenStream,
738) -> Result<TokenStream> {
739 let sig = get_fn_sig(tcx, fn_def_id)?;
740 let param_names_and_types: Vec<(Ident, Ty)> = {
741 let param_names = tcx.fn_arg_names(fn_def_id).iter().enumerate().map(|(i, name)| {
742 if name.as_str().is_empty() {
743 format_ident!("__param_{i}")
744 } else {
745 make_rs_ident(name.as_str())
746 }
747 });
748 let param_types = sig.inputs().iter().copied();
749 param_names.zip(param_types).collect_vec()
750 };
751
752 let mut thunk_params = param_names_and_types
753 .iter()
754 .map(|(param_name, ty)| {
755 let rs_type = format_ty_for_rs(tcx, *ty)
756 .with_context(|| format!("Error handling parameter `{param_name}`"))?;
757 Ok(if is_c_abi_compatible_by_value(*ty) {
758 quote! { #param_name: #rs_type }
759 } else {
760 quote! { #param_name: &mut ::core::mem::MaybeUninit<#rs_type> }
761 })
762 })
763 .collect::<Result<Vec<_>>>()?;
764
765 let mut thunk_ret_type = format_ty_for_rs(tcx, sig.output())?;
766 let mut thunk_body = {
767 let fn_args = param_names_and_types.iter().map(|(rs_name, ty)| {
768 if is_c_abi_compatible_by_value(*ty) {
769 quote! { #rs_name }
770 } else {
771 quote! { unsafe { #rs_name.assume_init_read() } }
772 }
773 });
774 quote! {
775 #fully_qualified_fn_name( #( #fn_args ),* )
776 }
777 };
778 if !is_c_abi_compatible_by_value(sig.output()) {
779 thunk_params.push(quote! {
780 __ret_slot: &mut ::core::mem::MaybeUninit<#thunk_ret_type>
781 });
782 thunk_ret_type = quote! { () };
783 thunk_body = quote! { __ret_slot.write(#thunk_body); };
784 };
785
786 let thunk_name = make_rs_ident(thunk_name);
787 Ok(quote! {
788 #[no_mangle]
789 extern "C" fn #thunk_name( #( #thunk_params ),* ) -> #thunk_ret_type {
790 #thunk_body
791 }
792 })
793}
794
Lukasz Anforowicz9924c432023-04-04 12:51:22 -0700795fn get_symbol_name<'tcx>(tcx: TyCtxt<'tcx>, def_id: LocalDefId) -> Result<&'tcx str> {
796 ensure!(
797 tcx.generics_of(def_id).count() == 0,
798 "Generic functions are not supported yet (b/259749023) - caller should filter them out",
799 );
800
801 // Call to `mono` is ok - `generics_of` have been checked above.
802 let instance = ty::Instance::mono(tcx, def_id.to_def_id());
803
804 Ok(tcx.symbol_name(instance).name)
805}
806
807fn get_thunk_name(symbol_name: &str) -> String {
808 format!("__crubit_thunk_{}", &escape_non_identifier_chars(symbol_name))
809}
810
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700811fn check_fn_sig(sig: &ty::FnSig) -> Result<()> {
812 if sig.c_variadic {
813 // TODO(b/254097223): Add support for variadic functions.
814 bail!("C variadic functions are not supported (b/254097223)");
815 }
816
817 match sig.unsafety {
818 Unsafety::Normal => (),
819 Unsafety::Unsafe => {
820 // TODO(b/254095482): Figure out how to handle `unsafe` functions.
821 bail!("Bindings for `unsafe` functions are not fully designed yet (b/254095482)");
822 }
823 }
824
825 Ok(())
826}
827
828/// Returns `Ok(())` if no thunk is required.
829/// Otherwise returns an error the describes why the thunk is needed.
830fn is_thunk_required(sig: &ty::FnSig) -> Result<()> {
831 match sig.abi {
832 // "C" ABI is okay: Before https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a
833 // Rust panic that "escapes" a "C" ABI function leads to Undefined Behavior. This is
834 // unfortunate, but Crubit's `panics_and_exceptions.md` documents that `-Cpanic=abort`
835 // is the only supported configuration.
836 //
837 // After https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a Rust panic that
838 // tries to "escape" a "C" ABI function will terminate the program. This is okay.
839 rustc_target::spec::abi::Abi::C { unwind: false } => (),
840
841 // "C-unwind" ABI is okay: After
842 // https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a new "C-unwind" ABI may be
843 // used by Rust functions that want to safely propagate Rust panics through frames that
844 // may belong to another language.
845 rustc_target::spec::abi::Abi::C { unwind: true } => (),
846
847 // All other ABIs trigger thunk generation. This covers Rust ABI functions, but also
848 // ABIs that theoretically are understood both by C++ and Rust (e.g. see
849 // `format_cc_call_conv_as_clang_attribute` in `rs_bindings_from_cc/src_code_gen.rs`).
850 _ => bail!("Calling convention other than `extern \"C\"` requires a thunk"),
851 };
852
853 ensure!(is_c_abi_compatible_by_value(sig.output()), "Return type requires a thunk");
854 for (i, param_ty) in sig.inputs().iter().enumerate() {
855 ensure!(is_c_abi_compatible_by_value(*param_ty), "Type of parameter #{i} requires a thunk",);
856 }
857
858 Ok(())
859}
860
Googlerfb204272022-12-02 00:52:05 -0800861/// Formats a function with the given `local_def_id`.
Lukasz Anforowicze4333062022-10-17 14:47:53 -0700862///
Googlerfb204272022-12-02 00:52:05 -0800863/// Will panic if `local_def_id`
Lukasz Anforowicz7123f212022-10-20 08:51:04 -0700864/// - is invalid
865/// - doesn't identify a function,
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700866fn format_fn(input: &Input, local_def_id: LocalDefId) -> Result<ApiSnippets> {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -0800867 let tcx = input.tcx;
Googler624580b2022-12-01 01:23:42 -0800868 let def_id: DefId = local_def_id.to_def_id(); // Convert LocalDefId to DefId.
Lukasz Anforowicz903fc632022-10-25 08:55:33 -0700869
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -0800870 ensure!(
871 tcx.generics_of(def_id).count() == 0,
872 "Generic functions are not supported yet (b/259749023)"
873 );
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -0800874
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -0700875 let sig = get_fn_sig(tcx, local_def_id)?;
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700876 check_fn_sig(&sig)?;
877 let needs_thunk = is_thunk_required(&sig).is_err();
Lukasz Anforowicz9924c432023-04-04 12:51:22 -0700878 let thunk_name = {
879 let symbol_name = get_symbol_name(tcx, local_def_id)?;
880 if needs_thunk { get_thunk_name(symbol_name) } else { symbol_name.to_string() }
881 };
Googler624580b2022-12-01 01:23:42 -0800882
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -0700883 let FullyQualifiedName { krate, mod_path, name } = FullyQualifiedName::new(tcx, def_id);
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800884 let fn_name = name.expect("Functions are assumed to always have a name");
885 let main_api_fn_name =
886 format_cc_ident(fn_name.as_str()).context("Error formatting function name")?;
Lukasz Anforowicza577d822022-12-12 15:00:46 -0800887
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800888 let mut main_api_prereqs = CcPrerequisites::default();
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700889 let main_api_ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut main_api_prereqs);
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800890 let is_static_method = match tcx.hir().get_by_def_id(local_def_id) {
891 Node::ImplItem(impl_item) => match &impl_item.kind {
892 ImplItemKind::Fn(fn_sig, _) => match fn_sig.decl.implicit_self {
893 ImplicitSelfKind::None => true,
894 _ => bail!("`self` parameter is not supported yet"),
895 },
896 _ => panic!("`format_fn` can only work with functions"),
897 },
898 Node::Item(_) => false, // Free function
899 other => panic!("Unexpected HIR node kind: {other:?}"),
900 };
901
902 struct Param<'tcx> {
903 cc_name: TokenStream,
904 cc_type: TokenStream,
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800905 ty: Ty<'tcx>,
906 }
907 let params = {
908 let names = tcx.fn_arg_names(def_id).iter();
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700909 let cc_types = format_param_types_for_cc(input, &sig)?;
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800910 names
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800911 .enumerate()
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700912 .zip(sig.inputs().iter())
913 .zip(cc_types.into_iter())
914 .map(|(((i, name), &ty), cc_type)| {
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800915 let cc_name = format_cc_ident(name.as_str())
916 .unwrap_or_else(|_err| format_cc_ident(&format!("__param_{i}")).unwrap());
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700917 let cc_type = cc_type.into_tokens(&mut main_api_prereqs);
918 Param { cc_name, cc_type, ty }
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800919 })
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700920 .collect_vec()
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800921 };
922 let main_api_params = params
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800923 .iter()
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800924 .map(|Param { cc_name, cc_type, .. }| quote! { #cc_type #cc_name })
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800925 .collect_vec();
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800926
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -0800927 let struct_name = match tcx.impl_of_method(def_id) {
928 Some(impl_id) => match tcx.impl_subject(impl_id) {
929 ty::ImplSubject::Inherent(ty) => match ty.kind() {
930 ty::TyKind::Adt(adt, substs) => {
931 assert_eq!(0, substs.len(), "Callers should filter out generics");
932 Some(tcx.item_name(adt.did()))
933 }
934 _ => panic!("Non-ADT `impl`s should be filtered by caller"),
935 },
936 ty::ImplSubject::Trait(_) => panic!("Trait methods should be filtered by caller"),
937 },
938 None => None,
939 };
Lukasz Anforowicz9924c432023-04-04 12:51:22 -0700940 let needs_definition = fn_name.as_str() != thunk_name;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800941 let main_api = {
942 let doc_comment = {
943 let doc_comment = format_doc_comment(tcx, local_def_id);
944 quote! { __NEWLINE__ #doc_comment }
945 };
946
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800947 let mut prereqs = main_api_prereqs.clone();
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800948 prereqs.move_defs_to_fwd_decls();
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800949
950 let static_ = if is_static_method {
951 quote! { static }
952 } else {
953 quote! {}
954 };
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -0800955 let extern_c_or_inline = if !needs_definition {
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800956 quote! { extern "C" }
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -0800957 } else {
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800958 quote! { inline }
959 };
960 CcSnippet {
961 prereqs,
962 tokens: quote! {
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -0800963 __NEWLINE__
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800964 #doc_comment
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800965 #static_ #extern_c_or_inline
966 #main_api_ret_type #main_api_fn_name ( #( #main_api_params ),* );
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -0800967 __NEWLINE__
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800968 },
969 }
970 };
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700971 let cc_details = if !needs_definition {
972 CcSnippet::default()
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800973 } else {
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700974 let thunk_name = format_cc_ident(&thunk_name).context("Error formatting thunk name")?;
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700975 let struct_name = match struct_name.as_ref() {
976 None => quote! {},
977 Some(symbol) => {
978 let name = format_cc_ident(symbol.as_str())
979 .expect("Caller of format_fn should verify struct via format_adt_core");
980 quote! { #name :: }
981 }
982 };
983
984 let mut prereqs = main_api_prereqs;
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700985 let thunk_decl =
986 format_thunk_decl(input, local_def_id, &thunk_name)?.into_tokens(&mut prereqs);
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700987
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700988 let mut thunk_args = params
989 .iter()
990 .map(|Param { cc_name, ty, .. }| {
991 if is_c_abi_compatible_by_value(*ty) {
992 quote! { #cc_name }
993 } else {
994 quote! { & #cc_name }
995 }
996 })
997 .collect_vec();
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700998 let impl_body: TokenStream;
999 if is_c_abi_compatible_by_value(sig.output()) {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001000 impl_body = quote! {
1001 return __crubit_internal :: #thunk_name( #( #thunk_args ),* );
1002 };
1003 } else {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001004 thunk_args.push(quote! { __ret_slot.Get() });
1005 impl_body = quote! {
1006 crubit::ReturnValueSlot<#main_api_ret_type> __ret_slot;
1007 __crubit_internal :: #thunk_name( #( #thunk_args ),* );
1008 return std::move(__ret_slot).AssumeInitAndTakeValue();
1009 };
1010 prereqs.includes.insert(CcInclude::utility());
1011 prereqs.includes.insert(input.support_header("internal/return_value_slot.h"));
1012 };
1013 CcSnippet {
1014 prereqs,
1015 tokens: quote! {
1016 __NEWLINE__
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07001017 #thunk_decl
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001018 inline #main_api_ret_type #struct_name #main_api_fn_name (
1019 #( #main_api_params ),* ) {
1020 #impl_body
1021 }
1022 __NEWLINE__
1023 },
1024 }
1025 };
1026
1027 let rs_details = if !needs_thunk {
1028 quote! {}
1029 } else {
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -07001030 let crate_name = make_rs_ident(krate.as_str());
1031 let mod_path = mod_path.format_for_rs();
1032 let fn_name = make_rs_ident(fn_name.as_str());
1033 let struct_name = match struct_name.as_ref() {
1034 None => quote! {},
1035 Some(symbol) => {
1036 let name = make_rs_ident(symbol.as_str());
1037 quote! { #name :: }
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001038 }
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08001039 };
Devin Jeanpierre81aec502023-04-04 15:33:39 -07001040 let fully_qualified_fn_name = quote! { :: #crate_name :: #mod_path #struct_name #fn_name };
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001041 format_thunk_impl(tcx, local_def_id, &thunk_name, fully_qualified_fn_name)?
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001042 };
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001043 Ok(ApiSnippets { main_api, cc_details, rs_details })
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001044}
1045
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001046/// Represents bindings for the "core" part of an algebraic data type (an ADT -
1047/// a struct, an enum, or a union) in a way that supports later injecting the
1048/// other parts like so:
1049///
1050/// ```
1051/// quote! {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001052/// #keyword #alignment #name final {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001053/// #core
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001054/// #decls_of_other_parts // (e.g. struct fields, methods, etc.)
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001055/// }
1056/// }
1057/// ```
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001058///
1059/// `keyword`, `name` are stored separately, to support formatting them as a
1060/// forward declaration - e.g. `struct SomeStruct`.
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001061struct AdtCoreBindings {
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001062 /// DefId of the ADT.
1063 def_id: DefId,
1064
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001065 /// C++ tag - e.g. `struct`, `class`, `enum`, or `union`. This isn't always
1066 /// a direct mapping from Rust (e.g. a Rust `enum` might end up being
1067 /// represented as an opaque C++ `struct`).
1068 keyword: TokenStream,
1069
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001070 /// C++ translation of the ADT identifier - e.g. `SomeStruct`.
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001071 ///
1072 /// A _short_ name is sufficient (i.e. there is no need to use a
1073 /// namespace-qualified name), for `CcSnippet`s that are emitted into
1074 /// the same namespace as the ADT. (This seems to be all the snippets
1075 /// today.)
1076 cc_short_name: TokenStream,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001077
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001078 /// Rust spelling of the ADT type - e.g.
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001079 /// `::some_crate::some_module::SomeStruct`.
1080 rs_fully_qualified_name: TokenStream,
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001081
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001082 /// `core` contains declarations of
1083 /// - the default constructor
1084 /// - the copy constructor
1085 /// - the move constructor
1086 /// - the copy assignment operator
1087 /// - the move assignment operator
1088 /// - the destructor
1089 core: TokenStream,
1090
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001091 alignment_in_bytes: u64,
1092 size_in_bytes: u64,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001093}
1094
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -08001095/// Like `TyCtxt::is_directly_public`, but works not only with `LocalDefId`, but
1096/// also with `DefId`.
1097fn is_directly_public(tcx: TyCtxt, def_id: DefId) -> bool {
1098 match def_id.as_local() {
1099 None => {
1100 // This mimics the checks in `try_print_visible_def_path_recur` in
1101 // `compiler/rustc_middle/src/ty/print/pretty.rs`.
1102 let actual_parent = tcx.opt_parent(def_id);
1103 let visible_parent = tcx.visible_parent_map(()).get(&def_id).copied();
1104 actual_parent == visible_parent
1105 }
1106 Some(local_def_id) => tcx.effective_visibilities(()).is_directly_public(local_def_id),
1107 }
1108}
1109
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001110fn get_layout<'tcx>(tcx: TyCtxt<'tcx>, ty: Ty<'tcx>) -> Result<Layout<'tcx>> {
1111 // TODO(b/259749095): Support non-empty set of generic parameters.
1112 let param_env = ty::ParamEnv::empty();
1113
1114 tcx.layout_of(param_env.and(ty)).map(|ty_and_layout| ty_and_layout.layout).map_err(
1115 |layout_err| {
1116 // Have to use `.map_err`, because `LayoutError` doesn't satisfy the
1117 // `anyhow::context::ext::StdError` trait bound.
1118 anyhow!("Error computing the layout: {layout_err}")
1119 },
1120 )
1121}
1122
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001123/// Formats the core of an algebraic data type (an ADT - a struct, an enum, or a
1124/// union) represented by `def_id`.
1125///
1126/// The "core" means things that are necessary for a succesful binding (e.g.
1127/// inability to generate a correct C++ destructor means that the ADT cannot
1128/// have any bindings). "core" excludes things that are A) infallible (e.g.
1129/// struct or union fields which can always be translated into private, opaque
1130/// blobs of bytes) or B) optional (e.g. a problematic instance method
1131/// can just be ignored, unlike a problematic destructor). The split between
1132/// fallible "core" and non-fallible "rest" is motivated by the need to avoid
1133/// cycles / infinite recursion (e.g. when processing fields that refer back to
1134/// the struct type, possible with an indirection of a pointer).
1135///
1136/// `format_adt_core` is used both to 1) format bindings for the core of an ADT,
1137/// and 2) check if formatting would have succeeded (e.g. when called from
1138/// `format_ty`). The 2nd case is needed for ADTs defined in any crate - this
1139/// is why the `def_id` parameter is a DefId rather than LocalDefId.
1140//
Lukasz Anforowicz1ab5e872022-12-05 10:07:00 -08001141// TODO(b/259724276): This function's results should be memoized.
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001142fn format_adt_core(tcx: TyCtxt, def_id: DefId) -> Result<AdtCoreBindings> {
Lukasz Anforowicz0c07b092023-03-03 10:34:15 -08001143 let ty = tcx.type_of(def_id).subst_identity();
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -08001144 assert!(ty.is_adt());
1145 assert!(is_directly_public(tcx, def_id), "Caller should verify");
1146
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001147 // TODO(b/259749095): Support non-empty set of generic parameters.
1148 let param_env = ty::ParamEnv::empty();
1149
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001150 if ty.needs_drop(tcx, param_env) {
1151 // TODO(b/258251148): Support custom `Drop` impls.
1152 bail!("`Drop` trait and \"drop glue\" are not supported yet (b/258251148)");
1153 }
1154
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08001155 let adt_def = ty.ty_adt_def().expect("`def_id` needs to identify an ADT");
1156 let keyword = match adt_def.adt_kind() {
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08001157 ty::AdtKind::Struct | ty::AdtKind::Enum => quote! { struct },
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08001158 ty::AdtKind::Union => quote! { union },
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08001159 };
1160
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001161 let item_name = tcx.item_name(def_id);
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001162 let rs_fully_qualified_name = format_ty_for_rs(tcx, ty)?;
1163 let cc_short_name =
1164 format_cc_ident(item_name.as_str()).context("Error formatting item name")?;
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001165
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001166 let layout = get_layout(tcx, ty)
1167 .with_context(|| format!("Error computing the layout of #{item_name}"))?;
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001168 let alignment_in_bytes = {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001169 // Only the ABI-mandated alignment is considered (i.e. `AbiAndPrefAlign::pref`
1170 // is ignored), because 1) Rust's `std::mem::align_of` returns the
1171 // ABI-mandated alignment and 2) the generated C++'s `alignas(...)`
1172 // should specify the minimal/mandatory alignment.
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001173 layout.align().abi.bytes()
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001174 };
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001175 let size_in_bytes = layout.size().bytes();
1176 ensure!(size_in_bytes != 0, "Zero-sized types (ZSTs) are not supported (b/258259459)");
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001177
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001178 let core = quote! {
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001179 // TODO(b/258249993): Provide `default` copy constructor and assignment operator if
1180 // the wrapped type is `Copy` on Rust side.
1181 // TODO(b/259741191): If the wrapped type implements the `Clone` trait, then we should
1182 // *consider* calling `clone` from the copy constructor and `clone_from` from the copy
1183 // assignment operator.
1184 #cc_short_name(const #cc_short_name&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001185
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001186 // The generated bindings have to follow Rust move semantics:
1187 // * All Rust types are memcpy-movable (e.g. <internal link>/constructors.html says
1188 // that "Every type must be ready for it to be blindly memcopied to somewhere else
1189 // in memory")
1190 // * The only valid operation on a moved-from non-`Copy` Rust struct is to assign to
1191 // it.
1192 //
1193 // The generated C++ bindings match the required semantics because they:
1194 // * Generate trivial` C++ move constructor and move assignment operator. Per
1195 // <internal link>/cpp/language/move_constructor#Trivial_move_constructor: "A trivial move
1196 // constructor is a constructor that performs the same action as the trivial copy
1197 // constructor, that is, makes a copy of the object representation as if by
1198 // std::memmove."
1199 // * Generate trivial C++ destructor. (Types that implement `Drop` trait or require
1200 // "drop glue" are not *yet* supported - this might eventually change as part of the
1201 // work tracked under b/258251148). Per
1202 // <internal link>/cpp/language/destructor#Trivial_destructor: "A trivial destructor is a
1203 // destructor that performs no action."
1204 //
1205 // In particular, note that the following C++ code and Rust code are exactly equivalent
1206 // (except that in Rust, reuse of `y` is forbidden at compile time, whereas in C++,
1207 // it's only prohibited by convention):
1208 // * C++, assumming trivial move constructor and trivial destructor:
1209 // `auto x = std::move(y);`
1210 // * Rust, assumming non-`Copy`, no custom `Drop` or drop glue:
1211 // `let x = y;`
1212 //
1213 // TODO(b/258251148): If the ADT provides a custom `Drop` impls or requires drop glue,
1214 // then extra care should be taken to ensure the C++ destructor can handle the
1215 // moved-from object in a way that meets Rust move semantics. For example, the
1216 // generated C++ move constructor might need to assign `Default::default()` to the
1217 // moved-from object.
1218 #cc_short_name(#cc_short_name&&) = default;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001219
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001220 // TODO(b/258235219): Providing assignment operators enables mutation which
1221 // may negatively interact with support for references. Therefore until we
1222 // have more confidence in our reference-handling-plans, we are deleting the
1223 // assignment operators.
1224 //
1225 // (Move assignment operator has another set of concerns and constraints - see the
1226 // comment for the move constructor above).
1227 #cc_short_name& operator=(const #cc_short_name&) = delete;
1228 #cc_short_name& operator=(#cc_short_name&&) = delete;
Lukasz Anforowicz554ed652023-01-12 15:41:58 -08001229
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001230 // TODO(b/258251148): Support custom `Drop` impls and drop glue.
1231 ~#cc_short_name() = default;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001232 };
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001233 Ok(AdtCoreBindings {
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001234 def_id,
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08001235 keyword,
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001236 cc_short_name,
1237 rs_fully_qualified_name,
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001238 core,
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001239 alignment_in_bytes,
1240 size_in_bytes,
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001241 })
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001242}
1243
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001244fn format_fields(input: &Input, core: &AdtCoreBindings) -> ApiSnippets {
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08001245 let tcx = input.tcx;
1246
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001247 // TODO(b/259749095): Support non-empty set of generic parameters.
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001248 let substs_ref = ty::List::empty().as_substs();
1249
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001250 struct FieldTypeInfo {
1251 size: u64,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001252 cc_type: CcSnippet,
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001253 }
1254 struct Field {
1255 type_info: Result<FieldTypeInfo>,
1256 cc_name: TokenStream,
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07001257 rs_name: TokenStream,
1258 is_public: bool,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001259 index: usize,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001260 offset: u64,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001261 offset_of_next_field: u64,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001262 }
1263 let ty = tcx.type_of(core.def_id).subst_identity();
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001264 let layout =
1265 get_layout(tcx, ty).expect("Layout should be already verified by `format_adt_core`");
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001266 let fields: Vec<Field> = if ty.is_enum() || ty.is_union() {
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001267 // Note that `#[repr(Rust)]` unions don't guarantee that all their fields
1268 // have offset 0.
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001269 vec![Field {
1270 type_info: Err(anyhow!(
1271 "No support for bindings of individual fields of \
1272 `union` (b/272801632) or `enum`"
1273 )),
1274 cc_name: quote! { __opaque_blob_of_bytes },
1275 rs_name: quote! { __opaque_blob_of_bytes },
1276 is_public: false,
1277 index: 0,
1278 offset: 0,
1279 offset_of_next_field: core.size_in_bytes,
1280 }]
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001281 } else {
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001282 let mut fields = ty
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001283 .ty_adt_def()
1284 .expect("`core.def_id` needs to identify an ADT")
1285 .all_fields()
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001286 .sorted_by_key(|f| tcx.def_span(f.did))
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001287 .enumerate()
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001288 .map(|(index, field_def)| {
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001289 let field_ty = field_def.ty(tcx, substs_ref);
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001290 let size = get_layout(tcx, field_ty).map(|layout| layout.size().bytes());
1291 let type_info = size.and_then(|size| {
Googler7a77a802023-04-21 08:32:50 -07001292 Ok(FieldTypeInfo {
1293 size,
1294 cc_type: format_ty_for_cc(input, field_ty, TypeLocation::Other)?,
1295 })
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001296 });
1297 let name = field_def.ident(tcx);
Devin Jeanpierre81aec502023-04-04 15:33:39 -07001298 let cc_name = format_cc_ident(name.as_str())
1299 .unwrap_or_else(|_err| format_ident!("__field{index}").into_token_stream());
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07001300 let rs_name = {
1301 let name_starts_with_digit = name
1302 .as_str()
1303 .chars()
1304 .next()
1305 .expect("Empty names are unexpected (here and in general)")
1306 .is_ascii_digit();
1307 if name_starts_with_digit {
1308 let index = Literal::usize_unsuffixed(index);
1309 quote! { #index }
1310 } else {
1311 let name = make_rs_ident(name.as_str());
1312 quote! { #name }
1313 }
1314 };
1315 let is_public = field_def.vis == ty::Visibility::Public;
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001316
1317 // `offset` and `offset_of_next_field` will be fixed by FieldsShape::Arbitrary
1318 // branch below.
1319 let offset = 0;
1320 let offset_of_next_field = 0;
1321
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001322 Field {
1323 type_info,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001324 cc_name,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001325 rs_name,
1326 is_public,
1327 index,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001328 offset,
1329 offset_of_next_field,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001330 }
1331 })
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001332 .collect_vec();
1333 match layout.fields() {
1334 FieldsShape::Arbitrary { offsets, .. } => {
1335 for (index, offset) in offsets.iter().enumerate() {
1336 // Documentation of `FieldsShape::Arbitrary says that the offsets are "ordered
1337 // to match the source definition order". We can coorelate them with elements
1338 // of the `fields` vector because we've explicitly `sorted_by_key` using
1339 // `def_span`.
1340 fields[index].offset = offset.bytes();
1341 }
Devin Jeanpierre9e15d0b2023-04-06 13:18:22 -07001342 // Sort by offset first; ZSTs in the same offset are sorted by source order.
1343 fields.sort_by_key(|field| (field.offset, field.index));
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001344 let next_offsets = fields
1345 .iter()
1346 .map(|Field { offset, .. }| *offset)
1347 .skip(1)
1348 .chain(once(core.size_in_bytes))
1349 .collect_vec();
1350 for (field, next_offset) in fields.iter_mut().zip(next_offsets) {
1351 field.offset_of_next_field = next_offset;
1352 }
1353 fields
1354 }
1355 unexpected => panic!("Unexpected FieldsShape: {unexpected:?}"),
1356 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001357 };
1358
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001359 let cc_details = if fields.is_empty() {
1360 CcSnippet::default()
1361 } else {
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001362 let adt_cc_name = &core.cc_short_name;
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001363 let cc_assertions: TokenStream = fields
1364 .iter()
1365 .map(|Field { cc_name, offset, .. }| {
1366 let offset = Literal::u64_unsuffixed(*offset);
1367 quote! { static_assert(#offset == offsetof(#adt_cc_name, #cc_name)); }
1368 })
1369 .collect();
1370 CcSnippet::new(quote! {
1371 inline void #adt_cc_name::__crubit_field_offset_assertions() {
1372 #cc_assertions
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08001373 }
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001374 })
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001375 };
1376 let rs_details: TokenStream = {
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001377 let adt_rs_name = &core.rs_fully_qualified_name;
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001378 fields
1379 .iter()
1380 .filter(|Field { is_public, .. }| *is_public)
1381 .map(|Field { rs_name, offset, .. }| {
1382 let expected_offset = Literal::u64_unsuffixed(*offset);
1383 let actual_offset = quote! { memoffset::offset_of!(#adt_rs_name, #rs_name) };
1384 quote! { const _: () = assert!(#actual_offset == #expected_offset); }
1385 })
1386 .collect()
1387 };
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08001388 let main_api = {
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001389 let assertions_method_decl = if fields.is_empty() {
1390 quote! {}
1391 } else {
1392 // We put the assertions in a method so that they can read private member
1393 // variables.
1394 quote! { inline static void __crubit_field_offset_assertions(); }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001395 };
1396
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001397 let mut prereqs = CcPrerequisites::default();
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001398 let fields: TokenStream = fields
1399 .into_iter()
1400 .map(|field| {
1401 let cc_name = field.cc_name;
1402 match field.type_info {
1403 Err(err) => {
Devin Jeanpierre9e15d0b2023-04-06 13:18:22 -07001404 let size = field.offset_of_next_field - field.offset;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001405 let msg =
1406 format!("Field type has been replaced with a blob of bytes: {err:#}");
Devin Jeanpierre9e15d0b2023-04-06 13:18:22 -07001407
1408 // Empty arrays are ill-formed, but also unnecessary for padding.
1409 if size > 0 {
1410 let size = Literal::u64_unsuffixed(size);
1411 quote! {
1412 __COMMENT__ #msg
1413 unsigned char #cc_name[#size];
1414 }
1415 } else {
1416 // TODO(b/258259459): finalize the approach here.
1417 // Possibly we should, rather than using no_unique_address, drop the
1418 // field entirely. This also requires removing the field's assertions,
1419 // added above.
1420 quote! {
1421 __COMMENT__ #msg
1422 [[no_unique_address]] struct{} #cc_name;
1423 }
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001424 }
1425 }
1426 Ok(FieldTypeInfo { cc_type, size }) => {
1427 let padding = field.offset_of_next_field - field.offset - size;
1428 let padding = if padding == 0 {
1429 quote! {}
1430 } else {
1431 let padding = Literal::u64_unsuffixed(padding);
1432 let ident = format_ident!("__padding{}", field.index);
1433 quote! { unsigned char #ident[#padding]; }
1434 };
1435 let cc_type = cc_type.into_tokens(&mut prereqs);
1436 quote! { #cc_type #cc_name; #padding }
1437 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001438 }
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001439 })
1440 .collect();
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001441
1442 CcSnippet {
1443 prereqs,
1444 tokens: quote! {
1445 // TODO(b/271002281): Preserve actual field visibility.
1446 private: __NEWLINE__
1447 #fields
1448 #assertions_method_decl
1449 },
1450 }
1451 };
1452
1453 ApiSnippets { main_api, cc_details, rs_details }
1454}
1455
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001456/// Finds the `Impl` of a trait impl for `self_ty`. Returns an error if the
1457/// impl wasn't found.
1458///
1459/// `self_ty` should specify a *local* type (i.e. type defined in the crate
1460/// being "compiled").
1461///
1462/// `trait_name` should specify the name of a `core` trait - e.g.
1463/// [`sym::Default`](https://doc.rust-lang.org/beta/nightly-rustc/rustc_span/symbol/sym/constant.Default.html) is a valid
1464/// argument.
1465fn find_core_trait_impl<'tcx>(
1466 tcx: TyCtxt<'tcx>,
1467 self_ty: Ty<'tcx>,
1468 trait_name: Symbol,
1469) -> Result<&'tcx Impl<'tcx>> {
1470 let trait_id = tcx
1471 .get_diagnostic_item(trait_name)
1472 .expect("`find_core_trait_impl` should only be called with `core`, always-present traits");
1473 // TODO(b/275387739): Eventually we might need to support blanket impls.
1474 let mut impls = tcx.non_blanket_impls_for_ty(trait_id, self_ty);
1475 let impl_id = impls.next();
1476 if impl_id.is_some() {
1477 assert_eq!(None, impls.next(), "Expecting only a single trait impl");
1478 }
1479 let impl_id =
1480 impl_id.ok_or_else(|| anyhow!("`{self_ty}` doesn't implement the `{trait_name}` trait"))?;
1481 let impl_id = impl_id.expect_local(); // Expecting that `self_ty` is a local type.
1482 match &tcx.hir().expect_item(impl_id).kind {
1483 ItemKind::Impl(impl_) => Ok(impl_),
1484 other => panic!("Unexpected `ItemKind` from `non_blanket_impls_for_ty`: {other:?}"),
1485 }
1486}
1487
1488/// Formats a default constructor for an ADT if possible (i.e. if the `Default`
1489/// trait is implemented for the ADT). Returns an error otherwise (e.g. if
1490/// there is no `Default` impl).
1491fn format_default_ctor(input: &Input, core: &AdtCoreBindings) -> Result<ApiSnippets> {
1492 let tcx = input.tcx;
1493 let ty = tcx.type_of(core.def_id).subst_identity();
1494
1495 let trait_impl = find_core_trait_impl(input.tcx, ty, sym::Default)?;
1496 assert_eq!(trait_impl.items.len(), 1, "Only the `default` method is expected");
1497 assert_eq!(trait_impl.items[0].ident.name.as_str(), "default");
1498 let cc_struct_name = &core.cc_short_name;
1499 let main_api = CcSnippet::new(quote! {
1500 __NEWLINE__ __COMMENT__ "Default::default"
1501 inline #cc_struct_name(); __NEWLINE__ __NEWLINE__
1502 });
1503 let fn_def_id = trait_impl.items[0].id.owner_id.def_id;
1504 let thunk_name = get_thunk_name(get_symbol_name(tcx, fn_def_id)?);
1505 let cc_details = {
1506 let thunk_name = format_cc_ident(&thunk_name)?;
1507 let CcSnippet { tokens: thunk_decl, prereqs } =
1508 format_thunk_decl(input, fn_def_id, &thunk_name)?;
1509 let tokens = quote! {
1510 #thunk_decl
1511 #cc_struct_name::#cc_struct_name() {
1512 __crubit_internal::#thunk_name(this);
1513 }
1514 };
1515 CcSnippet { tokens, prereqs }
1516 };
1517 let rs_details = {
1518 let struct_name = &core.rs_fully_qualified_name;
1519 let fully_qualified_fn_name =
1520 quote! { <#struct_name as ::core::default::Default>::default };
1521 format_thunk_impl(tcx, fn_def_id, &thunk_name, fully_qualified_fn_name)?
1522 };
1523 Ok(ApiSnippets { main_api, cc_details, rs_details })
1524}
1525
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001526/// Formats an algebraic data type (an ADT - a struct, an enum, or a union)
1527/// represented by `core`. This function is infallible - after
1528/// `format_adt_core` returns success we have committed to emitting C++ bindings
1529/// for the ADT.
1530fn format_adt(input: &Input, core: &AdtCoreBindings) -> ApiSnippets {
1531 let tcx = input.tcx;
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001532 let adt_cc_name = &core.cc_short_name;
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001533
1534 // `format_adt` should only be called for local ADTs.
1535 let local_def_id = core.def_id.expect_local();
1536
1537 let ApiSnippets {
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001538 main_api: default_ctor_main_api,
1539 cc_details: default_ctor_cc_details,
1540 rs_details: default_ctor_rs_details,
1541 } = format_default_ctor(input, core).unwrap_or_else(|err| {
1542 let msg = format!("{err:#}");
1543 ApiSnippets {
1544 main_api: CcSnippet::new(quote! {
1545 __NEWLINE__ __COMMENT__ #msg
1546 #adt_cc_name() = delete; __NEWLINE__
1547 }),
1548 ..Default::default()
1549 }
1550 });
1551
1552 let ApiSnippets {
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001553 main_api: fields_main_api,
1554 cc_details: fields_cc_details,
1555 rs_details: fields_rs_details,
1556 } = format_fields(input, core);
1557
1558 let ApiSnippets {
1559 main_api: impl_items_main_api,
1560 cc_details: impl_items_cc_details,
1561 rs_details: impl_items_rs_details,
1562 } = tcx
1563 .inherent_impls(core.def_id)
1564 .iter()
1565 .map(|impl_id| tcx.hir().expect_item(impl_id.expect_local()))
1566 .flat_map(|item| match &item.kind {
1567 ItemKind::Impl(impl_) => impl_.items,
1568 other => panic!("Unexpected `ItemKind` from `inherent_impls`: {other:?}"),
1569 })
1570 .sorted_by_key(|impl_item_ref| {
1571 let def_id = impl_item_ref.id.owner_id.def_id;
1572 tcx.def_span(def_id)
1573 })
1574 .filter_map(|impl_item_ref| {
1575 let def_id = impl_item_ref.id.owner_id.def_id;
1576 if !tcx.effective_visibilities(()).is_directly_public(def_id) {
1577 return None;
1578 }
1579 let result = match impl_item_ref.kind {
1580 AssocItemKind::Fn { .. } => format_fn(input, def_id).map(Some),
1581 other => Err(anyhow!("Unsupported `impl` item kind: {other:?}")),
1582 };
1583 result.unwrap_or_else(|err| Some(format_unsupported_def(tcx, def_id, err)))
1584 })
1585 .collect();
1586
1587 let alignment = Literal::u64_unsuffixed(core.alignment_in_bytes);
1588 let size = Literal::u64_unsuffixed(core.size_in_bytes);
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001589 let main_api = {
1590 let cc_packed_attribute = {
1591 let has_packed_attribute = tcx
1592 .get_attrs(core.def_id, rustc_span::symbol::sym::repr)
1593 .flat_map(|attr| rustc_attr::parse_repr_attr(tcx.sess(), attr))
1594 .any(|repr| matches!(repr, rustc_attr::ReprPacked { .. }));
1595 if has_packed_attribute {
1596 quote! { __attribute__((packed)) }
1597 } else {
1598 quote! {}
1599 }
1600 };
1601
1602 let doc_comment = format_doc_comment(tcx, core.def_id.expect_local());
1603 let keyword = &core.keyword;
1604 let core = &core.core;
1605
1606 let mut prereqs = CcPrerequisites::default();
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001607 let default_ctor_main_api = default_ctor_main_api.into_tokens(&mut prereqs);
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001608 let impl_items_main_api = if impl_items_main_api.tokens.is_empty() {
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001609 quote! {}
1610 } else {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001611 let tokens = impl_items_main_api.into_tokens(&mut prereqs);
1612 quote! { public: #tokens }
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001613 };
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001614 let fields_main_api = fields_main_api.into_tokens(&mut prereqs);
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08001615 prereqs.fwd_decls.remove(&local_def_id);
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001616
1617 CcSnippet {
1618 prereqs,
1619 tokens: quote! {
1620 __NEWLINE__ #doc_comment
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07001621 #keyword alignas(#alignment) #cc_packed_attribute #adt_cc_name final {
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001622 public:
1623 #default_ctor_main_api
1624 #core
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001625 #impl_items_main_api
1626 #fields_main_api
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001627 };
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001628 __NEWLINE__
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001629 },
1630 }
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001631 };
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001632 let cc_details = {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001633 let mut prereqs = CcPrerequisites::default();
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001634 let default_ctor_cc_details = default_ctor_cc_details.into_tokens(&mut prereqs);
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001635 let impl_items_cc_details = impl_items_cc_details.into_tokens(&mut prereqs);
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001636 let fields_cc_details = fields_cc_details.into_tokens(&mut prereqs);
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001637 prereqs.defs.insert(local_def_id);
1638 CcSnippet {
1639 prereqs,
1640 tokens: quote! {
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07001641 __NEWLINE__
1642 static_assert(
1643 sizeof(#adt_cc_name) == #size,
1644 "Verify that struct layout didn't change since this header got generated");
1645 static_assert(
1646 alignof(#adt_cc_name) == #alignment,
1647 "Verify that struct layout didn't change since this header got generated");
1648 __NEWLINE__
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001649 #default_ctor_cc_details
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001650 #impl_items_cc_details
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001651 #fields_cc_details
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001652 },
1653 }
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001654 };
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001655 let rs_details = {
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001656 let adt_rs_name = &core.rs_fully_qualified_name;
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001657 quote! {
1658 const _: () = assert!(::std::mem::size_of::<#adt_rs_name>() == #size);
1659 const _: () = assert!(::std::mem::align_of::<#adt_rs_name>() == #alignment);
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001660 #default_ctor_rs_details
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001661 #impl_items_rs_details
1662 #fields_rs_details
1663 }
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001664 };
1665 ApiSnippets { main_api, cc_details, rs_details }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001666}
1667
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001668/// Formats the forward declaration of an algebraic data type (an ADT - a
1669/// struct, an enum, or a union), returning something like
1670/// `quote!{ struct SomeStruct; }`.
1671///
1672/// Will panic if `def_id` doesn't identify an ADT that can be successfully
1673/// handled by `format_adt_core`.
1674fn format_fwd_decl(tcx: TyCtxt, def_id: LocalDefId) -> TokenStream {
1675 let def_id = def_id.to_def_id(); // LocalDefId -> DefId conversion.
1676
1677 // `format_fwd_decl` should only be called for items from
1678 // `CcPrerequisites::fwd_decls` and `fwd_decls` should only contain ADTs
1679 // that `format_adt_core` succeeds for.
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001680 let AdtCoreBindings { keyword, cc_short_name, .. } = format_adt_core(tcx, def_id)
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001681 .expect("`format_fwd_decl` should only be called if `format_adt_core` succeeded");
1682
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001683 quote! { #keyword #cc_short_name; }
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001684}
1685
Googler47fd9572023-01-20 09:57:32 -08001686fn format_source_location(tcx: TyCtxt, local_def_id: LocalDefId) -> String {
1687 let def_span = tcx.def_span(local_def_id);
1688 let rustc_span::FileLines { file, lines } =
1689 match tcx.sess().source_map().span_to_lines(def_span) {
1690 Ok(filelines) => filelines,
1691 Err(_) => return "unknown location".to_string(),
1692 };
1693 let file_name = file.name.prefer_local().to_string();
1694 // Note: line_index starts at 0, while CodeSearch starts indexing at 1.
1695 let line_number = lines[0].line_index + 1;
1696 let google3_prefix = {
1697 // If rustc_span::FileName isn't a 'real' file, then it's surrounded by by angle
Lukasz Anforowicz1f233912023-02-14 08:50:26 -08001698 // brackets, thus don't prepend "google3/" prefix.
1699 if file.name.is_real() { "google3/" } else { "" }
Googler47fd9572023-01-20 09:57:32 -08001700 };
Googler785e6b42023-01-23 12:11:36 -08001701 format!("{google3_prefix}{file_name};l={line_number}")
Googler47fd9572023-01-20 09:57:32 -08001702}
1703
1704/// Formats the doc comment (if any) associated with the item identified by
1705/// `local_def_id`, and appends the source location at which the item is
1706/// defined.
Googlerfb204272022-12-02 00:52:05 -08001707fn format_doc_comment(tcx: TyCtxt, local_def_id: LocalDefId) -> TokenStream {
1708 let hir_id = tcx.local_def_id_to_hir_id(local_def_id);
Googler47fd9572023-01-20 09:57:32 -08001709 let doc_comment = tcx
Googlerfb204272022-12-02 00:52:05 -08001710 .hir()
1711 .attrs(hir_id)
1712 .iter()
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -08001713 .filter_map(|attr| attr.doc_str())
Googler47fd9572023-01-20 09:57:32 -08001714 .map(|symbol| symbol.to_string())
Googler785e6b42023-01-23 12:11:36 -08001715 .chain(once(format!("Generated from: {}", format_source_location(tcx, local_def_id))))
Googler34f3d572022-12-02 00:53:37 -08001716 .join("\n\n");
Googler47fd9572023-01-20 09:57:32 -08001717 quote! { __COMMENT__ #doc_comment}
Googlerfb204272022-12-02 00:52:05 -08001718}
1719
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001720/// Formats a HIR item idenfied by `def_id`. Returns `None` if the item
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001721/// can be ignored. Returns an `Err` if the definition couldn't be formatted.
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001722///
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001723/// Will panic if `def_id` is invalid (i.e. doesn't identify a HIR item).
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001724fn format_item(input: &Input, def_id: LocalDefId) -> Result<Option<ApiSnippets>> {
Lukasz Anforowicz1382f392022-12-12 17:13:23 -08001725 // TODO(b/262052635): When adding support for re-exports we may need to change
Devin Jeanpierre81aec502023-04-04 15:33:39 -07001726 // `is_directly_public` below into `is_exported`. (OTOH such change *alone* is
1727 // undesirable, because it would mean exposing items from a private module.
1728 // Exposing a private module is undesirable, because it would mean that
1729 // changes of private implementation details of the crate could become
1730 // breaking changes for users of the generated C++ bindings.)
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001731 if !input.tcx.effective_visibilities(()).is_directly_public(def_id) {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001732 return Ok(None);
Lukasz Anforowicz8b68a552022-12-12 15:07:58 -08001733 }
1734
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001735 match input.tcx.hir().expect_item(def_id) {
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08001736 Item { kind: ItemKind::Struct(_, generics) |
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08001737 ItemKind::Enum(_, generics) |
1738 ItemKind::Union(_, generics),
1739 .. } if !generics.params.is_empty() => {
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08001740 bail!("Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08001741 },
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001742 Item { kind: ItemKind::Fn(..), .. } => format_fn(input, def_id).map(Some),
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08001743 Item { kind: ItemKind::Struct(..) | ItemKind::Enum(..) | ItemKind::Union(..), .. } =>
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001744 format_adt_core(input.tcx, def_id.to_def_id())
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001745 .map(|core| Some(format_adt(input, &core))),
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08001746 Item { kind: ItemKind::Impl(_), .. } | // Handled by `format_adt`
1747 Item { kind: ItemKind::Mod(_), .. } => // Handled by `format_crate`
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001748 Ok(None),
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001749 Item { kind, .. } => bail!("Unsupported rustc_hir::hir::ItemKind: {}", kind.descr()),
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001750 }
1751}
1752
1753/// Formats a C++ comment explaining why no bindings have been generated for
1754/// `local_def_id`.
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -07001755fn format_unsupported_def(
1756 tcx: TyCtxt,
1757 local_def_id: LocalDefId,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001758 err: anyhow::Error,
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001759) -> ApiSnippets {
Googler785e6b42023-01-23 12:11:36 -08001760 let source_loc = format_source_location(tcx, local_def_id);
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -07001761 let name = tcx.def_path_str(local_def_id.to_def_id());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001762
1763 // https://docs.rs/anyhow/latest/anyhow/struct.Error.html#display-representations
1764 // says: To print causes as well [...], use the alternate selector “{:#}”.
Googler785e6b42023-01-23 12:11:36 -08001765 let msg = format!("Error generating bindings for `{name}` defined at {source_loc}: {err:#}");
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001766 let main_api = CcSnippet::new(quote! { __NEWLINE__ __NEWLINE__ __COMMENT__ #msg __NEWLINE__ });
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001767
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001768 ApiSnippets { main_api, cc_details: CcSnippet::default(), rs_details: quote! {} }
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -07001769}
1770
Lukasz Anforowicz2ec13122022-11-10 12:39:04 -08001771/// Formats all public items from the Rust crate being compiled.
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08001772fn format_crate(input: &Input) -> Result<Output> {
1773 let tcx = input.tcx;
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001774 let mut cc_details_prereqs = CcPrerequisites::default();
1775 let mut cc_details: Vec<(LocalDefId, TokenStream)> = vec![];
1776 let mut rs_body = TokenStream::default();
Lukasz Anforowicz991b2a52023-03-23 07:28:41 -07001777 let mut main_apis = HashMap::<LocalDefId, CcSnippet>::new();
1778 let formatted_items = tcx
Lukasz Anforowicz2ec13122022-11-10 12:39:04 -08001779 .hir()
1780 .items()
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001781 .filter_map(|item_id| {
Lukasz Anforowicz2ec13122022-11-10 12:39:04 -08001782 let def_id: LocalDefId = item_id.owner_id.def_id;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001783 format_item(input, def_id)
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001784 .unwrap_or_else(|err| Some(format_unsupported_def(tcx, def_id, err)))
1785 .map(|api_snippets| (def_id, api_snippets))
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07001786 })
Lukasz Anforowicz991b2a52023-03-23 07:28:41 -07001787 .sorted_by_key(|(def_id, _)| tcx.def_span(*def_id));
1788 for (def_id, api_snippets) in formatted_items {
1789 let old_item = main_apis.insert(def_id, api_snippets.main_api);
1790 assert!(old_item.is_none(), "Duplicated key: {def_id:?}");
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001791
Lukasz Anforowicz991b2a52023-03-23 07:28:41 -07001792 // `cc_details` don't participate in the toposort, because
1793 // `CcPrerequisites::defs` always use `main_api` as the predecessor
1794 // - `chain`ing `cc_details` after `ordered_main_apis` trivially
1795 // meets the prerequisites.
1796 cc_details.push((def_id, api_snippets.cc_details.into_tokens(&mut cc_details_prereqs)));
1797 rs_body.extend(api_snippets.rs_details);
1798 }
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08001799
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001800 // Find the order of `main_apis` that 1) meets the requirements of
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08001801 // `CcPrerequisites::defs` and 2) makes a best effort attempt to keep the
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001802 // `main_apis` in the same order as the source order of the Rust APIs.
Lukasz Anforowicz2ecb27f2023-01-12 15:29:51 -08001803 let ordered_ids = {
1804 let toposort::TopoSortResult { ordered: ordered_ids, failed: failed_ids } = {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001805 let nodes = main_apis.keys().copied();
1806 let deps = main_apis.iter().flat_map(|(&successor, main_api)| {
1807 let predecessors = main_api.prereqs.defs.iter().map(|&def_id| def_id);
Lukasz Anforowicz2ecb27f2023-01-12 15:29:51 -08001808 predecessors.map(move |predecessor| toposort::Dependency { predecessor, successor })
1809 });
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001810 toposort::toposort(nodes, deps, move |lhs_id, rhs_id| {
1811 tcx.def_span(*lhs_id).cmp(&tcx.def_span(*rhs_id))
1812 })
Lukasz Anforowicz2ecb27f2023-01-12 15:29:51 -08001813 };
1814 assert_eq!(
1815 0,
1816 failed_ids.len(),
1817 "There are no known scenarios where CcPrerequisites::defs can form \
1818 a dependency cycle. These `LocalDefId`s form an unexpected cycle: {}",
1819 failed_ids.into_iter().map(|id| format!("{:?}", id)).join(",")
1820 );
1821 ordered_ids
Lukasz Anforowiczab563af2022-12-15 08:09:50 -08001822 };
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08001823
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001824 // Destructure/rebuild `main_apis` (in the same order as `ordered_ids`) into
1825 // `includes`, and `ordered_cc` (mixing in `fwd_decls` and `cc_details`).
1826 let (includes, ordered_cc) = {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001827 let mut already_declared = HashSet::new();
1828 let mut fwd_decls = HashSet::new();
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001829 let mut includes = cc_details_prereqs.includes;
1830 let mut ordered_main_apis: Vec<(LocalDefId, TokenStream)> = Vec::new();
1831 for def_id in ordered_ids.into_iter() {
1832 let CcSnippet {
1833 tokens: cc_tokens,
1834 prereqs: CcPrerequisites {
1835 includes: mut inner_includes,
1836 fwd_decls: inner_fwd_decls,
1837 .. // `defs` have already been utilized by `toposort` above
Lukasz Anforowicz7f31f802022-12-16 08:24:13 -08001838 }
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001839 } = main_apis.remove(&def_id).unwrap();
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001840
1841 fwd_decls.extend(inner_fwd_decls.difference(&already_declared).copied());
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001842 already_declared.insert(def_id);
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001843 already_declared.extend(inner_fwd_decls.into_iter());
1844
1845 includes.append(&mut inner_includes);
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001846 ordered_main_apis.push((def_id, cc_tokens));
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001847 }
1848
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001849 let fwd_decls = fwd_decls
1850 .into_iter()
1851 .sorted_by_key(|def_id| tcx.def_span(*def_id))
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07001852 .map(|local_def_id| (local_def_id, format_fwd_decl(tcx, local_def_id)));
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001853
1854 let ordered_cc: Vec<(NamespaceQualifier, TokenStream)> = fwd_decls
1855 .into_iter()
1856 .chain(ordered_main_apis.into_iter())
1857 .chain(cc_details.into_iter())
1858 .map(|(local_def_id, tokens)| {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001859 let mod_path = FullyQualifiedName::new(tcx, local_def_id.to_def_id()).mod_path;
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001860 (mod_path, tokens)
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001861 })
1862 .collect_vec();
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001863
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001864 (includes, ordered_cc)
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001865 };
Lukasz Anforowicza577d822022-12-12 15:00:46 -08001866
1867 // Generate top-level elements of the C++ header file.
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001868 let h_body = {
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001869 // TODO(b/254690602): Decide whether using `#crate_name` as the name of the
1870 // top-level namespace is okay (e.g. investigate if this name is globally
1871 // unique + ergonomic).
1872 let crate_name = format_cc_ident(tcx.crate_name(LOCAL_CRATE).as_str())?;
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001873
Lukasz Anforowicz54efc162022-12-16 15:58:44 -08001874 let includes = format_cc_includes(&includes);
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08001875 let ordered_cc = format_namespace_bound_cc_tokens(ordered_cc);
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001876 quote! {
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001877 #includes
1878 __NEWLINE__ __NEWLINE__
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001879 namespace #crate_name {
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001880 __NEWLINE__
Lukasz Anforowicz54efc162022-12-16 15:58:44 -08001881 #ordered_cc
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001882 __NEWLINE__
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001883 }
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001884 __NEWLINE__
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001885 }
1886 };
Lukasz Anforowicz7f31f802022-12-16 08:24:13 -08001887
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08001888 Ok(Output { h_body, rs_body })
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -07001889}
1890
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07001891#[cfg(test)]
Lukasz Anforowiczf018e4d2022-09-28 07:35:59 -07001892pub mod tests {
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08001893 use super::*;
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07001894
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001895 use anyhow::Result;
1896 use itertools::Itertools;
1897 use proc_macro2::TokenStream;
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001898 use quote::quote;
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07001899 use rustc_middle::ty::{Ty, TyCtxt};
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001900 use rustc_span::def_id::LocalDefId;
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07001901
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08001902 use crate::run_compiler::tests::run_compiler_for_testing;
Lukasz Anforowicz52274992023-03-08 12:29:28 -08001903 use code_gen_utils::format_cc_includes;
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08001904 use token_stream_matchers::{
1905 assert_cc_matches, assert_cc_not_matches, assert_rs_matches, assert_rs_not_matches,
1906 };
Lukasz Anforowicz2b38d272022-09-23 08:08:18 -07001907
Lukasz Anforowicz5bddf182022-09-30 16:06:59 -07001908 #[test]
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001909 #[should_panic(expected = "No items named `missing_name`.\n\
1910 Instead found:\n`bar`,\n`foo`,\n`m1`,\n`m2`,\n`std`")]
1911 fn test_find_def_id_by_name_panic_when_no_item_with_matching_name() {
1912 let test_src = r#"
1913 pub extern "C" fn foo() {}
1914
1915 pub mod m1 {
1916 pub fn bar() {}
1917 }
1918 pub mod m2 {
1919 pub fn bar() {}
1920 }
1921 "#;
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08001922 run_compiler_for_testing(test_src, |tcx| find_def_id_by_name(tcx, "missing_name"));
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001923 }
1924
1925 #[test]
1926 #[should_panic(expected = "More than one item named `some_name`")]
1927 fn test_find_def_id_by_name_panic_when_multiple_items_with_matching_name() {
1928 let test_src = r#"
1929 pub mod m1 {
1930 pub fn some_name() {}
1931 }
1932 pub mod m2 {
1933 pub fn some_name() {}
1934 }
1935 "#;
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08001936 run_compiler_for_testing(test_src, |tcx| find_def_id_by_name(tcx, "some_name"));
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001937 }
1938
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001939 /// This test covers only a single example of a function that should get a
1940 /// C++ binding. The test focuses on verification that the output from
1941 /// `format_fn` gets propagated all the way to `GenerateBindings::new`.
1942 /// Additional coverage of how functions are formatted is provided
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001943 /// by `test_format_item_..._fn_...` tests (which work at the `format_fn`
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001944 /// level).
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001945 #[test]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001946 fn test_generated_bindings_fn_no_mangle_extern_c() {
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07001947 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07001948 #[no_mangle]
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001949 pub extern "C" fn public_function() {
1950 println!("foo");
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07001951 }
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07001952 "#;
1953 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001954 let bindings = bindings.unwrap();
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001955 assert_cc_matches!(
1956 bindings.h_body,
1957 quote! {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001958 extern "C" void public_function();
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001959 }
Lukasz Anforowiczd9ff4ab2022-09-23 08:11:18 -07001960 );
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001961
1962 // No Rust thunks should be generated in this test scenario.
Devin Jeanpierre81aec502023-04-04 15:33:39 -07001963 assert_rs_not_matches!(bindings.rs_body, quote! { public_function });
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001964 });
1965 }
1966
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001967 /// `test_generated_bindings_fn_export_name` covers a scenario where
1968 /// `MixedSnippet::cc` is present but `MixedSnippet::rs` is empty
1969 /// (because no Rust thunks are needed).
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001970 #[test]
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001971 fn test_generated_bindings_fn_export_name() {
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001972 let test_src = r#"
1973 #[export_name = "export_name"]
1974 pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y }
1975 "#;
1976 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001977 let bindings = bindings.unwrap();
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001978 assert_cc_matches!(
1979 bindings.h_body,
1980 quote! {
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001981 namespace rust_out {
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08001982 ...
1983 inline double public_function(double x, double y);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001984 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07001985 extern "C" double export_name(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001986 }
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001987 inline double public_function(double x, double y) {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001988 return __crubit_internal::export_name(x, y);
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001989 }
1990 }
1991 }
1992 );
1993 });
1994 }
1995
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001996 /// The `test_generated_bindings_struct` test covers only a single example
1997 /// of an ADT (struct/enum/union) that should get a C++ binding.
1998 /// Additional coverage of how items are formatted is provided by
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001999 /// `test_format_item_..._struct_...`, `test_format_item_..._enum_...`,
2000 /// and `test_format_item_..._union_...` tests.
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002001 ///
2002 /// We don't want to duplicate coverage already provided by
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002003 /// `test_format_item_struct_with_fields`, but we do want to verify that
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002004 /// * `format_crate` will actually find and process the struct
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002005 /// (`test_format_item_...` doesn't cover this aspect - it uses a
2006 /// test-only `find_def_id_by_name` instead)
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002007 /// * The actual shape of the bindings still looks okay at this level.
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002008 #[test]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002009 fn test_generated_bindings_struct() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002010 let test_src = r#"
2011 pub struct Point {
2012 pub x: i32,
2013 pub y: i32,
2014 }
2015 "#;
2016 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002017 let bindings = bindings.unwrap();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002018 assert_cc_matches!(
2019 bindings.h_body,
2020 quote! {
2021 namespace rust_out {
Googler47fd9572023-01-20 09:57:32 -08002022 ...
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002023 struct alignas(4) Point final {
2024 // No point replicating test coverage of
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002025 // `test_format_item_struct_with_fields`.
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002026 ...
2027 };
2028 static_assert(sizeof(Point) == 8, ...);
2029 static_assert(alignof(Point) == 4, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07002030 ... // Other static_asserts are covered by
2031 // `test_format_item_struct_with_fields`
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002032 } // namespace rust_out
2033 }
2034 );
2035 assert_rs_matches!(
2036 bindings.rs_body,
2037 quote! {
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07002038 // No point replicating test coverage of
2039 // `test_format_item_struct_with_fields`.
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08002040 const _: () = assert!(::std::mem::size_of::<::rust_out::Point>() == 8);
2041 const _: () = assert!(::std::mem::align_of::<::rust_out::Point>() == 4);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07002042 const _: () = assert!( memoffset::offset_of!(::rust_out::Point, x) == 0);
2043 const _: () = assert!( memoffset::offset_of!(::rust_out::Point, y) == 4);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002044 }
2045 );
2046 });
2047 }
2048
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002049 /// The `test_generated_bindings_impl` test covers only a single example of
2050 /// a non-trait `impl`. Additional coverage of how items are formatted
2051 /// should be provided in the future by `test_format_item_...` tests.
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002052 ///
2053 /// We don't want to duplicate coverage already provided by
2054 /// `test_format_item_static_method`, but we do want to verify that
2055 /// * `format_crate` won't process the `impl` as a standalone HIR item
2056 /// * The actual shape of the bindings still looks okay at this level.
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002057 #[test]
2058 fn test_generated_bindings_impl() {
2059 let test_src = r#"
2060 pub struct SomeStruct(i32);
2061
2062 impl SomeStruct {
2063 pub fn public_static_method() -> i32 { 123 }
2064
2065 #[allow(dead_code)]
2066 fn private_static_method() -> i32 { 123 }
2067 }
2068 "#;
2069 test_generated_bindings(test_src, |bindings| {
2070 let bindings = bindings.unwrap();
2071 assert_cc_matches!(
2072 bindings.h_body,
2073 quote! {
2074 namespace rust_out {
2075 ...
2076 struct ... SomeStruct ... {
2077 // No point replicating test coverage of
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002078 // `test_format_item_static_method`.
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002079 ...
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002080 std::int32_t public_static_method();
2081 ...
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002082 };
2083 ...
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002084 std::int32_t SomeStruct::public_static_method() {
2085 ...
2086 }
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07002087 ...
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002088 } // namespace rust_out
2089 }
2090 );
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002091 assert_rs_matches!(
2092 bindings.rs_body,
2093 quote! {
2094 extern "C" fn ...() -> i32 {
2095 ::rust_out::SomeStruct::public_static_method()
2096 }
2097 }
2098 );
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002099 });
2100 }
2101
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002102 #[test]
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002103 fn test_generated_bindings_includes() {
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07002104 let test_src = r#"
2105 #[no_mangle]
2106 pub extern "C" fn public_function(i: i32, d: isize, u: u64) {
2107 dbg!(i);
2108 dbg!(d);
2109 dbg!(u);
2110 }
2111 "#;
2112 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002113 let bindings = bindings.unwrap();
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07002114 assert_cc_matches!(
2115 bindings.h_body,
2116 quote! {
2117 __HASH_TOKEN__ include <cstdint> ...
2118 namespace ... {
Googler47fd9572023-01-20 09:57:32 -08002119 ...
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07002120 extern "C" void public_function(
2121 std::int32_t i,
2122 std::intptr_t d,
2123 std::uint64_t u);
2124 }
2125 }
2126 );
2127 });
2128 }
2129
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002130 /// Tests that `toposort` is used to reorder item bindings.
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07002131 #[test]
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002132 fn test_generated_bindings_prereq_defs_field_deps_require_reordering() {
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002133 let test_src = r#"
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002134 // In the generated bindings `Outer` needs to come *after* `Inner`.
2135 pub struct Outer(Inner);
2136 pub struct Inner(bool);
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002137 "#;
2138 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002139 let bindings = bindings.unwrap();
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002140 assert_cc_matches!(
2141 bindings.h_body,
2142 quote! {
2143 namespace rust_out {
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002144 ...
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002145 struct alignas(1) Inner final {
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07002146 ... bool __field0; ...
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002147 };
2148 ...
2149 struct alignas(1) Outer final {
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07002150 ... ::rust_out::Inner __field0; ...
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002151 };
2152 ...
2153 } // namespace rust_out
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002154 }
2155 );
2156 });
2157 }
2158
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002159 /// Tests that a forward declaration is present when it is required to
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002160 /// preserve the original source order. In this test the
2161 /// `CcPrerequisites::fwd_decls` dependency comes from a pointer parameter.
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002162 #[test]
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002163 fn test_generated_bindings_prereq_fwd_decls_for_ptr_param() {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002164 let test_src = r#"
2165 // To preserve original API order we need to forward declare S.
2166 pub fn f(_: *const S) {}
2167 pub struct S(bool);
2168 "#;
2169 test_generated_bindings(test_src, |bindings| {
2170 let bindings = bindings.unwrap();
2171 assert_cc_matches!(
2172 bindings.h_body,
2173 quote! {
2174 namespace rust_out {
2175 ...
2176 // Verifing the presence of this forward declaration
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002177 // it the essence of this test. The order of the items
2178 // below also matters.
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002179 struct S;
2180 ...
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002181 inline void f(const ::rust_out::S* __param_0);
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002182 ...
2183 struct alignas(...) S final { ... }
2184 ...
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002185 inline void f(const ::rust_out::S* __param_0) { ... }
2186 ...
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002187 } // namespace rust_out
2188 }
2189 );
2190 });
2191 }
2192
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002193 /// Tests that a forward declaration is present when it is required to
2194 /// preserve the original source order. In this test the
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002195 /// `CcPrerequisites::fwd_decls` dependency comes from a
2196 /// function declaration that has a parameter that takes a struct by value.
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002197 #[test]
2198 fn test_generated_bindings_prereq_fwd_decls_for_cpp_fn_decl() {
2199 let test_src = r#"
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002200 #[no_mangle]
2201 pub extern "C" fn f(s: S) -> bool { s.0 }
2202
2203 #[repr(C)]
2204 pub struct S(bool);
2205 "#;
2206
2207 test_generated_bindings(test_src, |bindings| {
2208 let bindings = bindings.unwrap();
2209 assert_cc_matches!(
2210 bindings.h_body,
2211 quote! {
2212 namespace rust_out {
2213 ...
2214 // Verifing the presence of this forward declaration
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002215 // is the essence of this test. The order also matters:
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002216 // 1. The fwd decl of `S` should come first,
2217 // 2. Declaration of `f` and definition of `S` should come next
2218 // (in their original order - `f` first and then `S`).
2219 struct S;
2220 ...
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002221 // `CcPrerequisites` of `f` declaration below (the main api of `f`) should
2222 // include `S` as a `fwd_decls` edge, rather than as a `defs` edge.
2223 inline bool f(::rust_out::S s);
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002224 ...
2225 struct alignas(...) S final { ... }
2226 ...
2227 } // namespace rust_out
2228 }
2229 );
2230 });
2231 }
2232
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002233 /// This test verifies that a forward declaration for a given ADT is only
2234 /// emitted once (and not once for every API item that requires the
2235 /// forward declaration as a prerequisite).
2236 #[test]
2237 fn test_generated_bindings_prereq_fwd_decls_no_duplication() {
2238 let test_src = r#"
2239 // All three functions below require a forward declaration of S.
2240 pub fn f1(_: *const S) {}
2241 pub fn f2(_: *const S) {}
2242 pub fn f3(_: *const S) {}
2243
2244 pub struct S(bool);
2245
2246 // This function also includes S in its CcPrerequisites::fwd_decls
2247 // (although here it is not required, because the definition of S
2248 // is already available above).
2249 pub fn f4(_: *const S) {}
2250 "#;
2251 test_generated_bindings(test_src, |bindings| {
2252 let bindings = bindings.unwrap().h_body.to_string();
2253
2254 // Only a single forward declaration is expected.
2255 assert_eq!(1, bindings.matches("struct S ;").count(), "bindings = {bindings}");
2256 });
2257 }
2258
2259 /// This test verifies that forward declarations are emitted in a
2260 /// deterministic order. The particular order doesn't matter _that_
2261 /// much, but it definitely shouldn't change every time
2262 /// `cc_bindings_from_rs` is invoked again. The current order preserves
2263 /// the original source order of the Rust API items.
2264 #[test]
2265 fn test_generated_bindings_prereq_fwd_decls_deterministic_order() {
2266 let test_src = r#"
2267 // To try to mix things up, the bindings for the functions below
2268 // will *ask* for forward declarations in a different order:
2269 // * Different from the order in which the forward declarations
2270 // are expected to be *emitted* (the original source order).
2271 // * Different from alphabetical order.
2272 pub fn f1(_: *const b::S3) {}
2273 pub fn f2(_: *const a::S2) {}
2274 pub fn f3(_: *const a::S1) {}
2275
2276 pub mod a {
2277 pub struct S1(bool);
2278 pub struct S2(bool);
2279 }
2280
2281 pub mod b {
2282 pub struct S3(bool);
2283 }
2284 "#;
2285 test_generated_bindings(test_src, |bindings| {
2286 let bindings = bindings.unwrap();
2287 assert_cc_matches!(
2288 bindings.h_body,
2289 quote! {
2290 namespace rust_out {
2291 ...
2292 // Verifying that we get the same order in each test
2293 // run is the essence of this test.
2294 namespace a {
2295 struct S1;
2296 struct S2;
2297 }
2298 namespace b {
2299 struct S3;
2300 }
2301 ...
2302 inline void f1 ...
2303 inline void f2 ...
2304 inline void f3 ...
2305
2306 namespace a { ...
2307 struct alignas(...) S1 final { ... } ...
2308 struct alignas(...) S2 final { ... } ...
2309 } ...
2310 namespace b { ...
2311 struct alignas(...) S3 final { ... } ...
2312 } ...
2313 } // namespace rust_out
2314 }
2315 );
2316 });
2317 }
2318
2319 /// This test verifies that forward declarations are not emitted if they are
2320 /// not needed (e.g. if bindings the given `struct` or other ADT have
2321 /// already been defined earlier). In particular, we don't want to emit
2322 /// forward declarations for *all* `structs` (regardless if they are
2323 /// needed or not).
2324 #[test]
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08002325 fn test_generated_bindings_prereq_fwd_decls_not_needed_because_of_initial_order() {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002326 let test_src = r#"
2327 pub struct S(bool);
2328
2329 // S is already defined above - no need for forward declaration in C++.
2330 pub fn f(_s: *const S) {}
2331 "#;
2332 test_generated_bindings(test_src, |bindings| {
2333 let bindings = bindings.unwrap();
2334 assert_cc_not_matches!(bindings.h_body, quote! { struct S; });
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08002335 assert_cc_matches!(bindings.h_body, quote! { void f(const ::rust_out::S* _s); });
2336 });
2337 }
2338
2339 /// This test verifies that a method declaration doesn't ask for a forward
2340 /// declaration to the struct.
2341 #[test]
2342 fn test_generated_bindings_prereq_fwd_decls_not_needed_inside_struct_definition() {
2343 let test_src = r#"
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002344 #![allow(dead_code)]
2345
2346 pub struct S {
2347 // This shouldn't require a fwd decl of S.
2348 field: *const S,
2349 }
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08002350
2351 impl S {
2352 // This shouldn't require a fwd decl of S.
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002353 pub fn create() -> S { Self{ field: std::ptr::null() } }
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08002354 }
2355 "#;
2356 test_generated_bindings(test_src, |bindings| {
2357 let bindings = bindings.unwrap();
2358 assert_cc_not_matches!(bindings.h_body, quote! { struct S; });
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002359 assert_cc_matches!(
2360 bindings.h_body,
2361 quote! {
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07002362 static inline ::rust_out::S create(); ...
2363 const ::rust_out::S* field; ...
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002364 }
2365 );
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002366 });
2367 }
2368
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002369 #[test]
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002370 fn test_generated_bindings_module_basics() {
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002371 let test_src = r#"
2372 pub mod some_module {
2373 pub fn some_func() {}
2374 }
2375 "#;
2376 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002377 let bindings = bindings.unwrap();
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002378 assert_cc_matches!(
2379 bindings.h_body,
2380 quote! {
2381 namespace rust_out {
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002382 namespace some_module {
2383 ...
2384 inline void some_func() { ... }
2385 ...
2386 } // namespace some_module
2387 } // namespace rust_out
2388 }
2389 );
2390 assert_rs_matches!(
2391 bindings.rs_body,
2392 quote! {
2393 #[no_mangle]
2394 extern "C"
2395 fn ...() -> () {
2396 ::rust_out::some_module::some_func()
2397 }
2398 }
2399 );
2400 });
2401 }
2402
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002403 #[test]
2404 fn test_generated_bindings_module_name_is_cpp_reserved_keyword() {
2405 let test_src = r#"
2406 pub mod working_module {
2407 pub fn working_module_f1() {}
2408 pub fn working_module_f2() {}
2409 }
2410 pub mod reinterpret_cast {
2411 pub fn broken_module_f1() {}
2412 pub fn broken_module_f2() {}
2413 }
2414 "#;
2415 test_generated_bindings(test_src, |bindings| {
2416 let bindings = bindings.unwrap();
2417
2418 // Items in the broken module should be replaced with a comment explaining the
2419 // problem.
2420 let broken_module_msg = "Failed to format namespace name `reinterpret_cast`: \
2421 `reinterpret_cast` is a C++ reserved keyword \
2422 and can't be used as a C++ identifier";
2423 assert_cc_not_matches!(bindings.h_body, quote! { namespace reinterpret_cast });
2424 assert_cc_not_matches!(bindings.h_body, quote! { broken_module_f1 });
2425 assert_cc_not_matches!(bindings.h_body, quote! { broken_module_f2 });
2426
2427 // Items in the other module should still go through.
2428 assert_cc_matches!(
2429 bindings.h_body,
2430 quote! {
2431 namespace rust_out {
2432 namespace working_module {
2433 ...
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002434 inline void working_module_f1();
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002435 ...
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002436 inline void working_module_f2();
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002437 ...
2438 } // namespace some_module
2439
2440 __COMMENT__ #broken_module_msg
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002441 ...
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002442 } // namespace rust_out
2443 }
2444 );
2445 });
2446 }
2447
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002448 /// `test_generated_bindings_non_pub_items` verifies that non-public items
2449 /// are not present/propagated into the generated bindings.
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002450 #[test]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002451 fn test_generated_bindings_non_pub_items() {
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002452 let test_src = r#"
Lukasz Anforowicz88bde9b2022-10-14 14:48:07 -07002453 #![allow(dead_code)]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002454
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002455 extern "C" fn private_function() {
2456 println!("foo");
2457 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002458
2459 struct PrivateStruct {
2460 x: i32,
2461 y: i32,
2462 }
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002463
Lukasz Anforowicz14229762023-02-10 15:28:33 -08002464 pub struct PublicStruct(i32);
2465
2466 impl PublicStruct {
2467 fn private_method() {}
2468 }
2469
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002470 pub mod public_module {
2471 fn priv_func_in_pub_module() {}
2472 }
2473
2474 mod private_module {
2475 pub fn pub_func_in_priv_module() { priv_func_in_priv_module() }
2476 fn priv_func_in_priv_module() {}
2477 }
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002478 "#;
2479 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002480 let bindings = bindings.unwrap();
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002481 assert_cc_not_matches!(bindings.h_body, quote! { private_function });
Lukasz Anforowicze7a25002022-11-10 06:21:42 -08002482 assert_rs_not_matches!(bindings.rs_body, quote! { private_function });
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002483 assert_cc_not_matches!(bindings.h_body, quote! { PrivateStruct });
2484 assert_rs_not_matches!(bindings.rs_body, quote! { PrivateStruct });
Lukasz Anforowicz14229762023-02-10 15:28:33 -08002485 assert_cc_not_matches!(bindings.h_body, quote! { private_method });
2486 assert_rs_not_matches!(bindings.rs_body, quote! { private_method });
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002487 assert_cc_not_matches!(bindings.h_body, quote! { priv_func_in_priv_module });
2488 assert_rs_not_matches!(bindings.rs_body, quote! { priv_func_in_priv_module });
2489 assert_cc_not_matches!(bindings.h_body, quote! { priv_func_in_pub_module });
2490 assert_rs_not_matches!(bindings.rs_body, quote! { priv_func_in_pub_module });
Lukasz Anforowicz1382f392022-12-12 17:13:23 -08002491 assert_cc_not_matches!(bindings.h_body, quote! { private_module });
2492 assert_rs_not_matches!(bindings.rs_body, quote! { private_module });
2493 assert_cc_not_matches!(bindings.h_body, quote! { pub_func_in_priv_module });
2494 assert_rs_not_matches!(bindings.rs_body, quote! { pub_func_in_priv_module });
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002495 });
2496 }
2497
2498 #[test]
Lukasz Anforowicz4aa1ff92022-10-10 11:22:22 -07002499 fn test_generated_bindings_top_level_items() {
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002500 let test_src = "pub fn public_function() {}";
2501 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002502 let bindings = bindings.unwrap();
Devin Jeanpierre81aec502023-04-04 15:33:39 -07002503 let expected_comment_txt = "Automatically @generated C++ bindings for the following Rust crate:\n\
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002504 rust_out";
2505 assert_cc_matches!(
2506 bindings.h_body,
2507 quote! {
2508 __COMMENT__ #expected_comment_txt
Lukasz Anforowicz4aa1ff92022-10-10 11:22:22 -07002509 ...
Lukasz Anforowicz31b29cd2022-10-10 11:33:41 -07002510 __HASH_TOKEN__ pragma once
2511 ...
Lukasz Anforowicz4aa1ff92022-10-10 11:22:22 -07002512 namespace rust_out {
2513 ...
2514 }
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002515 }
2516 );
Lukasz Anforowicze7a25002022-11-10 06:21:42 -08002517 assert_cc_matches!(
2518 bindings.rs_body,
2519 quote! {
2520 __COMMENT__ #expected_comment_txt
2521 }
2522 );
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002523 })
2524 }
2525
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002526 /// The `test_generated_bindings_unsupported_item` test verifies how `Err`
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002527 /// from `format_item` is formatted as a C++ comment (in `format_crate`
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002528 /// and `format_unsupported_def`):
2529 /// - This test covers only a single example of an unsupported item.
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002530 /// Additional coverage is provided by `test_format_item_unsupported_...`
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002531 /// tests.
2532 /// - This test somewhat arbitrarily chooses an example of an unsupported
2533 /// item, trying to pick one that 1) will never be supported (b/254104998
2534 /// has some extra notes about APIs named after reserved C++ keywords) and
2535 /// 2) tests that the full error chain is included in the message.
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002536 #[test]
2537 fn test_generated_bindings_unsupported_item() {
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002538 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002539 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002540 pub extern "C" fn reinterpret_cast() {}
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002541 "#;
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002542 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002543 let bindings = bindings.unwrap();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002544 let expected_comment_txt = "Error generating bindings for `reinterpret_cast` \
Googler785e6b42023-01-23 12:11:36 -08002545 defined at <crubit_unittests.rs>;l=3: \
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002546 Error formatting function name: \
2547 `reinterpret_cast` is a C++ reserved keyword \
2548 and can't be used as a C++ identifier";
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002549 assert_cc_matches!(
2550 bindings.h_body,
2551 quote! {
2552 __COMMENT__ #expected_comment_txt
2553 }
2554 );
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07002555 })
2556 }
2557
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07002558 #[test]
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -08002559 fn test_generated_bindings_reimports() {
2560 let test_src = r#"
2561 #![allow(dead_code)]
2562 #![allow(unused_imports)]
2563 mod private_submodule1 {
2564 pub fn subfunction1() {}
2565 pub fn subfunction2() {}
2566 pub fn subfunction3() {}
2567 }
2568 mod private_submodule2 {
2569 pub fn subfunction8() {}
2570 pub fn subfunction9() {}
2571 }
2572
2573 // Public re-import.
2574 pub use private_submodule1::subfunction1;
2575
2576 // Private re-import.
2577 use private_submodule1::subfunction2;
2578
2579 // Re-import that renames.
2580 pub use private_submodule1::subfunction3 as public_function3;
2581
2582 // Re-import of multiple items via glob.
2583 pub use private_submodule2::*;
2584 "#;
2585 test_generated_bindings(test_src, |bindings| {
2586 let bindings = bindings.unwrap();
2587
2588 let failures = vec![(1, 15), (3, 21), (4, 24)];
2589 for (use_number, line_number) in failures.into_iter() {
2590 let expected_comment_txt = format!(
2591 "Error generating bindings for `{{use#{use_number}}}` defined at \
2592 <crubit_unittests.rs>;l={line_number}: \
2593 Unsupported rustc_hir::hir::ItemKind: `use` import"
2594 );
2595 assert_cc_matches!(
2596 bindings.h_body,
2597 quote! {
2598 __COMMENT__ #expected_comment_txt
2599 }
2600 );
2601 }
2602 });
2603 }
2604
2605 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002606 fn test_format_item_fn_extern_c_no_mangle_no_params_no_return_type() {
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07002607 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002608 #[no_mangle]
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07002609 pub extern "C" fn public_function() {}
2610 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002611 test_format_item(test_src, "public_function", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002612 let result = result.unwrap().unwrap();
2613 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002614 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002615 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002616 main_api.tokens,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002617 quote! {
2618 extern "C" void public_function();
2619 }
2620 );
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002621
2622 // Sufficient to just re-declare the Rust API in C++.
2623 // (i.e. there is no need to have a C++-side definition of `public_function`).
2624 assert!(result.cc_details.tokens.is_empty());
2625
2626 // There is no need to have a separate thunk for an `extern "C"` function.
2627 assert!(result.rs_details.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002628 });
2629 }
2630
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002631 /// The `test_format_item_fn_explicit_unit_return_type` test below is very
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002632 /// similar to the
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002633 /// `test_format_item_fn_extern_c_no_mangle_no_params_no_return_type` above,
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002634 /// except that the return type is explicitly spelled out. There is no
2635 /// difference in `ty::FnSig` so our code behaves exactly the same, but the
2636 /// test has been planned based on earlier, hir-focused approach and having
2637 /// this extra test coverage shouldn't hurt. (`hir::FnSig`
2638 /// and `hir::FnRetTy` _would_ see a difference between the two tests, even
2639 /// though there is no different in the current `bindings.rs` code).
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002640 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002641 fn test_format_item_fn_explicit_unit_return_type() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002642 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002643 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002644 pub extern "C" fn explicit_unit_return_type() -> () {}
2645 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002646 test_format_item(test_src, "explicit_unit_return_type", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002647 let result = result.unwrap().unwrap();
2648 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002649 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002650 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002651 main_api.tokens,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002652 quote! {
2653 extern "C" void explicit_unit_return_type();
2654 }
2655 );
2656 });
2657 }
2658
2659 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002660 fn test_format_item_fn_never_return_type() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002661 let test_src = r#"
2662 #[no_mangle]
2663 pub extern "C" fn never_returning_function() -> ! {
2664 panic!("This function panics and therefore never returns");
2665 }
2666 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002667 test_format_item(test_src, "never_returning_function", |result| {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002668 // TODO(b/254507801): The function should be annotated with the `[[noreturn]]`
2669 // attribute.
2670 // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more
2671 // details).
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002672 let result = result.unwrap().unwrap();
2673 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002674 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002675 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002676 main_api.tokens,
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002677 quote! {
2678 extern "C" void never_returning_function();
2679 }
2680 );
2681 })
2682 }
2683
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002684 /// `test_format_item_fn_mangling` checks that bindings can be generated for
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002685 /// `extern "C"` functions that do *not* have `#[no_mangle]` attribute. The
2686 /// test elides away the mangled name in the `assert_cc_matches` checks
2687 /// below, but end-to-end test coverage should eventually be provided by
2688 /// `test/functions` (see b/262904507).
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002689 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002690 fn test_format_item_fn_mangling() {
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002691 let test_src = r#"
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002692 pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y }
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002693 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002694 test_format_item(test_src, "public_function", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002695 let result = result.unwrap().unwrap();
2696 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002697 assert!(main_api.prereqs.is_empty());
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002698 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002699 main_api.tokens,
2700 quote! {
2701 inline double public_function(double x, double y);
2702 }
2703 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002704 assert!(result.rs_details.is_empty());
2705 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002706 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002707 result.cc_details.tokens,
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002708 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002709 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07002710 extern "C" double ...(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002711 }
Googler47fd9572023-01-20 09:57:32 -08002712 ...
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002713 inline double public_function(double x, double y) {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002714 return __crubit_internal::...(x, y);
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002715 }
2716 }
2717 );
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002718 });
2719 }
2720
2721 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002722 fn test_format_item_fn_export_name() {
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002723 let test_src = r#"
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002724 #[export_name = "export_name"]
2725 pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y }
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002726 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002727 test_format_item(test_src, "public_function", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002728 let result = result.unwrap().unwrap();
2729 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002730 assert!(main_api.prereqs.is_empty());
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002731 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002732 main_api.tokens,
2733 quote! {
2734 inline double public_function(double x, double y);
2735 }
2736 );
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002737
2738 // There is no need to have a separate thunk for an `extern "C"` function.
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002739 assert!(result.rs_details.is_empty());
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002740
2741 // We generate a C++-side definition of `public_function` so that we
2742 // can call a differently-named (but same-signature) `export_name` function.
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002743 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002744 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002745 result.cc_details.tokens,
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002746 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002747 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07002748 extern "C" double export_name(double, double);
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002749 }
Googler47fd9572023-01-20 09:57:32 -08002750 ...
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002751 inline double public_function(double x, double y) {
2752 return __crubit_internal::export_name(x, y);
2753 }
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002754 }
2755 );
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002756 });
2757 }
2758
2759 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002760 fn test_format_item_unsupported_fn_unsafe() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002761 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002762 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002763 pub unsafe extern "C" fn foo() {}
2764 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002765 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002766 let err = result.unwrap_err();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002767 assert_eq!(
2768 err,
2769 "Bindings for `unsafe` functions \
2770 are not fully designed yet (b/254095482)"
2771 );
2772 });
2773 }
2774
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002775 /// `test_format_item_fn_const` tests how bindings for an `const fn` are
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002776 /// generated.
2777 ///
Googler47fd9572023-01-20 09:57:32 -08002778 /// Right now the `const` qualifier is ignored, but one can imagine that in
2779 /// the (very) long-term future such functions (including their bodies)
2780 /// could be translated into C++ `consteval` functions.
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002781 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002782 fn test_format_item_fn_const() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002783 let test_src = r#"
2784 pub const fn foo(i: i32) -> i32 { i * 42 }
2785 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002786 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002787 // TODO(b/254095787): Update test expectations below once `const fn` from Rust
2788 // is translated into a `consteval` C++ function.
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002789 let result = result.unwrap().unwrap();
2790 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002791 assert!(!main_api.prereqs.is_empty());
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002792 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002793 main_api.tokens,
2794 quote! {
2795 inline std::int32_t foo(std::int32_t i);
2796 }
2797 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002798 assert!(!result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002799 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002800 result.cc_details.tokens,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002801 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002802 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07002803 extern "C" std::int32_t ...( std::int32_t);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002804 }
Googler47fd9572023-01-20 09:57:32 -08002805 ...
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002806 inline std::int32_t foo(std::int32_t i) {
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08002807 return __crubit_internal::...(i);
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002808 }
2809 }
2810 );
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002811 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002812 result.rs_details,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002813 quote! {
2814 #[no_mangle]
2815 extern "C"
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08002816 fn ...(i: i32) -> i32 {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08002817 ::rust_out::foo(i)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002818 }
2819 }
2820 );
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002821 });
2822 }
2823
2824 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002825 fn test_format_item_fn_with_c_unwind_abi() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002826 // See also https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html
2827 let test_src = r#"
2828 #![feature(c_unwind)]
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002829
2830 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002831 pub extern "C-unwind" fn may_throw() {}
2832 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002833 test_format_item(test_src, "may_throw", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002834 let result = result.unwrap().unwrap();
2835 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002836 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002837 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002838 main_api.tokens,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002839 quote! {
2840 extern "C" void may_throw();
2841 }
2842 );
2843 });
2844 }
2845
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002846 /// This test mainly verifies that `format_item` correctly propagates
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002847 /// `CcPrerequisites` of parameter types and return type.
2848 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002849 fn test_format_item_fn_cc_prerequisites_if_cpp_definition_needed() {
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002850 let test_src = r#"
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002851 pub fn foo(_i: i32) -> S { panic!("foo") }
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002852 pub struct S(i32);
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002853 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002854 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002855 let result = result.unwrap().unwrap();
2856 let main_api = &result.main_api;
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002857
2858 // Minimal coverage, just to double-check that the test setup works.
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002859 //
2860 // Note that this is a definition, and therefore `S` should be defined
2861 // earlier (not just forward declared).
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002862 assert_cc_matches!(main_api.tokens, quote! { S foo(std::int32_t _i);});
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002863 assert_cc_matches!(result.cc_details.tokens, quote! { S foo(std::int32_t _i) { ... }});
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002864
2865 // Main checks: `CcPrerequisites::includes`.
2866 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002867 format_cc_includes(&main_api.prereqs.includes),
2868 quote! { include <cstdint> }
2869 );
2870 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002871 format_cc_includes(&result.cc_details.prereqs.includes),
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002872 quote! { include <cstdint> }
2873 );
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002874
2875 // Main checks: `CcPrerequisites::defs` and `CcPrerequisites::fwd_decls`.
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002876 //
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002877 // Verifying the actual def_id is tricky, because `test_format_item` doesn't
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002878 // expose `tcx` to the verification function (and therefore calling
2879 // `find_def_id_by_name` is not easily possible).
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002880 //
2881 // Note that `main_api` and `impl_details` have different expectations.
2882 assert_eq!(0, main_api.prereqs.defs.len());
2883 assert_eq!(1, main_api.prereqs.fwd_decls.len());
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002884 assert_eq!(1, result.cc_details.prereqs.defs.len());
2885 assert_eq!(0, result.cc_details.prereqs.fwd_decls.len());
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002886 });
2887 }
2888
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002889 /// This test verifies that `format_item` uses `CcPrerequisites::fwd_decls`
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002890 /// rather than `CcPrerequisites::defs` for function declarations in the
2891 /// `main_api`.
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002892 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002893 fn test_format_item_fn_cc_prerequisites_if_only_cpp_declaration_needed() {
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002894 let test_src = r#"
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002895 #[no_mangle]
2896 pub extern "C" fn foo(s: S) -> bool { s.0 }
2897
2898 #[repr(C)]
2899 pub struct S(bool);
2900 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002901 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002902 let result = result.unwrap().unwrap();
2903 let main_api = &result.main_api;
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002904
2905 // Minimal coverage, just to double-check that the test setup works.
2906 //
2907 // Note that this is only a function *declaration* (not a function definition -
2908 // there is no function body), and therefore `S` just needs to be
2909 // forward-declared earlier.
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002910 assert_cc_matches!(main_api.tokens, quote! { inline bool foo(::rust_out::S s); });
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002911
2912 // Main checks: `CcPrerequisites::defs` and `CcPrerequisites::fwd_decls`.
2913 //
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002914 // Verifying the actual def_id is tricky, because `test_format_item` doesn't
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002915 // expose `tcx` to the verification function (and therefore calling
2916 // `find_def_id_by_name` is not easily possible).
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002917 assert_eq!(0, main_api.prereqs.defs.len());
2918 assert_eq!(1, main_api.prereqs.fwd_decls.len());
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002919 });
2920 }
2921
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002922 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002923 fn test_format_item_fn_with_type_aliased_return_type() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002924 // Type aliases disappear at the `rustc_middle::ty::Ty` level and therefore in
2925 // the short-term the generated bindings also ignore type aliases.
2926 //
2927 // TODO(b/254096006): Consider preserving `type` aliases when generating
2928 // bindings.
2929 let test_src = r#"
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07002930 type MyTypeAlias = f64;
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002931
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002932 #[no_mangle]
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07002933 pub extern "C" fn type_aliased_return() -> MyTypeAlias { 42.0 }
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002934 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002935 test_format_item(test_src, "type_aliased_return", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002936 let result = result.unwrap().unwrap();
2937 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002938 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002939 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002940 main_api.tokens,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002941 quote! {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07002942 extern "C" double type_aliased_return();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002943 }
2944 );
2945 });
2946 }
2947
2948 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002949 fn test_format_item_fn_with_doc_comment_with_unmangled_name() {
Googler624580b2022-12-01 01:23:42 -08002950 let test_src = r#"
2951 /// Outer line doc.
2952 /** Outer block doc that spans lines.
2953 */
2954 #[doc = "Doc comment via doc attribute."]
2955 #[no_mangle]
2956 pub extern "C" fn fn_with_doc_comment_with_unmangled_name() {}
2957 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002958 test_format_item(test_src, "fn_with_doc_comment_with_unmangled_name", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002959 let result = result.unwrap().unwrap();
2960 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002961 assert!(main_api.prereqs.is_empty());
Googler624580b2022-12-01 01:23:42 -08002962 let doc_comments = [
2963 " Outer line doc.",
Googler34f3d572022-12-02 00:53:37 -08002964 "",
Googler624580b2022-12-01 01:23:42 -08002965 " Outer block doc that spans lines.",
2966 " ",
Googler34f3d572022-12-02 00:53:37 -08002967 "",
Googler624580b2022-12-01 01:23:42 -08002968 "Doc comment via doc attribute.",
Googler47fd9572023-01-20 09:57:32 -08002969 "",
2970 "Generated from: <crubit_unittests.rs>;l=7",
Googler624580b2022-12-01 01:23:42 -08002971 ]
2972 .join("\n");
2973 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002974 main_api.tokens,
Googler624580b2022-12-01 01:23:42 -08002975 quote! {
2976 __COMMENT__ #doc_comments
2977 extern "C" void fn_with_doc_comment_with_unmangled_name();
2978 }
2979 );
2980 });
2981 }
2982
2983 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002984 fn test_format_item_fn_with_inner_doc_comment_with_unmangled_name() {
Googler624580b2022-12-01 01:23:42 -08002985 let test_src = r#"
2986 /// Outer doc comment.
2987 #[no_mangle]
2988 pub extern "C" fn fn_with_inner_doc_comment_with_unmangled_name() {
2989 //! Inner doc comment.
2990 }
2991 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002992 test_format_item(test_src, "fn_with_inner_doc_comment_with_unmangled_name", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002993 let result = result.unwrap().unwrap();
2994 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002995 assert!(main_api.prereqs.is_empty());
Googler47fd9572023-01-20 09:57:32 -08002996 let doc_comments = [
2997 " Outer doc comment.",
2998 " Inner doc comment.",
2999 "Generated from: <crubit_unittests.rs>;l=4",
3000 ]
3001 .join("\n\n");
Googler624580b2022-12-01 01:23:42 -08003002 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003003 main_api.tokens,
Googler624580b2022-12-01 01:23:42 -08003004 quote! {
3005 __COMMENT__ #doc_comments
3006 extern "C" void fn_with_inner_doc_comment_with_unmangled_name();
3007 }
3008 );
3009 });
3010 }
3011
3012 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003013 fn test_format_item_fn_with_doc_comment_with_mangled_name() {
Googler624580b2022-12-01 01:23:42 -08003014 let test_src = r#"
3015 /// Doc comment of a function with mangled name.
3016 pub extern "C" fn fn_with_doc_comment_with_mangled_name() {}
3017 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003018 test_format_item(test_src, "fn_with_doc_comment_with_mangled_name", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003019 let result = result.unwrap().unwrap();
3020 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003021 assert!(main_api.prereqs.is_empty());
Googler47fd9572023-01-20 09:57:32 -08003022 let comment = " Doc comment of a function with mangled name.\n\n\
3023 Generated from: <crubit_unittests.rs>;l=3";
Googler624580b2022-12-01 01:23:42 -08003024 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003025 main_api.tokens,
Googler624580b2022-12-01 01:23:42 -08003026 quote! {
Googler624580b2022-12-01 01:23:42 -08003027 __COMMENT__ #comment
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003028 inline void fn_with_doc_comment_with_mangled_name();
Googler624580b2022-12-01 01:23:42 -08003029 }
3030 );
3031 });
3032 }
3033
3034 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003035 fn test_format_item_unsupported_fn_name_is_reserved_cpp_keyword() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003036 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07003037 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003038 pub extern "C" fn reinterpret_cast() -> () {}
3039 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003040 test_format_item(test_src, "reinterpret_cast", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003041 let err = result.unwrap_err();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003042 assert_eq!(
3043 err,
3044 "Error formatting function name: \
3045 `reinterpret_cast` is a C++ reserved keyword \
3046 and can't be used as a C++ identifier"
3047 );
3048 });
3049 }
3050
3051 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003052 fn test_format_item_unsupported_fn_ret_type() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003053 let test_src = r#"
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08003054 pub fn foo() -> (i32, i32) { (123, 456) }
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003055 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003056 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003057 let err = result.unwrap_err();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003058 assert_eq!(
3059 err,
3060 "Error formatting function return type: \
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08003061 Tuples are not supported yet: (i32, i32) (b/254099023)"
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003062 );
3063 });
3064 }
3065
3066 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003067 fn test_format_item_unsupported_fn_with_late_bound_lifetimes() {
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003068 // TODO(b/258235219): Expect success after adding support for references.
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003069 let test_src = r#"
3070 pub fn foo(arg: &i32) -> &i32 { arg }
3071
3072 // Lifetime inference translates the above into:
3073 // pub fn foo<'a>(arg: &'a i32) -> &'a i32 { ... }
3074 // leaving 'a lifetime late-bound (it is bound with a lifetime
3075 // taken from each of the callsites). In other words, we can't
3076 // just call `no_bound_vars` on this `FnSig`'s `Binder`.
3077 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003078 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003079 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003080 assert_eq!(err, "Generic functions are not supported yet (b/259749023)");
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003081 });
3082 }
3083
3084 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003085 fn test_format_item_unsupported_generic_fn() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003086 let test_src = r#"
3087 use std::default::Default;
3088 use std::fmt::Display;
3089 pub fn generic_function<T: Default + Display>() {
3090 println!("{}", T::default());
3091 }
3092 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003093 test_format_item(test_src, "generic_function", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003094 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003095 assert_eq!(err, "Generic functions are not supported yet (b/259749023)");
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07003096 });
3097 }
3098
3099 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003100 fn test_format_item_unsupported_generic_struct() {
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003101 let test_src = r#"
3102 pub struct Point<T> {
3103 pub x: T,
3104 pub y: T,
3105 }
3106 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003107 test_format_item(test_src, "Point", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003108 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003109 assert_eq!(err, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003110 });
3111 }
3112
3113 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003114 fn test_format_item_unsupported_generic_enum() {
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003115 let test_src = r#"
3116 pub enum Point<T> {
3117 Cartesian{x: T, y: T},
3118 Polar{angle: T, dist: T},
3119 }
3120 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003121 test_format_item(test_src, "Point", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003122 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003123 assert_eq!(err, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003124 });
3125 }
3126
3127 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003128 fn test_format_item_unsupported_generic_union() {
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003129 let test_src = r#"
3130 pub union SomeUnion<T> {
3131 pub x: std::mem::ManuallyDrop<T>,
3132 pub y: i32,
3133 }
3134 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003135 test_format_item(test_src, "SomeUnion", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003136 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003137 assert_eq!(err, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003138 });
3139 }
3140
3141 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003142 fn test_format_item_unsupported_fn_async() {
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07003143 let test_src = r#"
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003144 pub async fn async_function() {}
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07003145 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003146 test_format_item(test_src, "async_function", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003147 let err = result.unwrap_err();
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003148 assert_eq!(
3149 err,
3150 "Error formatting function return type: \
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003151 The following Rust type is not supported yet: \
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003152 impl std::future::Future<Output = ()>"
3153 );
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003154 });
3155 }
3156
3157 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003158 fn test_format_item_fn_rust_abi() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003159 let test_src = r#"
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003160 pub fn add(x: f64, y: f64) -> f64 { x * y }
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003161 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003162 test_format_item(test_src, "add", |result| {
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003163 // TODO(b/261074843): Re-add thunk name verification once we are using stable
3164 // name mangling (which may be coming in Q1 2023). (This might mean
3165 // reverting cl/492333432 + manual review and tweaks.)
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003166 let result = result.unwrap().unwrap();
3167 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003168 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003169 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003170 main_api.tokens,
3171 quote! {
3172 inline double add(double x, double y);
3173 }
3174 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003175 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003176 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003177 result.cc_details.tokens,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003178 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003179 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003180 extern "C" double ...(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003181 }
Googler47fd9572023-01-20 09:57:32 -08003182 ...
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003183 inline double add(double x, double y) {
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08003184 return __crubit_internal::...(x, y);
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003185 }
3186 }
3187 );
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003188 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003189 result.rs_details,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003190 quote! {
3191 #[no_mangle]
3192 extern "C"
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08003193 fn ...(x: f64, y: f64) -> f64 {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003194 ::rust_out::add(x, y)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003195 }
3196 }
3197 );
3198 });
3199 }
3200
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003201 #[test]
3202 fn test_format_item_fn_rust_abi_with_param_taking_struct_by_value() {
3203 let test_src = r#"
3204 pub struct S(i32);
3205 pub fn into_i32(s: S) -> i32 { s.0 }
3206 "#;
3207 test_format_item(test_src, "into_i32", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003208 let result = result.unwrap().unwrap();
3209 let main_api = &result.main_api;
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003210 assert_cc_matches!(
3211 main_api.tokens,
3212 quote! {
3213 inline std::int32_t into_i32(::rust_out::S s);
3214 }
3215 );
3216 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003217 result.cc_details.tokens,
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003218 quote! {
3219 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003220 extern "C" std::int32_t ...(::rust_out::S*);
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003221 }
3222 ...
3223 inline std::int32_t into_i32(::rust_out::S s) {
Lukasz Anforowiczd082f352023-03-09 17:46:11 -08003224 return __crubit_internal::...(&s);
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003225 }
3226 }
3227 );
3228 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003229 result.rs_details,
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003230 quote! {
3231 #[no_mangle]
3232 extern "C"
Lukasz Anforowiczd082f352023-03-09 17:46:11 -08003233 fn ...(s: &mut ::core::mem::MaybeUninit<::rust_out::S>) -> i32 {
3234 ::rust_out::into_i32(unsafe { s.assume_init_read() })
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003235 }
3236 }
3237 );
3238 });
3239 }
3240
3241 #[test]
3242 fn test_format_item_fn_rust_abi_returning_struct_by_value() {
3243 let test_src = r#"
3244 pub struct S(i32);
3245 pub fn create(i: i32) -> S { S(i) }
3246 "#;
3247 test_format_item(test_src, "create", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003248 let result = result.unwrap().unwrap();
3249 let main_api = &result.main_api;
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003250 assert_cc_matches!(
3251 main_api.tokens,
3252 quote! {
3253 inline ::rust_out::S create(std::int32_t i);
3254 }
3255 );
3256 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003257 result.cc_details.tokens,
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003258 quote! {
3259 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003260 extern "C" void ...(std::int32_t, ::rust_out::S* __ret_ptr);
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003261 }
3262 ...
3263 inline ::rust_out::S create(std::int32_t i) {
Lukasz Anforowicza3b7db02023-03-09 17:34:05 -08003264 crubit::ReturnValueSlot<::rust_out::S> __ret_slot;
3265 __crubit_internal::...(i, __ret_slot.Get());
3266 return std::move(__ret_slot).AssumeInitAndTakeValue();
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003267 }
3268 }
3269 );
3270 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003271 result.rs_details,
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003272 quote! {
3273 #[no_mangle]
3274 extern "C"
Lukasz Anforowicza3b7db02023-03-09 17:34:05 -08003275 fn ...(
3276 i: i32,
3277 __ret_slot: &mut ::core::mem::MaybeUninit<::rust_out::S>
3278 ) -> () {
3279 __ret_slot.write(::rust_out::create(i));
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003280 }
3281 }
3282 );
3283 });
3284 }
3285
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003286 /// `test_format_item_fn_rust_abi` tests a function call that is not a
3287 /// C-ABI, and is not the default Rust ABI. It can't use `"stdcall"`,
3288 /// because it is not supported on the targets where Crubit's tests run.
3289 /// So, it ended up using `"vectorcall"`.
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003290 ///
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003291 /// This test almost entirely replicates `test_format_item_fn_rust_abi`,
Googler47fd9572023-01-20 09:57:32 -08003292 /// except for the `extern "vectorcall"` part in the `test_src` test
3293 /// input.
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003294 ///
Googler47fd9572023-01-20 09:57:32 -08003295 /// This test verifies the current behavior that gives reasonable and
3296 /// functional FFI bindings. OTOH, in the future we may decide to avoid
3297 /// having the extra thunk for cases where the given non-C-ABI function
3298 /// call convention is supported by both C++ and Rust
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003299 /// (see also `format_cc_call_conv_as_clang_attribute` in
3300 /// `rs_bindings_from_cc/src_code_gen.rs`)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003301 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003302 fn test_format_item_fn_vectorcall_abi() {
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003303 let test_src = r#"
3304 #![feature(abi_vectorcall)]
3305 pub extern "vectorcall" fn add(x: f64, y: f64) -> f64 { x * y }
3306 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003307 test_format_item(test_src, "add", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003308 let result = result.unwrap().unwrap();
3309 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003310 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003311 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003312 main_api.tokens,
3313 quote! {
3314 inline double add(double x, double y);
3315 }
3316 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003317 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003318 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003319 result.cc_details.tokens,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003320 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003321 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003322 extern "C" double ...(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003323 }
Googler47fd9572023-01-20 09:57:32 -08003324 ...
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003325 inline double add(double x, double y) {
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08003326 return __crubit_internal::...(x, y);
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003327 }
3328 }
3329 );
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003330 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003331 result.rs_details,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003332 quote! {
3333 #[no_mangle]
3334 extern "C"
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08003335 fn ...(x: f64, y: f64) -> f64 {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003336 ::rust_out::add(x, y)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003337 }
3338 }
3339 );
3340 });
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003341 }
3342
3343 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003344 fn test_format_item_unsupported_fn_variadic() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003345 let test_src = r#"
3346 #![feature(c_variadic)]
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07003347
3348 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003349 pub unsafe extern "C" fn variadic_function(_fmt: *const u8, ...) {}
3350 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003351 test_format_item(test_src, "variadic_function", |result| {
Lukasz Anforowicz13794df2022-10-21 07:56:34 -07003352 // TODO(b/254097223): Add support for variadic functions.
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003353 let err = result.unwrap_err();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003354 assert_eq!(err, "C variadic functions are not supported (b/254097223)");
3355 });
3356 }
3357
3358 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003359 fn test_format_item_fn_params() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003360 let test_src = r#"
3361 #[allow(unused_variables)]
3362 #[no_mangle]
3363 pub extern "C" fn foo(b: bool, f: f64) {}
3364 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003365 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003366 let result = result.unwrap().unwrap();
3367 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003368 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003369 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003370 main_api.tokens,
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003371 quote! {
Googler47fd9572023-01-20 09:57:32 -08003372 ...
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003373 extern "C" void foo(bool b, double f);
3374 }
3375 );
3376 });
3377 }
3378
3379 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003380 fn test_format_item_fn_param_name_reserved_keyword() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003381 let test_src = r#"
3382 #[allow(unused_variables)]
3383 #[no_mangle]
3384 pub extern "C" fn some_function(reinterpret_cast: f64) {}
3385 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003386 test_format_item(test_src, "some_function", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003387 let result = result.unwrap().unwrap();
3388 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003389 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003390 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003391 main_api.tokens,
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003392 quote! {
Googler47fd9572023-01-20 09:57:32 -08003393 ...
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003394 extern "C" void some_function(double __param_0);
3395 }
3396 );
3397 });
3398 }
3399
3400 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003401 fn test_format_item_fn_with_multiple_anonymous_parameter_names() {
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003402 let test_src = r#"
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003403 pub fn foo(_: f64, _: f64) {}
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003404 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003405 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003406 let result = result.unwrap().unwrap();
3407 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003408 assert!(main_api.prereqs.is_empty());
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003409 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003410 main_api.tokens,
3411 quote! {
3412 inline void foo(double __param_0, double __param_1);
3413 }
3414 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003415 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003416 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003417 result.cc_details.tokens,
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003418 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003419 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003420 extern "C" void ...(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003421 }
Googler47fd9572023-01-20 09:57:32 -08003422 ...
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003423 inline void foo(double __param_0, double __param_1) {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003424 return __crubit_internal::...(__param_0, __param_1);
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003425 }
3426 }
3427 );
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003428 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003429 result.rs_details,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003430 quote! {
3431 #[no_mangle]
3432 extern "C" fn ...(__param_0: f64, __param_1: f64) -> () {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003433 ::rust_out::foo(__param_0, __param_1)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003434 }
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003435 }
3436 );
3437 });
3438 }
3439
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003440 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003441 fn test_format_item_fn_with_destructuring_parameter_name() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003442 let test_src = r#"
3443 pub struct S {
3444 pub f1: i32,
3445 pub f2: i32,
3446 }
3447
3448 // This test mostly focuses on the weird parameter "name" below.
3449 // See also
3450 // https://doc.rust-lang.org/reference/items/functions.html#function-parameters
3451 // which points out that function parameters are just irrefutable patterns.
3452 pub fn func(S{f1, f2}: S) -> i32 { f1 + f2 }
3453 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003454 test_format_item(test_src, "func", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003455 let result = result.unwrap().unwrap();
3456 let main_api = &result.main_api;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003457 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003458 main_api.tokens,
3459 quote! {
3460 inline std::int32_t func(::rust_out::S __param_0);
3461 }
3462 );
3463 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003464 result.cc_details.tokens,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003465 quote! {
3466 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003467 extern "C" std::int32_t ...(::rust_out::S*);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003468 }
Googler47fd9572023-01-20 09:57:32 -08003469 ...
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003470 inline std::int32_t func(::rust_out::S __param_0) {
Lukasz Anforowiczd082f352023-03-09 17:46:11 -08003471 return __crubit_internal::...(&__param_0);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003472 }
3473 }
3474 );
3475 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003476 result.rs_details,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003477 quote! {
3478 #[no_mangle]
Lukasz Anforowiczd082f352023-03-09 17:46:11 -08003479 extern "C" fn ...(
3480 __param_0: &mut ::core::mem::MaybeUninit<::rust_out::S>
3481 ) -> i32 {
3482 ::rust_out::func(unsafe {__param_0.assume_init_read() })
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003483 }
3484 }
3485 );
3486 });
3487 }
3488
3489 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003490 fn test_format_item_unsupported_fn_param_type() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003491 let test_src = r#"
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08003492 pub fn foo(_param: (i32, i32)) {}
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003493 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003494 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003495 let err = result.unwrap_err();
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003496 assert_eq!(
3497 err,
3498 "Error handling parameter #0: \
3499 Tuples are not supported yet: (i32, i32) (b/254099023)"
3500 );
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003501 });
3502 }
3503
3504 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003505 fn test_format_item_unsupported_fn_param_type_unit() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003506 let test_src = r#"
3507 #[no_mangle]
3508 pub fn fn_with_params(_param: ()) {}
3509 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003510 test_format_item(test_src, "fn_with_params", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003511 let err = result.unwrap_err();
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003512 assert_eq!(
3513 err,
3514 "Error handling parameter #0: \
3515 `()` / `void` is only supported as a return type (b/254507801)"
3516 );
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003517 });
3518 }
3519
3520 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003521 fn test_format_item_unsupported_fn_param_type_never() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003522 let test_src = r#"
3523 #![feature(never_type)]
3524
3525 #[no_mangle]
3526 pub extern "C" fn fn_with_params(_param: !) {}
3527 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003528 test_format_item(test_src, "fn_with_params", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003529 let err = result.unwrap_err();
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003530 assert_eq!(
3531 err,
Lukasz Anforowicza691cf52023-03-08 12:24:33 -08003532 "Error handling parameter #0: \
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003533 The never type `!` is only supported as a return type (b/254507801)"
3534 );
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003535 });
3536 }
3537
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003538 /// This is a test for a regular struct - a struct with named fields.
3539 /// https://doc.rust-lang.org/reference/items/structs.html refers to this kind of struct as
3540 /// `StructStruct` or "nominal struct type".
3541 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003542 fn test_format_item_struct_with_fields() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003543 let test_src = r#"
3544 pub struct SomeStruct {
3545 pub x: i32,
3546 pub y: i32,
3547 }
3548
3549 const _: () = assert!(std::mem::size_of::<SomeStruct>() == 8);
3550 const _: () = assert!(std::mem::align_of::<SomeStruct>() == 4);
3551 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003552 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003553 let result = result.unwrap().unwrap();
3554 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003555 assert!(!main_api.prereqs.is_empty());
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003556 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003557 main_api.tokens,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003558 quote! {
Googler47fd9572023-01-20 09:57:32 -08003559 ...
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003560 struct alignas(4) SomeStruct final {
3561 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07003562 __COMMENT__ "`SomeStruct` doesn't implement the `Default` trait"
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003563 SomeStruct() = delete;
3564
3565 // In this test there is no `Copy` implementation / derive.
3566 SomeStruct(const SomeStruct&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003567
3568 // All Rust types are trivially-movable.
3569 SomeStruct(SomeStruct&&) = default;
Lukasz Anforowicz554ed652023-01-12 15:41:58 -08003570
3571 // Assignment operators are disabled for now.
3572 SomeStruct& operator=(const SomeStruct&) = delete;
3573 SomeStruct& operator=(SomeStruct&&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003574
3575 // In this test there is no custom `Drop`, so C++ can also
3576 // just use the `default` destructor.
3577 ~SomeStruct() = default;
3578 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003579 ... std::int32_t x;
3580 ... std::int32_t y;
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003581 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003582 };
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003583 }
3584 );
3585 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003586 result.cc_details.tokens,
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003587 quote! {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003588 static_assert(sizeof(SomeStruct) == 8, ...);
3589 static_assert(alignof(SomeStruct) == 4, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003590 inline void SomeStruct::__crubit_field_offset_assertions() {
3591 static_assert(0 == offsetof(SomeStruct, x));
3592 static_assert(4 == offsetof(SomeStruct, y));
3593 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003594 }
3595 );
3596 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003597 result.rs_details,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003598 quote! {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003599 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8);
3600 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003601 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, x) == 0);
3602 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, y) == 4);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003603 }
3604 );
3605 });
3606 }
3607
3608 /// This is a test for `TupleStruct` or "tuple struct" - for more details
3609 /// please refer to https://doc.rust-lang.org/reference/items/structs.html
3610 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003611 fn test_format_item_struct_with_tuple() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003612 let test_src = r#"
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003613 pub struct TupleStruct(pub i32, pub i32);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003614 const _: () = assert!(std::mem::size_of::<TupleStruct>() == 8);
3615 const _: () = assert!(std::mem::align_of::<TupleStruct>() == 4);
3616 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003617 test_format_item(test_src, "TupleStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003618 let result = result.unwrap().unwrap();
3619 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003620 assert!(!main_api.prereqs.is_empty());
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003621 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003622 main_api.tokens,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003623 quote! {
Googler47fd9572023-01-20 09:57:32 -08003624 ...
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003625 struct alignas(4) TupleStruct final {
3626 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07003627 __COMMENT__ "`TupleStruct` doesn't implement the `Default` trait"
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003628 TupleStruct() = delete;
3629
3630 // In this test there is no `Copy` implementation / derive.
3631 TupleStruct(const TupleStruct&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003632
3633 // All Rust types are trivially-movable.
3634 TupleStruct(TupleStruct&&) = default;
Lukasz Anforowicz554ed652023-01-12 15:41:58 -08003635
3636 // Assignment operators are disabled for now.
3637 TupleStruct& operator=(const TupleStruct&) = delete;
3638 TupleStruct& operator=(TupleStruct&&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003639
3640 // In this test there is no custom `Drop`, so C++ can also
3641 // just use the `default` destructor.
3642 ~TupleStruct() = default;
3643 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003644 ... std::int32_t __field0;
3645 ... std::int32_t __field1;
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003646 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003647 };
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003648 }
3649 );
3650 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003651 result.cc_details.tokens,
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003652 quote! {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003653 static_assert(sizeof(TupleStruct) == 8, ...);
3654 static_assert(alignof(TupleStruct) == 4, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003655 inline void TupleStruct::__crubit_field_offset_assertions() {
3656 static_assert(0 == offsetof(TupleStruct, __field0));
3657 static_assert(4 == offsetof(TupleStruct, __field1));
3658 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003659 }
3660 );
3661 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003662 result.rs_details,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003663 quote! {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003664 const _: () = assert!(::std::mem::size_of::<::rust_out::TupleStruct>() == 8);
3665 const _: () = assert!(::std::mem::align_of::<::rust_out::TupleStruct>() == 4);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003666 const _: () = assert!( memoffset::offset_of!(::rust_out::TupleStruct, 0) == 0);
3667 const _: () = assert!( memoffset::offset_of!(::rust_out::TupleStruct, 1) == 4);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003668 }
3669 );
3670 });
3671 }
3672
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003673 /// This test the scenario where Rust lays out field in a different order
3674 /// than the source order.
3675 #[test]
3676 fn test_format_item_struct_with_reordered_field_offsets() {
3677 let test_src = r#"
3678 pub struct SomeStruct {
3679 pub field1: i16,
3680 pub field2: i32,
3681 pub field3: i16,
3682 }
3683
3684 const _: () = assert!(std::mem::size_of::<SomeStruct>() == 8);
3685 const _: () = assert!(std::mem::align_of::<SomeStruct>() == 4);
3686 "#;
3687 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003688 let result = result.unwrap().unwrap();
3689 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003690 assert!(!main_api.prereqs.is_empty());
3691 assert_cc_matches!(
3692 main_api.tokens,
3693 quote! {
3694 ...
3695 struct alignas(4) SomeStruct final {
3696 ...
3697 private:
3698 // The particular order below is not guaranteed,
3699 // so we may need to adjust this test assertion
3700 // (if Rust changes how it lays out the fields).
3701 ... std::int32_t field2;
3702 ... std::int16_t field1;
3703 ... std::int16_t field3;
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003704 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003705 };
3706 }
3707 );
3708 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003709 result.cc_details.tokens,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003710 quote! {
3711 static_assert(sizeof(SomeStruct) == 8, ...);
3712 static_assert(alignof(SomeStruct) == 4, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003713 inline void SomeStruct::__crubit_field_offset_assertions() {
3714 static_assert(0 == offsetof(SomeStruct, field2));
3715 static_assert(4 == offsetof(SomeStruct, field1));
3716 static_assert(6 == offsetof(SomeStruct, field3));
3717 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003718 }
3719 );
3720 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003721 result.rs_details,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003722 quote! {
3723 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8);
3724 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003725 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field2)
3726 == 0);
3727 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field1)
3728 == 4);
3729 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field3)
3730 == 6);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003731 }
3732 );
3733 });
3734 }
3735
3736 #[test]
3737 fn test_format_item_struct_with_packed_layout() {
3738 let test_src = r#"
3739 #[repr(packed(1))]
3740 pub struct SomeStruct {
3741 pub field1: u16,
3742 pub field2: u32,
3743 }
3744 const _: () = assert!(::std::mem::size_of::<SomeStruct>() == 6);
3745 const _: () = assert!(::std::mem::align_of::<SomeStruct>() == 1);
3746 "#;
3747 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003748 let result = result.unwrap().unwrap();
3749 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003750 assert!(!main_api.prereqs.is_empty());
3751 assert_cc_matches!(
3752 main_api.tokens,
3753 quote! {
3754 ...
3755 struct alignas(1) __attribute__((packed)) SomeStruct final {
3756 ...
3757 std::uint16_t field1;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003758 std::uint32_t field2;
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003759 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003760 };
3761 }
3762 );
3763 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003764 result.cc_details.tokens,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003765 quote! {
3766 static_assert(sizeof(SomeStruct) == 6, ...);
3767 static_assert(alignof(SomeStruct) == 1, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003768 inline void SomeStruct::__crubit_field_offset_assertions() {
3769 static_assert(0 == offsetof(SomeStruct, field1));
3770 static_assert(2 == offsetof(SomeStruct, field2));
3771 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003772 }
3773 );
3774 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003775 result.rs_details,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003776 quote! {
3777 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 6);
3778 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 1);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003779 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field1)
3780 == 0);
3781 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field2)
3782 == 2);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003783 }
3784 );
3785 });
3786 }
3787
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003788 #[test]
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07003789 fn test_format_item_struct_with_explicit_padding_in_generated_code() {
3790 let test_src = r#"
3791 pub struct SomeStruct {
3792 pub f1: u8,
3793 pub f2: u32,
3794 }
3795 const _: () = assert!(::std::mem::size_of::<SomeStruct>() == 8);
3796 const _: () = assert!(::std::mem::align_of::<SomeStruct>() == 4);
3797 "#;
3798 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003799 let result = result.unwrap().unwrap();
3800 let main_api = &result.main_api;
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07003801 assert!(!main_api.prereqs.is_empty());
3802 assert_cc_matches!(
3803 main_api.tokens,
3804 quote! {
3805 ...
3806 struct alignas(4) SomeStruct final {
3807 ...
3808 std::uint32_t f2;
3809 std::uint8_t f1;
3810 unsigned char __padding0[3];
3811 inline static void __crubit_field_offset_assertions();
3812 };
3813 }
3814 );
3815 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003816 result.cc_details.tokens,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07003817 quote! {
3818 static_assert(sizeof(SomeStruct) == 8, ...);
3819 static_assert(alignof(SomeStruct) == 4, ...);
3820 inline void SomeStruct::__crubit_field_offset_assertions() {
3821 static_assert(0 == offsetof(SomeStruct, f2));
3822 static_assert(4 == offsetof(SomeStruct, f1));
3823 }
3824 }
3825 );
3826 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003827 result.rs_details,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07003828 quote! {
3829 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8);
3830 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
3831 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, f2) == 0);
3832 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, f1) == 4);
3833 }
3834 );
3835 });
3836 }
3837
3838 #[test]
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003839 fn test_format_item_static_method() {
3840 let test_src = r#"
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003841 /// No-op `f32` placeholder is used, because ZSTs are not supported
3842 /// (b/258259459).
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003843 pub struct Math(f32);
3844
3845 impl Math {
3846 pub fn add_i32(x: f32, y: f32) -> f32 {
3847 x + y
3848 }
3849 }
3850 "#;
3851 test_format_item(test_src, "Math", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003852 let result = result.unwrap().unwrap();
3853 let main_api = &result.main_api;
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003854 assert!(main_api.prereqs.is_empty());
3855 assert_cc_matches!(
3856 main_api.tokens,
3857 quote! {
3858 ...
3859 struct ... Math final {
3860 ...
3861 public:
3862 ...
3863 static inline float add_i32(float x, float y);
3864 ...
3865 };
3866 }
3867 );
3868 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003869 result.cc_details.tokens,
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003870 quote! {
3871 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003872 extern "C" float ... (float, float);
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003873 }
3874 inline float Math::add_i32(float x, float y) {
3875 return __crubit_internal::...(x, y);
3876 }
3877 }
3878 );
3879 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003880 result.rs_details,
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003881 quote! {
3882 #[no_mangle]
3883 extern "C" fn ...(x: f32, y: f32) -> f32 {
3884 ::rust_out::Math::add_i32(x, y)
3885 }
3886 }
3887 );
3888 });
3889 }
3890
3891 #[test]
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003892 fn test_format_item_static_method_with_generic_type_parameters() {
3893 let test_src = r#"
3894 /// No-op `f32` placeholder is used, because ZSTs are not supported
3895 /// (b/258259459).
3896 pub struct SomeStruct(f32);
3897
3898 impl SomeStruct {
3899 // To make this testcase distinct / non-overlapping wrt
3900 // test_format_item_static_method_with_generic_lifetime_parameters
3901 // `t` is taken by value below.
3902 pub fn generic_method<T: Clone>(t: T) -> T {
3903 t.clone()
3904 }
3905 }
3906 "#;
3907 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003908 let result = result.unwrap().unwrap();
3909 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003910 assert!(main_api.prereqs.is_empty());
3911 let unsupported_msg = "Error generating bindings for `SomeStruct::generic_method` \
3912 defined at <crubit_unittests.rs>;l=10: \
3913 Generic functions are not supported yet (b/259749023)";
3914 assert_cc_matches!(
3915 main_api.tokens,
3916 quote! {
3917 ...
3918 struct ... SomeStruct final {
3919 ...
3920 __COMMENT__ #unsupported_msg
3921 ...
3922 };
3923 ...
3924 }
3925 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003926 assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::generic_method },);
3927 assert_rs_not_matches!(result.rs_details, quote! { generic_method },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003928 });
3929 }
3930
3931 #[test]
3932 fn test_format_item_static_method_with_generic_lifetime_parameters() {
3933 let test_src = r#"
3934 /// No-op `f32` placeholder is used, because ZSTs are not supported
3935 /// (b/258259459).
3936 pub struct SomeStruct(f32);
3937
3938 impl SomeStruct {
3939 pub fn fn_taking_reference<'a>(x: &'a i32) -> i32 { *x }
3940 }
3941 "#;
3942 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003943 let result = result.unwrap().unwrap();
3944 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003945 assert!(main_api.prereqs.is_empty());
3946 let unsupported_msg = "Error generating bindings for `SomeStruct::fn_taking_reference` \
3947 defined at <crubit_unittests.rs>;l=7: \
3948 Generic functions are not supported yet (b/259749023)";
3949 assert_cc_matches!(
3950 main_api.tokens,
3951 quote! {
3952 ...
3953 struct ... SomeStruct final {
3954 ...
3955 __COMMENT__ #unsupported_msg
3956 ...
3957 };
3958 ...
3959 }
3960 );
3961 assert_cc_not_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003962 result.cc_details.tokens,
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003963 quote! { SomeStruct::fn_taking_reference },
3964 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003965 assert_rs_not_matches!(result.rs_details, quote! { fn_taking_reference },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003966 });
3967 }
3968
3969 #[test]
3970 fn test_format_item_method_taking_self_by_value() {
3971 let test_src = r#"
3972 pub struct SomeStruct(f32);
3973
3974 impl SomeStruct {
3975 pub fn into_f32(self) -> f32 {
3976 self.0
3977 }
3978 }
3979 "#;
3980 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003981 let result = result.unwrap().unwrap();
3982 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003983 assert!(main_api.prereqs.is_empty());
3984 let unsupported_msg = "Error generating bindings for `SomeStruct::into_f32` \
3985 defined at <crubit_unittests.rs>;l=5: \
3986 `self` parameter is not supported yet";
3987 assert_cc_matches!(
3988 main_api.tokens,
3989 quote! {
3990 ...
3991 struct ... SomeStruct final {
3992 ...
3993 __COMMENT__ #unsupported_msg
3994 ...
3995 };
3996 ...
3997 }
3998 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003999 assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::into_f32 },);
4000 assert_rs_not_matches!(result.rs_details, quote! { into_f32 },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004001 });
4002 }
4003
4004 #[test]
4005 fn test_format_item_method_taking_self_by_const_ref() {
4006 let test_src = r#"
4007 pub struct SomeStruct(f32);
4008
4009 impl SomeStruct {
4010 pub fn get_f32(&self) -> f32 {
4011 self.0
4012 }
4013 }
4014 "#;
4015 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004016 let result = result.unwrap().unwrap();
4017 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004018 assert!(main_api.prereqs.is_empty());
4019 let unsupported_msg = "Error generating bindings for `SomeStruct::get_f32` \
4020 defined at <crubit_unittests.rs>;l=5: \
4021 Generic functions are not supported yet (b/259749023)";
4022 assert_cc_matches!(
4023 main_api.tokens,
4024 quote! {
4025 ...
4026 struct ... SomeStruct final {
4027 ...
4028 __COMMENT__ #unsupported_msg
4029 ...
4030 };
4031 ...
4032 }
4033 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004034 assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::get_f32 },);
4035 assert_rs_not_matches!(result.rs_details, quote! { get_f32 },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004036 });
4037 }
4038
4039 #[test]
4040 fn test_format_item_method_taking_self_by_mutable_ref() {
4041 let test_src = r#"
4042 pub struct SomeStruct(f32);
4043
4044 impl SomeStruct {
4045 pub fn set_f32(&mut self, new_value: f32) {
4046 self.0 = new_value;
4047 }
4048 }
4049 "#;
4050 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004051 let result = result.unwrap().unwrap();
4052 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004053 assert!(main_api.prereqs.is_empty());
4054 let unsupported_msg = "Error generating bindings for `SomeStruct::set_f32` \
4055 defined at <crubit_unittests.rs>;l=5: \
4056 Generic functions are not supported yet (b/259749023)";
4057 assert_cc_matches!(
4058 main_api.tokens,
4059 quote! {
4060 ...
4061 struct ... SomeStruct final {
4062 ...
4063 __COMMENT__ #unsupported_msg
4064 ...
4065 };
4066 ...
4067 }
4068 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004069 assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::set_f32 },);
4070 assert_rs_not_matches!(result.rs_details, quote! { set_f32 },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004071 });
4072 }
4073
4074 #[test]
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07004075 fn test_format_item_struct_with_default_constructor() {
4076 let test_src = r#"
4077 #[derive(Default)]
4078 pub struct Point(i32, i32);
4079 "#;
4080 test_format_item(test_src, "Point", |result| {
4081 let result = result.unwrap().unwrap();
4082 let main_api = &result.main_api;
4083 assert_cc_matches!(
4084 main_api.tokens,
4085 quote! {
4086 ...
4087 struct ... Point final {
4088 ...
4089 public:
4090 __COMMENT__ "Default::default"
4091 inline Point();
4092 ...
4093 };
4094 }
4095 );
4096 assert_cc_matches!(
4097 result.cc_details.tokens,
4098 quote! {
4099 namespace __crubit_internal {
4100 extern "C" void ...(::rust_out::Point* __ret_ptr);
4101 }
4102 Point::Point() {
4103 ...(this);
4104 }
4105 }
4106 );
4107 assert_rs_matches!(
4108 result.rs_details,
4109 quote! {
4110 #[no_mangle]
4111 extern "C" fn ...(
4112 __ret_slot: &mut ::core::mem::MaybeUninit<::rust_out::Point>
4113 ) -> () {
4114 __ret_slot.write(<::rust_out::Point as ::core::default::Default>::default());
4115 }
4116 }
4117 );
4118 });
4119 }
4120
4121 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004122 fn test_format_item_unsupported_struct_with_name_that_is_reserved_keyword() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004123 let test_src = r#"
4124 #[allow(non_camel_case_types)]
4125 pub struct reinterpret_cast {
4126 pub x: i32,
4127 pub y: i32,
4128 }
4129 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004130 test_format_item(test_src, "reinterpret_cast", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004131 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004132 assert_eq!(
4133 err,
4134 "Error formatting item name: \
4135 `reinterpret_cast` is a C++ reserved keyword \
4136 and can't be used as a C++ identifier"
4137 );
4138 });
4139 }
4140
4141 #[test]
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004142 fn test_format_item_struct_with_unsupported_field_type() {
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004143 let test_src = r#"
4144 pub struct SomeStruct {
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004145 pub successful_field: i32,
4146 pub unsupported_field: Option<[i32; 3]>,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004147 }
4148 "#;
4149 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004150 let result = result.unwrap().unwrap();
4151 let main_api = &result.main_api;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004152 let broken_field_msg = "Field type has been replaced with a blob of bytes: \
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004153 Generic types are not supported yet (b/259749095)";
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004154 assert_cc_matches!(
4155 main_api.tokens,
4156 quote! {
4157 ...
4158 struct ... SomeStruct final {
4159 ...
4160 private:
4161 __COMMENT__ #broken_field_msg
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004162 unsigned char unsupported_field[16];
4163 std::int32_t successful_field;
4164 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004165 };
4166 ...
4167 }
4168 );
4169 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004170 result.cc_details.tokens,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004171 quote! {
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004172 static_assert(sizeof(SomeStruct) == 20, ...);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004173 static_assert(alignof(SomeStruct) == 4, ...);
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004174 inline void SomeStruct::__crubit_field_offset_assertions() {
4175 static_assert(0 == offsetof(SomeStruct, unsupported_field));
4176 static_assert(16 == offsetof(SomeStruct, successful_field));
4177 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004178 }
4179 );
4180 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004181 result.rs_details,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004182 quote! {
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004183 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 20);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004184 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004185 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4186 unsupported_field) == 0);
4187 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4188 successful_field) == 16);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004189 }
4190 );
4191 });
4192 }
4193
4194 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004195 fn test_format_item_unsupported_struct_with_custom_drop_impl() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004196 let test_src = r#"
4197 pub struct StructWithCustomDropImpl {
4198 pub x: i32,
4199 pub y: i32,
4200 }
4201
4202 impl Drop for StructWithCustomDropImpl {
4203 fn drop(&mut self) {}
4204 }
4205 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004206 test_format_item(test_src, "StructWithCustomDropImpl", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004207 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004208 assert_eq!(err, "`Drop` trait and \"drop glue\" are not supported yet (b/258251148)");
4209 });
4210 }
4211
4212 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004213 fn test_format_item_unsupported_struct_with_custom_drop_glue() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004214 let test_src = r#"
4215 #![allow(dead_code)]
4216
4217 // `i32` is present to avoid hitting the ZST checks related to (b/258259459)
4218 struct StructWithCustomDropImpl(i32);
4219
4220 impl Drop for StructWithCustomDropImpl {
4221 fn drop(&mut self) {
4222 println!("dropping!");
4223 }
4224 }
4225
4226 pub struct StructRequiringCustomDropGlue {
4227 field: StructWithCustomDropImpl,
4228 }
4229 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004230 test_format_item(test_src, "StructRequiringCustomDropGlue", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004231 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004232 assert_eq!(err, "`Drop` trait and \"drop glue\" are not supported yet (b/258251148)");
4233 });
4234 }
4235
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004236 /// This test covers how ZSTs (zero-sized-types) are handled.
4237 /// https://doc.rust-lang.org/reference/items/structs.html refers to this kind of struct as a
4238 /// "unit-like struct".
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004239 #[test]
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004240 fn test_format_item_unsupported_struct_zero_sized_type_with_no_fields() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004241 let test_src = r#"
4242 pub struct ZeroSizedType1;
4243 pub struct ZeroSizedType2();
4244 pub struct ZeroSizedType3{}
4245 "#;
4246 for name in ["ZeroSizedType1", "ZeroSizedType2", "ZeroSizedType3"] {
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004247 test_format_item(test_src, name, |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004248 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004249 assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)");
4250 });
4251 }
4252 }
4253
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004254 #[test]
4255 fn test_format_item_unsupported_struct_with_only_zero_sized_type_fields() {
4256 let test_src = r#"
4257 pub struct ZeroSizedType;
4258 pub struct SomeStruct {
4259 pub zst1: ZeroSizedType,
4260 pub zst2: ZeroSizedType,
4261 }
4262 "#;
4263 test_format_item(test_src, "SomeStruct", |result| {
4264 let err = result.unwrap_err();
4265 assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)",);
4266 });
4267 }
4268
Devin Jeanpierre9e15d0b2023-04-06 13:18:22 -07004269 #[test]
4270 fn test_format_item_unsupported_struct_with_some_zero_sized_type_fields() {
4271 let test_src = r#"
4272 pub struct ZeroSizedType;
4273 pub struct SomeStruct {
4274 pub zst1: ZeroSizedType,
4275 pub zst2: ZeroSizedType,
4276 pub successful_field: i32,
4277 }
4278 "#;
4279 test_format_item(test_src, "SomeStruct", |result| {
4280 let result = result.unwrap().unwrap();
4281 let main_api = &result.main_api;
4282 let broken_field_msg = "Field type has been replaced with a blob of bytes: \
4283 Failed to generate bindings for the definition of `ZeroSizedType`: \
4284 Zero-sized types (ZSTs) are not supported (b/258259459)";
4285 assert_cc_matches!(
4286 main_api.tokens,
4287 quote! {
4288 ...
4289 struct ... SomeStruct final {
4290 ...
4291 private:
4292 __COMMENT__ #broken_field_msg
4293 [[no_unique_address]] struct{} zst1;
4294 __COMMENT__ #broken_field_msg
4295 [[no_unique_address]] struct{} zst2;
4296 std::int32_t successful_field;
4297 inline static void __crubit_field_offset_assertions();
4298 };
4299 ...
4300 }
4301 );
4302 assert_cc_matches!(
4303 result.cc_details.tokens,
4304 quote! {
4305 static_assert(sizeof(SomeStruct) == 4, ...);
4306 static_assert(alignof(SomeStruct) == 4, ...);
4307 inline void SomeStruct::__crubit_field_offset_assertions() {
4308 static_assert(0 == offsetof(SomeStruct, zst1));
4309 static_assert(0 == offsetof(SomeStruct, zst2));
4310 static_assert(0 == offsetof(SomeStruct, successful_field));
4311 }
4312 }
4313 );
4314 assert_rs_matches!(
4315 result.rs_details,
4316 quote! {
4317 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 4);
4318 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
4319 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4320 zst1) == 0);
4321 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4322 zst2) == 0);
4323 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4324 successful_field) == 0);
4325 }
4326 );
4327 });
4328 }
4329
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004330 /// This is a test for an enum that only has `EnumItemDiscriminant` items
4331 /// (and doesn't have `EnumItemTuple` or `EnumItemStruct` items). See
4332 /// also https://doc.rust-lang.org/reference/items/enumerations.html
4333 #[test]
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004334 fn test_format_item_enum_with_only_discriminant_items() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004335 let test_src = r#"
4336 pub enum SomeEnum {
4337 Red,
4338 Green = 123,
4339 Blue,
4340 }
4341
4342 const _: () = assert!(std::mem::size_of::<SomeEnum>() == 1);
4343 const _: () = assert!(std::mem::align_of::<SomeEnum>() == 1);
4344 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004345 test_format_item(test_src, "SomeEnum", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004346 let result = result.unwrap().unwrap();
4347 let main_api = &result.main_api;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004348 let no_fields_msg = "Field type has been replaced with a blob of bytes: \
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004349 No support for bindings of individual fields of \
4350 `union` (b/272801632) or `enum`";
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004351 assert!(main_api.prereqs.is_empty());
4352 assert_cc_matches!(
4353 main_api.tokens,
4354 quote! {
4355 ...
4356 struct alignas(1) SomeEnum final {
4357 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07004358 __COMMENT__ "`SomeEnum` doesn't implement the `Default` trait"
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004359 SomeEnum() = delete;
4360
4361 // In this test there is no `Copy` implementation / derive.
4362 SomeEnum(const SomeEnum&) = delete;
4363
4364 // All Rust types are trivially-movable.
4365 SomeEnum(SomeEnum&&) = default;
4366
4367 // Assignment operators are disabled for now.
4368 SomeEnum& operator=(const SomeEnum&) = delete;
4369 SomeEnum& operator=(SomeEnum&&) = delete;
4370
4371 // In this test there is no custom `Drop`, so C++ can also
4372 // just use the `default` destructor.
4373 ~SomeEnum() = default;
4374 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004375 __COMMENT__ #no_fields_msg
4376 unsigned char __opaque_blob_of_bytes[1];
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004377 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004378 };
4379 }
4380 );
4381 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004382 result.cc_details.tokens,
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004383 quote! {
4384 static_assert(sizeof(SomeEnum) == 1, ...);
4385 static_assert(alignof(SomeEnum) == 1, ...);
4386 }
4387 );
4388 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004389 result.rs_details,
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004390 quote! {
4391 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeEnum>() == 1);
4392 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeEnum>() == 1);
4393 }
4394 );
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004395 });
4396 }
4397
4398 /// This is a test for an enum that has `EnumItemTuple` and `EnumItemStruct`
4399 /// items. See also https://doc.rust-lang.org/reference/items/enumerations.html
4400 #[test]
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004401 fn test_format_item_enum_with_tuple_and_struct_items() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004402 let test_src = r#"
4403 pub enum Point {
4404 Cartesian(f32, f32),
4405 Polar{ dist: f32, angle: f32 },
4406 }
4407
4408 const _: () = assert!(std::mem::size_of::<Point>() == 12);
4409 const _: () = assert!(std::mem::align_of::<Point>() == 4);
4410 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004411 test_format_item(test_src, "Point", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004412 let result = result.unwrap().unwrap();
4413 let main_api = &result.main_api;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004414 let no_fields_msg = "Field type has been replaced with a blob of bytes: \
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004415 No support for bindings of individual fields of \
4416 `union` (b/272801632) or `enum`";
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004417 assert!(main_api.prereqs.is_empty());
4418 assert_cc_matches!(
4419 main_api.tokens,
4420 quote! {
4421 ...
4422 struct alignas(4) Point final {
4423 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07004424 __COMMENT__ "`Point` doesn't implement the `Default` trait"
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004425 Point() = delete;
4426
4427 // In this test there is no `Copy` implementation / derive.
4428 Point(const Point&) = delete;
4429
4430 // All Rust types are trivially-movable.
4431 Point(Point&&) = default;
4432
4433 // Assignment operators are disabled for now.
4434 Point& operator=(const Point&) = delete;
4435 Point& operator=(Point&&) = delete;
4436
4437 // In this test there is no custom `Drop`, so C++ can also
4438 // just use the `default` destructor.
4439 ~Point() = default;
4440 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004441 __COMMENT__ #no_fields_msg
4442 unsigned char __opaque_blob_of_bytes[12];
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004443 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004444 };
4445 }
4446 );
4447 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004448 result.cc_details.tokens,
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004449 quote! {
4450 static_assert(sizeof(Point) == 12, ...);
4451 static_assert(alignof(Point) == 4, ...);
4452 }
4453 );
4454 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004455 result.rs_details,
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004456 quote! {
4457 const _: () = assert!(::std::mem::size_of::<::rust_out::Point>() == 12);
4458 const _: () = assert!(::std::mem::align_of::<::rust_out::Point>() == 4);
4459 }
4460 );
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004461 });
4462 }
4463
4464 /// This test covers how zero-variant enums are handled. See also
4465 /// https://doc.rust-lang.org/reference/items/enumerations.html#zero-variant-enums
4466 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004467 fn test_format_item_unsupported_enum_zero_variants() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004468 let test_src = r#"
4469 pub enum ZeroVariantEnum {}
4470 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004471 test_format_item(test_src, "ZeroVariantEnum", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004472 let err = result.unwrap_err();
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004473 assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)");
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004474 });
4475 }
4476
4477 /// This is a test for a `union`. See also
4478 /// https://doc.rust-lang.org/reference/items/unions.html
4479 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004480 fn test_format_item_union() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004481 let test_src = r#"
4482 pub union SomeUnion {
4483 pub i: i32,
4484 pub f: f64,
4485 }
4486
4487 const _: () = assert!(std::mem::size_of::<SomeUnion>() == 8);
4488 const _: () = assert!(std::mem::align_of::<SomeUnion>() == 8);
4489 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004490 test_format_item(test_src, "SomeUnion", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004491 let result = result.unwrap().unwrap();
4492 let main_api = &result.main_api;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004493 let no_fields_msg = "Field type has been replaced with a blob of bytes: \
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004494 No support for bindings of individual fields of \
4495 `union` (b/272801632) or `enum`";
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004496 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004497 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004498 main_api.tokens,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004499 quote! {
Googler47fd9572023-01-20 09:57:32 -08004500 ...
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08004501 union alignas(8) SomeUnion final {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004502 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07004503 __COMMENT__ "`SomeUnion` doesn't implement the `Default` trait"
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004504 SomeUnion() = delete;
4505
4506 // In this test there is no `Copy` implementation / derive.
4507 SomeUnion(const SomeUnion&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004508
4509 // All Rust types are trivially-movable.
4510 SomeUnion(SomeUnion&&) = default;
Lukasz Anforowicz554ed652023-01-12 15:41:58 -08004511
4512 // Assignment operators are disabled for now.
4513 SomeUnion& operator=(const SomeUnion&) = delete;
4514 SomeUnion& operator=(SomeUnion&&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004515
4516 // In this test there is no custom `Drop`, so C++ can also
4517 // just use the `default` destructor.
4518 ~SomeUnion() = default;
4519 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004520 __COMMENT__ #no_fields_msg
4521 unsigned char __opaque_blob_of_bytes[8];
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004522 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004523 };
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004524 }
4525 );
4526 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004527 result.cc_details.tokens,
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004528 quote! {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004529 static_assert(sizeof(SomeUnion) == 8, ...);
4530 static_assert(alignof(SomeUnion) == 8, ...);
4531 }
4532 );
4533 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004534 result.rs_details,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004535 quote! {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08004536 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeUnion>() == 8);
4537 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeUnion>() == 8);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004538 }
4539 );
4540 });
4541 }
4542
Lukasz Anforowicze4333062022-10-17 14:47:53 -07004543 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004544 fn test_format_item_doc_comments_union() {
Googler04329a12022-12-02 00:56:33 -08004545 let test_src = r#"
4546 /// Doc for some union.
4547 pub union SomeUnionWithDocs {
4548 /// Doc for a field in a union.
4549 pub i: i32,
4550 pub f: f64
4551 }
4552 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004553 test_format_item(test_src, "SomeUnionWithDocs", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004554 let result = result.unwrap().unwrap();
4555 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004556 let comment = " Doc for some union.\n\n\
4557 Generated from: <crubit_unittests.rs>;l=3";
Googler04329a12022-12-02 00:56:33 -08004558 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004559 main_api.tokens,
Googler04329a12022-12-02 00:56:33 -08004560 quote! {
4561 __COMMENT__ #comment
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08004562 union ... SomeUnionWithDocs final {
Googler04329a12022-12-02 00:56:33 -08004563 ...
4564 }
4565 ...
4566 }
4567 );
4568 });
4569 }
4570
4571 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004572 fn test_format_item_doc_comments_enum() {
Googler04329a12022-12-02 00:56:33 -08004573 let test_src = r#"
4574 /** Doc for some enum. */
4575 pub enum SomeEnumWithDocs {
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004576 Kind1(i32),
Googler04329a12022-12-02 00:56:33 -08004577 }
4578 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004579 test_format_item(test_src, "SomeEnumWithDocs", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004580 let result = result.unwrap().unwrap();
4581 let main_api = &result.main_api;
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004582 let comment = " Doc for some enum. \n\n\
4583 Generated from: <crubit_unittests.rs>;l=3";
4584 assert_cc_matches!(
4585 main_api.tokens,
4586 quote! {
4587 __COMMENT__ #comment
4588 struct ... SomeEnumWithDocs final {
4589 ...
4590 }
4591 ...
4592 }
4593 );
Googler04329a12022-12-02 00:56:33 -08004594 });
4595 }
4596
4597 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004598 fn test_format_item_doc_comments_struct() {
Googler04329a12022-12-02 00:56:33 -08004599 let test_src = r#"
4600 #![allow(dead_code)]
4601 #[doc = "Doc for some struct."]
4602 pub struct SomeStructWithDocs {
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08004603 #[doc = "Doc for first field."]
Googler04329a12022-12-02 00:56:33 -08004604 some_field : i32,
4605 }
4606 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004607 test_format_item(test_src, "SomeStructWithDocs", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004608 let result = result.unwrap().unwrap();
4609 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004610 let comment = "Doc for some struct.\n\n\
4611 Generated from: <crubit_unittests.rs>;l=4";
Googler04329a12022-12-02 00:56:33 -08004612 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004613 main_api.tokens,
Googler04329a12022-12-02 00:56:33 -08004614 quote! {
4615 __COMMENT__ #comment
4616 struct ... SomeStructWithDocs final {
4617 ...
4618 }
4619 ...
4620 }
4621 );
4622 });
4623 }
4624
4625 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004626 fn test_format_item_doc_comments_tuple_struct() {
Googler04329a12022-12-02 00:56:33 -08004627 let test_src = r#"
4628 /// Doc for some tuple struct.
4629 pub struct SomeTupleStructWithDocs(i32);
4630 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004631 test_format_item(test_src, "SomeTupleStructWithDocs", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004632 let result = result.unwrap().unwrap();
4633 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004634 let comment = " Doc for some tuple struct.\n\n\
4635 Generated from: <crubit_unittests.rs>;l=3";
Googler04329a12022-12-02 00:56:33 -08004636 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004637 main_api.tokens,
Googler04329a12022-12-02 00:56:33 -08004638 quote! {
4639 __COMMENT__ #comment
4640 struct ... SomeTupleStructWithDocs final {
4641 ...
4642 }
4643 ...
4644 },
4645 );
4646 });
4647 }
4648
4649 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004650 fn test_format_item_source_loc_macro_rules() {
Googler47fd9572023-01-20 09:57:32 -08004651 let test_src = r#"
4652 macro_rules! some_tuple_struct_macro_for_testing_source_loc {
4653 () => {
4654 /// Some doc on SomeTupleStructMacroForTesingSourceLoc.
4655 pub struct SomeTupleStructMacroForTesingSourceLoc(i32);
4656 };
4657 }
4658
4659 some_tuple_struct_macro_for_testing_source_loc!();
4660 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004661 test_format_item(test_src, "SomeTupleStructMacroForTesingSourceLoc", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004662 let result = result.unwrap().unwrap();
4663 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004664 let source_loc_comment = " Some doc on SomeTupleStructMacroForTesingSourceLoc.\n\n\
4665 Generated from: <crubit_unittests.rs>;l=5";
4666 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004667 main_api.tokens,
Googler47fd9572023-01-20 09:57:32 -08004668 quote! {
4669 __COMMENT__ #source_loc_comment
4670 struct ... SomeTupleStructMacroForTesingSourceLoc final {
4671 ...
4672 }
4673 ...
4674 },
4675 );
4676 });
4677 }
4678
4679 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004680 fn test_format_item_source_loc_with_no_doc_comment() {
Googler47fd9572023-01-20 09:57:32 -08004681 let test_src = r#"
4682 pub struct SomeTupleStructWithNoDocComment(i32);
4683 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004684 test_format_item(test_src, "SomeTupleStructWithNoDocComment", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004685 let result = result.unwrap().unwrap();
4686 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004687 let comment = "Generated from: <crubit_unittests.rs>;l=2";
4688 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004689 main_api.tokens,
Googler47fd9572023-01-20 09:57:32 -08004690 quote! {
4691 __COMMENT__ #comment
4692 struct ... SomeTupleStructWithNoDocComment final {
4693 ...
4694 }
4695 ...
4696 },
4697 );
4698 });
4699 }
4700
4701 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004702 fn test_format_item_unsupported_static_value() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07004703 let test_src = r#"
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004704 #[no_mangle]
4705 pub static STATIC_VALUE: i32 = 42;
Lukasz Anforowicze4333062022-10-17 14:47:53 -07004706 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004707 test_format_item(test_src, "STATIC_VALUE", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004708 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004709 assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: static item");
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07004710 });
4711 }
4712
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004713 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004714 fn test_format_item_unsupported_const_value() {
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004715 let test_src = r#"
4716 pub const CONST_VALUE: i32 = 42;
4717 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004718 test_format_item(test_src, "CONST_VALUE", |result| {
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004719 let err = result.unwrap_err();
4720 assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: constant item");
4721 });
4722 }
4723
4724 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004725 fn test_format_item_unsupported_type_alias() {
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004726 let test_src = r#"
4727 pub type TypeAlias = i32;
4728 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004729 test_format_item(test_src, "TypeAlias", |result| {
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004730 // TODO(b/254096006): Add support for type alias definitions.
4731 let err = result.unwrap_err();
4732 assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: type alias");
4733 });
4734 }
4735
Lukasz Anforowicz14229762023-02-10 15:28:33 -08004736 #[test]
4737 fn test_format_item_unsupported_impl_item_const_value() {
4738 let test_src = r#"
4739 pub struct SomeStruct(i32);
4740
4741 impl SomeStruct {
4742 pub const CONST_VALUE: i32 = 42;
4743 }
4744 "#;
4745 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004746 let result = result.unwrap().unwrap();
4747 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004748 assert!(!main_api.prereqs.is_empty());
Lukasz Anforowicz14229762023-02-10 15:28:33 -08004749 let unsupported_msg = "Error generating bindings for `SomeStruct::CONST_VALUE` \
4750 defined at <crubit_unittests.rs>;l=5: \
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004751 Unsupported `impl` item kind: Const";
Lukasz Anforowicz14229762023-02-10 15:28:33 -08004752 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004753 main_api.tokens,
Lukasz Anforowicz14229762023-02-10 15:28:33 -08004754 quote! {
4755 ...
4756 struct alignas(4) SomeStruct final {
4757 ...
4758 __COMMENT__ #unsupported_msg
4759 ...
4760 };
4761 ...
4762 }
4763 );
4764 });
4765 }
4766
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08004767 /// `test_format_ret_ty_for_cc_successes` provides test coverage for cases
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07004768 /// where `format_ty_for_cc` takes `TypeLocation::FnReturn` and returns
4769 /// an `Ok(...)`. Additional testcases are covered by
4770 /// `test_format_ty_for_cc_successes`.
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004771 #[test]
Lukasz Anforowicz8a68f502022-11-15 08:43:43 -08004772 fn test_format_ret_ty_for_cc_successes() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004773 let testcases = [
4774 // ( <Rust type>, <expected C++ type> )
4775 ("bool", "bool"), // TyKind::Bool
4776 ("()", "void"),
4777 // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more
4778 // details).
4779 ("!", "void"),
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07004780 (
4781 "extern \"C\" fn (f32, f32) -> f32",
4782 "crubit :: type_identity_t < float (float , float) > &",
4783 ),
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004784 ];
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004785 test_ty(&testcases, quote! {}, |desc, tcx, ty, expected| {
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07004786 let actual = {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08004787 let input = bindings_input_for_tests(tcx);
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07004788 let cc_snippet = format_ty_for_cc(&input, ty, TypeLocation::FnReturn).unwrap();
Lukasz Anforowicz3744e502022-12-02 08:40:38 -08004789 cc_snippet.tokens.to_string()
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07004790 };
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004791 let expected = expected.parse::<TokenStream>().unwrap().to_string();
4792 assert_eq!(actual, expected, "{desc}");
4793 });
4794 }
4795
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004796 /// `test_format_ty_for_cc_successes` provides test coverage for cases where
4797 /// `format_ty_for_cc` returns an `Ok(...)`.
4798 ///
4799 /// Note that using `std::int8_t` (instead of `::std::int8_t`) has been an
4800 /// explicit decision. The "Google C++ Style Guide" suggests to "avoid
4801 /// nested namespaces that match well-known top-level namespaces" and "in
4802 /// particular, [...] not create any nested std namespaces.". It
4803 /// seems desirable if the generated bindings conform to this aspect of the
4804 /// style guide, because it makes things easier for *users* of these
4805 /// bindings.
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004806 #[test]
Lukasz Anforowicz8a68f502022-11-15 08:43:43 -08004807 fn test_format_ty_for_cc_successes() {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004808 let testcases = [
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004809 // ( <Rust type>, (<expected C++ type>,
4810 // <expected #include>,
4811 // <expected prereq def>,
4812 // <expected prereq fwd decl>) )
4813 ("bool", ("bool", "", "", "")),
4814 ("f32", ("float", "", "", "")),
4815 ("f64", ("double", "", "", "")),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08004816 ("i8", ("std::int8_t", "<cstdint>", "", "")),
4817 ("i16", ("std::int16_t", "<cstdint>", "", "")),
4818 ("i32", ("std::int32_t", "<cstdint>", "", "")),
4819 ("i64", ("std::int64_t", "<cstdint>", "", "")),
4820 ("isize", ("std::intptr_t", "<cstdint>", "", "")),
4821 ("u8", ("std::uint8_t", "<cstdint>", "", "")),
4822 ("u16", ("std::uint16_t", "<cstdint>", "", "")),
4823 ("u32", ("std::uint32_t", "<cstdint>", "", "")),
4824 ("u64", ("std::uint64_t", "<cstdint>", "", "")),
4825 ("usize", ("std::uintptr_t", "<cstdint>", "", "")),
Lukasz Anforowiczec0b64e2023-02-17 14:31:12 -08004826 ("char", ("rs_std::rs_char", "\"crubit/support/for/tests/rs_std/rs_char.h\"", "", "")),
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004827 ("SomeStruct", ("::rust_out::SomeStruct", "", "SomeStruct", "")),
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004828 ("SomeEnum", ("::rust_out::SomeEnum", "", "SomeEnum", "")),
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004829 ("SomeUnion", ("::rust_out::SomeUnion", "", "SomeUnion", "")),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08004830 ("*const i32", ("const std::int32_t*", "<cstdint>", "", "")),
4831 ("*mut i32", ("std::int32_t*", "<cstdint>", "", "")),
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004832 // `SomeStruct` is a `fwd_decls` prerequisite (not `defs` prerequisite):
4833 ("*mut SomeStruct", ("::rust_out::SomeStruct*", "", "", "SomeStruct")),
4834 // Testing propagation of deeper/nested `fwd_decls`:
4835 ("*mut *mut SomeStruct", (":: rust_out :: SomeStruct * *", "", "", "SomeStruct")),
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07004836 (
4837 "extern \"C\" fn (f32, f32) -> f32",
4838 (
4839 "crubit :: type_identity_t < float (float , float) > *",
4840 "\"crubit/support/for/tests/internal/cxx20_backports.h\"",
4841 "",
4842 "",
4843 ),
4844 ),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004845 // Extra parens/sugar are expected to be ignored:
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004846 ("(bool)", ("bool", "", "", "")),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004847 ];
Lukasz Anforowicz40472722022-11-08 13:29:08 -08004848 let preamble = quote! {
4849 #![allow(unused_parens)]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004850
4851 pub struct SomeStruct {
4852 pub x: i32,
4853 pub y: i32,
4854 }
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004855 pub enum SomeEnum {
4856 Cartesian{x: f64, y: f64},
4857 Polar{angle: f64, dist: f64},
4858 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004859 pub union SomeUnion {
4860 pub x: i32,
4861 pub y: i32,
4862 }
Lukasz Anforowicz40472722022-11-08 13:29:08 -08004863 };
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08004864 test_ty(
4865 &testcases,
4866 preamble,
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004867 |desc, tcx, ty,
4868 (expected_tokens, expected_include, expected_prereq_def, expected_prereq_fwd_decl)| {
4869 let (actual_tokens, actual_prereqs) = {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08004870 let input = bindings_input_for_tests(tcx);
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07004871 let s = format_ty_for_cc(&input, ty, TypeLocation::Other).unwrap();
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004872 (s.tokens.to_string(), s.prereqs)
4873 };
4874 let (actual_includes, actual_prereq_defs, actual_prereq_fwd_decls) =
4875 (actual_prereqs.includes, actual_prereqs.defs, actual_prereqs.fwd_decls);
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07004876
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08004877 let expected_tokens = expected_tokens.parse::<TokenStream>().unwrap().to_string();
4878 assert_eq!(actual_tokens, expected_tokens, "{desc}");
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07004879
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004880 if expected_include.is_empty() {
4881 assert!(actual_includes.is_empty());
4882 } else {
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08004883 let expected_include: TokenStream = expected_include.parse().unwrap();
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004884 assert_cc_matches!(
4885 format_cc_includes(&actual_includes),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08004886 quote! { __HASH_TOKEN__ include #expected_include }
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004887 );
4888 }
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08004889
4890 if expected_prereq_def.is_empty() {
4891 assert!(actual_prereq_defs.is_empty());
4892 } else {
4893 let expected_def_id = find_def_id_by_name(tcx, expected_prereq_def);
4894 assert_eq!(1, actual_prereq_defs.len());
4895 assert_eq!(expected_def_id, actual_prereq_defs.into_iter().next().unwrap());
4896 }
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004897
4898 if expected_prereq_fwd_decl.is_empty() {
4899 assert!(actual_prereq_fwd_decls.is_empty());
4900 } else {
4901 let expected_def_id = find_def_id_by_name(tcx, expected_prereq_fwd_decl);
4902 assert_eq!(1, actual_prereq_fwd_decls.len());
4903 assert_eq!(expected_def_id,
4904 actual_prereq_fwd_decls.into_iter().next().unwrap());
4905 }
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08004906 },
4907 );
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004908 }
4909
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004910 /// `test_format_ty_for_cc_failures` provides test coverage for cases where
4911 /// `format_ty_for_cc` returns an `Err(...)`.
4912 ///
4913 /// It seems okay to have no test coverage for now for the following types
4914 /// (which should never be encountered when generating bindings and where
4915 /// `format_ty_for_cc` should panic):
4916 /// - TyKind::Closure
4917 /// - TyKind::Error
4918 /// - TyKind::FnDef
4919 /// - TyKind::Infer
4920 ///
Lukasz Anforowicz0182c5c2022-12-29 10:08:50 -08004921 /// TODO(lukasza): Add test coverage (here and in the "for_rs" flavours)
4922 /// for:
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004923 /// - TyKind::Bound
4924 /// - TyKind::Dynamic (`dyn Eq`)
4925 /// - TyKind::Foreign (`extern type T`)
4926 /// - https://doc.rust-lang.org/beta/unstable-book/language-features/generators.html:
4927 /// TyKind::Generator, TyKind::GeneratorWitness
4928 /// - TyKind::Param
4929 /// - TyKind::Placeholder
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004930 #[test]
Lukasz Anforowicz8a68f502022-11-15 08:43:43 -08004931 fn test_format_ty_for_cc_failures() {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004932 let testcases = [
4933 // ( <Rust type>, <expected error message> )
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004934 (
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004935 "()", // Empty TyKind::Tuple
Devin Jeanpierre81aec502023-04-04 15:33:39 -07004936 "`()` / `void` is only supported as a return type (b/254507801)",
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004937 ),
4938 (
4939 // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more
4940 // details).
4941 "!", // TyKind::Never
Devin Jeanpierre81aec502023-04-04 15:33:39 -07004942 "The never type `!` is only supported as a return type (b/254507801)",
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004943 ),
4944 (
4945 "(i32, i32)", // Non-empty TyKind::Tuple
Lukasz Anforowicz13794df2022-10-21 07:56:34 -07004946 "Tuples are not supported yet: (i32, i32) (b/254099023)",
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004947 ),
4948 (
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004949 "&'static i32", // TyKind::Ref
4950 "The following Rust type is not supported yet: &'static i32",
4951 ),
4952 (
4953 "[i32; 42]", // TyKind::Array
4954 "The following Rust type is not supported yet: [i32; 42]",
4955 ),
4956 (
4957 "&'static [i32]", // TyKind::Slice (nested underneath TyKind::Ref)
4958 "The following Rust type is not supported yet: &'static [i32]",
4959 ),
4960 (
4961 "&'static str", // TyKind::Str (nested underneath TyKind::Ref)
4962 "The following Rust type is not supported yet: &'static str",
4963 ),
4964 (
Lukasz Anforowicz0182c5c2022-12-29 10:08:50 -08004965 "impl Eq", // TyKind::Alias
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004966 "The following Rust type is not supported yet: impl std::cmp::Eq",
4967 ),
4968 (
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07004969 "fn(i32) -> i32", // TyKind::FnPtr (default ABI = "Rust")
4970 "Function pointers can't have a thunk: \
4971 Calling convention other than `extern \"C\"` requires a thunk",
4972 ),
4973 (
4974 "extern \"C\" fn (SomeStruct, f32) -> f32",
4975 "Function pointers can't have a thunk: Type of parameter #0 requires a thunk",
4976 ),
4977 (
4978 "extern \"C\" fn (f32, f32) -> SomeStruct",
4979 "Function pointers can't have a thunk: Return type requires a thunk",
4980 ),
4981 (
4982 "unsafe fn(i32) -> i32",
4983 "Bindings for `unsafe` functions are not fully designed yet (b/254095482)",
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004984 ),
4985 // TODO(b/254094650): Consider mapping this to Clang's (and GCC's) `__int128`
4986 // or to `absl::in128`.
4987 ("i128", "C++ doesn't have a standard equivalent of `i128` (b/254094650)"),
4988 ("u128", "C++ doesn't have a standard equivalent of `u128` (b/254094650)"),
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004989 (
4990 "StructWithCustomDrop",
4991 "Failed to generate bindings for the definition of `StructWithCustomDrop`: \
Devin Jeanpierre81aec502023-04-04 15:33:39 -07004992 `Drop` trait and \"drop glue\" are not supported yet (b/258251148)",
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004993 ),
Devin Jeanpierre81aec502023-04-04 15:33:39 -07004994 ("ConstGenericStruct<42>", "Generic types are not supported yet (b/259749095)"),
4995 ("TypeGenericStruct<u8>", "Generic types are not supported yet (b/259749095)"),
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004996 (
4997 // This double-checks that TyKind::Adt(..., substs) are present
4998 // even if the type parameter argument is not explicitly specified
4999 // (here it comes from the default: `...Struct<T = u8>`).
5000 "TypeGenericStruct",
5001 "Generic types are not supported yet (b/259749095)",
5002 ),
Googler7a77a802023-04-21 08:32:50 -07005003 ("LifetimeGenericStruct<'static>", "Generic types are not supported yet (b/259749095)"),
Lukasz Anforowiczcfe84552023-04-20 13:38:54 -07005004 (
5005 "std::cmp::Ordering",
5006 "Type `std::cmp::Ordering` comes from the `core` crate, \
5007 but no `--other-crate-bindings` were specified for this crate",
5008 ),
Googler7a77a802023-04-21 08:32:50 -07005009 ("Option<i8>", "Generic types are not supported yet (b/259749095)"),
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -08005010 (
5011 "PublicReexportOfStruct",
5012 "Not directly public type (re-exports are not supported yet - b/262052635)",
5013 ),
5014 (
5015 // This testcase is like `PublicReexportOfStruct`, but the private type and the
5016 // re-export are in another crate. When authoring this test
5017 // `core::alloc::LayoutError` was a public re-export of
5018 // `core::alloc::layout::LayoutError`:
5019 // `https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=d2b5528af9b33b25abe44cc4646d65e3`
5020 // TODO(b/258261328): Once cross-crate bindings are supported we should try
5021 // to test them via a test crate that we control (rather than testing via
5022 // implementation details of the std crate).
5023 "core::alloc::LayoutError",
5024 "Not directly public type (re-exports are not supported yet - b/262052635)",
5025 ),
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07005026 (
5027 "*const Option<i8>",
5028 "Failed to format the pointee of the pointer type `std::option::Option<i8>`: \
5029 Generic types are not supported yet (b/259749095)",
5030 ),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005031 ];
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005032 let preamble = quote! {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005033 #![feature(never_type)]
5034
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07005035 pub struct SomeStruct {
5036 pub x: i32,
5037 pub y: i32,
5038 }
5039
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005040 pub struct StructWithCustomDrop {
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005041 pub x: i32,
5042 pub y: i32,
5043 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005044
5045 impl Drop for StructWithCustomDrop {
5046 fn drop(&mut self) {}
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005047 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005048
5049 pub struct ConstGenericStruct<const N: usize> {
5050 pub arr: [u8; N],
5051 }
5052
5053 pub struct TypeGenericStruct<T = u8> {
5054 pub t: T,
5055 }
5056
5057 pub struct LifetimeGenericStruct<'a> {
5058 pub reference: &'a u8,
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005059 }
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -08005060
5061 mod private_submodule {
5062 pub struct PublicStructInPrivateModule;
5063 }
5064 pub use private_submodule::PublicStructInPrivateModule
5065 as PublicReexportOfStruct;
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005066 };
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005067 test_ty(&testcases, preamble, |desc, tcx, ty, expected_msg| {
5068 let input = bindings_input_for_tests(tcx);
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07005069 let anyhow_err = format_ty_for_cc(&input, ty, TypeLocation::Other).unwrap_err();
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005070 let actual_msg = format!("{anyhow_err:#}");
5071 assert_eq!(&actual_msg, *expected_msg, "{desc}");
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005072 });
5073 }
5074
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005075 #[test]
5076 fn test_format_ty_for_rs_successes() {
5077 // Test coverage for cases where `format_ty_for_rs` returns an `Ok(...)`.
5078 let testcases = [
5079 // ( <Rust type>, <expected Rust spelling for ..._cc_api_impl.rs> )
5080 ("bool", "bool"),
5081 ("f32", "f32"),
5082 ("f64", "f64"),
5083 ("i8", "i8"),
5084 ("i16", "i16"),
5085 ("i32", "i32"),
5086 ("i64", "i64"),
5087 ("i128", "i128"),
5088 ("isize", "isize"),
5089 ("u8", "u8"),
5090 ("u16", "u16"),
5091 ("u32", "u32"),
5092 ("u64", "u64"),
5093 ("u128", "u128"),
5094 ("usize", "usize"),
5095 ("char", "char"),
5096 ("!", "!"),
5097 ("()", "()"),
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08005098 // ADTs:
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08005099 ("SomeStruct", "::rust_out::SomeStruct"),
5100 ("SomeEnum", "::rust_out::SomeEnum"),
5101 ("SomeUnion", "::rust_out::SomeUnion"),
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08005102 // Type from another crate:
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08005103 ("std::cmp::Ordering", "::core::cmp::Ordering"),
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08005104 // `const` and `mut` pointers:
5105 ("*const i32", "*const i32"),
5106 ("*mut i32", "*mut i32"),
5107 // Pointer to an ADT:
5108 ("*mut SomeStruct", "* mut :: rust_out :: SomeStruct"),
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07005109 ("extern \"C\" fn(i32) -> i32", "extern \"C\" fn(i32) -> i32"),
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005110 ];
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005111 let preamble = quote! {
5112 #![feature(never_type)]
5113
5114 pub struct SomeStruct {
5115 pub x: i32,
5116 pub y: i32,
5117 }
5118 pub enum SomeEnum {
5119 Cartesian{x: f64, y: f64},
5120 Polar{angle: f64, dist: f64},
5121 }
5122 pub union SomeUnion {
5123 pub x: i32,
5124 pub y: i32,
5125 }
5126 };
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08005127 test_ty(&testcases, preamble, |desc, tcx, ty, expected_tokens| {
5128 let actual_tokens = format_ty_for_rs(tcx, ty).unwrap().to_string();
5129 let expected_tokens = expected_tokens.parse::<TokenStream>().unwrap().to_string();
5130 assert_eq!(actual_tokens, expected_tokens, "{desc}");
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005131 });
5132 }
5133
5134 #[test]
5135 fn test_format_ty_for_rs_failures() {
5136 // This test provides coverage for cases where `format_ty_for_rs` returns an
5137 // `Err(...)`.
5138 let testcases = [
5139 // ( <Rust type>, <expected error message> )
5140 (
5141 "(i32, i32)", // Non-empty TyKind::Tuple
5142 "Tuples are not supported yet: (i32, i32) (b/254099023)",
5143 ),
5144 (
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005145 "&'static i32", // TyKind::Ref
5146 "The following Rust type is not supported yet: &'static i32",
5147 ),
5148 (
5149 "[i32; 42]", // TyKind::Array
5150 "The following Rust type is not supported yet: [i32; 42]",
5151 ),
5152 (
5153 "&'static [i32]", // TyKind::Slice (nested underneath TyKind::Ref)
5154 "The following Rust type is not supported yet: &'static [i32]",
5155 ),
5156 (
5157 "&'static str", // TyKind::Str (nested underneath TyKind::Ref)
5158 "The following Rust type is not supported yet: &'static str",
5159 ),
5160 (
Lukasz Anforowicz0182c5c2022-12-29 10:08:50 -08005161 "impl Eq", // TyKind::Alias
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005162 "The following Rust type is not supported yet: impl std::cmp::Eq",
5163 ),
5164 (
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08005165 "Option<i8>", // TyKind::Adt - generic + different crate
5166 "Generic types are not supported yet (b/259749095)",
5167 ),
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005168 ];
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005169 let preamble = quote! {};
5170 test_ty(&testcases, preamble, |desc, tcx, ty, expected_err| {
5171 let anyhow_err = format_ty_for_rs(tcx, ty).unwrap_err();
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005172 let actual_err = format!("{anyhow_err:#}");
5173 assert_eq!(&actual_err, *expected_err, "{desc}");
5174 });
5175 }
5176
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005177 fn test_ty<TestFn, Expectation>(
5178 testcases: &[(&str, Expectation)],
5179 preamble: TokenStream,
5180 test_fn: TestFn,
5181 ) where
Lukasz Anforowiczd16b6bf2022-11-22 18:35:08 -08005182 TestFn: for<'tcx> Fn(
5183 /* testcase_description: */ &str,
5184 TyCtxt<'tcx>,
5185 Ty<'tcx>,
5186 &Expectation,
5187 ) + Sync,
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005188 Expectation: Sync,
5189 {
Lukasz Anforowiczd0f0a842022-11-03 12:40:13 -07005190 for (index, (input, expected)) in testcases.iter().enumerate() {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005191 let desc = format!("test #{index}: test input: `{input}`");
5192 let input = {
5193 let ty_tokens: TokenStream = input.parse().unwrap();
5194 let input = quote! {
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005195 #preamble
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005196 pub fn test_function() -> #ty_tokens { panic!("") }
5197 };
5198 input.to_string()
5199 };
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08005200 run_compiler_for_testing(input, |tcx| {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005201 let def_id = find_def_id_by_name(tcx, "test_function");
Lukasz Anforowicz087dff72023-02-17 12:13:32 -08005202 let ty = tcx
5203 .fn_sig(def_id.to_def_id())
5204 .subst_identity()
5205 .no_bound_vars()
5206 .unwrap()
5207 .output();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005208 test_fn(&desc, tcx, ty, expected);
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005209 });
5210 }
5211 }
5212
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08005213 /// Tests invoking `format_item` on the item with the specified `name` from
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005214 /// the given Rust `source`. Returns the result of calling
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08005215 /// `test_function` with `format_item`'s result as an argument.
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005216 /// (`test_function` should typically `assert!` that it got the expected
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08005217 /// result from `format_item`.)
5218 fn test_format_item<F, T>(source: &str, name: &str, test_function: F) -> T
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005219 where
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07005220 F: FnOnce(Result<Option<ApiSnippets>, String>) -> T + Send,
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005221 T: Send,
5222 {
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08005223 run_compiler_for_testing(source, |tcx| {
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005224 let def_id = find_def_id_by_name(tcx, name);
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08005225 let result = format_item(&bindings_input_for_tests(tcx), def_id);
Lukasz Anforowicze4333062022-10-17 14:47:53 -07005226
5227 // https://docs.rs/anyhow/latest/anyhow/struct.Error.html#display-representations says:
5228 // To print causes as well [...], use the alternate selector “{:#}”.
5229 let result = result.map_err(|anyhow_err| format!("{anyhow_err:#}"));
5230
5231 test_function(result)
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005232 })
5233 }
5234
5235 /// Finds the definition id of a Rust item with the specified `name`.
5236 /// Panics if no such item is found, or if there is more than one match.
5237 fn find_def_id_by_name(tcx: TyCtxt, name: &str) -> LocalDefId {
5238 let hir_items = || tcx.hir().items().map(|item_id| tcx.hir().item(item_id));
5239 let items_with_matching_name =
5240 hir_items().filter(|item| item.ident.name.as_str() == name).collect_vec();
Lukasz Anforowiczd0f0a842022-11-03 12:40:13 -07005241 match *items_with_matching_name.as_slice() {
5242 [] => {
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005243 let found_names = hir_items()
5244 .map(|item| item.ident.name.as_str())
5245 .filter(|s| !s.is_empty())
5246 .sorted()
5247 .dedup()
5248 .map(|name| format!("`{name}`"))
Lukasz Anforowiczd0f0a842022-11-03 12:40:13 -07005249 .join(",\n");
5250 panic!("No items named `{name}`.\nInstead found:\n{found_names}");
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005251 }
Lukasz Anforowicz61cb1e32022-11-04 09:08:35 -07005252 [item] => item.owner_id.def_id,
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005253 _ => panic!("More than one item named `{name}`"),
5254 }
5255 }
5256
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005257 fn bindings_input_for_tests(tcx: TyCtxt) -> Input {
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08005258 Input {
5259 tcx,
5260 crubit_support_path: "crubit/support/for/tests".into(),
Lukasz Anforowiczcfe84552023-04-20 13:38:54 -07005261 crate_name_to_include_path: Default::default(),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08005262 _features: (),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08005263 }
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005264 }
5265
5266 /// Tests invoking `generate_bindings` on the given Rust `source`.
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005267 /// Returns the result of calling `test_function` with the generated
5268 /// bindings as an argument. (`test_function` should typically `assert!`
5269 /// that it got the expected `GeneratedBindings`.)
5270 fn test_generated_bindings<F, T>(source: &str, test_function: F) -> T
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07005271 where
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005272 F: FnOnce(Result<Output>) -> T + Send,
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07005273 T: Send,
5274 {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005275 run_compiler_for_testing(source, |tcx| {
5276 test_function(generate_bindings(&bindings_input_for_tests(tcx)))
5277 })
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07005278 }
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07005279}