blob: a241ac47b94d9eb1b12ccb92bff87992fb524162 [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 }),
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -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)");
373 },
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700374 }
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 }),
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -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)");
382 },
383 }
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())
485 .ok_or_else(|| anyhow!(
486 "Type `{ty}` comes from the `{other_crate_name}` crate, \
487 but no `--other-crate-bindings` were specified for this crate"))?;
488 prereqs.includes.insert(include.clone());
489 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800490
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -0800491 // Verify if definition of `ty` can be succesfully imported and bail otherwise.
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700492 format_adt_core(input.tcx, def_id).with_context(|| {
493 format!("Failed to generate bindings for the definition of `{ty}`")
494 })?;
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -0800495
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800496 CcSnippet {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -0800497 tokens: FullyQualifiedName::new(input.tcx, def_id).format_for_cc()?,
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800498 prereqs,
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800499 }
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700500 }
501
502 ty::TyKind::RawPtr(ty::TypeAndMut { ty, mutbl }) => {
503 let const_qualifier = match mutbl {
504 Mutability::Mut => quote! {},
505 Mutability::Not => quote! { const },
506 };
507 let CcSnippet { tokens, mut prereqs } =
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700508 format_ty_for_cc(input, *ty, TypeLocation::Other).with_context(|| {
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700509 format!("Failed to format the pointee of the pointer type `{ty}`")
510 })?;
511 prereqs.move_defs_to_fwd_decls();
512 CcSnippet { prereqs, tokens: quote! { #const_qualifier #tokens * } }
513 }
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800514
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700515 ty::TyKind::FnPtr(sig) => {
516 let sig = match sig.no_bound_vars() {
517 None => bail!("Generic functions are not supported yet (b/259749023)"),
518 Some(sig) => sig,
519 };
520 check_fn_sig(&sig)?;
521 is_thunk_required(&sig).context("Function pointers can't have a thunk")?;
522
523 // `is_thunk_required` check above implies `extern "C"` (or `"C-unwind"`).
524 // This assertion reinforces that the generated C++ code doesn't need
525 // to use calling convention attributes like `_stdcall`, etc.
526 assert!(matches!(sig.abi, rustc_target::spec::abi::Abi::C { .. }));
527
528 // C++ references are not rebindable and therefore can't be used to replicate semantics
529 // of Rust field types (or, say, element types of Rust arrays). Because of this, C++
530 // references are only used for top-level return types and parameter types (and
531 // pointers are used in other locations).
532 let ptr_or_ref_sigil = match location {
533 TypeLocation::FnReturn | TypeLocation::FnParam => quote!{ & },
534 TypeLocation::Other => quote!{ * },
535 };
536
537 let mut prereqs = CcPrerequisites::default();
538 prereqs.includes.insert(input.support_header("internal/cxx20_backports.h"));
539 let ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut prereqs);
540 let param_types = format_param_types_for_cc(input, &sig)?
541 .into_iter()
542 .map(|snippet| snippet.into_tokens(&mut prereqs));
543 let tokens = quote! {
544 crubit::type_identity_t<
545 #ret_type( #( #param_types ),* )
546 > #ptr_or_ref_sigil
547 };
548
549 CcSnippet { tokens, prereqs }
550 },
551
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800552 // TODO(b/260268230, b/260729464): When recursively processing nested types (e.g. an
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -0800553 // element type of an Array, a referent of a Ref, a parameter type of an FnPtr, etc), one
554 // should also 1) propagate `CcPrerequisites::defs`, 2) cover `CcPrerequisites::defs` in
555 // `test_format_ty_for_cc...`. For ptr/ref it might be possible to use
556 // `CcPrerequisites::move_defs_to_fwd_decls`.
Lukasz Anforowicz087dff72023-02-17 12:13:32 -0800557 _ => bail!("The following Rust type is not supported yet: {ty}"),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -0700558 })
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -0700559}
560
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700561fn format_ret_ty_for_cc<'tcx>(input: &Input<'tcx>, sig: &ty::FnSig<'tcx>) -> Result<CcSnippet> {
562 format_ty_for_cc(input, sig.output(), TypeLocation::FnReturn)
563 .context("Error formatting function return type")
564}
565
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700566fn format_param_types_for_cc<'tcx>(
567 input: &Input<'tcx>,
568 sig: &ty::FnSig<'tcx>,
569) -> Result<Vec<CcSnippet>> {
570 sig.inputs()
571 .iter()
572 .enumerate()
573 .map(|(i, &ty)| {
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700574 Ok(format_ty_for_cc(input, ty, TypeLocation::FnParam)
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700575 .with_context(|| format!("Error handling parameter #{i}"))?)
576 })
577 .collect()
578}
579
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800580/// Formats `ty` for Rust - to be used in `..._cc_api_impl.rs` (e.g. as a type
581/// of a parameter in a Rust thunk). Because `..._cc_api_impl.rs` is a
582/// distinct, separate crate, the returned `TokenStream` uses crate-qualified
583/// names whenever necessary - for example: `target_crate::SomeStruct` rather
584/// than just `SomeStruct`.
Lukasz Anforowicz1ab5e872022-12-05 10:07:00 -0800585//
586// TODO(b/259724276): This function's results should be memoized.
Lukasz Anforowicz75894832022-11-22 18:25:28 -0800587fn format_ty_for_rs(tcx: TyCtxt, ty: Ty) -> Result<TokenStream> {
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -0800588 Ok(match ty.kind() {
589 ty::TyKind::Bool
590 | ty::TyKind::Float(_)
591 | ty::TyKind::Char
592 | ty::TyKind::Int(_)
593 | ty::TyKind::Uint(_)
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700594 | ty::TyKind::FnPtr(_)
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -0800595 | ty::TyKind::Never => ty
596 .to_string()
597 .parse()
598 .expect("rustc_middle::ty::Ty::to_string() should produce no parsing errors"),
599 ty::TyKind::Tuple(types) => {
600 if types.len() == 0 {
601 quote! { () }
602 } else {
603 // TODO(b/254099023): Add support for tuples.
604 bail!("Tuples are not supported yet: {} (b/254099023)", ty);
605 }
606 }
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800607 ty::TyKind::Adt(adt, substs) => {
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -0800608 ensure!(substs.len() == 0, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -0800609 FullyQualifiedName::new(tcx, adt.did()).format_for_rs()
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700610 }
611 ty::TyKind::RawPtr(ty::TypeAndMut { ty, mutbl }) => {
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800612 let qualifier = match mutbl {
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700613 Mutability::Mut => quote! { mut },
614 Mutability::Not => quote! { const },
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -0800615 };
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700616 let ty = format_ty_for_rs(tcx, *ty).with_context(|| {
617 format!("Failed to format the pointee of the pointer type `{ty}`")
618 })?;
619 quote! { * #qualifier #ty }
620 }
Lukasz Anforowicz087dff72023-02-17 12:13:32 -0800621 _ => bail!("The following Rust type is not supported yet: {ty}"),
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -0800622 })
623}
624
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700625#[derive(Debug, Default)]
626struct ApiSnippets {
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800627 /// Main API - for example:
628 /// - A C++ declaration of a function (with a doc comment),
629 /// - A C++ definition of a struct (with a doc comment).
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700630 main_api: CcSnippet,
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800631
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700632 /// C++ implementation details - for example:
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800633 /// - A C++ declaration of an `extern "C"` thunk,
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700634 /// - C++ `static_assert`s about struct size, aligment, and field offsets.
635 cc_details: CcSnippet,
636
637 /// Rust implementation details - for exmaple:
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800638 /// - A Rust implementation of an `extern "C"` thunk,
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700639 /// - Rust `assert!`s about struct size, aligment, and field offsets.
640 rs_details: TokenStream,
Lukasz Anforowicz33f46302023-02-10 15:38:05 -0800641}
642
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700643impl FromIterator<ApiSnippets> for ApiSnippets {
644 fn from_iter<I: IntoIterator<Item = ApiSnippets>>(iter: I) -> Self {
645 let mut result = ApiSnippets::default();
646 for ApiSnippets { main_api, cc_details, rs_details } in iter.into_iter() {
647 result.main_api += main_api;
648 result.cc_details += cc_details;
649 result.rs_details.extend(rs_details);
650 }
651 result
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -0800652 }
653}
654
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -0700655fn get_fn_sig<'tcx>(tcx: TyCtxt<'tcx>, fn_def_id: LocalDefId) -> Result<ty::FnSig<'tcx>> {
656 match tcx.fn_sig(fn_def_id).subst_identity().no_bound_vars() {
657 None => bail!("Generic functions are not supported yet (b/259749023)"),
658 Some(sig) => Ok(sig),
659 }
660}
661
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700662/// Formats a C++ function declaration of a thunk that wraps a Rust function
663/// identified by `fn_def_id`. `format_thunk_impl` may panic if `fn_def_id`
664/// doesn't identify a function.
Lukasz Anforowicz9924c432023-04-04 12:51:22 -0700665fn format_thunk_decl(
666 input: &Input,
667 fn_def_id: LocalDefId,
668 thunk_name: &TokenStream,
669) -> Result<CcSnippet> {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700670 let tcx = input.tcx;
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700671
672 let mut prereqs = CcPrerequisites::default();
673 let sig = get_fn_sig(tcx, fn_def_id)?;
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700674 let main_api_ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut prereqs);
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700675
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700676 let mut thunk_params = {
677 let cc_types = format_param_types_for_cc(input, &sig)?;
678 sig.inputs()
679 .iter()
680 .zip(cc_types.into_iter())
681 .map(|(&ty, cc_type)| -> Result<TokenStream> {
682 let cc_type = cc_type.into_tokens(&mut prereqs);
683 if is_c_abi_compatible_by_value(ty) {
684 Ok(quote! { #cc_type })
685 } else {
686 // Rust thunk will move a value via memcpy - we need to `ensure` that
687 // invoking the C++ destructor (on the moved-away value) is safe.
688 // TODO(b/259749095): Support generic structs (with non-empty ParamEnv).
689 ensure!(
690 !ty.needs_drop(tcx, ty::ParamEnv::empty()),
691 "Only trivially-movable and trivially-destructible types \
692 may be passed by value over the FFI boundary"
693 );
694 Ok(quote! { #cc_type* })
695 }
696 })
697 .collect::<Result<Vec<_>>>()?
698 };
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700699
700 let thunk_ret_type: TokenStream;
701 if is_c_abi_compatible_by_value(sig.output()) {
702 thunk_ret_type = main_api_ret_type.clone();
703 } else {
704 thunk_ret_type = quote! { void };
705 thunk_params.push(quote! { #main_api_ret_type* __ret_ptr });
706 prereqs.includes.insert(CcInclude::utility());
707 prereqs.includes.insert(input.support_header("internal/return_value_slot.h"));
708 };
709 Ok(CcSnippet {
710 prereqs,
711 tokens: quote! {
712 namespace __crubit_internal {
713 extern "C" #thunk_ret_type #thunk_name ( #( #thunk_params ),* );
714 }
715 },
716 })
717}
718
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -0700719/// Formats a thunk implementation in Rust that provides an `extern "C"` ABI for
720/// calling a Rust function identified by `fn_def_id`. `format_thunk_impl` may
721/// panic if `fn_def_id` doesn't identify a function.
722///
723/// `fully_qualified_fn_name` specifies how the thunk can identify the function
724/// to call. Examples of valid arguments:
725/// - `::crate_name::some_module::free_function`
726/// - `::crate_name::some_module::SomeStruct::method`
727/// - `<::create_name::some_module::SomeStruct as
728/// ::core::default::Default>::default`
729fn format_thunk_impl(
730 tcx: TyCtxt,
731 fn_def_id: LocalDefId,
732 thunk_name: &str,
733 fully_qualified_fn_name: TokenStream,
734) -> Result<TokenStream> {
735 let sig = get_fn_sig(tcx, fn_def_id)?;
736 let param_names_and_types: Vec<(Ident, Ty)> = {
737 let param_names = tcx.fn_arg_names(fn_def_id).iter().enumerate().map(|(i, name)| {
738 if name.as_str().is_empty() {
739 format_ident!("__param_{i}")
740 } else {
741 make_rs_ident(name.as_str())
742 }
743 });
744 let param_types = sig.inputs().iter().copied();
745 param_names.zip(param_types).collect_vec()
746 };
747
748 let mut thunk_params = param_names_and_types
749 .iter()
750 .map(|(param_name, ty)| {
751 let rs_type = format_ty_for_rs(tcx, *ty)
752 .with_context(|| format!("Error handling parameter `{param_name}`"))?;
753 Ok(if is_c_abi_compatible_by_value(*ty) {
754 quote! { #param_name: #rs_type }
755 } else {
756 quote! { #param_name: &mut ::core::mem::MaybeUninit<#rs_type> }
757 })
758 })
759 .collect::<Result<Vec<_>>>()?;
760
761 let mut thunk_ret_type = format_ty_for_rs(tcx, sig.output())?;
762 let mut thunk_body = {
763 let fn_args = param_names_and_types.iter().map(|(rs_name, ty)| {
764 if is_c_abi_compatible_by_value(*ty) {
765 quote! { #rs_name }
766 } else {
767 quote! { unsafe { #rs_name.assume_init_read() } }
768 }
769 });
770 quote! {
771 #fully_qualified_fn_name( #( #fn_args ),* )
772 }
773 };
774 if !is_c_abi_compatible_by_value(sig.output()) {
775 thunk_params.push(quote! {
776 __ret_slot: &mut ::core::mem::MaybeUninit<#thunk_ret_type>
777 });
778 thunk_ret_type = quote! { () };
779 thunk_body = quote! { __ret_slot.write(#thunk_body); };
780 };
781
782 let thunk_name = make_rs_ident(thunk_name);
783 Ok(quote! {
784 #[no_mangle]
785 extern "C" fn #thunk_name( #( #thunk_params ),* ) -> #thunk_ret_type {
786 #thunk_body
787 }
788 })
789}
790
Lukasz Anforowicz9924c432023-04-04 12:51:22 -0700791fn get_symbol_name<'tcx>(tcx: TyCtxt<'tcx>, def_id: LocalDefId) -> Result<&'tcx str> {
792 ensure!(
793 tcx.generics_of(def_id).count() == 0,
794 "Generic functions are not supported yet (b/259749023) - caller should filter them out",
795 );
796
797 // Call to `mono` is ok - `generics_of` have been checked above.
798 let instance = ty::Instance::mono(tcx, def_id.to_def_id());
799
800 Ok(tcx.symbol_name(instance).name)
801}
802
803fn get_thunk_name(symbol_name: &str) -> String {
804 format!("__crubit_thunk_{}", &escape_non_identifier_chars(symbol_name))
805}
806
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700807fn check_fn_sig(sig: &ty::FnSig) -> Result<()> {
808 if sig.c_variadic {
809 // TODO(b/254097223): Add support for variadic functions.
810 bail!("C variadic functions are not supported (b/254097223)");
811 }
812
813 match sig.unsafety {
814 Unsafety::Normal => (),
815 Unsafety::Unsafe => {
816 // TODO(b/254095482): Figure out how to handle `unsafe` functions.
817 bail!("Bindings for `unsafe` functions are not fully designed yet (b/254095482)");
818 }
819 }
820
821 Ok(())
822}
823
824/// Returns `Ok(())` if no thunk is required.
825/// Otherwise returns an error the describes why the thunk is needed.
826fn is_thunk_required(sig: &ty::FnSig) -> Result<()> {
827 match sig.abi {
828 // "C" ABI is okay: Before https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a
829 // Rust panic that "escapes" a "C" ABI function leads to Undefined Behavior. This is
830 // unfortunate, but Crubit's `panics_and_exceptions.md` documents that `-Cpanic=abort`
831 // is the only supported configuration.
832 //
833 // After https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a Rust panic that
834 // tries to "escape" a "C" ABI function will terminate the program. This is okay.
835 rustc_target::spec::abi::Abi::C { unwind: false } => (),
836
837 // "C-unwind" ABI is okay: After
838 // https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a new "C-unwind" ABI may be
839 // used by Rust functions that want to safely propagate Rust panics through frames that
840 // may belong to another language.
841 rustc_target::spec::abi::Abi::C { unwind: true } => (),
842
843 // All other ABIs trigger thunk generation. This covers Rust ABI functions, but also
844 // ABIs that theoretically are understood both by C++ and Rust (e.g. see
845 // `format_cc_call_conv_as_clang_attribute` in `rs_bindings_from_cc/src_code_gen.rs`).
846 _ => bail!("Calling convention other than `extern \"C\"` requires a thunk"),
847 };
848
849 ensure!(is_c_abi_compatible_by_value(sig.output()), "Return type requires a thunk");
850 for (i, param_ty) in sig.inputs().iter().enumerate() {
851 ensure!(is_c_abi_compatible_by_value(*param_ty), "Type of parameter #{i} requires a thunk",);
852 }
853
854 Ok(())
855}
856
Googlerfb204272022-12-02 00:52:05 -0800857/// Formats a function with the given `local_def_id`.
Lukasz Anforowicze4333062022-10-17 14:47:53 -0700858///
Googlerfb204272022-12-02 00:52:05 -0800859/// Will panic if `local_def_id`
Lukasz Anforowicz7123f212022-10-20 08:51:04 -0700860/// - is invalid
861/// - doesn't identify a function,
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700862fn format_fn(input: &Input, local_def_id: LocalDefId) -> Result<ApiSnippets> {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -0800863 let tcx = input.tcx;
Googler624580b2022-12-01 01:23:42 -0800864 let def_id: DefId = local_def_id.to_def_id(); // Convert LocalDefId to DefId.
Lukasz Anforowicz903fc632022-10-25 08:55:33 -0700865
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -0800866 ensure!(
867 tcx.generics_of(def_id).count() == 0,
868 "Generic functions are not supported yet (b/259749023)"
869 );
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -0800870
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -0700871 let sig = get_fn_sig(tcx, local_def_id)?;
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -0700872 check_fn_sig(&sig)?;
873 let needs_thunk = is_thunk_required(&sig).is_err();
Lukasz Anforowicz9924c432023-04-04 12:51:22 -0700874 let thunk_name = {
875 let symbol_name = get_symbol_name(tcx, local_def_id)?;
876 if needs_thunk { get_thunk_name(symbol_name) } else { symbol_name.to_string() }
877 };
Googler624580b2022-12-01 01:23:42 -0800878
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -0700879 let FullyQualifiedName { krate, mod_path, name } = FullyQualifiedName::new(tcx, def_id);
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800880 let fn_name = name.expect("Functions are assumed to always have a name");
881 let main_api_fn_name =
882 format_cc_ident(fn_name.as_str()).context("Error formatting function name")?;
Lukasz Anforowicza577d822022-12-12 15:00:46 -0800883
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800884 let mut main_api_prereqs = CcPrerequisites::default();
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -0700885 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 -0800886 let is_static_method = match tcx.hir().get_by_def_id(local_def_id) {
887 Node::ImplItem(impl_item) => match &impl_item.kind {
888 ImplItemKind::Fn(fn_sig, _) => match fn_sig.decl.implicit_self {
889 ImplicitSelfKind::None => true,
890 _ => bail!("`self` parameter is not supported yet"),
891 },
892 _ => panic!("`format_fn` can only work with functions"),
893 },
894 Node::Item(_) => false, // Free function
895 other => panic!("Unexpected HIR node kind: {other:?}"),
896 };
897
898 struct Param<'tcx> {
899 cc_name: TokenStream,
900 cc_type: TokenStream,
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800901 ty: Ty<'tcx>,
902 }
903 let params = {
904 let names = tcx.fn_arg_names(def_id).iter();
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700905 let cc_types = format_param_types_for_cc(input, &sig)?;
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800906 names
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800907 .enumerate()
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700908 .zip(sig.inputs().iter())
909 .zip(cc_types.into_iter())
910 .map(|(((i, name), &ty), cc_type)| {
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800911 let cc_name = format_cc_ident(name.as_str())
912 .unwrap_or_else(|_err| format_cc_ident(&format!("__param_{i}")).unwrap());
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700913 let cc_type = cc_type.into_tokens(&mut main_api_prereqs);
914 Param { cc_name, cc_type, ty }
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800915 })
Lukasz Anforowiczc6ff4532023-04-13 15:16:30 -0700916 .collect_vec()
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800917 };
918 let main_api_params = params
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800919 .iter()
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800920 .map(|Param { cc_name, cc_type, .. }| quote! { #cc_type #cc_name })
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800921 .collect_vec();
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800922
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -0800923 let struct_name = match tcx.impl_of_method(def_id) {
924 Some(impl_id) => match tcx.impl_subject(impl_id) {
925 ty::ImplSubject::Inherent(ty) => match ty.kind() {
926 ty::TyKind::Adt(adt, substs) => {
927 assert_eq!(0, substs.len(), "Callers should filter out generics");
928 Some(tcx.item_name(adt.did()))
929 }
930 _ => panic!("Non-ADT `impl`s should be filtered by caller"),
931 },
932 ty::ImplSubject::Trait(_) => panic!("Trait methods should be filtered by caller"),
933 },
934 None => None,
935 };
Lukasz Anforowicz9924c432023-04-04 12:51:22 -0700936 let needs_definition = fn_name.as_str() != thunk_name;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800937 let main_api = {
938 let doc_comment = {
939 let doc_comment = format_doc_comment(tcx, local_def_id);
940 quote! { __NEWLINE__ #doc_comment }
941 };
942
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800943 let mut prereqs = main_api_prereqs.clone();
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800944 prereqs.move_defs_to_fwd_decls();
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800945
946 let static_ = if is_static_method {
947 quote! { static }
948 } else {
949 quote! {}
950 };
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -0800951 let extern_c_or_inline = if !needs_definition {
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800952 quote! { extern "C" }
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -0800953 } else {
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800954 quote! { inline }
955 };
956 CcSnippet {
957 prereqs,
958 tokens: quote! {
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -0800959 __NEWLINE__
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800960 #doc_comment
Lukasz Anforowicza691cf52023-03-08 12:24:33 -0800961 #static_ #extern_c_or_inline
962 #main_api_ret_type #main_api_fn_name ( #( #main_api_params ),* );
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -0800963 __NEWLINE__
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800964 },
965 }
966 };
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700967 let cc_details = if !needs_definition {
968 CcSnippet::default()
Lukasz Anforowicz6753d402023-02-10 16:41:23 -0800969 } else {
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700970 let thunk_name = format_cc_ident(&thunk_name).context("Error formatting thunk name")?;
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700971 let struct_name = match struct_name.as_ref() {
972 None => quote! {},
973 Some(symbol) => {
974 let name = format_cc_ident(symbol.as_str())
975 .expect("Caller of format_fn should verify struct via format_adt_core");
976 quote! { #name :: }
977 }
978 };
979
980 let mut prereqs = main_api_prereqs;
Devin Jeanpierre81aec502023-04-04 15:33:39 -0700981 let thunk_decl =
982 format_thunk_decl(input, local_def_id, &thunk_name)?.into_tokens(&mut prereqs);
Lukasz Anforowicz73edf152023-04-04 12:05:00 -0700983
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700984 let mut thunk_args = params
985 .iter()
986 .map(|Param { cc_name, ty, .. }| {
987 if is_c_abi_compatible_by_value(*ty) {
988 quote! { #cc_name }
989 } else {
990 quote! { & #cc_name }
991 }
992 })
993 .collect_vec();
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700994 let impl_body: TokenStream;
995 if is_c_abi_compatible_by_value(sig.output()) {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -0700996 impl_body = quote! {
997 return __crubit_internal :: #thunk_name( #( #thunk_args ),* );
998 };
999 } else {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001000 thunk_args.push(quote! { __ret_slot.Get() });
1001 impl_body = quote! {
1002 crubit::ReturnValueSlot<#main_api_ret_type> __ret_slot;
1003 __crubit_internal :: #thunk_name( #( #thunk_args ),* );
1004 return std::move(__ret_slot).AssumeInitAndTakeValue();
1005 };
1006 prereqs.includes.insert(CcInclude::utility());
1007 prereqs.includes.insert(input.support_header("internal/return_value_slot.h"));
1008 };
1009 CcSnippet {
1010 prereqs,
1011 tokens: quote! {
1012 __NEWLINE__
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07001013 #thunk_decl
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001014 inline #main_api_ret_type #struct_name #main_api_fn_name (
1015 #( #main_api_params ),* ) {
1016 #impl_body
1017 }
1018 __NEWLINE__
1019 },
1020 }
1021 };
1022
1023 let rs_details = if !needs_thunk {
1024 quote! {}
1025 } else {
Lukasz Anforowicz6b1ee8c2023-04-04 11:58:29 -07001026 let crate_name = make_rs_ident(krate.as_str());
1027 let mod_path = mod_path.format_for_rs();
1028 let fn_name = make_rs_ident(fn_name.as_str());
1029 let struct_name = match struct_name.as_ref() {
1030 None => quote! {},
1031 Some(symbol) => {
1032 let name = make_rs_ident(symbol.as_str());
1033 quote! { #name :: }
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001034 }
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08001035 };
Devin Jeanpierre81aec502023-04-04 15:33:39 -07001036 let fully_qualified_fn_name = quote! { :: #crate_name :: #mod_path #struct_name #fn_name };
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001037 format_thunk_impl(tcx, local_def_id, &thunk_name, fully_qualified_fn_name)?
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001038 };
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001039 Ok(ApiSnippets { main_api, cc_details, rs_details })
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001040}
1041
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001042/// Represents bindings for the "core" part of an algebraic data type (an ADT -
1043/// a struct, an enum, or a union) in a way that supports later injecting the
1044/// other parts like so:
1045///
1046/// ```
1047/// quote! {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001048/// #keyword #alignment #name final {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001049/// #core
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001050/// #decls_of_other_parts // (e.g. struct fields, methods, etc.)
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001051/// }
1052/// }
1053/// ```
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001054///
1055/// `keyword`, `name` are stored separately, to support formatting them as a
1056/// forward declaration - e.g. `struct SomeStruct`.
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001057struct AdtCoreBindings {
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001058 /// DefId of the ADT.
1059 def_id: DefId,
1060
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001061 /// C++ tag - e.g. `struct`, `class`, `enum`, or `union`. This isn't always
1062 /// a direct mapping from Rust (e.g. a Rust `enum` might end up being
1063 /// represented as an opaque C++ `struct`).
1064 keyword: TokenStream,
1065
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001066 /// C++ translation of the ADT identifier - e.g. `SomeStruct`.
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001067 ///
1068 /// A _short_ name is sufficient (i.e. there is no need to use a
1069 /// namespace-qualified name), for `CcSnippet`s that are emitted into
1070 /// the same namespace as the ADT. (This seems to be all the snippets
1071 /// today.)
1072 cc_short_name: TokenStream,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001073
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001074 /// Rust spelling of the ADT type - e.g.
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001075 /// `::some_crate::some_module::SomeStruct`.
1076 rs_fully_qualified_name: TokenStream,
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001077
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001078 /// `core` contains declarations of
1079 /// - the default constructor
1080 /// - the copy constructor
1081 /// - the move constructor
1082 /// - the copy assignment operator
1083 /// - the move assignment operator
1084 /// - the destructor
1085 core: TokenStream,
1086
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001087 alignment_in_bytes: u64,
1088 size_in_bytes: u64,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001089}
1090
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -08001091/// Like `TyCtxt::is_directly_public`, but works not only with `LocalDefId`, but
1092/// also with `DefId`.
1093fn is_directly_public(tcx: TyCtxt, def_id: DefId) -> bool {
1094 match def_id.as_local() {
1095 None => {
1096 // This mimics the checks in `try_print_visible_def_path_recur` in
1097 // `compiler/rustc_middle/src/ty/print/pretty.rs`.
1098 let actual_parent = tcx.opt_parent(def_id);
1099 let visible_parent = tcx.visible_parent_map(()).get(&def_id).copied();
1100 actual_parent == visible_parent
1101 }
1102 Some(local_def_id) => tcx.effective_visibilities(()).is_directly_public(local_def_id),
1103 }
1104}
1105
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001106fn get_layout<'tcx>(tcx: TyCtxt<'tcx>, ty: Ty<'tcx>) -> Result<Layout<'tcx>> {
1107 // TODO(b/259749095): Support non-empty set of generic parameters.
1108 let param_env = ty::ParamEnv::empty();
1109
1110 tcx.layout_of(param_env.and(ty)).map(|ty_and_layout| ty_and_layout.layout).map_err(
1111 |layout_err| {
1112 // Have to use `.map_err`, because `LayoutError` doesn't satisfy the
1113 // `anyhow::context::ext::StdError` trait bound.
1114 anyhow!("Error computing the layout: {layout_err}")
1115 },
1116 )
1117}
1118
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001119/// Formats the core of an algebraic data type (an ADT - a struct, an enum, or a
1120/// union) represented by `def_id`.
1121///
1122/// The "core" means things that are necessary for a succesful binding (e.g.
1123/// inability to generate a correct C++ destructor means that the ADT cannot
1124/// have any bindings). "core" excludes things that are A) infallible (e.g.
1125/// struct or union fields which can always be translated into private, opaque
1126/// blobs of bytes) or B) optional (e.g. a problematic instance method
1127/// can just be ignored, unlike a problematic destructor). The split between
1128/// fallible "core" and non-fallible "rest" is motivated by the need to avoid
1129/// cycles / infinite recursion (e.g. when processing fields that refer back to
1130/// the struct type, possible with an indirection of a pointer).
1131///
1132/// `format_adt_core` is used both to 1) format bindings for the core of an ADT,
1133/// and 2) check if formatting would have succeeded (e.g. when called from
1134/// `format_ty`). The 2nd case is needed for ADTs defined in any crate - this
1135/// is why the `def_id` parameter is a DefId rather than LocalDefId.
1136//
Lukasz Anforowicz1ab5e872022-12-05 10:07:00 -08001137// TODO(b/259724276): This function's results should be memoized.
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001138fn format_adt_core(tcx: TyCtxt, def_id: DefId) -> Result<AdtCoreBindings> {
Lukasz Anforowicz0c07b092023-03-03 10:34:15 -08001139 let ty = tcx.type_of(def_id).subst_identity();
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -08001140 assert!(ty.is_adt());
1141 assert!(is_directly_public(tcx, def_id), "Caller should verify");
1142
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001143 // TODO(b/259749095): Support non-empty set of generic parameters.
1144 let param_env = ty::ParamEnv::empty();
1145
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001146 if ty.needs_drop(tcx, param_env) {
1147 // TODO(b/258251148): Support custom `Drop` impls.
1148 bail!("`Drop` trait and \"drop glue\" are not supported yet (b/258251148)");
1149 }
1150
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08001151 let adt_def = ty.ty_adt_def().expect("`def_id` needs to identify an ADT");
1152 let keyword = match adt_def.adt_kind() {
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08001153 ty::AdtKind::Struct | ty::AdtKind::Enum => quote! { struct },
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08001154 ty::AdtKind::Union => quote! { union },
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08001155 };
1156
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001157 let item_name = tcx.item_name(def_id);
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001158 let rs_fully_qualified_name = format_ty_for_rs(tcx, ty)?;
1159 let cc_short_name =
1160 format_cc_ident(item_name.as_str()).context("Error formatting item name")?;
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001161
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001162 let layout = get_layout(tcx, ty)
1163 .with_context(|| format!("Error computing the layout of #{item_name}"))?;
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001164 let alignment_in_bytes = {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001165 // Only the ABI-mandated alignment is considered (i.e. `AbiAndPrefAlign::pref`
1166 // is ignored), because 1) Rust's `std::mem::align_of` returns the
1167 // ABI-mandated alignment and 2) the generated C++'s `alignas(...)`
1168 // should specify the minimal/mandatory alignment.
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001169 layout.align().abi.bytes()
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001170 };
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001171 let size_in_bytes = layout.size().bytes();
1172 ensure!(size_in_bytes != 0, "Zero-sized types (ZSTs) are not supported (b/258259459)");
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001173
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001174 let core = quote! {
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001175 // TODO(b/258249993): Provide `default` copy constructor and assignment operator if
1176 // the wrapped type is `Copy` on Rust side.
1177 // TODO(b/259741191): If the wrapped type implements the `Clone` trait, then we should
1178 // *consider* calling `clone` from the copy constructor and `clone_from` from the copy
1179 // assignment operator.
1180 #cc_short_name(const #cc_short_name&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001181
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001182 // The generated bindings have to follow Rust move semantics:
1183 // * All Rust types are memcpy-movable (e.g. <internal link>/constructors.html says
1184 // that "Every type must be ready for it to be blindly memcopied to somewhere else
1185 // in memory")
1186 // * The only valid operation on a moved-from non-`Copy` Rust struct is to assign to
1187 // it.
1188 //
1189 // The generated C++ bindings match the required semantics because they:
1190 // * Generate trivial` C++ move constructor and move assignment operator. Per
1191 // <internal link>/cpp/language/move_constructor#Trivial_move_constructor: "A trivial move
1192 // constructor is a constructor that performs the same action as the trivial copy
1193 // constructor, that is, makes a copy of the object representation as if by
1194 // std::memmove."
1195 // * Generate trivial C++ destructor. (Types that implement `Drop` trait or require
1196 // "drop glue" are not *yet* supported - this might eventually change as part of the
1197 // work tracked under b/258251148). Per
1198 // <internal link>/cpp/language/destructor#Trivial_destructor: "A trivial destructor is a
1199 // destructor that performs no action."
1200 //
1201 // In particular, note that the following C++ code and Rust code are exactly equivalent
1202 // (except that in Rust, reuse of `y` is forbidden at compile time, whereas in C++,
1203 // it's only prohibited by convention):
1204 // * C++, assumming trivial move constructor and trivial destructor:
1205 // `auto x = std::move(y);`
1206 // * Rust, assumming non-`Copy`, no custom `Drop` or drop glue:
1207 // `let x = y;`
1208 //
1209 // TODO(b/258251148): If the ADT provides a custom `Drop` impls or requires drop glue,
1210 // then extra care should be taken to ensure the C++ destructor can handle the
1211 // moved-from object in a way that meets Rust move semantics. For example, the
1212 // generated C++ move constructor might need to assign `Default::default()` to the
1213 // moved-from object.
1214 #cc_short_name(#cc_short_name&&) = default;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001215
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001216 // TODO(b/258235219): Providing assignment operators enables mutation which
1217 // may negatively interact with support for references. Therefore until we
1218 // have more confidence in our reference-handling-plans, we are deleting the
1219 // assignment operators.
1220 //
1221 // (Move assignment operator has another set of concerns and constraints - see the
1222 // comment for the move constructor above).
1223 #cc_short_name& operator=(const #cc_short_name&) = delete;
1224 #cc_short_name& operator=(#cc_short_name&&) = delete;
Lukasz Anforowicz554ed652023-01-12 15:41:58 -08001225
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001226 // TODO(b/258251148): Support custom `Drop` impls and drop glue.
1227 ~#cc_short_name() = default;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001228 };
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001229 Ok(AdtCoreBindings {
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001230 def_id,
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08001231 keyword,
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001232 cc_short_name,
1233 rs_fully_qualified_name,
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001234 core,
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001235 alignment_in_bytes,
1236 size_in_bytes,
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001237 })
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001238}
1239
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001240fn format_fields(input: &Input, core: &AdtCoreBindings) -> ApiSnippets {
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08001241 let tcx = input.tcx;
1242
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001243 // TODO(b/259749095): Support non-empty set of generic parameters.
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001244 let substs_ref = ty::List::empty().as_substs();
1245
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001246 struct FieldTypeInfo {
1247 size: u64,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001248 cc_type: CcSnippet,
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001249 }
1250 struct Field {
1251 type_info: Result<FieldTypeInfo>,
1252 cc_name: TokenStream,
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07001253 rs_name: TokenStream,
1254 is_public: bool,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001255 index: usize,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001256 offset: u64,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001257 offset_of_next_field: u64,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001258 }
1259 let ty = tcx.type_of(core.def_id).subst_identity();
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001260 let layout =
1261 get_layout(tcx, ty).expect("Layout should be already verified by `format_adt_core`");
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001262 let fields: Vec<Field> = if ty.is_enum() || ty.is_union() {
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001263 // Note that `#[repr(Rust)]` unions don't guarantee that all their fields
1264 // have offset 0.
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001265 vec![Field {
1266 type_info: Err(anyhow!(
1267 "No support for bindings of individual fields of \
1268 `union` (b/272801632) or `enum`"
1269 )),
1270 cc_name: quote! { __opaque_blob_of_bytes },
1271 rs_name: quote! { __opaque_blob_of_bytes },
1272 is_public: false,
1273 index: 0,
1274 offset: 0,
1275 offset_of_next_field: core.size_in_bytes,
1276 }]
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001277 } else {
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001278 let mut fields = ty
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001279 .ty_adt_def()
1280 .expect("`core.def_id` needs to identify an ADT")
1281 .all_fields()
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001282 .sorted_by_key(|f| tcx.def_span(f.did))
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001283 .enumerate()
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001284 .map(|(index, field_def)| {
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001285 let field_ty = field_def.ty(tcx, substs_ref);
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001286 let size = get_layout(tcx, field_ty).map(|layout| layout.size().bytes());
1287 let type_info = size.and_then(|size| {
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07001288 Ok(FieldTypeInfo { size, cc_type: format_ty_for_cc(input, field_ty, TypeLocation::Other)? })
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001289 });
1290 let name = field_def.ident(tcx);
Devin Jeanpierre81aec502023-04-04 15:33:39 -07001291 let cc_name = format_cc_ident(name.as_str())
1292 .unwrap_or_else(|_err| format_ident!("__field{index}").into_token_stream());
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07001293 let rs_name = {
1294 let name_starts_with_digit = name
1295 .as_str()
1296 .chars()
1297 .next()
1298 .expect("Empty names are unexpected (here and in general)")
1299 .is_ascii_digit();
1300 if name_starts_with_digit {
1301 let index = Literal::usize_unsuffixed(index);
1302 quote! { #index }
1303 } else {
1304 let name = make_rs_ident(name.as_str());
1305 quote! { #name }
1306 }
1307 };
1308 let is_public = field_def.vis == ty::Visibility::Public;
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001309
1310 // `offset` and `offset_of_next_field` will be fixed by FieldsShape::Arbitrary
1311 // branch below.
1312 let offset = 0;
1313 let offset_of_next_field = 0;
1314
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001315 Field {
1316 type_info,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001317 cc_name,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001318 rs_name,
1319 is_public,
1320 index,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07001321 offset,
1322 offset_of_next_field,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001323 }
1324 })
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001325 .collect_vec();
1326 match layout.fields() {
1327 FieldsShape::Arbitrary { offsets, .. } => {
1328 for (index, offset) in offsets.iter().enumerate() {
1329 // Documentation of `FieldsShape::Arbitrary says that the offsets are "ordered
1330 // to match the source definition order". We can coorelate them with elements
1331 // of the `fields` vector because we've explicitly `sorted_by_key` using
1332 // `def_span`.
1333 fields[index].offset = offset.bytes();
1334 }
Devin Jeanpierre9e15d0b2023-04-06 13:18:22 -07001335 // Sort by offset first; ZSTs in the same offset are sorted by source order.
1336 fields.sort_by_key(|field| (field.offset, field.index));
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001337 let next_offsets = fields
1338 .iter()
1339 .map(|Field { offset, .. }| *offset)
1340 .skip(1)
1341 .chain(once(core.size_in_bytes))
1342 .collect_vec();
1343 for (field, next_offset) in fields.iter_mut().zip(next_offsets) {
1344 field.offset_of_next_field = next_offset;
1345 }
1346 fields
1347 }
1348 unexpected => panic!("Unexpected FieldsShape: {unexpected:?}"),
1349 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001350 };
1351
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001352 let cc_details = if fields.is_empty() {
1353 CcSnippet::default()
1354 } else {
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001355 let adt_cc_name = &core.cc_short_name;
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001356 let cc_assertions: TokenStream = fields
1357 .iter()
1358 .map(|Field { cc_name, offset, .. }| {
1359 let offset = Literal::u64_unsuffixed(*offset);
1360 quote! { static_assert(#offset == offsetof(#adt_cc_name, #cc_name)); }
1361 })
1362 .collect();
1363 CcSnippet::new(quote! {
1364 inline void #adt_cc_name::__crubit_field_offset_assertions() {
1365 #cc_assertions
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08001366 }
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001367 })
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001368 };
1369 let rs_details: TokenStream = {
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001370 let adt_rs_name = &core.rs_fully_qualified_name;
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001371 fields
1372 .iter()
1373 .filter(|Field { is_public, .. }| *is_public)
1374 .map(|Field { rs_name, offset, .. }| {
1375 let expected_offset = Literal::u64_unsuffixed(*offset);
1376 let actual_offset = quote! { memoffset::offset_of!(#adt_rs_name, #rs_name) };
1377 quote! { const _: () = assert!(#actual_offset == #expected_offset); }
1378 })
1379 .collect()
1380 };
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08001381 let main_api = {
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001382 let assertions_method_decl = if fields.is_empty() {
1383 quote! {}
1384 } else {
1385 // We put the assertions in a method so that they can read private member
1386 // variables.
1387 quote! { inline static void __crubit_field_offset_assertions(); }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001388 };
1389
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001390 let mut prereqs = CcPrerequisites::default();
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001391 let fields: TokenStream = fields
1392 .into_iter()
1393 .map(|field| {
1394 let cc_name = field.cc_name;
1395 match field.type_info {
1396 Err(err) => {
Devin Jeanpierre9e15d0b2023-04-06 13:18:22 -07001397 let size = field.offset_of_next_field - field.offset;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001398 let msg =
1399 format!("Field type has been replaced with a blob of bytes: {err:#}");
Devin Jeanpierre9e15d0b2023-04-06 13:18:22 -07001400
1401 // Empty arrays are ill-formed, but also unnecessary for padding.
1402 if size > 0 {
1403 let size = Literal::u64_unsuffixed(size);
1404 quote! {
1405 __COMMENT__ #msg
1406 unsigned char #cc_name[#size];
1407 }
1408 } else {
1409 // TODO(b/258259459): finalize the approach here.
1410 // Possibly we should, rather than using no_unique_address, drop the
1411 // field entirely. This also requires removing the field's assertions,
1412 // added above.
1413 quote! {
1414 __COMMENT__ #msg
1415 [[no_unique_address]] struct{} #cc_name;
1416 }
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001417 }
1418 }
1419 Ok(FieldTypeInfo { cc_type, size }) => {
1420 let padding = field.offset_of_next_field - field.offset - size;
1421 let padding = if padding == 0 {
1422 quote! {}
1423 } else {
1424 let padding = Literal::u64_unsuffixed(padding);
1425 let ident = format_ident!("__padding{}", field.index);
1426 quote! { unsigned char #ident[#padding]; }
1427 };
1428 let cc_type = cc_type.into_tokens(&mut prereqs);
1429 quote! { #cc_type #cc_name; #padding }
1430 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001431 }
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07001432 })
1433 .collect();
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001434
1435 CcSnippet {
1436 prereqs,
1437 tokens: quote! {
1438 // TODO(b/271002281): Preserve actual field visibility.
1439 private: __NEWLINE__
1440 #fields
1441 #assertions_method_decl
1442 },
1443 }
1444 };
1445
1446 ApiSnippets { main_api, cc_details, rs_details }
1447}
1448
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001449/// Finds the `Impl` of a trait impl for `self_ty`. Returns an error if the
1450/// impl wasn't found.
1451///
1452/// `self_ty` should specify a *local* type (i.e. type defined in the crate
1453/// being "compiled").
1454///
1455/// `trait_name` should specify the name of a `core` trait - e.g.
1456/// [`sym::Default`](https://doc.rust-lang.org/beta/nightly-rustc/rustc_span/symbol/sym/constant.Default.html) is a valid
1457/// argument.
1458fn find_core_trait_impl<'tcx>(
1459 tcx: TyCtxt<'tcx>,
1460 self_ty: Ty<'tcx>,
1461 trait_name: Symbol,
1462) -> Result<&'tcx Impl<'tcx>> {
1463 let trait_id = tcx
1464 .get_diagnostic_item(trait_name)
1465 .expect("`find_core_trait_impl` should only be called with `core`, always-present traits");
1466 // TODO(b/275387739): Eventually we might need to support blanket impls.
1467 let mut impls = tcx.non_blanket_impls_for_ty(trait_id, self_ty);
1468 let impl_id = impls.next();
1469 if impl_id.is_some() {
1470 assert_eq!(None, impls.next(), "Expecting only a single trait impl");
1471 }
1472 let impl_id =
1473 impl_id.ok_or_else(|| anyhow!("`{self_ty}` doesn't implement the `{trait_name}` trait"))?;
1474 let impl_id = impl_id.expect_local(); // Expecting that `self_ty` is a local type.
1475 match &tcx.hir().expect_item(impl_id).kind {
1476 ItemKind::Impl(impl_) => Ok(impl_),
1477 other => panic!("Unexpected `ItemKind` from `non_blanket_impls_for_ty`: {other:?}"),
1478 }
1479}
1480
1481/// Formats a default constructor for an ADT if possible (i.e. if the `Default`
1482/// trait is implemented for the ADT). Returns an error otherwise (e.g. if
1483/// there is no `Default` impl).
1484fn format_default_ctor(input: &Input, core: &AdtCoreBindings) -> Result<ApiSnippets> {
1485 let tcx = input.tcx;
1486 let ty = tcx.type_of(core.def_id).subst_identity();
1487
1488 let trait_impl = find_core_trait_impl(input.tcx, ty, sym::Default)?;
1489 assert_eq!(trait_impl.items.len(), 1, "Only the `default` method is expected");
1490 assert_eq!(trait_impl.items[0].ident.name.as_str(), "default");
1491 let cc_struct_name = &core.cc_short_name;
1492 let main_api = CcSnippet::new(quote! {
1493 __NEWLINE__ __COMMENT__ "Default::default"
1494 inline #cc_struct_name(); __NEWLINE__ __NEWLINE__
1495 });
1496 let fn_def_id = trait_impl.items[0].id.owner_id.def_id;
1497 let thunk_name = get_thunk_name(get_symbol_name(tcx, fn_def_id)?);
1498 let cc_details = {
1499 let thunk_name = format_cc_ident(&thunk_name)?;
1500 let CcSnippet { tokens: thunk_decl, prereqs } =
1501 format_thunk_decl(input, fn_def_id, &thunk_name)?;
1502 let tokens = quote! {
1503 #thunk_decl
1504 #cc_struct_name::#cc_struct_name() {
1505 __crubit_internal::#thunk_name(this);
1506 }
1507 };
1508 CcSnippet { tokens, prereqs }
1509 };
1510 let rs_details = {
1511 let struct_name = &core.rs_fully_qualified_name;
1512 let fully_qualified_fn_name =
1513 quote! { <#struct_name as ::core::default::Default>::default };
1514 format_thunk_impl(tcx, fn_def_id, &thunk_name, fully_qualified_fn_name)?
1515 };
1516 Ok(ApiSnippets { main_api, cc_details, rs_details })
1517}
1518
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001519/// Formats an algebraic data type (an ADT - a struct, an enum, or a union)
1520/// represented by `core`. This function is infallible - after
1521/// `format_adt_core` returns success we have committed to emitting C++ bindings
1522/// for the ADT.
1523fn format_adt(input: &Input, core: &AdtCoreBindings) -> ApiSnippets {
1524 let tcx = input.tcx;
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001525 let adt_cc_name = &core.cc_short_name;
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001526
1527 // `format_adt` should only be called for local ADTs.
1528 let local_def_id = core.def_id.expect_local();
1529
1530 let ApiSnippets {
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001531 main_api: default_ctor_main_api,
1532 cc_details: default_ctor_cc_details,
1533 rs_details: default_ctor_rs_details,
1534 } = format_default_ctor(input, core).unwrap_or_else(|err| {
1535 let msg = format!("{err:#}");
1536 ApiSnippets {
1537 main_api: CcSnippet::new(quote! {
1538 __NEWLINE__ __COMMENT__ #msg
1539 #adt_cc_name() = delete; __NEWLINE__
1540 }),
1541 ..Default::default()
1542 }
1543 });
1544
1545 let ApiSnippets {
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001546 main_api: fields_main_api,
1547 cc_details: fields_cc_details,
1548 rs_details: fields_rs_details,
1549 } = format_fields(input, core);
1550
1551 let ApiSnippets {
1552 main_api: impl_items_main_api,
1553 cc_details: impl_items_cc_details,
1554 rs_details: impl_items_rs_details,
1555 } = tcx
1556 .inherent_impls(core.def_id)
1557 .iter()
1558 .map(|impl_id| tcx.hir().expect_item(impl_id.expect_local()))
1559 .flat_map(|item| match &item.kind {
1560 ItemKind::Impl(impl_) => impl_.items,
1561 other => panic!("Unexpected `ItemKind` from `inherent_impls`: {other:?}"),
1562 })
1563 .sorted_by_key(|impl_item_ref| {
1564 let def_id = impl_item_ref.id.owner_id.def_id;
1565 tcx.def_span(def_id)
1566 })
1567 .filter_map(|impl_item_ref| {
1568 let def_id = impl_item_ref.id.owner_id.def_id;
1569 if !tcx.effective_visibilities(()).is_directly_public(def_id) {
1570 return None;
1571 }
1572 let result = match impl_item_ref.kind {
1573 AssocItemKind::Fn { .. } => format_fn(input, def_id).map(Some),
1574 other => Err(anyhow!("Unsupported `impl` item kind: {other:?}")),
1575 };
1576 result.unwrap_or_else(|err| Some(format_unsupported_def(tcx, def_id, err)))
1577 })
1578 .collect();
1579
1580 let alignment = Literal::u64_unsuffixed(core.alignment_in_bytes);
1581 let size = Literal::u64_unsuffixed(core.size_in_bytes);
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001582 let main_api = {
1583 let cc_packed_attribute = {
1584 let has_packed_attribute = tcx
1585 .get_attrs(core.def_id, rustc_span::symbol::sym::repr)
1586 .flat_map(|attr| rustc_attr::parse_repr_attr(tcx.sess(), attr))
1587 .any(|repr| matches!(repr, rustc_attr::ReprPacked { .. }));
1588 if has_packed_attribute {
1589 quote! { __attribute__((packed)) }
1590 } else {
1591 quote! {}
1592 }
1593 };
1594
1595 let doc_comment = format_doc_comment(tcx, core.def_id.expect_local());
1596 let keyword = &core.keyword;
1597 let core = &core.core;
1598
1599 let mut prereqs = CcPrerequisites::default();
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001600 let default_ctor_main_api = default_ctor_main_api.into_tokens(&mut prereqs);
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001601 let impl_items_main_api = if impl_items_main_api.tokens.is_empty() {
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001602 quote! {}
1603 } else {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001604 let tokens = impl_items_main_api.into_tokens(&mut prereqs);
1605 quote! { public: #tokens }
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001606 };
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001607 let fields_main_api = fields_main_api.into_tokens(&mut prereqs);
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08001608 prereqs.fwd_decls.remove(&local_def_id);
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001609
1610 CcSnippet {
1611 prereqs,
1612 tokens: quote! {
1613 __NEWLINE__ #doc_comment
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07001614 #keyword alignas(#alignment) #cc_packed_attribute #adt_cc_name final {
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001615 public:
1616 #default_ctor_main_api
1617 #core
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001618 #impl_items_main_api
1619 #fields_main_api
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001620 };
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001621 __NEWLINE__
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001622 },
1623 }
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001624 };
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001625 let cc_details = {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001626 let mut prereqs = CcPrerequisites::default();
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001627 let default_ctor_cc_details = default_ctor_cc_details.into_tokens(&mut prereqs);
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001628 let impl_items_cc_details = impl_items_cc_details.into_tokens(&mut prereqs);
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001629 let fields_cc_details = fields_cc_details.into_tokens(&mut prereqs);
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001630 prereqs.defs.insert(local_def_id);
1631 CcSnippet {
1632 prereqs,
1633 tokens: quote! {
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07001634 __NEWLINE__
1635 static_assert(
1636 sizeof(#adt_cc_name) == #size,
1637 "Verify that struct layout didn't change since this header got generated");
1638 static_assert(
1639 alignof(#adt_cc_name) == #alignment,
1640 "Verify that struct layout didn't change since this header got generated");
1641 __NEWLINE__
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001642 #default_ctor_cc_details
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001643 #impl_items_cc_details
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001644 #fields_cc_details
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001645 },
1646 }
Lukasz Anforowicz3b579542023-02-06 11:23:49 -08001647 };
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001648 let rs_details = {
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001649 let adt_rs_name = &core.rs_fully_qualified_name;
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001650 quote! {
1651 const _: () = assert!(::std::mem::size_of::<#adt_rs_name>() == #size);
1652 const _: () = assert!(::std::mem::align_of::<#adt_rs_name>() == #alignment);
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07001653 #default_ctor_rs_details
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07001654 #impl_items_rs_details
1655 #fields_rs_details
1656 }
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001657 };
1658 ApiSnippets { main_api, cc_details, rs_details }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001659}
1660
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001661/// Formats the forward declaration of an algebraic data type (an ADT - a
1662/// struct, an enum, or a union), returning something like
1663/// `quote!{ struct SomeStruct; }`.
1664///
1665/// Will panic if `def_id` doesn't identify an ADT that can be successfully
1666/// handled by `format_adt_core`.
1667fn format_fwd_decl(tcx: TyCtxt, def_id: LocalDefId) -> TokenStream {
1668 let def_id = def_id.to_def_id(); // LocalDefId -> DefId conversion.
1669
1670 // `format_fwd_decl` should only be called for items from
1671 // `CcPrerequisites::fwd_decls` and `fwd_decls` should only contain ADTs
1672 // that `format_adt_core` succeeds for.
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001673 let AdtCoreBindings { keyword, cc_short_name, .. } = format_adt_core(tcx, def_id)
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001674 .expect("`format_fwd_decl` should only be called if `format_adt_core` succeeded");
1675
Lukasz Anforowiczce56a802023-04-04 12:18:19 -07001676 quote! { #keyword #cc_short_name; }
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001677}
1678
Googler47fd9572023-01-20 09:57:32 -08001679fn format_source_location(tcx: TyCtxt, local_def_id: LocalDefId) -> String {
1680 let def_span = tcx.def_span(local_def_id);
1681 let rustc_span::FileLines { file, lines } =
1682 match tcx.sess().source_map().span_to_lines(def_span) {
1683 Ok(filelines) => filelines,
1684 Err(_) => return "unknown location".to_string(),
1685 };
1686 let file_name = file.name.prefer_local().to_string();
1687 // Note: line_index starts at 0, while CodeSearch starts indexing at 1.
1688 let line_number = lines[0].line_index + 1;
1689 let google3_prefix = {
1690 // If rustc_span::FileName isn't a 'real' file, then it's surrounded by by angle
Lukasz Anforowicz1f233912023-02-14 08:50:26 -08001691 // brackets, thus don't prepend "google3/" prefix.
1692 if file.name.is_real() { "google3/" } else { "" }
Googler47fd9572023-01-20 09:57:32 -08001693 };
Googler785e6b42023-01-23 12:11:36 -08001694 format!("{google3_prefix}{file_name};l={line_number}")
Googler47fd9572023-01-20 09:57:32 -08001695}
1696
1697/// Formats the doc comment (if any) associated with the item identified by
1698/// `local_def_id`, and appends the source location at which the item is
1699/// defined.
Googlerfb204272022-12-02 00:52:05 -08001700fn format_doc_comment(tcx: TyCtxt, local_def_id: LocalDefId) -> TokenStream {
1701 let hir_id = tcx.local_def_id_to_hir_id(local_def_id);
Googler47fd9572023-01-20 09:57:32 -08001702 let doc_comment = tcx
Googlerfb204272022-12-02 00:52:05 -08001703 .hir()
1704 .attrs(hir_id)
1705 .iter()
Lukasz Anforowiczdf363ee2022-12-16 14:56:38 -08001706 .filter_map(|attr| attr.doc_str())
Googler47fd9572023-01-20 09:57:32 -08001707 .map(|symbol| symbol.to_string())
Googler785e6b42023-01-23 12:11:36 -08001708 .chain(once(format!("Generated from: {}", format_source_location(tcx, local_def_id))))
Googler34f3d572022-12-02 00:53:37 -08001709 .join("\n\n");
Googler47fd9572023-01-20 09:57:32 -08001710 quote! { __COMMENT__ #doc_comment}
Googlerfb204272022-12-02 00:52:05 -08001711}
1712
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001713/// Formats a HIR item idenfied by `def_id`. Returns `None` if the item
Lukasz Anforowicz14229762023-02-10 15:28:33 -08001714/// can be ignored. Returns an `Err` if the definition couldn't be formatted.
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001715///
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001716/// Will panic if `def_id` is invalid (i.e. doesn't identify a HIR item).
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001717fn format_item(input: &Input, def_id: LocalDefId) -> Result<Option<ApiSnippets>> {
Lukasz Anforowicz1382f392022-12-12 17:13:23 -08001718 // TODO(b/262052635): When adding support for re-exports we may need to change
Devin Jeanpierre81aec502023-04-04 15:33:39 -07001719 // `is_directly_public` below into `is_exported`. (OTOH such change *alone* is
1720 // undesirable, because it would mean exposing items from a private module.
1721 // Exposing a private module is undesirable, because it would mean that
1722 // changes of private implementation details of the crate could become
1723 // breaking changes for users of the generated C++ bindings.)
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001724 if !input.tcx.effective_visibilities(()).is_directly_public(def_id) {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001725 return Ok(None);
Lukasz Anforowicz8b68a552022-12-12 15:07:58 -08001726 }
1727
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001728 match input.tcx.hir().expect_item(def_id) {
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08001729 Item { kind: ItemKind::Struct(_, generics) |
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08001730 ItemKind::Enum(_, generics) |
1731 ItemKind::Union(_, generics),
1732 .. } if !generics.params.is_empty() => {
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08001733 bail!("Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08001734 },
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001735 Item { kind: ItemKind::Fn(..), .. } => format_fn(input, def_id).map(Some),
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08001736 Item { kind: ItemKind::Struct(..) | ItemKind::Enum(..) | ItemKind::Union(..), .. } =>
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07001737 format_adt_core(input.tcx, def_id.to_def_id())
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001738 .map(|core| Some(format_adt(input, &core))),
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08001739 Item { kind: ItemKind::Impl(_), .. } | // Handled by `format_adt`
1740 Item { kind: ItemKind::Mod(_), .. } => // Handled by `format_crate`
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001741 Ok(None),
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001742 Item { kind, .. } => bail!("Unsupported rustc_hir::hir::ItemKind: {}", kind.descr()),
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001743 }
1744}
1745
1746/// Formats a C++ comment explaining why no bindings have been generated for
1747/// `local_def_id`.
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -07001748fn format_unsupported_def(
1749 tcx: TyCtxt,
1750 local_def_id: LocalDefId,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001751 err: anyhow::Error,
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001752) -> ApiSnippets {
Googler785e6b42023-01-23 12:11:36 -08001753 let source_loc = format_source_location(tcx, local_def_id);
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -07001754 let name = tcx.def_path_str(local_def_id.to_def_id());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001755
1756 // https://docs.rs/anyhow/latest/anyhow/struct.Error.html#display-representations
1757 // says: To print causes as well [...], use the alternate selector “{:#}”.
Googler785e6b42023-01-23 12:11:36 -08001758 let msg = format!("Error generating bindings for `{name}` defined at {source_loc}: {err:#}");
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001759 let main_api = CcSnippet::new(quote! { __NEWLINE__ __NEWLINE__ __COMMENT__ #msg __NEWLINE__ });
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001760
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001761 ApiSnippets { main_api, cc_details: CcSnippet::default(), rs_details: quote! {} }
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -07001762}
1763
Lukasz Anforowicz2ec13122022-11-10 12:39:04 -08001764/// Formats all public items from the Rust crate being compiled.
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08001765fn format_crate(input: &Input) -> Result<Output> {
1766 let tcx = input.tcx;
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001767 let mut cc_details_prereqs = CcPrerequisites::default();
1768 let mut cc_details: Vec<(LocalDefId, TokenStream)> = vec![];
1769 let mut rs_body = TokenStream::default();
Lukasz Anforowicz991b2a52023-03-23 07:28:41 -07001770 let mut main_apis = HashMap::<LocalDefId, CcSnippet>::new();
1771 let formatted_items = tcx
Lukasz Anforowicz2ec13122022-11-10 12:39:04 -08001772 .hir()
1773 .items()
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001774 .filter_map(|item_id| {
Lukasz Anforowicz2ec13122022-11-10 12:39:04 -08001775 let def_id: LocalDefId = item_id.owner_id.def_id;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001776 format_item(input, def_id)
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001777 .unwrap_or_else(|err| Some(format_unsupported_def(tcx, def_id, err)))
1778 .map(|api_snippets| (def_id, api_snippets))
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07001779 })
Lukasz Anforowicz991b2a52023-03-23 07:28:41 -07001780 .sorted_by_key(|(def_id, _)| tcx.def_span(*def_id));
1781 for (def_id, api_snippets) in formatted_items {
1782 let old_item = main_apis.insert(def_id, api_snippets.main_api);
1783 assert!(old_item.is_none(), "Duplicated key: {def_id:?}");
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001784
Lukasz Anforowicz991b2a52023-03-23 07:28:41 -07001785 // `cc_details` don't participate in the toposort, because
1786 // `CcPrerequisites::defs` always use `main_api` as the predecessor
1787 // - `chain`ing `cc_details` after `ordered_main_apis` trivially
1788 // meets the prerequisites.
1789 cc_details.push((def_id, api_snippets.cc_details.into_tokens(&mut cc_details_prereqs)));
1790 rs_body.extend(api_snippets.rs_details);
1791 }
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08001792
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001793 // Find the order of `main_apis` that 1) meets the requirements of
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08001794 // `CcPrerequisites::defs` and 2) makes a best effort attempt to keep the
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001795 // `main_apis` in the same order as the source order of the Rust APIs.
Lukasz Anforowicz2ecb27f2023-01-12 15:29:51 -08001796 let ordered_ids = {
1797 let toposort::TopoSortResult { ordered: ordered_ids, failed: failed_ids } = {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001798 let nodes = main_apis.keys().copied();
1799 let deps = main_apis.iter().flat_map(|(&successor, main_api)| {
1800 let predecessors = main_api.prereqs.defs.iter().map(|&def_id| def_id);
Lukasz Anforowicz2ecb27f2023-01-12 15:29:51 -08001801 predecessors.map(move |predecessor| toposort::Dependency { predecessor, successor })
1802 });
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001803 toposort::toposort(nodes, deps, move |lhs_id, rhs_id| {
1804 tcx.def_span(*lhs_id).cmp(&tcx.def_span(*rhs_id))
1805 })
Lukasz Anforowicz2ecb27f2023-01-12 15:29:51 -08001806 };
1807 assert_eq!(
1808 0,
1809 failed_ids.len(),
1810 "There are no known scenarios where CcPrerequisites::defs can form \
1811 a dependency cycle. These `LocalDefId`s form an unexpected cycle: {}",
1812 failed_ids.into_iter().map(|id| format!("{:?}", id)).join(",")
1813 );
1814 ordered_ids
Lukasz Anforowiczab563af2022-12-15 08:09:50 -08001815 };
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08001816
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001817 // Destructure/rebuild `main_apis` (in the same order as `ordered_ids`) into
1818 // `includes`, and `ordered_cc` (mixing in `fwd_decls` and `cc_details`).
1819 let (includes, ordered_cc) = {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001820 let mut already_declared = HashSet::new();
1821 let mut fwd_decls = HashSet::new();
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001822 let mut includes = cc_details_prereqs.includes;
1823 let mut ordered_main_apis: Vec<(LocalDefId, TokenStream)> = Vec::new();
1824 for def_id in ordered_ids.into_iter() {
1825 let CcSnippet {
1826 tokens: cc_tokens,
1827 prereqs: CcPrerequisites {
1828 includes: mut inner_includes,
1829 fwd_decls: inner_fwd_decls,
1830 .. // `defs` have already been utilized by `toposort` above
Lukasz Anforowicz7f31f802022-12-16 08:24:13 -08001831 }
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001832 } = main_apis.remove(&def_id).unwrap();
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001833
1834 fwd_decls.extend(inner_fwd_decls.difference(&already_declared).copied());
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001835 already_declared.insert(def_id);
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001836 already_declared.extend(inner_fwd_decls.into_iter());
1837
1838 includes.append(&mut inner_includes);
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001839 ordered_main_apis.push((def_id, cc_tokens));
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001840 }
1841
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001842 let fwd_decls = fwd_decls
1843 .into_iter()
1844 .sorted_by_key(|def_id| tcx.def_span(*def_id))
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07001845 .map(|local_def_id| (local_def_id, format_fwd_decl(tcx, local_def_id)));
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001846
1847 let ordered_cc: Vec<(NamespaceQualifier, TokenStream)> = fwd_decls
1848 .into_iter()
1849 .chain(ordered_main_apis.into_iter())
1850 .chain(cc_details.into_iter())
1851 .map(|(local_def_id, tokens)| {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001852 let mod_path = FullyQualifiedName::new(tcx, local_def_id.to_def_id()).mod_path;
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001853 (mod_path, tokens)
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001854 })
1855 .collect_vec();
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001856
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07001857 (includes, ordered_cc)
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08001858 };
Lukasz Anforowicza577d822022-12-12 15:00:46 -08001859
1860 // Generate top-level elements of the C++ header file.
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001861 let h_body = {
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001862 // TODO(b/254690602): Decide whether using `#crate_name` as the name of the
1863 // top-level namespace is okay (e.g. investigate if this name is globally
1864 // unique + ergonomic).
1865 let crate_name = format_cc_ident(tcx.crate_name(LOCAL_CRATE).as_str())?;
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001866
Lukasz Anforowicz54efc162022-12-16 15:58:44 -08001867 let includes = format_cc_includes(&includes);
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08001868 let ordered_cc = format_namespace_bound_cc_tokens(ordered_cc);
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001869 quote! {
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001870 #includes
1871 __NEWLINE__ __NEWLINE__
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001872 namespace #crate_name {
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001873 __NEWLINE__
Lukasz Anforowicz54efc162022-12-16 15:58:44 -08001874 #ordered_cc
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001875 __NEWLINE__
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001876 }
Lukasz Anforowicza0502fb2023-02-13 15:33:18 -08001877 __NEWLINE__
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001878 }
1879 };
Lukasz Anforowicz7f31f802022-12-16 08:24:13 -08001880
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08001881 Ok(Output { h_body, rs_body })
Lukasz Anforowiczed1c4f02022-09-29 11:11:20 -07001882}
1883
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07001884#[cfg(test)]
Lukasz Anforowiczf018e4d2022-09-28 07:35:59 -07001885pub mod tests {
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08001886 use super::*;
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07001887
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001888 use anyhow::Result;
1889 use itertools::Itertools;
1890 use proc_macro2::TokenStream;
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001891 use quote::quote;
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07001892 use rustc_middle::ty::{Ty, TyCtxt};
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001893 use rustc_span::def_id::LocalDefId;
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07001894
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08001895 use crate::run_compiler::tests::run_compiler_for_testing;
Lukasz Anforowicz52274992023-03-08 12:29:28 -08001896 use code_gen_utils::format_cc_includes;
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08001897 use token_stream_matchers::{
1898 assert_cc_matches, assert_cc_not_matches, assert_rs_matches, assert_rs_not_matches,
1899 };
Lukasz Anforowicz2b38d272022-09-23 08:08:18 -07001900
Lukasz Anforowicz5bddf182022-09-30 16:06:59 -07001901 #[test]
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001902 #[should_panic(expected = "No items named `missing_name`.\n\
1903 Instead found:\n`bar`,\n`foo`,\n`m1`,\n`m2`,\n`std`")]
1904 fn test_find_def_id_by_name_panic_when_no_item_with_matching_name() {
1905 let test_src = r#"
1906 pub extern "C" fn foo() {}
1907
1908 pub mod m1 {
1909 pub fn bar() {}
1910 }
1911 pub mod m2 {
1912 pub fn bar() {}
1913 }
1914 "#;
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08001915 run_compiler_for_testing(test_src, |tcx| find_def_id_by_name(tcx, "missing_name"));
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001916 }
1917
1918 #[test]
1919 #[should_panic(expected = "More than one item named `some_name`")]
1920 fn test_find_def_id_by_name_panic_when_multiple_items_with_matching_name() {
1921 let test_src = r#"
1922 pub mod m1 {
1923 pub fn some_name() {}
1924 }
1925 pub mod m2 {
1926 pub fn some_name() {}
1927 }
1928 "#;
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08001929 run_compiler_for_testing(test_src, |tcx| find_def_id_by_name(tcx, "some_name"));
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001930 }
1931
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001932 /// This test covers only a single example of a function that should get a
1933 /// C++ binding. The test focuses on verification that the output from
1934 /// `format_fn` gets propagated all the way to `GenerateBindings::new`.
1935 /// Additional coverage of how functions are formatted is provided
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001936 /// by `test_format_item_..._fn_...` tests (which work at the `format_fn`
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001937 /// level).
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07001938 #[test]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08001939 fn test_generated_bindings_fn_no_mangle_extern_c() {
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07001940 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07001941 #[no_mangle]
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001942 pub extern "C" fn public_function() {
1943 println!("foo");
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07001944 }
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07001945 "#;
1946 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001947 let bindings = bindings.unwrap();
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001948 assert_cc_matches!(
1949 bindings.h_body,
1950 quote! {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07001951 extern "C" void public_function();
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001952 }
Lukasz Anforowiczd9ff4ab2022-09-23 08:11:18 -07001953 );
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001954
1955 // No Rust thunks should be generated in this test scenario.
Devin Jeanpierre81aec502023-04-04 15:33:39 -07001956 assert_rs_not_matches!(bindings.rs_body, quote! { public_function });
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001957 });
1958 }
1959
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001960 /// `test_generated_bindings_fn_export_name` covers a scenario where
1961 /// `MixedSnippet::cc` is present but `MixedSnippet::rs` is empty
1962 /// (because no Rust thunks are needed).
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07001963 #[test]
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001964 fn test_generated_bindings_fn_export_name() {
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001965 let test_src = r#"
1966 #[export_name = "export_name"]
1967 pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y }
1968 "#;
1969 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001970 let bindings = bindings.unwrap();
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001971 assert_cc_matches!(
1972 bindings.h_body,
1973 quote! {
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001974 namespace rust_out {
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08001975 ...
1976 inline double public_function(double x, double y);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001977 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07001978 extern "C" double export_name(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001979 }
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001980 inline double public_function(double x, double y) {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08001981 return __crubit_internal::export_name(x, y);
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07001982 }
1983 }
1984 }
1985 );
1986 });
1987 }
1988
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001989 /// The `test_generated_bindings_struct` test covers only a single example
1990 /// of an ADT (struct/enum/union) that should get a C++ binding.
1991 /// Additional coverage of how items are formatted is provided by
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001992 /// `test_format_item_..._struct_...`, `test_format_item_..._enum_...`,
1993 /// and `test_format_item_..._union_...` tests.
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001994 ///
1995 /// We don't want to duplicate coverage already provided by
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001996 /// `test_format_item_struct_with_fields`, but we do want to verify that
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08001997 /// * `format_crate` will actually find and process the struct
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08001998 /// (`test_format_item_...` doesn't cover this aspect - it uses a
1999 /// test-only `find_def_id_by_name` instead)
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002000 /// * The actual shape of the bindings still looks okay at this level.
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002001 #[test]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002002 fn test_generated_bindings_struct() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002003 let test_src = r#"
2004 pub struct Point {
2005 pub x: i32,
2006 pub y: i32,
2007 }
2008 "#;
2009 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002010 let bindings = bindings.unwrap();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002011 assert_cc_matches!(
2012 bindings.h_body,
2013 quote! {
2014 namespace rust_out {
Googler47fd9572023-01-20 09:57:32 -08002015 ...
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002016 struct alignas(4) Point final {
2017 // No point replicating test coverage of
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002018 // `test_format_item_struct_with_fields`.
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002019 ...
2020 };
2021 static_assert(sizeof(Point) == 8, ...);
2022 static_assert(alignof(Point) == 4, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07002023 ... // Other static_asserts are covered by
2024 // `test_format_item_struct_with_fields`
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002025 } // namespace rust_out
2026 }
2027 );
2028 assert_rs_matches!(
2029 bindings.rs_body,
2030 quote! {
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07002031 // No point replicating test coverage of
2032 // `test_format_item_struct_with_fields`.
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08002033 const _: () = assert!(::std::mem::size_of::<::rust_out::Point>() == 8);
2034 const _: () = assert!(::std::mem::align_of::<::rust_out::Point>() == 4);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07002035 const _: () = assert!( memoffset::offset_of!(::rust_out::Point, x) == 0);
2036 const _: () = assert!( memoffset::offset_of!(::rust_out::Point, y) == 4);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002037 }
2038 );
2039 });
2040 }
2041
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002042 /// The `test_generated_bindings_impl` test covers only a single example of
2043 /// a non-trait `impl`. Additional coverage of how items are formatted
2044 /// should be provided in the future by `test_format_item_...` tests.
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002045 ///
2046 /// We don't want to duplicate coverage already provided by
2047 /// `test_format_item_static_method`, but we do want to verify that
2048 /// * `format_crate` won't process the `impl` as a standalone HIR item
2049 /// * The actual shape of the bindings still looks okay at this level.
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002050 #[test]
2051 fn test_generated_bindings_impl() {
2052 let test_src = r#"
2053 pub struct SomeStruct(i32);
2054
2055 impl SomeStruct {
2056 pub fn public_static_method() -> i32 { 123 }
2057
2058 #[allow(dead_code)]
2059 fn private_static_method() -> i32 { 123 }
2060 }
2061 "#;
2062 test_generated_bindings(test_src, |bindings| {
2063 let bindings = bindings.unwrap();
2064 assert_cc_matches!(
2065 bindings.h_body,
2066 quote! {
2067 namespace rust_out {
2068 ...
2069 struct ... SomeStruct ... {
2070 // No point replicating test coverage of
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002071 // `test_format_item_static_method`.
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002072 ...
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002073 std::int32_t public_static_method();
2074 ...
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002075 };
2076 ...
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002077 std::int32_t SomeStruct::public_static_method() {
2078 ...
2079 }
Lukasz Anforowiczb7145e62023-03-23 09:00:37 -07002080 ...
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002081 } // namespace rust_out
2082 }
2083 );
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08002084 assert_rs_matches!(
2085 bindings.rs_body,
2086 quote! {
2087 extern "C" fn ...() -> i32 {
2088 ::rust_out::SomeStruct::public_static_method()
2089 }
2090 }
2091 );
Lukasz Anforowicz93513ec2023-02-10 14:21:20 -08002092 });
2093 }
2094
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002095 #[test]
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002096 fn test_generated_bindings_includes() {
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07002097 let test_src = r#"
2098 #[no_mangle]
2099 pub extern "C" fn public_function(i: i32, d: isize, u: u64) {
2100 dbg!(i);
2101 dbg!(d);
2102 dbg!(u);
2103 }
2104 "#;
2105 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002106 let bindings = bindings.unwrap();
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07002107 assert_cc_matches!(
2108 bindings.h_body,
2109 quote! {
2110 __HASH_TOKEN__ include <cstdint> ...
2111 namespace ... {
Googler47fd9572023-01-20 09:57:32 -08002112 ...
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07002113 extern "C" void public_function(
2114 std::int32_t i,
2115 std::intptr_t d,
2116 std::uint64_t u);
2117 }
2118 }
2119 );
2120 });
2121 }
2122
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002123 /// Tests that `toposort` is used to reorder item bindings.
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07002124 #[test]
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002125 fn test_generated_bindings_prereq_defs_field_deps_require_reordering() {
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002126 let test_src = r#"
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002127 // In the generated bindings `Outer` needs to come *after* `Inner`.
2128 pub struct Outer(Inner);
2129 pub struct Inner(bool);
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002130 "#;
2131 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002132 let bindings = bindings.unwrap();
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002133 assert_cc_matches!(
2134 bindings.h_body,
2135 quote! {
2136 namespace rust_out {
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002137 ...
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002138 struct alignas(1) Inner final {
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07002139 ... bool __field0; ...
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002140 };
2141 ...
2142 struct alignas(1) Outer final {
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07002143 ... ::rust_out::Inner __field0; ...
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002144 };
2145 ...
2146 } // namespace rust_out
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002147 }
2148 );
2149 });
2150 }
2151
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002152 /// Tests that a forward declaration is present when it is required to
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002153 /// preserve the original source order. In this test the
2154 /// `CcPrerequisites::fwd_decls` dependency comes from a pointer parameter.
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002155 #[test]
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002156 fn test_generated_bindings_prereq_fwd_decls_for_ptr_param() {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002157 let test_src = r#"
2158 // To preserve original API order we need to forward declare S.
2159 pub fn f(_: *const S) {}
2160 pub struct S(bool);
2161 "#;
2162 test_generated_bindings(test_src, |bindings| {
2163 let bindings = bindings.unwrap();
2164 assert_cc_matches!(
2165 bindings.h_body,
2166 quote! {
2167 namespace rust_out {
2168 ...
2169 // Verifing the presence of this forward declaration
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002170 // it the essence of this test. The order of the items
2171 // below also matters.
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002172 struct S;
2173 ...
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002174 inline void f(const ::rust_out::S* __param_0);
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002175 ...
2176 struct alignas(...) S final { ... }
2177 ...
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002178 inline void f(const ::rust_out::S* __param_0) { ... }
2179 ...
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002180 } // namespace rust_out
2181 }
2182 );
2183 });
2184 }
2185
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002186 /// Tests that a forward declaration is present when it is required to
2187 /// preserve the original source order. In this test the
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002188 /// `CcPrerequisites::fwd_decls` dependency comes from a
2189 /// function declaration that has a parameter that takes a struct by value.
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002190 #[test]
2191 fn test_generated_bindings_prereq_fwd_decls_for_cpp_fn_decl() {
2192 let test_src = r#"
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002193 #[no_mangle]
2194 pub extern "C" fn f(s: S) -> bool { s.0 }
2195
2196 #[repr(C)]
2197 pub struct S(bool);
2198 "#;
2199
2200 test_generated_bindings(test_src, |bindings| {
2201 let bindings = bindings.unwrap();
2202 assert_cc_matches!(
2203 bindings.h_body,
2204 quote! {
2205 namespace rust_out {
2206 ...
2207 // Verifing the presence of this forward declaration
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002208 // is the essence of this test. The order also matters:
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002209 // 1. The fwd decl of `S` should come first,
2210 // 2. Declaration of `f` and definition of `S` should come next
2211 // (in their original order - `f` first and then `S`).
2212 struct S;
2213 ...
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002214 // `CcPrerequisites` of `f` declaration below (the main api of `f`) should
2215 // include `S` as a `fwd_decls` edge, rather than as a `defs` edge.
2216 inline bool f(::rust_out::S s);
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002217 ...
2218 struct alignas(...) S final { ... }
2219 ...
2220 } // namespace rust_out
2221 }
2222 );
2223 });
2224 }
2225
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002226 /// This test verifies that a forward declaration for a given ADT is only
2227 /// emitted once (and not once for every API item that requires the
2228 /// forward declaration as a prerequisite).
2229 #[test]
2230 fn test_generated_bindings_prereq_fwd_decls_no_duplication() {
2231 let test_src = r#"
2232 // All three functions below require a forward declaration of S.
2233 pub fn f1(_: *const S) {}
2234 pub fn f2(_: *const S) {}
2235 pub fn f3(_: *const S) {}
2236
2237 pub struct S(bool);
2238
2239 // This function also includes S in its CcPrerequisites::fwd_decls
2240 // (although here it is not required, because the definition of S
2241 // is already available above).
2242 pub fn f4(_: *const S) {}
2243 "#;
2244 test_generated_bindings(test_src, |bindings| {
2245 let bindings = bindings.unwrap().h_body.to_string();
2246
2247 // Only a single forward declaration is expected.
2248 assert_eq!(1, bindings.matches("struct S ;").count(), "bindings = {bindings}");
2249 });
2250 }
2251
2252 /// This test verifies that forward declarations are emitted in a
2253 /// deterministic order. The particular order doesn't matter _that_
2254 /// much, but it definitely shouldn't change every time
2255 /// `cc_bindings_from_rs` is invoked again. The current order preserves
2256 /// the original source order of the Rust API items.
2257 #[test]
2258 fn test_generated_bindings_prereq_fwd_decls_deterministic_order() {
2259 let test_src = r#"
2260 // To try to mix things up, the bindings for the functions below
2261 // will *ask* for forward declarations in a different order:
2262 // * Different from the order in which the forward declarations
2263 // are expected to be *emitted* (the original source order).
2264 // * Different from alphabetical order.
2265 pub fn f1(_: *const b::S3) {}
2266 pub fn f2(_: *const a::S2) {}
2267 pub fn f3(_: *const a::S1) {}
2268
2269 pub mod a {
2270 pub struct S1(bool);
2271 pub struct S2(bool);
2272 }
2273
2274 pub mod b {
2275 pub struct S3(bool);
2276 }
2277 "#;
2278 test_generated_bindings(test_src, |bindings| {
2279 let bindings = bindings.unwrap();
2280 assert_cc_matches!(
2281 bindings.h_body,
2282 quote! {
2283 namespace rust_out {
2284 ...
2285 // Verifying that we get the same order in each test
2286 // run is the essence of this test.
2287 namespace a {
2288 struct S1;
2289 struct S2;
2290 }
2291 namespace b {
2292 struct S3;
2293 }
2294 ...
2295 inline void f1 ...
2296 inline void f2 ...
2297 inline void f3 ...
2298
2299 namespace a { ...
2300 struct alignas(...) S1 final { ... } ...
2301 struct alignas(...) S2 final { ... } ...
2302 } ...
2303 namespace b { ...
2304 struct alignas(...) S3 final { ... } ...
2305 } ...
2306 } // namespace rust_out
2307 }
2308 );
2309 });
2310 }
2311
2312 /// This test verifies that forward declarations are not emitted if they are
2313 /// not needed (e.g. if bindings the given `struct` or other ADT have
2314 /// already been defined earlier). In particular, we don't want to emit
2315 /// forward declarations for *all* `structs` (regardless if they are
2316 /// needed or not).
2317 #[test]
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08002318 fn test_generated_bindings_prereq_fwd_decls_not_needed_because_of_initial_order() {
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002319 let test_src = r#"
2320 pub struct S(bool);
2321
2322 // S is already defined above - no need for forward declaration in C++.
2323 pub fn f(_s: *const S) {}
2324 "#;
2325 test_generated_bindings(test_src, |bindings| {
2326 let bindings = bindings.unwrap();
2327 assert_cc_not_matches!(bindings.h_body, quote! { struct S; });
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08002328 assert_cc_matches!(bindings.h_body, quote! { void f(const ::rust_out::S* _s); });
2329 });
2330 }
2331
2332 /// This test verifies that a method declaration doesn't ask for a forward
2333 /// declaration to the struct.
2334 #[test]
2335 fn test_generated_bindings_prereq_fwd_decls_not_needed_inside_struct_definition() {
2336 let test_src = r#"
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002337 #![allow(dead_code)]
2338
2339 pub struct S {
2340 // This shouldn't require a fwd decl of S.
2341 field: *const S,
2342 }
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08002343
2344 impl S {
2345 // This shouldn't require a fwd decl of S.
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002346 pub fn create() -> S { Self{ field: std::ptr::null() } }
Lukasz Anforowicz64b04ba2023-02-10 17:19:05 -08002347 }
2348 "#;
2349 test_generated_bindings(test_src, |bindings| {
2350 let bindings = bindings.unwrap();
2351 assert_cc_not_matches!(bindings.h_body, quote! { struct S; });
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002352 assert_cc_matches!(
2353 bindings.h_body,
2354 quote! {
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07002355 static inline ::rust_out::S create(); ...
2356 const ::rust_out::S* field; ...
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07002357 }
2358 );
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08002359 });
2360 }
2361
Lukasz Anforowicz816cbaa2022-12-07 09:31:30 -08002362 #[test]
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002363 fn test_generated_bindings_module_basics() {
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002364 let test_src = r#"
2365 pub mod some_module {
2366 pub fn some_func() {}
2367 }
2368 "#;
2369 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002370 let bindings = bindings.unwrap();
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002371 assert_cc_matches!(
2372 bindings.h_body,
2373 quote! {
2374 namespace rust_out {
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002375 namespace some_module {
2376 ...
2377 inline void some_func() { ... }
2378 ...
2379 } // namespace some_module
2380 } // namespace rust_out
2381 }
2382 );
2383 assert_rs_matches!(
2384 bindings.rs_body,
2385 quote! {
2386 #[no_mangle]
2387 extern "C"
2388 fn ...() -> () {
2389 ::rust_out::some_module::some_func()
2390 }
2391 }
2392 );
2393 });
2394 }
2395
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002396 #[test]
2397 fn test_generated_bindings_module_name_is_cpp_reserved_keyword() {
2398 let test_src = r#"
2399 pub mod working_module {
2400 pub fn working_module_f1() {}
2401 pub fn working_module_f2() {}
2402 }
2403 pub mod reinterpret_cast {
2404 pub fn broken_module_f1() {}
2405 pub fn broken_module_f2() {}
2406 }
2407 "#;
2408 test_generated_bindings(test_src, |bindings| {
2409 let bindings = bindings.unwrap();
2410
2411 // Items in the broken module should be replaced with a comment explaining the
2412 // problem.
2413 let broken_module_msg = "Failed to format namespace name `reinterpret_cast`: \
2414 `reinterpret_cast` is a C++ reserved keyword \
2415 and can't be used as a C++ identifier";
2416 assert_cc_not_matches!(bindings.h_body, quote! { namespace reinterpret_cast });
2417 assert_cc_not_matches!(bindings.h_body, quote! { broken_module_f1 });
2418 assert_cc_not_matches!(bindings.h_body, quote! { broken_module_f2 });
2419
2420 // Items in the other module should still go through.
2421 assert_cc_matches!(
2422 bindings.h_body,
2423 quote! {
2424 namespace rust_out {
2425 namespace working_module {
2426 ...
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002427 inline void working_module_f1();
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002428 ...
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002429 inline void working_module_f2();
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002430 ...
2431 } // namespace some_module
2432
2433 __COMMENT__ #broken_module_msg
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002434 ...
Lukasz Anforowicze1da5372023-01-03 12:31:14 -08002435 } // namespace rust_out
2436 }
2437 );
2438 });
2439 }
2440
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002441 /// `test_generated_bindings_non_pub_items` verifies that non-public items
2442 /// are not present/propagated into the generated bindings.
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002443 #[test]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002444 fn test_generated_bindings_non_pub_items() {
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002445 let test_src = r#"
Lukasz Anforowicz88bde9b2022-10-14 14:48:07 -07002446 #![allow(dead_code)]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002447
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002448 extern "C" fn private_function() {
2449 println!("foo");
2450 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002451
2452 struct PrivateStruct {
2453 x: i32,
2454 y: i32,
2455 }
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002456
Lukasz Anforowicz14229762023-02-10 15:28:33 -08002457 pub struct PublicStruct(i32);
2458
2459 impl PublicStruct {
2460 fn private_method() {}
2461 }
2462
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002463 pub mod public_module {
2464 fn priv_func_in_pub_module() {}
2465 }
2466
2467 mod private_module {
2468 pub fn pub_func_in_priv_module() { priv_func_in_priv_module() }
2469 fn priv_func_in_priv_module() {}
2470 }
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002471 "#;
2472 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002473 let bindings = bindings.unwrap();
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002474 assert_cc_not_matches!(bindings.h_body, quote! { private_function });
Lukasz Anforowicze7a25002022-11-10 06:21:42 -08002475 assert_rs_not_matches!(bindings.rs_body, quote! { private_function });
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002476 assert_cc_not_matches!(bindings.h_body, quote! { PrivateStruct });
2477 assert_rs_not_matches!(bindings.rs_body, quote! { PrivateStruct });
Lukasz Anforowicz14229762023-02-10 15:28:33 -08002478 assert_cc_not_matches!(bindings.h_body, quote! { private_method });
2479 assert_rs_not_matches!(bindings.rs_body, quote! { private_method });
Lukasz Anforowicza577d822022-12-12 15:00:46 -08002480 assert_cc_not_matches!(bindings.h_body, quote! { priv_func_in_priv_module });
2481 assert_rs_not_matches!(bindings.rs_body, quote! { priv_func_in_priv_module });
2482 assert_cc_not_matches!(bindings.h_body, quote! { priv_func_in_pub_module });
2483 assert_rs_not_matches!(bindings.rs_body, quote! { priv_func_in_pub_module });
Lukasz Anforowicz1382f392022-12-12 17:13:23 -08002484 assert_cc_not_matches!(bindings.h_body, quote! { private_module });
2485 assert_rs_not_matches!(bindings.rs_body, quote! { private_module });
2486 assert_cc_not_matches!(bindings.h_body, quote! { pub_func_in_priv_module });
2487 assert_rs_not_matches!(bindings.rs_body, quote! { pub_func_in_priv_module });
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002488 });
2489 }
2490
2491 #[test]
Lukasz Anforowicz4aa1ff92022-10-10 11:22:22 -07002492 fn test_generated_bindings_top_level_items() {
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002493 let test_src = "pub fn public_function() {}";
2494 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002495 let bindings = bindings.unwrap();
Devin Jeanpierre81aec502023-04-04 15:33:39 -07002496 let expected_comment_txt = "Automatically @generated C++ bindings for the following Rust crate:\n\
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002497 rust_out";
2498 assert_cc_matches!(
2499 bindings.h_body,
2500 quote! {
2501 __COMMENT__ #expected_comment_txt
Lukasz Anforowicz4aa1ff92022-10-10 11:22:22 -07002502 ...
Lukasz Anforowicz31b29cd2022-10-10 11:33:41 -07002503 __HASH_TOKEN__ pragma once
2504 ...
Lukasz Anforowicz4aa1ff92022-10-10 11:22:22 -07002505 namespace rust_out {
2506 ...
2507 }
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002508 }
2509 );
Lukasz Anforowicze7a25002022-11-10 06:21:42 -08002510 assert_cc_matches!(
2511 bindings.rs_body,
2512 quote! {
2513 __COMMENT__ #expected_comment_txt
2514 }
2515 );
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002516 })
2517 }
2518
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002519 /// The `test_generated_bindings_unsupported_item` test verifies how `Err`
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002520 /// from `format_item` is formatted as a C++ comment (in `format_crate`
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002521 /// and `format_unsupported_def`):
2522 /// - This test covers only a single example of an unsupported item.
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002523 /// Additional coverage is provided by `test_format_item_unsupported_...`
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002524 /// tests.
2525 /// - This test somewhat arbitrarily chooses an example of an unsupported
2526 /// item, trying to pick one that 1) will never be supported (b/254104998
2527 /// has some extra notes about APIs named after reserved C++ keywords) and
2528 /// 2) tests that the full error chain is included in the message.
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002529 #[test]
2530 fn test_generated_bindings_unsupported_item() {
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002531 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002532 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002533 pub extern "C" fn reinterpret_cast() {}
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002534 "#;
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002535 test_generated_bindings(test_src, |bindings| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002536 let bindings = bindings.unwrap();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002537 let expected_comment_txt = "Error generating bindings for `reinterpret_cast` \
Googler785e6b42023-01-23 12:11:36 -08002538 defined at <crubit_unittests.rs>;l=3: \
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002539 Error formatting function name: \
2540 `reinterpret_cast` is a C++ reserved keyword \
2541 and can't be used as a C++ identifier";
Lukasz Anforowicz2f2510a2022-09-29 13:06:44 -07002542 assert_cc_matches!(
2543 bindings.h_body,
2544 quote! {
2545 __COMMENT__ #expected_comment_txt
2546 }
2547 );
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07002548 })
2549 }
2550
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07002551 #[test]
Lukasz Anforowiczce17f3f2023-02-27 11:32:14 -08002552 fn test_generated_bindings_reimports() {
2553 let test_src = r#"
2554 #![allow(dead_code)]
2555 #![allow(unused_imports)]
2556 mod private_submodule1 {
2557 pub fn subfunction1() {}
2558 pub fn subfunction2() {}
2559 pub fn subfunction3() {}
2560 }
2561 mod private_submodule2 {
2562 pub fn subfunction8() {}
2563 pub fn subfunction9() {}
2564 }
2565
2566 // Public re-import.
2567 pub use private_submodule1::subfunction1;
2568
2569 // Private re-import.
2570 use private_submodule1::subfunction2;
2571
2572 // Re-import that renames.
2573 pub use private_submodule1::subfunction3 as public_function3;
2574
2575 // Re-import of multiple items via glob.
2576 pub use private_submodule2::*;
2577 "#;
2578 test_generated_bindings(test_src, |bindings| {
2579 let bindings = bindings.unwrap();
2580
2581 let failures = vec![(1, 15), (3, 21), (4, 24)];
2582 for (use_number, line_number) in failures.into_iter() {
2583 let expected_comment_txt = format!(
2584 "Error generating bindings for `{{use#{use_number}}}` defined at \
2585 <crubit_unittests.rs>;l={line_number}: \
2586 Unsupported rustc_hir::hir::ItemKind: `use` import"
2587 );
2588 assert_cc_matches!(
2589 bindings.h_body,
2590 quote! {
2591 __COMMENT__ #expected_comment_txt
2592 }
2593 );
2594 }
2595 });
2596 }
2597
2598 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002599 fn test_format_item_fn_extern_c_no_mangle_no_params_no_return_type() {
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07002600 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002601 #[no_mangle]
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07002602 pub extern "C" fn public_function() {}
2603 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002604 test_format_item(test_src, "public_function", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002605 let result = result.unwrap().unwrap();
2606 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002607 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002608 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002609 main_api.tokens,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002610 quote! {
2611 extern "C" void public_function();
2612 }
2613 );
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002614
2615 // Sufficient to just re-declare the Rust API in C++.
2616 // (i.e. there is no need to have a C++-side definition of `public_function`).
2617 assert!(result.cc_details.tokens.is_empty());
2618
2619 // There is no need to have a separate thunk for an `extern "C"` function.
2620 assert!(result.rs_details.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002621 });
2622 }
2623
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002624 /// The `test_format_item_fn_explicit_unit_return_type` test below is very
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002625 /// similar to the
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002626 /// `test_format_item_fn_extern_c_no_mangle_no_params_no_return_type` above,
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002627 /// except that the return type is explicitly spelled out. There is no
2628 /// difference in `ty::FnSig` so our code behaves exactly the same, but the
2629 /// test has been planned based on earlier, hir-focused approach and having
2630 /// this extra test coverage shouldn't hurt. (`hir::FnSig`
2631 /// and `hir::FnRetTy` _would_ see a difference between the two tests, even
2632 /// though there is no different in the current `bindings.rs` code).
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002633 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002634 fn test_format_item_fn_explicit_unit_return_type() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002635 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002636 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002637 pub extern "C" fn explicit_unit_return_type() -> () {}
2638 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002639 test_format_item(test_src, "explicit_unit_return_type", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002640 let result = result.unwrap().unwrap();
2641 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002642 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002643 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002644 main_api.tokens,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002645 quote! {
2646 extern "C" void explicit_unit_return_type();
2647 }
2648 );
2649 });
2650 }
2651
2652 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002653 fn test_format_item_fn_never_return_type() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002654 let test_src = r#"
2655 #[no_mangle]
2656 pub extern "C" fn never_returning_function() -> ! {
2657 panic!("This function panics and therefore never returns");
2658 }
2659 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002660 test_format_item(test_src, "never_returning_function", |result| {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002661 // TODO(b/254507801): The function should be annotated with the `[[noreturn]]`
2662 // attribute.
2663 // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more
2664 // details).
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002665 let result = result.unwrap().unwrap();
2666 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002667 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002668 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002669 main_api.tokens,
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002670 quote! {
2671 extern "C" void never_returning_function();
2672 }
2673 );
2674 })
2675 }
2676
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002677 /// `test_format_item_fn_mangling` checks that bindings can be generated for
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002678 /// `extern "C"` functions that do *not* have `#[no_mangle]` attribute. The
2679 /// test elides away the mangled name in the `assert_cc_matches` checks
2680 /// below, but end-to-end test coverage should eventually be provided by
2681 /// `test/functions` (see b/262904507).
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07002682 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002683 fn test_format_item_fn_mangling() {
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002684 let test_src = r#"
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002685 pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y }
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002686 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002687 test_format_item(test_src, "public_function", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002688 let result = result.unwrap().unwrap();
2689 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002690 assert!(main_api.prereqs.is_empty());
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002691 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002692 main_api.tokens,
2693 quote! {
2694 inline double public_function(double x, double y);
2695 }
2696 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002697 assert!(result.rs_details.is_empty());
2698 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002699 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002700 result.cc_details.tokens,
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002701 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002702 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07002703 extern "C" double ...(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002704 }
Googler47fd9572023-01-20 09:57:32 -08002705 ...
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002706 inline double public_function(double x, double y) {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08002707 return __crubit_internal::...(x, y);
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002708 }
2709 }
2710 );
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002711 });
2712 }
2713
2714 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002715 fn test_format_item_fn_export_name() {
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002716 let test_src = r#"
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002717 #[export_name = "export_name"]
2718 pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y }
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002719 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002720 test_format_item(test_src, "public_function", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002721 let result = result.unwrap().unwrap();
2722 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002723 assert!(main_api.prereqs.is_empty());
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002724 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002725 main_api.tokens,
2726 quote! {
2727 inline double public_function(double x, double y);
2728 }
2729 );
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002730
2731 // There is no need to have a separate thunk for an `extern "C"` function.
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002732 assert!(result.rs_details.is_empty());
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002733
2734 // We generate a C++-side definition of `public_function` so that we
2735 // can call a differently-named (but same-signature) `export_name` function.
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002736 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002737 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002738 result.cc_details.tokens,
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002739 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002740 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07002741 extern "C" double export_name(double, double);
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002742 }
Googler47fd9572023-01-20 09:57:32 -08002743 ...
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002744 inline double public_function(double x, double y) {
2745 return __crubit_internal::export_name(x, y);
2746 }
Lukasz Anforowiczf04bc522022-11-04 09:19:51 -07002747 }
2748 );
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002749 });
2750 }
2751
2752 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002753 fn test_format_item_unsupported_fn_unsafe() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002754 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002755 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002756 pub unsafe extern "C" fn foo() {}
2757 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002758 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002759 let err = result.unwrap_err();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002760 assert_eq!(
2761 err,
2762 "Bindings for `unsafe` functions \
2763 are not fully designed yet (b/254095482)"
2764 );
2765 });
2766 }
2767
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002768 /// `test_format_item_fn_const` tests how bindings for an `const fn` are
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08002769 /// generated.
2770 ///
Googler47fd9572023-01-20 09:57:32 -08002771 /// Right now the `const` qualifier is ignored, but one can imagine that in
2772 /// the (very) long-term future such functions (including their bodies)
2773 /// could be translated into C++ `consteval` functions.
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002774 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002775 fn test_format_item_fn_const() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002776 let test_src = r#"
2777 pub const fn foo(i: i32) -> i32 { i * 42 }
2778 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002779 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002780 // TODO(b/254095787): Update test expectations below once `const fn` from Rust
2781 // is translated into a `consteval` C++ function.
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002782 let result = result.unwrap().unwrap();
2783 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002784 assert!(!main_api.prereqs.is_empty());
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002785 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002786 main_api.tokens,
2787 quote! {
2788 inline std::int32_t foo(std::int32_t i);
2789 }
2790 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002791 assert!(!result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002792 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002793 result.cc_details.tokens,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002794 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002795 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07002796 extern "C" std::int32_t ...( std::int32_t);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08002797 }
Googler47fd9572023-01-20 09:57:32 -08002798 ...
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002799 inline std::int32_t foo(std::int32_t i) {
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08002800 return __crubit_internal::...(i);
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002801 }
2802 }
2803 );
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002804 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002805 result.rs_details,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002806 quote! {
2807 #[no_mangle]
2808 extern "C"
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08002809 fn ...(i: i32) -> i32 {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08002810 ::rust_out::foo(i)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08002811 }
2812 }
2813 );
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002814 });
2815 }
2816
2817 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002818 fn test_format_item_fn_with_c_unwind_abi() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002819 // See also https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html
2820 let test_src = r#"
2821 #![feature(c_unwind)]
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002822
2823 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002824 pub extern "C-unwind" fn may_throw() {}
2825 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002826 test_format_item(test_src, "may_throw", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002827 let result = result.unwrap().unwrap();
2828 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002829 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002830 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002831 main_api.tokens,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002832 quote! {
2833 extern "C" void may_throw();
2834 }
2835 );
2836 });
2837 }
2838
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002839 /// This test mainly verifies that `format_item` correctly propagates
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002840 /// `CcPrerequisites` of parameter types and return type.
2841 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002842 fn test_format_item_fn_cc_prerequisites_if_cpp_definition_needed() {
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002843 let test_src = r#"
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002844 pub fn foo(_i: i32) -> S { panic!("foo") }
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002845 pub struct S(i32);
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002846 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002847 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002848 let result = result.unwrap().unwrap();
2849 let main_api = &result.main_api;
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002850
2851 // Minimal coverage, just to double-check that the test setup works.
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002852 //
2853 // Note that this is a definition, and therefore `S` should be defined
2854 // earlier (not just forward declared).
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002855 assert_cc_matches!(main_api.tokens, quote! { S foo(std::int32_t _i);});
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002856 assert_cc_matches!(result.cc_details.tokens, quote! { S foo(std::int32_t _i) { ... }});
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002857
2858 // Main checks: `CcPrerequisites::includes`.
2859 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002860 format_cc_includes(&main_api.prereqs.includes),
2861 quote! { include <cstdint> }
2862 );
2863 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002864 format_cc_includes(&result.cc_details.prereqs.includes),
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002865 quote! { include <cstdint> }
2866 );
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002867
2868 // Main checks: `CcPrerequisites::defs` and `CcPrerequisites::fwd_decls`.
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002869 //
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002870 // Verifying the actual def_id is tricky, because `test_format_item` doesn't
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002871 // expose `tcx` to the verification function (and therefore calling
2872 // `find_def_id_by_name` is not easily possible).
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08002873 //
2874 // Note that `main_api` and `impl_details` have different expectations.
2875 assert_eq!(0, main_api.prereqs.defs.len());
2876 assert_eq!(1, main_api.prereqs.fwd_decls.len());
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002877 assert_eq!(1, result.cc_details.prereqs.defs.len());
2878 assert_eq!(0, result.cc_details.prereqs.fwd_decls.len());
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002879 });
2880 }
2881
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002882 /// This test verifies that `format_item` uses `CcPrerequisites::fwd_decls`
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002883 /// rather than `CcPrerequisites::defs` for function declarations in the
2884 /// `main_api`.
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002885 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002886 fn test_format_item_fn_cc_prerequisites_if_only_cpp_declaration_needed() {
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002887 let test_src = r#"
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002888 #[no_mangle]
2889 pub extern "C" fn foo(s: S) -> bool { s.0 }
2890
2891 #[repr(C)]
2892 pub struct S(bool);
2893 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002894 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002895 let result = result.unwrap().unwrap();
2896 let main_api = &result.main_api;
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002897
2898 // Minimal coverage, just to double-check that the test setup works.
2899 //
2900 // Note that this is only a function *declaration* (not a function definition -
2901 // there is no function body), and therefore `S` just needs to be
2902 // forward-declared earlier.
Lukasz Anforowicz10b1a292023-04-03 16:19:08 -07002903 assert_cc_matches!(main_api.tokens, quote! { inline bool foo(::rust_out::S s); });
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002904
2905 // Main checks: `CcPrerequisites::defs` and `CcPrerequisites::fwd_decls`.
2906 //
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002907 // Verifying the actual def_id is tricky, because `test_format_item` doesn't
Lukasz Anforowicz7a7b81f2023-01-12 15:50:46 -08002908 // expose `tcx` to the verification function (and therefore calling
2909 // `find_def_id_by_name` is not easily possible).
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002910 assert_eq!(0, main_api.prereqs.defs.len());
2911 assert_eq!(1, main_api.prereqs.fwd_decls.len());
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08002912 });
2913 }
2914
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002915 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002916 fn test_format_item_fn_with_type_aliased_return_type() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002917 // Type aliases disappear at the `rustc_middle::ty::Ty` level and therefore in
2918 // the short-term the generated bindings also ignore type aliases.
2919 //
2920 // TODO(b/254096006): Consider preserving `type` aliases when generating
2921 // bindings.
2922 let test_src = r#"
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07002923 type MyTypeAlias = f64;
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002924
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07002925 #[no_mangle]
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07002926 pub extern "C" fn type_aliased_return() -> MyTypeAlias { 42.0 }
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002927 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002928 test_format_item(test_src, "type_aliased_return", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002929 let result = result.unwrap().unwrap();
2930 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002931 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002932 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002933 main_api.tokens,
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002934 quote! {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07002935 extern "C" double type_aliased_return();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07002936 }
2937 );
2938 });
2939 }
2940
2941 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002942 fn test_format_item_fn_with_doc_comment_with_unmangled_name() {
Googler624580b2022-12-01 01:23:42 -08002943 let test_src = r#"
2944 /// Outer line doc.
2945 /** Outer block doc that spans lines.
2946 */
2947 #[doc = "Doc comment via doc attribute."]
2948 #[no_mangle]
2949 pub extern "C" fn fn_with_doc_comment_with_unmangled_name() {}
2950 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002951 test_format_item(test_src, "fn_with_doc_comment_with_unmangled_name", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002952 let result = result.unwrap().unwrap();
2953 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002954 assert!(main_api.prereqs.is_empty());
Googler624580b2022-12-01 01:23:42 -08002955 let doc_comments = [
2956 " Outer line doc.",
Googler34f3d572022-12-02 00:53:37 -08002957 "",
Googler624580b2022-12-01 01:23:42 -08002958 " Outer block doc that spans lines.",
2959 " ",
Googler34f3d572022-12-02 00:53:37 -08002960 "",
Googler624580b2022-12-01 01:23:42 -08002961 "Doc comment via doc attribute.",
Googler47fd9572023-01-20 09:57:32 -08002962 "",
2963 "Generated from: <crubit_unittests.rs>;l=7",
Googler624580b2022-12-01 01:23:42 -08002964 ]
2965 .join("\n");
2966 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002967 main_api.tokens,
Googler624580b2022-12-01 01:23:42 -08002968 quote! {
2969 __COMMENT__ #doc_comments
2970 extern "C" void fn_with_doc_comment_with_unmangled_name();
2971 }
2972 );
2973 });
2974 }
2975
2976 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002977 fn test_format_item_fn_with_inner_doc_comment_with_unmangled_name() {
Googler624580b2022-12-01 01:23:42 -08002978 let test_src = r#"
2979 /// Outer doc comment.
2980 #[no_mangle]
2981 pub extern "C" fn fn_with_inner_doc_comment_with_unmangled_name() {
2982 //! Inner doc comment.
2983 }
2984 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08002985 test_format_item(test_src, "fn_with_inner_doc_comment_with_unmangled_name", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07002986 let result = result.unwrap().unwrap();
2987 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002988 assert!(main_api.prereqs.is_empty());
Googler47fd9572023-01-20 09:57:32 -08002989 let doc_comments = [
2990 " Outer doc comment.",
2991 " Inner doc comment.",
2992 "Generated from: <crubit_unittests.rs>;l=4",
2993 ]
2994 .join("\n\n");
Googler624580b2022-12-01 01:23:42 -08002995 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08002996 main_api.tokens,
Googler624580b2022-12-01 01:23:42 -08002997 quote! {
2998 __COMMENT__ #doc_comments
2999 extern "C" void fn_with_inner_doc_comment_with_unmangled_name();
3000 }
3001 );
3002 });
3003 }
3004
3005 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003006 fn test_format_item_fn_with_doc_comment_with_mangled_name() {
Googler624580b2022-12-01 01:23:42 -08003007 let test_src = r#"
3008 /// Doc comment of a function with mangled name.
3009 pub extern "C" fn fn_with_doc_comment_with_mangled_name() {}
3010 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003011 test_format_item(test_src, "fn_with_doc_comment_with_mangled_name", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003012 let result = result.unwrap().unwrap();
3013 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003014 assert!(main_api.prereqs.is_empty());
Googler47fd9572023-01-20 09:57:32 -08003015 let comment = " Doc comment of a function with mangled name.\n\n\
3016 Generated from: <crubit_unittests.rs>;l=3";
Googler624580b2022-12-01 01:23:42 -08003017 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003018 main_api.tokens,
Googler624580b2022-12-01 01:23:42 -08003019 quote! {
Googler624580b2022-12-01 01:23:42 -08003020 __COMMENT__ #comment
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003021 inline void fn_with_doc_comment_with_mangled_name();
Googler624580b2022-12-01 01:23:42 -08003022 }
3023 );
3024 });
3025 }
3026
3027 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003028 fn test_format_item_unsupported_fn_name_is_reserved_cpp_keyword() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003029 let test_src = r#"
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07003030 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003031 pub extern "C" fn reinterpret_cast() -> () {}
3032 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003033 test_format_item(test_src, "reinterpret_cast", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003034 let err = result.unwrap_err();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003035 assert_eq!(
3036 err,
3037 "Error formatting function name: \
3038 `reinterpret_cast` is a C++ reserved keyword \
3039 and can't be used as a C++ identifier"
3040 );
3041 });
3042 }
3043
3044 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003045 fn test_format_item_unsupported_fn_ret_type() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003046 let test_src = r#"
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08003047 pub fn foo() -> (i32, i32) { (123, 456) }
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003048 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003049 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003050 let err = result.unwrap_err();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003051 assert_eq!(
3052 err,
3053 "Error formatting function return type: \
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08003054 Tuples are not supported yet: (i32, i32) (b/254099023)"
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003055 );
3056 });
3057 }
3058
3059 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003060 fn test_format_item_unsupported_fn_with_late_bound_lifetimes() {
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003061 // TODO(b/258235219): Expect success after adding support for references.
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003062 let test_src = r#"
3063 pub fn foo(arg: &i32) -> &i32 { arg }
3064
3065 // Lifetime inference translates the above into:
3066 // pub fn foo<'a>(arg: &'a i32) -> &'a i32 { ... }
3067 // leaving 'a lifetime late-bound (it is bound with a lifetime
3068 // taken from each of the callsites). In other words, we can't
3069 // just call `no_bound_vars` on this `FnSig`'s `Binder`.
3070 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003071 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003072 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003073 assert_eq!(err, "Generic functions are not supported yet (b/259749023)");
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003074 });
3075 }
3076
3077 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003078 fn test_format_item_unsupported_generic_fn() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003079 let test_src = r#"
3080 use std::default::Default;
3081 use std::fmt::Display;
3082 pub fn generic_function<T: Default + Display>() {
3083 println!("{}", T::default());
3084 }
3085 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003086 test_format_item(test_src, "generic_function", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003087 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003088 assert_eq!(err, "Generic functions are not supported yet (b/259749023)");
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07003089 });
3090 }
3091
3092 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003093 fn test_format_item_unsupported_generic_struct() {
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003094 let test_src = r#"
3095 pub struct Point<T> {
3096 pub x: T,
3097 pub y: T,
3098 }
3099 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003100 test_format_item(test_src, "Point", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003101 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003102 assert_eq!(err, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003103 });
3104 }
3105
3106 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003107 fn test_format_item_unsupported_generic_enum() {
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003108 let test_src = r#"
3109 pub enum Point<T> {
3110 Cartesian{x: T, y: T},
3111 Polar{angle: T, dist: T},
3112 }
3113 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003114 test_format_item(test_src, "Point", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003115 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003116 assert_eq!(err, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003117 });
3118 }
3119
3120 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003121 fn test_format_item_unsupported_generic_union() {
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003122 let test_src = r#"
3123 pub union SomeUnion<T> {
3124 pub x: std::mem::ManuallyDrop<T>,
3125 pub y: i32,
3126 }
3127 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003128 test_format_item(test_src, "SomeUnion", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003129 let err = result.unwrap_err();
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003130 assert_eq!(err, "Generic types are not supported yet (b/259749095)");
Lukasz Anforowicz27914f52022-11-08 10:55:03 -08003131 });
3132 }
3133
3134 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003135 fn test_format_item_unsupported_fn_async() {
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07003136 let test_src = r#"
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003137 pub async fn async_function() {}
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07003138 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003139 test_format_item(test_src, "async_function", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003140 let err = result.unwrap_err();
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003141 assert_eq!(
3142 err,
3143 "Error formatting function return type: \
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003144 The following Rust type is not supported yet: \
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003145 impl std::future::Future<Output = ()>"
3146 );
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003147 });
3148 }
3149
3150 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003151 fn test_format_item_fn_rust_abi() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003152 let test_src = r#"
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003153 pub fn add(x: f64, y: f64) -> f64 { x * y }
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003154 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003155 test_format_item(test_src, "add", |result| {
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003156 // TODO(b/261074843): Re-add thunk name verification once we are using stable
3157 // name mangling (which may be coming in Q1 2023). (This might mean
3158 // reverting cl/492333432 + manual review and tweaks.)
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003159 let result = result.unwrap().unwrap();
3160 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003161 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003162 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003163 main_api.tokens,
3164 quote! {
3165 inline double add(double x, double y);
3166 }
3167 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003168 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003169 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003170 result.cc_details.tokens,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003171 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003172 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003173 extern "C" double ...(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003174 }
Googler47fd9572023-01-20 09:57:32 -08003175 ...
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003176 inline double add(double x, double y) {
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08003177 return __crubit_internal::...(x, y);
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003178 }
3179 }
3180 );
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003181 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003182 result.rs_details,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003183 quote! {
3184 #[no_mangle]
3185 extern "C"
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08003186 fn ...(x: f64, y: f64) -> f64 {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003187 ::rust_out::add(x, y)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003188 }
3189 }
3190 );
3191 });
3192 }
3193
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003194 #[test]
3195 fn test_format_item_fn_rust_abi_with_param_taking_struct_by_value() {
3196 let test_src = r#"
3197 pub struct S(i32);
3198 pub fn into_i32(s: S) -> i32 { s.0 }
3199 "#;
3200 test_format_item(test_src, "into_i32", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003201 let result = result.unwrap().unwrap();
3202 let main_api = &result.main_api;
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003203 assert_cc_matches!(
3204 main_api.tokens,
3205 quote! {
3206 inline std::int32_t into_i32(::rust_out::S s);
3207 }
3208 );
3209 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003210 result.cc_details.tokens,
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003211 quote! {
3212 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003213 extern "C" std::int32_t ...(::rust_out::S*);
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003214 }
3215 ...
3216 inline std::int32_t into_i32(::rust_out::S s) {
Lukasz Anforowiczd082f352023-03-09 17:46:11 -08003217 return __crubit_internal::...(&s);
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003218 }
3219 }
3220 );
3221 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003222 result.rs_details,
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003223 quote! {
3224 #[no_mangle]
3225 extern "C"
Lukasz Anforowiczd082f352023-03-09 17:46:11 -08003226 fn ...(s: &mut ::core::mem::MaybeUninit<::rust_out::S>) -> i32 {
3227 ::rust_out::into_i32(unsafe { s.assume_init_read() })
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003228 }
3229 }
3230 );
3231 });
3232 }
3233
3234 #[test]
3235 fn test_format_item_fn_rust_abi_returning_struct_by_value() {
3236 let test_src = r#"
3237 pub struct S(i32);
3238 pub fn create(i: i32) -> S { S(i) }
3239 "#;
3240 test_format_item(test_src, "create", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003241 let result = result.unwrap().unwrap();
3242 let main_api = &result.main_api;
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003243 assert_cc_matches!(
3244 main_api.tokens,
3245 quote! {
3246 inline ::rust_out::S create(std::int32_t i);
3247 }
3248 );
3249 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003250 result.cc_details.tokens,
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003251 quote! {
3252 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003253 extern "C" void ...(std::int32_t, ::rust_out::S* __ret_ptr);
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003254 }
3255 ...
3256 inline ::rust_out::S create(std::int32_t i) {
Lukasz Anforowicza3b7db02023-03-09 17:34:05 -08003257 crubit::ReturnValueSlot<::rust_out::S> __ret_slot;
3258 __crubit_internal::...(i, __ret_slot.Get());
3259 return std::move(__ret_slot).AssumeInitAndTakeValue();
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003260 }
3261 }
3262 );
3263 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003264 result.rs_details,
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003265 quote! {
3266 #[no_mangle]
3267 extern "C"
Lukasz Anforowicza3b7db02023-03-09 17:34:05 -08003268 fn ...(
3269 i: i32,
3270 __ret_slot: &mut ::core::mem::MaybeUninit<::rust_out::S>
3271 ) -> () {
3272 __ret_slot.write(::rust_out::create(i));
Lukasz Anforowicz52274992023-03-08 12:29:28 -08003273 }
3274 }
3275 );
3276 });
3277 }
3278
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003279 /// `test_format_item_fn_rust_abi` tests a function call that is not a
3280 /// C-ABI, and is not the default Rust ABI. It can't use `"stdcall"`,
3281 /// because it is not supported on the targets where Crubit's tests run.
3282 /// So, it ended up using `"vectorcall"`.
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003283 ///
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003284 /// This test almost entirely replicates `test_format_item_fn_rust_abi`,
Googler47fd9572023-01-20 09:57:32 -08003285 /// except for the `extern "vectorcall"` part in the `test_src` test
3286 /// input.
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003287 ///
Googler47fd9572023-01-20 09:57:32 -08003288 /// This test verifies the current behavior that gives reasonable and
3289 /// functional FFI bindings. OTOH, in the future we may decide to avoid
3290 /// having the extra thunk for cases where the given non-C-ABI function
3291 /// call convention is supported by both C++ and Rust
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003292 /// (see also `format_cc_call_conv_as_clang_attribute` in
3293 /// `rs_bindings_from_cc/src_code_gen.rs`)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003294 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003295 fn test_format_item_fn_vectorcall_abi() {
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003296 let test_src = r#"
3297 #![feature(abi_vectorcall)]
3298 pub extern "vectorcall" fn add(x: f64, y: f64) -> f64 { x * y }
3299 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003300 test_format_item(test_src, "add", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003301 let result = result.unwrap().unwrap();
3302 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003303 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003304 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003305 main_api.tokens,
3306 quote! {
3307 inline double add(double x, double y);
3308 }
3309 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003310 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003311 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003312 result.cc_details.tokens,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003313 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003314 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003315 extern "C" double ...(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003316 }
Googler47fd9572023-01-20 09:57:32 -08003317 ...
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003318 inline double add(double x, double y) {
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08003319 return __crubit_internal::...(x, y);
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003320 }
3321 }
3322 );
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003323 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003324 result.rs_details,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003325 quote! {
3326 #[no_mangle]
3327 extern "C"
Lukasz Anforowiczb4beb392022-12-01 16:49:11 -08003328 fn ...(x: f64, y: f64) -> f64 {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003329 ::rust_out::add(x, y)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003330 }
3331 }
3332 );
3333 });
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003334 }
3335
3336 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003337 fn test_format_item_unsupported_fn_variadic() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003338 let test_src = r#"
3339 #![feature(c_variadic)]
Lukasz Anforowicz7123f212022-10-20 08:51:04 -07003340
3341 #[no_mangle]
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003342 pub unsafe extern "C" fn variadic_function(_fmt: *const u8, ...) {}
3343 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003344 test_format_item(test_src, "variadic_function", |result| {
Lukasz Anforowicz13794df2022-10-21 07:56:34 -07003345 // TODO(b/254097223): Add support for variadic functions.
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003346 let err = result.unwrap_err();
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003347 assert_eq!(err, "C variadic functions are not supported (b/254097223)");
3348 });
3349 }
3350
3351 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003352 fn test_format_item_fn_params() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003353 let test_src = r#"
3354 #[allow(unused_variables)]
3355 #[no_mangle]
3356 pub extern "C" fn foo(b: bool, f: f64) {}
3357 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003358 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003359 let result = result.unwrap().unwrap();
3360 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003361 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003362 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003363 main_api.tokens,
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003364 quote! {
Googler47fd9572023-01-20 09:57:32 -08003365 ...
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003366 extern "C" void foo(bool b, double f);
3367 }
3368 );
3369 });
3370 }
3371
3372 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003373 fn test_format_item_fn_param_name_reserved_keyword() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003374 let test_src = r#"
3375 #[allow(unused_variables)]
3376 #[no_mangle]
3377 pub extern "C" fn some_function(reinterpret_cast: f64) {}
3378 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003379 test_format_item(test_src, "some_function", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003380 let result = result.unwrap().unwrap();
3381 let main_api = &result.main_api;
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003382 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003383 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003384 main_api.tokens,
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003385 quote! {
Googler47fd9572023-01-20 09:57:32 -08003386 ...
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003387 extern "C" void some_function(double __param_0);
3388 }
3389 );
3390 });
3391 }
3392
3393 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003394 fn test_format_item_fn_with_multiple_anonymous_parameter_names() {
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003395 let test_src = r#"
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003396 pub fn foo(_: f64, _: f64) {}
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003397 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003398 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003399 let result = result.unwrap().unwrap();
3400 let main_api = &result.main_api;
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003401 assert!(main_api.prereqs.is_empty());
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003402 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003403 main_api.tokens,
3404 quote! {
3405 inline void foo(double __param_0, double __param_1);
3406 }
3407 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003408 assert!(result.cc_details.prereqs.is_empty());
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003409 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003410 result.cc_details.tokens,
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003411 quote! {
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003412 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003413 extern "C" void ...(double, double);
Lukasz Anforowiczbad99632022-11-18 16:39:13 -08003414 }
Googler47fd9572023-01-20 09:57:32 -08003415 ...
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003416 inline void foo(double __param_0, double __param_1) {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003417 return __crubit_internal::...(__param_0, __param_1);
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003418 }
3419 }
3420 );
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003421 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003422 result.rs_details,
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003423 quote! {
3424 #[no_mangle]
3425 extern "C" fn ...(__param_0: f64, __param_1: f64) -> () {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003426 ::rust_out::foo(__param_0, __param_1)
Lukasz Anforowicz2f849cf2022-11-17 15:52:39 -08003427 }
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003428 }
3429 );
3430 });
3431 }
3432
Lukasz Anforowiczc51aeb12022-11-07 10:56:18 -08003433 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003434 fn test_format_item_fn_with_destructuring_parameter_name() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003435 let test_src = r#"
3436 pub struct S {
3437 pub f1: i32,
3438 pub f2: i32,
3439 }
3440
3441 // This test mostly focuses on the weird parameter "name" below.
3442 // See also
3443 // https://doc.rust-lang.org/reference/items/functions.html#function-parameters
3444 // which points out that function parameters are just irrefutable patterns.
3445 pub fn func(S{f1, f2}: S) -> i32 { f1 + f2 }
3446 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003447 test_format_item(test_src, "func", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003448 let result = result.unwrap().unwrap();
3449 let main_api = &result.main_api;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003450 assert_cc_matches!(
Lukasz Anforowicz6753d402023-02-10 16:41:23 -08003451 main_api.tokens,
3452 quote! {
3453 inline std::int32_t func(::rust_out::S __param_0);
3454 }
3455 );
3456 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003457 result.cc_details.tokens,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003458 quote! {
3459 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003460 extern "C" std::int32_t ...(::rust_out::S*);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003461 }
Googler47fd9572023-01-20 09:57:32 -08003462 ...
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003463 inline std::int32_t func(::rust_out::S __param_0) {
Lukasz Anforowiczd082f352023-03-09 17:46:11 -08003464 return __crubit_internal::...(&__param_0);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003465 }
3466 }
3467 );
3468 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003469 result.rs_details,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003470 quote! {
3471 #[no_mangle]
Lukasz Anforowiczd082f352023-03-09 17:46:11 -08003472 extern "C" fn ...(
3473 __param_0: &mut ::core::mem::MaybeUninit<::rust_out::S>
3474 ) -> i32 {
3475 ::rust_out::func(unsafe {__param_0.assume_init_read() })
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003476 }
3477 }
3478 );
3479 });
3480 }
3481
3482 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003483 fn test_format_item_unsupported_fn_param_type() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003484 let test_src = r#"
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08003485 pub fn foo(_param: (i32, i32)) {}
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003486 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003487 test_format_item(test_src, "foo", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003488 let err = result.unwrap_err();
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003489 assert_eq!(
3490 err,
3491 "Error handling parameter #0: \
3492 Tuples are not supported yet: (i32, i32) (b/254099023)"
3493 );
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003494 });
3495 }
3496
3497 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003498 fn test_format_item_unsupported_fn_param_type_unit() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003499 let test_src = r#"
3500 #[no_mangle]
3501 pub fn fn_with_params(_param: ()) {}
3502 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003503 test_format_item(test_src, "fn_with_params", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003504 let err = result.unwrap_err();
Devin Jeanpierre81aec502023-04-04 15:33:39 -07003505 assert_eq!(
3506 err,
3507 "Error handling parameter #0: \
3508 `()` / `void` is only supported as a return type (b/254507801)"
3509 );
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003510 });
3511 }
3512
3513 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003514 fn test_format_item_unsupported_fn_param_type_never() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003515 let test_src = r#"
3516 #![feature(never_type)]
3517
3518 #[no_mangle]
3519 pub extern "C" fn fn_with_params(_param: !) {}
3520 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003521 test_format_item(test_src, "fn_with_params", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08003522 let err = result.unwrap_err();
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003523 assert_eq!(
3524 err,
Lukasz Anforowicza691cf52023-03-08 12:24:33 -08003525 "Error handling parameter #0: \
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07003526 The never type `!` is only supported as a return type (b/254507801)"
3527 );
Lukasz Anforowicze4333062022-10-17 14:47:53 -07003528 });
3529 }
3530
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003531 /// This is a test for a regular struct - a struct with named fields.
3532 /// https://doc.rust-lang.org/reference/items/structs.html refers to this kind of struct as
3533 /// `StructStruct` or "nominal struct type".
3534 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003535 fn test_format_item_struct_with_fields() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003536 let test_src = r#"
3537 pub struct SomeStruct {
3538 pub x: i32,
3539 pub y: i32,
3540 }
3541
3542 const _: () = assert!(std::mem::size_of::<SomeStruct>() == 8);
3543 const _: () = assert!(std::mem::align_of::<SomeStruct>() == 4);
3544 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003545 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003546 let result = result.unwrap().unwrap();
3547 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003548 assert!(!main_api.prereqs.is_empty());
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003549 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003550 main_api.tokens,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003551 quote! {
Googler47fd9572023-01-20 09:57:32 -08003552 ...
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003553 struct alignas(4) SomeStruct final {
3554 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07003555 __COMMENT__ "`SomeStruct` doesn't implement the `Default` trait"
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003556 SomeStruct() = delete;
3557
3558 // In this test there is no `Copy` implementation / derive.
3559 SomeStruct(const SomeStruct&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003560
3561 // All Rust types are trivially-movable.
3562 SomeStruct(SomeStruct&&) = default;
Lukasz Anforowicz554ed652023-01-12 15:41:58 -08003563
3564 // Assignment operators are disabled for now.
3565 SomeStruct& operator=(const SomeStruct&) = delete;
3566 SomeStruct& operator=(SomeStruct&&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003567
3568 // In this test there is no custom `Drop`, so C++ can also
3569 // just use the `default` destructor.
3570 ~SomeStruct() = default;
3571 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003572 ... std::int32_t x;
3573 ... std::int32_t y;
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003574 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003575 };
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003576 }
3577 );
3578 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003579 result.cc_details.tokens,
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003580 quote! {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003581 static_assert(sizeof(SomeStruct) == 8, ...);
3582 static_assert(alignof(SomeStruct) == 4, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003583 inline void SomeStruct::__crubit_field_offset_assertions() {
3584 static_assert(0 == offsetof(SomeStruct, x));
3585 static_assert(4 == offsetof(SomeStruct, y));
3586 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003587 }
3588 );
3589 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003590 result.rs_details,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003591 quote! {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003592 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8);
3593 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003594 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, x) == 0);
3595 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, y) == 4);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003596 }
3597 );
3598 });
3599 }
3600
3601 /// This is a test for `TupleStruct` or "tuple struct" - for more details
3602 /// please refer to https://doc.rust-lang.org/reference/items/structs.html
3603 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003604 fn test_format_item_struct_with_tuple() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003605 let test_src = r#"
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003606 pub struct TupleStruct(pub i32, pub i32);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003607 const _: () = assert!(std::mem::size_of::<TupleStruct>() == 8);
3608 const _: () = assert!(std::mem::align_of::<TupleStruct>() == 4);
3609 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08003610 test_format_item(test_src, "TupleStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003611 let result = result.unwrap().unwrap();
3612 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003613 assert!(!main_api.prereqs.is_empty());
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003614 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003615 main_api.tokens,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003616 quote! {
Googler47fd9572023-01-20 09:57:32 -08003617 ...
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003618 struct alignas(4) TupleStruct final {
3619 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07003620 __COMMENT__ "`TupleStruct` doesn't implement the `Default` trait"
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003621 TupleStruct() = delete;
3622
3623 // In this test there is no `Copy` implementation / derive.
3624 TupleStruct(const TupleStruct&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003625
3626 // All Rust types are trivially-movable.
3627 TupleStruct(TupleStruct&&) = default;
Lukasz Anforowicz554ed652023-01-12 15:41:58 -08003628
3629 // Assignment operators are disabled for now.
3630 TupleStruct& operator=(const TupleStruct&) = delete;
3631 TupleStruct& operator=(TupleStruct&&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003632
3633 // In this test there is no custom `Drop`, so C++ can also
3634 // just use the `default` destructor.
3635 ~TupleStruct() = default;
3636 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003637 ... std::int32_t __field0;
3638 ... std::int32_t __field1;
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003639 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003640 };
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003641 }
3642 );
3643 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003644 result.cc_details.tokens,
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08003645 quote! {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003646 static_assert(sizeof(TupleStruct) == 8, ...);
3647 static_assert(alignof(TupleStruct) == 4, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003648 inline void TupleStruct::__crubit_field_offset_assertions() {
3649 static_assert(0 == offsetof(TupleStruct, __field0));
3650 static_assert(4 == offsetof(TupleStruct, __field1));
3651 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003652 }
3653 );
3654 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003655 result.rs_details,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003656 quote! {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08003657 const _: () = assert!(::std::mem::size_of::<::rust_out::TupleStruct>() == 8);
3658 const _: () = assert!(::std::mem::align_of::<::rust_out::TupleStruct>() == 4);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003659 const _: () = assert!( memoffset::offset_of!(::rust_out::TupleStruct, 0) == 0);
3660 const _: () = assert!( memoffset::offset_of!(::rust_out::TupleStruct, 1) == 4);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003661 }
3662 );
3663 });
3664 }
3665
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003666 /// This test the scenario where Rust lays out field in a different order
3667 /// than the source order.
3668 #[test]
3669 fn test_format_item_struct_with_reordered_field_offsets() {
3670 let test_src = r#"
3671 pub struct SomeStruct {
3672 pub field1: i16,
3673 pub field2: i32,
3674 pub field3: i16,
3675 }
3676
3677 const _: () = assert!(std::mem::size_of::<SomeStruct>() == 8);
3678 const _: () = assert!(std::mem::align_of::<SomeStruct>() == 4);
3679 "#;
3680 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003681 let result = result.unwrap().unwrap();
3682 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003683 assert!(!main_api.prereqs.is_empty());
3684 assert_cc_matches!(
3685 main_api.tokens,
3686 quote! {
3687 ...
3688 struct alignas(4) SomeStruct final {
3689 ...
3690 private:
3691 // The particular order below is not guaranteed,
3692 // so we may need to adjust this test assertion
3693 // (if Rust changes how it lays out the fields).
3694 ... std::int32_t field2;
3695 ... std::int16_t field1;
3696 ... std::int16_t field3;
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003697 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003698 };
3699 }
3700 );
3701 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003702 result.cc_details.tokens,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003703 quote! {
3704 static_assert(sizeof(SomeStruct) == 8, ...);
3705 static_assert(alignof(SomeStruct) == 4, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003706 inline void SomeStruct::__crubit_field_offset_assertions() {
3707 static_assert(0 == offsetof(SomeStruct, field2));
3708 static_assert(4 == offsetof(SomeStruct, field1));
3709 static_assert(6 == offsetof(SomeStruct, field3));
3710 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003711 }
3712 );
3713 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003714 result.rs_details,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003715 quote! {
3716 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8);
3717 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003718 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field2)
3719 == 0);
3720 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field1)
3721 == 4);
3722 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field3)
3723 == 6);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003724 }
3725 );
3726 });
3727 }
3728
3729 #[test]
3730 fn test_format_item_struct_with_packed_layout() {
3731 let test_src = r#"
3732 #[repr(packed(1))]
3733 pub struct SomeStruct {
3734 pub field1: u16,
3735 pub field2: u32,
3736 }
3737 const _: () = assert!(::std::mem::size_of::<SomeStruct>() == 6);
3738 const _: () = assert!(::std::mem::align_of::<SomeStruct>() == 1);
3739 "#;
3740 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003741 let result = result.unwrap().unwrap();
3742 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003743 assert!(!main_api.prereqs.is_empty());
3744 assert_cc_matches!(
3745 main_api.tokens,
3746 quote! {
3747 ...
3748 struct alignas(1) __attribute__((packed)) SomeStruct final {
3749 ...
3750 std::uint16_t field1;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003751 std::uint32_t field2;
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003752 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003753 };
3754 }
3755 );
3756 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003757 result.cc_details.tokens,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003758 quote! {
3759 static_assert(sizeof(SomeStruct) == 6, ...);
3760 static_assert(alignof(SomeStruct) == 1, ...);
Lukasz Anforowicze3eb1cf2023-03-14 09:56:00 -07003761 inline void SomeStruct::__crubit_field_offset_assertions() {
3762 static_assert(0 == offsetof(SomeStruct, field1));
3763 static_assert(2 == offsetof(SomeStruct, field2));
3764 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003765 }
3766 );
3767 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003768 result.rs_details,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003769 quote! {
3770 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 6);
3771 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 1);
Lukasz Anforowiczcf60f522023-03-14 10:03:55 -07003772 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field1)
3773 == 0);
3774 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field2)
3775 == 2);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07003776 }
3777 );
3778 });
3779 }
3780
Lukasz Anforowicz75894832022-11-22 18:25:28 -08003781 #[test]
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07003782 fn test_format_item_struct_with_explicit_padding_in_generated_code() {
3783 let test_src = r#"
3784 pub struct SomeStruct {
3785 pub f1: u8,
3786 pub f2: u32,
3787 }
3788 const _: () = assert!(::std::mem::size_of::<SomeStruct>() == 8);
3789 const _: () = assert!(::std::mem::align_of::<SomeStruct>() == 4);
3790 "#;
3791 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003792 let result = result.unwrap().unwrap();
3793 let main_api = &result.main_api;
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07003794 assert!(!main_api.prereqs.is_empty());
3795 assert_cc_matches!(
3796 main_api.tokens,
3797 quote! {
3798 ...
3799 struct alignas(4) SomeStruct final {
3800 ...
3801 std::uint32_t f2;
3802 std::uint8_t f1;
3803 unsigned char __padding0[3];
3804 inline static void __crubit_field_offset_assertions();
3805 };
3806 }
3807 );
3808 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003809 result.cc_details.tokens,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07003810 quote! {
3811 static_assert(sizeof(SomeStruct) == 8, ...);
3812 static_assert(alignof(SomeStruct) == 4, ...);
3813 inline void SomeStruct::__crubit_field_offset_assertions() {
3814 static_assert(0 == offsetof(SomeStruct, f2));
3815 static_assert(4 == offsetof(SomeStruct, f1));
3816 }
3817 }
3818 );
3819 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003820 result.rs_details,
Lukasz Anforowicz735bfb82023-03-15 16:18:44 -07003821 quote! {
3822 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8);
3823 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
3824 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, f2) == 0);
3825 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, f1) == 4);
3826 }
3827 );
3828 });
3829 }
3830
3831 #[test]
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003832 fn test_format_item_static_method() {
3833 let test_src = r#"
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003834 /// No-op `f32` placeholder is used, because ZSTs are not supported
3835 /// (b/258259459).
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003836 pub struct Math(f32);
3837
3838 impl Math {
3839 pub fn add_i32(x: f32, y: f32) -> f32 {
3840 x + y
3841 }
3842 }
3843 "#;
3844 test_format_item(test_src, "Math", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003845 let result = result.unwrap().unwrap();
3846 let main_api = &result.main_api;
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003847 assert!(main_api.prereqs.is_empty());
3848 assert_cc_matches!(
3849 main_api.tokens,
3850 quote! {
3851 ...
3852 struct ... Math final {
3853 ...
3854 public:
3855 ...
3856 static inline float add_i32(float x, float y);
3857 ...
3858 };
3859 }
3860 );
3861 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003862 result.cc_details.tokens,
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003863 quote! {
3864 namespace __crubit_internal {
Lukasz Anforowicz73edf152023-04-04 12:05:00 -07003865 extern "C" float ... (float, float);
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003866 }
3867 inline float Math::add_i32(float x, float y) {
3868 return __crubit_internal::...(x, y);
3869 }
3870 }
3871 );
3872 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003873 result.rs_details,
Lukasz Anforowicz43f06c62023-02-10 17:14:19 -08003874 quote! {
3875 #[no_mangle]
3876 extern "C" fn ...(x: f32, y: f32) -> f32 {
3877 ::rust_out::Math::add_i32(x, y)
3878 }
3879 }
3880 );
3881 });
3882 }
3883
3884 #[test]
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003885 fn test_format_item_static_method_with_generic_type_parameters() {
3886 let test_src = r#"
3887 /// No-op `f32` placeholder is used, because ZSTs are not supported
3888 /// (b/258259459).
3889 pub struct SomeStruct(f32);
3890
3891 impl SomeStruct {
3892 // To make this testcase distinct / non-overlapping wrt
3893 // test_format_item_static_method_with_generic_lifetime_parameters
3894 // `t` is taken by value below.
3895 pub fn generic_method<T: Clone>(t: T) -> T {
3896 t.clone()
3897 }
3898 }
3899 "#;
3900 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003901 let result = result.unwrap().unwrap();
3902 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003903 assert!(main_api.prereqs.is_empty());
3904 let unsupported_msg = "Error generating bindings for `SomeStruct::generic_method` \
3905 defined at <crubit_unittests.rs>;l=10: \
3906 Generic functions are not supported yet (b/259749023)";
3907 assert_cc_matches!(
3908 main_api.tokens,
3909 quote! {
3910 ...
3911 struct ... SomeStruct final {
3912 ...
3913 __COMMENT__ #unsupported_msg
3914 ...
3915 };
3916 ...
3917 }
3918 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003919 assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::generic_method },);
3920 assert_rs_not_matches!(result.rs_details, quote! { generic_method },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003921 });
3922 }
3923
3924 #[test]
3925 fn test_format_item_static_method_with_generic_lifetime_parameters() {
3926 let test_src = r#"
3927 /// No-op `f32` placeholder is used, because ZSTs are not supported
3928 /// (b/258259459).
3929 pub struct SomeStruct(f32);
3930
3931 impl SomeStruct {
3932 pub fn fn_taking_reference<'a>(x: &'a i32) -> i32 { *x }
3933 }
3934 "#;
3935 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003936 let result = result.unwrap().unwrap();
3937 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003938 assert!(main_api.prereqs.is_empty());
3939 let unsupported_msg = "Error generating bindings for `SomeStruct::fn_taking_reference` \
3940 defined at <crubit_unittests.rs>;l=7: \
3941 Generic functions are not supported yet (b/259749023)";
3942 assert_cc_matches!(
3943 main_api.tokens,
3944 quote! {
3945 ...
3946 struct ... SomeStruct final {
3947 ...
3948 __COMMENT__ #unsupported_msg
3949 ...
3950 };
3951 ...
3952 }
3953 );
3954 assert_cc_not_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003955 result.cc_details.tokens,
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003956 quote! { SomeStruct::fn_taking_reference },
3957 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003958 assert_rs_not_matches!(result.rs_details, quote! { fn_taking_reference },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003959 });
3960 }
3961
3962 #[test]
3963 fn test_format_item_method_taking_self_by_value() {
3964 let test_src = r#"
3965 pub struct SomeStruct(f32);
3966
3967 impl SomeStruct {
3968 pub fn into_f32(self) -> f32 {
3969 self.0
3970 }
3971 }
3972 "#;
3973 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003974 let result = result.unwrap().unwrap();
3975 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003976 assert!(main_api.prereqs.is_empty());
3977 let unsupported_msg = "Error generating bindings for `SomeStruct::into_f32` \
3978 defined at <crubit_unittests.rs>;l=5: \
3979 `self` parameter is not supported yet";
3980 assert_cc_matches!(
3981 main_api.tokens,
3982 quote! {
3983 ...
3984 struct ... SomeStruct final {
3985 ...
3986 __COMMENT__ #unsupported_msg
3987 ...
3988 };
3989 ...
3990 }
3991 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07003992 assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::into_f32 },);
3993 assert_rs_not_matches!(result.rs_details, quote! { into_f32 },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08003994 });
3995 }
3996
3997 #[test]
3998 fn test_format_item_method_taking_self_by_const_ref() {
3999 let test_src = r#"
4000 pub struct SomeStruct(f32);
4001
4002 impl SomeStruct {
4003 pub fn get_f32(&self) -> f32 {
4004 self.0
4005 }
4006 }
4007 "#;
4008 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004009 let result = result.unwrap().unwrap();
4010 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004011 assert!(main_api.prereqs.is_empty());
4012 let unsupported_msg = "Error generating bindings for `SomeStruct::get_f32` \
4013 defined at <crubit_unittests.rs>;l=5: \
4014 Generic functions are not supported yet (b/259749023)";
4015 assert_cc_matches!(
4016 main_api.tokens,
4017 quote! {
4018 ...
4019 struct ... SomeStruct final {
4020 ...
4021 __COMMENT__ #unsupported_msg
4022 ...
4023 };
4024 ...
4025 }
4026 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004027 assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::get_f32 },);
4028 assert_rs_not_matches!(result.rs_details, quote! { get_f32 },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004029 });
4030 }
4031
4032 #[test]
4033 fn test_format_item_method_taking_self_by_mutable_ref() {
4034 let test_src = r#"
4035 pub struct SomeStruct(f32);
4036
4037 impl SomeStruct {
4038 pub fn set_f32(&mut self, new_value: f32) {
4039 self.0 = new_value;
4040 }
4041 }
4042 "#;
4043 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004044 let result = result.unwrap().unwrap();
4045 let main_api = &result.main_api;
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004046 assert!(main_api.prereqs.is_empty());
4047 let unsupported_msg = "Error generating bindings for `SomeStruct::set_f32` \
4048 defined at <crubit_unittests.rs>;l=5: \
4049 Generic functions are not supported yet (b/259749023)";
4050 assert_cc_matches!(
4051 main_api.tokens,
4052 quote! {
4053 ...
4054 struct ... SomeStruct final {
4055 ...
4056 __COMMENT__ #unsupported_msg
4057 ...
4058 };
4059 ...
4060 }
4061 );
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004062 assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::set_f32 },);
4063 assert_rs_not_matches!(result.rs_details, quote! { set_f32 },);
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004064 });
4065 }
4066
4067 #[test]
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07004068 fn test_format_item_struct_with_default_constructor() {
4069 let test_src = r#"
4070 #[derive(Default)]
4071 pub struct Point(i32, i32);
4072 "#;
4073 test_format_item(test_src, "Point", |result| {
4074 let result = result.unwrap().unwrap();
4075 let main_api = &result.main_api;
4076 assert_cc_matches!(
4077 main_api.tokens,
4078 quote! {
4079 ...
4080 struct ... Point final {
4081 ...
4082 public:
4083 __COMMENT__ "Default::default"
4084 inline Point();
4085 ...
4086 };
4087 }
4088 );
4089 assert_cc_matches!(
4090 result.cc_details.tokens,
4091 quote! {
4092 namespace __crubit_internal {
4093 extern "C" void ...(::rust_out::Point* __ret_ptr);
4094 }
4095 Point::Point() {
4096 ...(this);
4097 }
4098 }
4099 );
4100 assert_rs_matches!(
4101 result.rs_details,
4102 quote! {
4103 #[no_mangle]
4104 extern "C" fn ...(
4105 __ret_slot: &mut ::core::mem::MaybeUninit<::rust_out::Point>
4106 ) -> () {
4107 __ret_slot.write(<::rust_out::Point as ::core::default::Default>::default());
4108 }
4109 }
4110 );
4111 });
4112 }
4113
4114 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004115 fn test_format_item_unsupported_struct_with_name_that_is_reserved_keyword() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004116 let test_src = r#"
4117 #[allow(non_camel_case_types)]
4118 pub struct reinterpret_cast {
4119 pub x: i32,
4120 pub y: i32,
4121 }
4122 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004123 test_format_item(test_src, "reinterpret_cast", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004124 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004125 assert_eq!(
4126 err,
4127 "Error formatting item name: \
4128 `reinterpret_cast` is a C++ reserved keyword \
4129 and can't be used as a C++ identifier"
4130 );
4131 });
4132 }
4133
4134 #[test]
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004135 fn test_format_item_struct_with_unsupported_field_type() {
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004136 let test_src = r#"
4137 pub struct SomeStruct {
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004138 pub successful_field: i32,
4139 pub unsupported_field: Option<[i32; 3]>,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004140 }
4141 "#;
4142 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004143 let result = result.unwrap().unwrap();
4144 let main_api = &result.main_api;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004145 let broken_field_msg = "Field type has been replaced with a blob of bytes: \
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004146 Generic types are not supported yet (b/259749095)";
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004147 assert_cc_matches!(
4148 main_api.tokens,
4149 quote! {
4150 ...
4151 struct ... SomeStruct final {
4152 ...
4153 private:
4154 __COMMENT__ #broken_field_msg
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004155 unsigned char unsupported_field[16];
4156 std::int32_t successful_field;
4157 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004158 };
4159 ...
4160 }
4161 );
4162 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004163 result.cc_details.tokens,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004164 quote! {
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004165 static_assert(sizeof(SomeStruct) == 20, ...);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004166 static_assert(alignof(SomeStruct) == 4, ...);
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004167 inline void SomeStruct::__crubit_field_offset_assertions() {
4168 static_assert(0 == offsetof(SomeStruct, unsupported_field));
4169 static_assert(16 == offsetof(SomeStruct, successful_field));
4170 }
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004171 }
4172 );
4173 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004174 result.rs_details,
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004175 quote! {
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004176 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 20);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004177 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004178 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4179 unsupported_field) == 0);
4180 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4181 successful_field) == 16);
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004182 }
4183 );
4184 });
4185 }
4186
4187 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004188 fn test_format_item_unsupported_struct_with_custom_drop_impl() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004189 let test_src = r#"
4190 pub struct StructWithCustomDropImpl {
4191 pub x: i32,
4192 pub y: i32,
4193 }
4194
4195 impl Drop for StructWithCustomDropImpl {
4196 fn drop(&mut self) {}
4197 }
4198 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004199 test_format_item(test_src, "StructWithCustomDropImpl", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004200 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004201 assert_eq!(err, "`Drop` trait and \"drop glue\" are not supported yet (b/258251148)");
4202 });
4203 }
4204
4205 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004206 fn test_format_item_unsupported_struct_with_custom_drop_glue() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004207 let test_src = r#"
4208 #![allow(dead_code)]
4209
4210 // `i32` is present to avoid hitting the ZST checks related to (b/258259459)
4211 struct StructWithCustomDropImpl(i32);
4212
4213 impl Drop for StructWithCustomDropImpl {
4214 fn drop(&mut self) {
4215 println!("dropping!");
4216 }
4217 }
4218
4219 pub struct StructRequiringCustomDropGlue {
4220 field: StructWithCustomDropImpl,
4221 }
4222 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004223 test_format_item(test_src, "StructRequiringCustomDropGlue", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004224 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004225 assert_eq!(err, "`Drop` trait and \"drop glue\" are not supported yet (b/258251148)");
4226 });
4227 }
4228
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004229 /// This test covers how ZSTs (zero-sized-types) are handled.
4230 /// https://doc.rust-lang.org/reference/items/structs.html refers to this kind of struct as a
4231 /// "unit-like struct".
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004232 #[test]
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004233 fn test_format_item_unsupported_struct_zero_sized_type_with_no_fields() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004234 let test_src = r#"
4235 pub struct ZeroSizedType1;
4236 pub struct ZeroSizedType2();
4237 pub struct ZeroSizedType3{}
4238 "#;
4239 for name in ["ZeroSizedType1", "ZeroSizedType2", "ZeroSizedType3"] {
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004240 test_format_item(test_src, name, |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004241 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004242 assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)");
4243 });
4244 }
4245 }
4246
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004247 #[test]
4248 fn test_format_item_unsupported_struct_with_only_zero_sized_type_fields() {
4249 let test_src = r#"
4250 pub struct ZeroSizedType;
4251 pub struct SomeStruct {
4252 pub zst1: ZeroSizedType,
4253 pub zst2: ZeroSizedType,
4254 }
4255 "#;
4256 test_format_item(test_src, "SomeStruct", |result| {
4257 let err = result.unwrap_err();
4258 assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)",);
4259 });
4260 }
4261
Devin Jeanpierre9e15d0b2023-04-06 13:18:22 -07004262 #[test]
4263 fn test_format_item_unsupported_struct_with_some_zero_sized_type_fields() {
4264 let test_src = r#"
4265 pub struct ZeroSizedType;
4266 pub struct SomeStruct {
4267 pub zst1: ZeroSizedType,
4268 pub zst2: ZeroSizedType,
4269 pub successful_field: i32,
4270 }
4271 "#;
4272 test_format_item(test_src, "SomeStruct", |result| {
4273 let result = result.unwrap().unwrap();
4274 let main_api = &result.main_api;
4275 let broken_field_msg = "Field type has been replaced with a blob of bytes: \
4276 Failed to generate bindings for the definition of `ZeroSizedType`: \
4277 Zero-sized types (ZSTs) are not supported (b/258259459)";
4278 assert_cc_matches!(
4279 main_api.tokens,
4280 quote! {
4281 ...
4282 struct ... SomeStruct final {
4283 ...
4284 private:
4285 __COMMENT__ #broken_field_msg
4286 [[no_unique_address]] struct{} zst1;
4287 __COMMENT__ #broken_field_msg
4288 [[no_unique_address]] struct{} zst2;
4289 std::int32_t successful_field;
4290 inline static void __crubit_field_offset_assertions();
4291 };
4292 ...
4293 }
4294 );
4295 assert_cc_matches!(
4296 result.cc_details.tokens,
4297 quote! {
4298 static_assert(sizeof(SomeStruct) == 4, ...);
4299 static_assert(alignof(SomeStruct) == 4, ...);
4300 inline void SomeStruct::__crubit_field_offset_assertions() {
4301 static_assert(0 == offsetof(SomeStruct, zst1));
4302 static_assert(0 == offsetof(SomeStruct, zst2));
4303 static_assert(0 == offsetof(SomeStruct, successful_field));
4304 }
4305 }
4306 );
4307 assert_rs_matches!(
4308 result.rs_details,
4309 quote! {
4310 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 4);
4311 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4);
4312 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4313 zst1) == 0);
4314 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4315 zst2) == 0);
4316 const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct,
4317 successful_field) == 0);
4318 }
4319 );
4320 });
4321 }
4322
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004323 /// This is a test for an enum that only has `EnumItemDiscriminant` items
4324 /// (and doesn't have `EnumItemTuple` or `EnumItemStruct` items). See
4325 /// also https://doc.rust-lang.org/reference/items/enumerations.html
4326 #[test]
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004327 fn test_format_item_enum_with_only_discriminant_items() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004328 let test_src = r#"
4329 pub enum SomeEnum {
4330 Red,
4331 Green = 123,
4332 Blue,
4333 }
4334
4335 const _: () = assert!(std::mem::size_of::<SomeEnum>() == 1);
4336 const _: () = assert!(std::mem::align_of::<SomeEnum>() == 1);
4337 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004338 test_format_item(test_src, "SomeEnum", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004339 let result = result.unwrap().unwrap();
4340 let main_api = &result.main_api;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004341 let no_fields_msg = "Field type has been replaced with a blob of bytes: \
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004342 No support for bindings of individual fields of \
4343 `union` (b/272801632) or `enum`";
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004344 assert!(main_api.prereqs.is_empty());
4345 assert_cc_matches!(
4346 main_api.tokens,
4347 quote! {
4348 ...
4349 struct alignas(1) SomeEnum final {
4350 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07004351 __COMMENT__ "`SomeEnum` doesn't implement the `Default` trait"
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004352 SomeEnum() = delete;
4353
4354 // In this test there is no `Copy` implementation / derive.
4355 SomeEnum(const SomeEnum&) = delete;
4356
4357 // All Rust types are trivially-movable.
4358 SomeEnum(SomeEnum&&) = default;
4359
4360 // Assignment operators are disabled for now.
4361 SomeEnum& operator=(const SomeEnum&) = delete;
4362 SomeEnum& operator=(SomeEnum&&) = delete;
4363
4364 // In this test there is no custom `Drop`, so C++ can also
4365 // just use the `default` destructor.
4366 ~SomeEnum() = default;
4367 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004368 __COMMENT__ #no_fields_msg
4369 unsigned char __opaque_blob_of_bytes[1];
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004370 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004371 };
4372 }
4373 );
4374 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004375 result.cc_details.tokens,
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004376 quote! {
4377 static_assert(sizeof(SomeEnum) == 1, ...);
4378 static_assert(alignof(SomeEnum) == 1, ...);
4379 }
4380 );
4381 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004382 result.rs_details,
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004383 quote! {
4384 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeEnum>() == 1);
4385 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeEnum>() == 1);
4386 }
4387 );
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004388 });
4389 }
4390
4391 /// This is a test for an enum that has `EnumItemTuple` and `EnumItemStruct`
4392 /// items. See also https://doc.rust-lang.org/reference/items/enumerations.html
4393 #[test]
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004394 fn test_format_item_enum_with_tuple_and_struct_items() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004395 let test_src = r#"
4396 pub enum Point {
4397 Cartesian(f32, f32),
4398 Polar{ dist: f32, angle: f32 },
4399 }
4400
4401 const _: () = assert!(std::mem::size_of::<Point>() == 12);
4402 const _: () = assert!(std::mem::align_of::<Point>() == 4);
4403 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004404 test_format_item(test_src, "Point", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004405 let result = result.unwrap().unwrap();
4406 let main_api = &result.main_api;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004407 let no_fields_msg = "Field type has been replaced with a blob of bytes: \
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004408 No support for bindings of individual fields of \
4409 `union` (b/272801632) or `enum`";
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004410 assert!(main_api.prereqs.is_empty());
4411 assert_cc_matches!(
4412 main_api.tokens,
4413 quote! {
4414 ...
4415 struct alignas(4) Point final {
4416 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07004417 __COMMENT__ "`Point` doesn't implement the `Default` trait"
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004418 Point() = delete;
4419
4420 // In this test there is no `Copy` implementation / derive.
4421 Point(const Point&) = delete;
4422
4423 // All Rust types are trivially-movable.
4424 Point(Point&&) = default;
4425
4426 // Assignment operators are disabled for now.
4427 Point& operator=(const Point&) = delete;
4428 Point& operator=(Point&&) = delete;
4429
4430 // In this test there is no custom `Drop`, so C++ can also
4431 // just use the `default` destructor.
4432 ~Point() = default;
4433 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004434 __COMMENT__ #no_fields_msg
4435 unsigned char __opaque_blob_of_bytes[12];
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004436 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004437 };
4438 }
4439 );
4440 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004441 result.cc_details.tokens,
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004442 quote! {
4443 static_assert(sizeof(Point) == 12, ...);
4444 static_assert(alignof(Point) == 4, ...);
4445 }
4446 );
4447 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004448 result.rs_details,
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004449 quote! {
4450 const _: () = assert!(::std::mem::size_of::<::rust_out::Point>() == 12);
4451 const _: () = assert!(::std::mem::align_of::<::rust_out::Point>() == 4);
4452 }
4453 );
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004454 });
4455 }
4456
4457 /// This test covers how zero-variant enums are handled. See also
4458 /// https://doc.rust-lang.org/reference/items/enumerations.html#zero-variant-enums
4459 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004460 fn test_format_item_unsupported_enum_zero_variants() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004461 let test_src = r#"
4462 pub enum ZeroVariantEnum {}
4463 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004464 test_format_item(test_src, "ZeroVariantEnum", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004465 let err = result.unwrap_err();
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004466 assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)");
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004467 });
4468 }
4469
4470 /// This is a test for a `union`. See also
4471 /// https://doc.rust-lang.org/reference/items/unions.html
4472 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004473 fn test_format_item_union() {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004474 let test_src = r#"
4475 pub union SomeUnion {
4476 pub i: i32,
4477 pub f: f64,
4478 }
4479
4480 const _: () = assert!(std::mem::size_of::<SomeUnion>() == 8);
4481 const _: () = assert!(std::mem::align_of::<SomeUnion>() == 8);
4482 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004483 test_format_item(test_src, "SomeUnion", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004484 let result = result.unwrap().unwrap();
4485 let main_api = &result.main_api;
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004486 let no_fields_msg = "Field type has been replaced with a blob of bytes: \
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004487 No support for bindings of individual fields of \
4488 `union` (b/272801632) or `enum`";
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004489 assert!(main_api.prereqs.is_empty());
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004490 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004491 main_api.tokens,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004492 quote! {
Googler47fd9572023-01-20 09:57:32 -08004493 ...
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08004494 union alignas(8) SomeUnion final {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004495 public:
Lukasz Anforowicz9924c432023-04-04 12:51:22 -07004496 __COMMENT__ "`SomeUnion` doesn't implement the `Default` trait"
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004497 SomeUnion() = delete;
4498
4499 // In this test there is no `Copy` implementation / derive.
4500 SomeUnion(const SomeUnion&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004501
4502 // All Rust types are trivially-movable.
4503 SomeUnion(SomeUnion&&) = default;
Lukasz Anforowicz554ed652023-01-12 15:41:58 -08004504
4505 // Assignment operators are disabled for now.
4506 SomeUnion& operator=(const SomeUnion&) = delete;
4507 SomeUnion& operator=(SomeUnion&&) = delete;
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004508
4509 // In this test there is no custom `Drop`, so C++ can also
4510 // just use the `default` destructor.
4511 ~SomeUnion() = default;
4512 private:
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004513 __COMMENT__ #no_fields_msg
4514 unsigned char __opaque_blob_of_bytes[8];
Lukasz Anforowicz672c5152023-03-15 16:23:00 -07004515 inline static void __crubit_field_offset_assertions();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004516 };
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004517 }
4518 );
4519 assert_cc_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004520 result.cc_details.tokens,
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004521 quote! {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004522 static_assert(sizeof(SomeUnion) == 8, ...);
4523 static_assert(alignof(SomeUnion) == 8, ...);
4524 }
4525 );
4526 assert_rs_matches!(
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004527 result.rs_details,
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004528 quote! {
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08004529 const _: () = assert!(::std::mem::size_of::<::rust_out::SomeUnion>() == 8);
4530 const _: () = assert!(::std::mem::align_of::<::rust_out::SomeUnion>() == 8);
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004531 }
4532 );
4533 });
4534 }
4535
Lukasz Anforowicze4333062022-10-17 14:47:53 -07004536 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004537 fn test_format_item_doc_comments_union() {
Googler04329a12022-12-02 00:56:33 -08004538 let test_src = r#"
4539 /// Doc for some union.
4540 pub union SomeUnionWithDocs {
4541 /// Doc for a field in a union.
4542 pub i: i32,
4543 pub f: f64
4544 }
4545 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004546 test_format_item(test_src, "SomeUnionWithDocs", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004547 let result = result.unwrap().unwrap();
4548 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004549 let comment = " Doc for some union.\n\n\
4550 Generated from: <crubit_unittests.rs>;l=3";
Googler04329a12022-12-02 00:56:33 -08004551 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004552 main_api.tokens,
Googler04329a12022-12-02 00:56:33 -08004553 quote! {
4554 __COMMENT__ #comment
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08004555 union ... SomeUnionWithDocs final {
Googler04329a12022-12-02 00:56:33 -08004556 ...
4557 }
4558 ...
4559 }
4560 );
4561 });
4562 }
4563
4564 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004565 fn test_format_item_doc_comments_enum() {
Googler04329a12022-12-02 00:56:33 -08004566 let test_src = r#"
4567 /** Doc for some enum. */
4568 pub enum SomeEnumWithDocs {
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004569 Kind1(i32),
Googler04329a12022-12-02 00:56:33 -08004570 }
4571 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004572 test_format_item(test_src, "SomeEnumWithDocs", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004573 let result = result.unwrap().unwrap();
4574 let main_api = &result.main_api;
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004575 let comment = " Doc for some enum. \n\n\
4576 Generated from: <crubit_unittests.rs>;l=3";
4577 assert_cc_matches!(
4578 main_api.tokens,
4579 quote! {
4580 __COMMENT__ #comment
4581 struct ... SomeEnumWithDocs final {
4582 ...
4583 }
4584 ...
4585 }
4586 );
Googler04329a12022-12-02 00:56:33 -08004587 });
4588 }
4589
4590 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004591 fn test_format_item_doc_comments_struct() {
Googler04329a12022-12-02 00:56:33 -08004592 let test_src = r#"
4593 #![allow(dead_code)]
4594 #[doc = "Doc for some struct."]
4595 pub struct SomeStructWithDocs {
Lukasz Anforowiczcc7a76b2023-02-28 14:19:42 -08004596 #[doc = "Doc for first field."]
Googler04329a12022-12-02 00:56:33 -08004597 some_field : i32,
4598 }
4599 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004600 test_format_item(test_src, "SomeStructWithDocs", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004601 let result = result.unwrap().unwrap();
4602 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004603 let comment = "Doc for some struct.\n\n\
4604 Generated from: <crubit_unittests.rs>;l=4";
Googler04329a12022-12-02 00:56:33 -08004605 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004606 main_api.tokens,
Googler04329a12022-12-02 00:56:33 -08004607 quote! {
4608 __COMMENT__ #comment
4609 struct ... SomeStructWithDocs final {
4610 ...
4611 }
4612 ...
4613 }
4614 );
4615 });
4616 }
4617
4618 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004619 fn test_format_item_doc_comments_tuple_struct() {
Googler04329a12022-12-02 00:56:33 -08004620 let test_src = r#"
4621 /// Doc for some tuple struct.
4622 pub struct SomeTupleStructWithDocs(i32);
4623 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004624 test_format_item(test_src, "SomeTupleStructWithDocs", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004625 let result = result.unwrap().unwrap();
4626 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004627 let comment = " Doc for some tuple struct.\n\n\
4628 Generated from: <crubit_unittests.rs>;l=3";
Googler04329a12022-12-02 00:56:33 -08004629 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004630 main_api.tokens,
Googler04329a12022-12-02 00:56:33 -08004631 quote! {
4632 __COMMENT__ #comment
4633 struct ... SomeTupleStructWithDocs final {
4634 ...
4635 }
4636 ...
4637 },
4638 );
4639 });
4640 }
4641
4642 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004643 fn test_format_item_source_loc_macro_rules() {
Googler47fd9572023-01-20 09:57:32 -08004644 let test_src = r#"
4645 macro_rules! some_tuple_struct_macro_for_testing_source_loc {
4646 () => {
4647 /// Some doc on SomeTupleStructMacroForTesingSourceLoc.
4648 pub struct SomeTupleStructMacroForTesingSourceLoc(i32);
4649 };
4650 }
4651
4652 some_tuple_struct_macro_for_testing_source_loc!();
4653 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004654 test_format_item(test_src, "SomeTupleStructMacroForTesingSourceLoc", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004655 let result = result.unwrap().unwrap();
4656 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004657 let source_loc_comment = " Some doc on SomeTupleStructMacroForTesingSourceLoc.\n\n\
4658 Generated from: <crubit_unittests.rs>;l=5";
4659 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004660 main_api.tokens,
Googler47fd9572023-01-20 09:57:32 -08004661 quote! {
4662 __COMMENT__ #source_loc_comment
4663 struct ... SomeTupleStructMacroForTesingSourceLoc final {
4664 ...
4665 }
4666 ...
4667 },
4668 );
4669 });
4670 }
4671
4672 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004673 fn test_format_item_source_loc_with_no_doc_comment() {
Googler47fd9572023-01-20 09:57:32 -08004674 let test_src = r#"
4675 pub struct SomeTupleStructWithNoDocComment(i32);
4676 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004677 test_format_item(test_src, "SomeTupleStructWithNoDocComment", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004678 let result = result.unwrap().unwrap();
4679 let main_api = &result.main_api;
Googler47fd9572023-01-20 09:57:32 -08004680 let comment = "Generated from: <crubit_unittests.rs>;l=2";
4681 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004682 main_api.tokens,
Googler47fd9572023-01-20 09:57:32 -08004683 quote! {
4684 __COMMENT__ #comment
4685 struct ... SomeTupleStructWithNoDocComment final {
4686 ...
4687 }
4688 ...
4689 },
4690 );
4691 });
4692 }
4693
4694 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004695 fn test_format_item_unsupported_static_value() {
Lukasz Anforowicze4333062022-10-17 14:47:53 -07004696 let test_src = r#"
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004697 #[no_mangle]
4698 pub static STATIC_VALUE: i32 = 42;
Lukasz Anforowicze4333062022-10-17 14:47:53 -07004699 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004700 test_format_item(test_src, "STATIC_VALUE", |result| {
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004701 let err = result.unwrap_err();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004702 assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: static item");
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07004703 });
4704 }
4705
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004706 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004707 fn test_format_item_unsupported_const_value() {
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004708 let test_src = r#"
4709 pub const CONST_VALUE: i32 = 42;
4710 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004711 test_format_item(test_src, "CONST_VALUE", |result| {
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004712 let err = result.unwrap_err();
4713 assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: constant item");
4714 });
4715 }
4716
4717 #[test]
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004718 fn test_format_item_unsupported_type_alias() {
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004719 let test_src = r#"
4720 pub type TypeAlias = i32;
4721 "#;
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08004722 test_format_item(test_src, "TypeAlias", |result| {
Lukasz Anforowicz08b8ae12023-02-06 10:51:09 -08004723 // TODO(b/254096006): Add support for type alias definitions.
4724 let err = result.unwrap_err();
4725 assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: type alias");
4726 });
4727 }
4728
Lukasz Anforowicz14229762023-02-10 15:28:33 -08004729 #[test]
4730 fn test_format_item_unsupported_impl_item_const_value() {
4731 let test_src = r#"
4732 pub struct SomeStruct(i32);
4733
4734 impl SomeStruct {
4735 pub const CONST_VALUE: i32 = 42;
4736 }
4737 "#;
4738 test_format_item(test_src, "SomeStruct", |result| {
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07004739 let result = result.unwrap().unwrap();
4740 let main_api = &result.main_api;
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07004741 assert!(!main_api.prereqs.is_empty());
Lukasz Anforowicz14229762023-02-10 15:28:33 -08004742 let unsupported_msg = "Error generating bindings for `SomeStruct::CONST_VALUE` \
4743 defined at <crubit_unittests.rs>;l=5: \
Lukasz Anforowicz56d94fc2023-02-17 12:23:20 -08004744 Unsupported `impl` item kind: Const";
Lukasz Anforowicz14229762023-02-10 15:28:33 -08004745 assert_cc_matches!(
Lukasz Anforowicz0b8a02f2023-02-10 15:55:40 -08004746 main_api.tokens,
Lukasz Anforowicz14229762023-02-10 15:28:33 -08004747 quote! {
4748 ...
4749 struct alignas(4) SomeStruct final {
4750 ...
4751 __COMMENT__ #unsupported_msg
4752 ...
4753 };
4754 ...
4755 }
4756 );
4757 });
4758 }
4759
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08004760 /// `test_format_ret_ty_for_cc_successes` provides test coverage for cases
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07004761 /// where `format_ty_for_cc` takes `TypeLocation::FnReturn` and returns
4762 /// an `Ok(...)`. Additional testcases are covered by
4763 /// `test_format_ty_for_cc_successes`.
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004764 #[test]
Lukasz Anforowicz8a68f502022-11-15 08:43:43 -08004765 fn test_format_ret_ty_for_cc_successes() {
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004766 let testcases = [
4767 // ( <Rust type>, <expected C++ type> )
4768 ("bool", "bool"), // TyKind::Bool
4769 ("()", "void"),
4770 // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more
4771 // details).
4772 ("!", "void"),
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07004773 (
4774 "extern \"C\" fn (f32, f32) -> f32",
4775 "crubit :: type_identity_t < float (float , float) > &",
4776 ),
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004777 ];
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004778 test_ty(&testcases, quote! {}, |desc, tcx, ty, expected| {
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07004779 let actual = {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08004780 let input = bindings_input_for_tests(tcx);
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07004781 let cc_snippet = format_ty_for_cc(&input, ty, TypeLocation::FnReturn).unwrap();
Lukasz Anforowicz3744e502022-12-02 08:40:38 -08004782 cc_snippet.tokens.to_string()
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07004783 };
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004784 let expected = expected.parse::<TokenStream>().unwrap().to_string();
4785 assert_eq!(actual, expected, "{desc}");
4786 });
4787 }
4788
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004789 /// `test_format_ty_for_cc_successes` provides test coverage for cases where
4790 /// `format_ty_for_cc` returns an `Ok(...)`.
4791 ///
4792 /// Note that using `std::int8_t` (instead of `::std::int8_t`) has been an
4793 /// explicit decision. The "Google C++ Style Guide" suggests to "avoid
4794 /// nested namespaces that match well-known top-level namespaces" and "in
4795 /// particular, [...] not create any nested std namespaces.". It
4796 /// seems desirable if the generated bindings conform to this aspect of the
4797 /// style guide, because it makes things easier for *users* of these
4798 /// bindings.
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004799 #[test]
Lukasz Anforowicz8a68f502022-11-15 08:43:43 -08004800 fn test_format_ty_for_cc_successes() {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004801 let testcases = [
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004802 // ( <Rust type>, (<expected C++ type>,
4803 // <expected #include>,
4804 // <expected prereq def>,
4805 // <expected prereq fwd decl>) )
4806 ("bool", ("bool", "", "", "")),
4807 ("f32", ("float", "", "", "")),
4808 ("f64", ("double", "", "", "")),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08004809 ("i8", ("std::int8_t", "<cstdint>", "", "")),
4810 ("i16", ("std::int16_t", "<cstdint>", "", "")),
4811 ("i32", ("std::int32_t", "<cstdint>", "", "")),
4812 ("i64", ("std::int64_t", "<cstdint>", "", "")),
4813 ("isize", ("std::intptr_t", "<cstdint>", "", "")),
4814 ("u8", ("std::uint8_t", "<cstdint>", "", "")),
4815 ("u16", ("std::uint16_t", "<cstdint>", "", "")),
4816 ("u32", ("std::uint32_t", "<cstdint>", "", "")),
4817 ("u64", ("std::uint64_t", "<cstdint>", "", "")),
4818 ("usize", ("std::uintptr_t", "<cstdint>", "", "")),
Lukasz Anforowiczec0b64e2023-02-17 14:31:12 -08004819 ("char", ("rs_std::rs_char", "\"crubit/support/for/tests/rs_std/rs_char.h\"", "", "")),
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004820 ("SomeStruct", ("::rust_out::SomeStruct", "", "SomeStruct", "")),
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004821 ("SomeEnum", ("::rust_out::SomeEnum", "", "SomeEnum", "")),
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004822 ("SomeUnion", ("::rust_out::SomeUnion", "", "SomeUnion", "")),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08004823 ("*const i32", ("const std::int32_t*", "<cstdint>", "", "")),
4824 ("*mut i32", ("std::int32_t*", "<cstdint>", "", "")),
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004825 // `SomeStruct` is a `fwd_decls` prerequisite (not `defs` prerequisite):
4826 ("*mut SomeStruct", ("::rust_out::SomeStruct*", "", "", "SomeStruct")),
4827 // Testing propagation of deeper/nested `fwd_decls`:
4828 ("*mut *mut SomeStruct", (":: rust_out :: SomeStruct * *", "", "", "SomeStruct")),
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07004829 (
4830 "extern \"C\" fn (f32, f32) -> f32",
4831 (
4832 "crubit :: type_identity_t < float (float , float) > *",
4833 "\"crubit/support/for/tests/internal/cxx20_backports.h\"",
4834 "",
4835 "",
4836 ),
4837 ),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004838 // Extra parens/sugar are expected to be ignored:
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004839 ("(bool)", ("bool", "", "", "")),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004840 ];
Lukasz Anforowicz40472722022-11-08 13:29:08 -08004841 let preamble = quote! {
4842 #![allow(unused_parens)]
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004843
4844 pub struct SomeStruct {
4845 pub x: i32,
4846 pub y: i32,
4847 }
Lukasz Anforowicz32ad9562023-03-10 14:48:25 -08004848 pub enum SomeEnum {
4849 Cartesian{x: f64, y: f64},
4850 Polar{angle: f64, dist: f64},
4851 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004852 pub union SomeUnion {
4853 pub x: i32,
4854 pub y: i32,
4855 }
Lukasz Anforowicz40472722022-11-08 13:29:08 -08004856 };
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08004857 test_ty(
4858 &testcases,
4859 preamble,
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004860 |desc, tcx, ty,
4861 (expected_tokens, expected_include, expected_prereq_def, expected_prereq_fwd_decl)| {
4862 let (actual_tokens, actual_prereqs) = {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08004863 let input = bindings_input_for_tests(tcx);
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07004864 let s = format_ty_for_cc(&input, ty, TypeLocation::Other).unwrap();
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004865 (s.tokens.to_string(), s.prereqs)
4866 };
4867 let (actual_includes, actual_prereq_defs, actual_prereq_fwd_decls) =
4868 (actual_prereqs.includes, actual_prereqs.defs, actual_prereqs.fwd_decls);
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07004869
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08004870 let expected_tokens = expected_tokens.parse::<TokenStream>().unwrap().to_string();
4871 assert_eq!(actual_tokens, expected_tokens, "{desc}");
Lukasz Anforowiczed17d052022-11-02 12:07:28 -07004872
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004873 if expected_include.is_empty() {
4874 assert!(actual_includes.is_empty());
4875 } else {
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08004876 let expected_include: TokenStream = expected_include.parse().unwrap();
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004877 assert_cc_matches!(
4878 format_cc_includes(&actual_includes),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08004879 quote! { __HASH_TOKEN__ include #expected_include }
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004880 );
4881 }
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08004882
4883 if expected_prereq_def.is_empty() {
4884 assert!(actual_prereq_defs.is_empty());
4885 } else {
4886 let expected_def_id = find_def_id_by_name(tcx, expected_prereq_def);
4887 assert_eq!(1, actual_prereq_defs.len());
4888 assert_eq!(expected_def_id, actual_prereq_defs.into_iter().next().unwrap());
4889 }
Lukasz Anforowiczf4501b42023-01-07 09:41:29 -08004890
4891 if expected_prereq_fwd_decl.is_empty() {
4892 assert!(actual_prereq_fwd_decls.is_empty());
4893 } else {
4894 let expected_def_id = find_def_id_by_name(tcx, expected_prereq_fwd_decl);
4895 assert_eq!(1, actual_prereq_fwd_decls.len());
4896 assert_eq!(expected_def_id,
4897 actual_prereq_fwd_decls.into_iter().next().unwrap());
4898 }
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08004899 },
4900 );
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004901 }
4902
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004903 /// `test_format_ty_for_cc_failures` provides test coverage for cases where
4904 /// `format_ty_for_cc` returns an `Err(...)`.
4905 ///
4906 /// It seems okay to have no test coverage for now for the following types
4907 /// (which should never be encountered when generating bindings and where
4908 /// `format_ty_for_cc` should panic):
4909 /// - TyKind::Closure
4910 /// - TyKind::Error
4911 /// - TyKind::FnDef
4912 /// - TyKind::Infer
4913 ///
Lukasz Anforowicz0182c5c2022-12-29 10:08:50 -08004914 /// TODO(lukasza): Add test coverage (here and in the "for_rs" flavours)
4915 /// for:
Lukasz Anforowicz4c19ad92022-12-16 15:23:14 -08004916 /// - TyKind::Bound
4917 /// - TyKind::Dynamic (`dyn Eq`)
4918 /// - TyKind::Foreign (`extern type T`)
4919 /// - https://doc.rust-lang.org/beta/unstable-book/language-features/generators.html:
4920 /// TyKind::Generator, TyKind::GeneratorWitness
4921 /// - TyKind::Param
4922 /// - TyKind::Placeholder
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004923 #[test]
Lukasz Anforowicz8a68f502022-11-15 08:43:43 -08004924 fn test_format_ty_for_cc_failures() {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004925 let testcases = [
4926 // ( <Rust type>, <expected error message> )
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004927 (
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004928 "()", // Empty TyKind::Tuple
Devin Jeanpierre81aec502023-04-04 15:33:39 -07004929 "`()` / `void` is only supported as a return type (b/254507801)",
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004930 ),
4931 (
4932 // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more
4933 // details).
4934 "!", // TyKind::Never
Devin Jeanpierre81aec502023-04-04 15:33:39 -07004935 "The never type `!` is only supported as a return type (b/254507801)",
Lukasz Anforowicz903fc632022-10-25 08:55:33 -07004936 ),
4937 (
4938 "(i32, i32)", // Non-empty TyKind::Tuple
Lukasz Anforowicz13794df2022-10-21 07:56:34 -07004939 "Tuples are not supported yet: (i32, i32) (b/254099023)",
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004940 ),
4941 (
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004942 "&'static i32", // TyKind::Ref
4943 "The following Rust type is not supported yet: &'static i32",
4944 ),
4945 (
4946 "[i32; 42]", // TyKind::Array
4947 "The following Rust type is not supported yet: [i32; 42]",
4948 ),
4949 (
4950 "&'static [i32]", // TyKind::Slice (nested underneath TyKind::Ref)
4951 "The following Rust type is not supported yet: &'static [i32]",
4952 ),
4953 (
4954 "&'static str", // TyKind::Str (nested underneath TyKind::Ref)
4955 "The following Rust type is not supported yet: &'static str",
4956 ),
4957 (
Lukasz Anforowicz0182c5c2022-12-29 10:08:50 -08004958 "impl Eq", // TyKind::Alias
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004959 "The following Rust type is not supported yet: impl std::cmp::Eq",
4960 ),
4961 (
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07004962 "fn(i32) -> i32", // TyKind::FnPtr (default ABI = "Rust")
4963 "Function pointers can't have a thunk: \
4964 Calling convention other than `extern \"C\"` requires a thunk",
4965 ),
4966 (
4967 "extern \"C\" fn (SomeStruct, f32) -> f32",
4968 "Function pointers can't have a thunk: Type of parameter #0 requires a thunk",
4969 ),
4970 (
4971 "extern \"C\" fn (f32, f32) -> SomeStruct",
4972 "Function pointers can't have a thunk: Return type requires a thunk",
4973 ),
4974 (
4975 "unsafe fn(i32) -> i32",
4976 "Bindings for `unsafe` functions are not fully designed yet (b/254095482)",
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07004977 ),
4978 // TODO(b/254094650): Consider mapping this to Clang's (and GCC's) `__int128`
4979 // or to `absl::in128`.
4980 ("i128", "C++ doesn't have a standard equivalent of `i128` (b/254094650)"),
4981 ("u128", "C++ doesn't have a standard equivalent of `u128` (b/254094650)"),
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004982 (
4983 "StructWithCustomDrop",
4984 "Failed to generate bindings for the definition of `StructWithCustomDrop`: \
Devin Jeanpierre81aec502023-04-04 15:33:39 -07004985 `Drop` trait and \"drop glue\" are not supported yet (b/258251148)",
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004986 ),
Devin Jeanpierre81aec502023-04-04 15:33:39 -07004987 ("ConstGenericStruct<42>", "Generic types are not supported yet (b/259749095)"),
4988 ("TypeGenericStruct<u8>", "Generic types are not supported yet (b/259749095)"),
Lukasz Anforowicz75894832022-11-22 18:25:28 -08004989 (
4990 // This double-checks that TyKind::Adt(..., substs) are present
4991 // even if the type parameter argument is not explicitly specified
4992 // (here it comes from the default: `...Struct<T = u8>`).
4993 "TypeGenericStruct",
4994 "Generic types are not supported yet (b/259749095)",
4995 ),
Lukasz Anforowiczcfe84552023-04-20 13:38:54 -07004996 (
4997 "LifetimeGenericStruct<'static>",
4998 "Generic types are not supported yet (b/259749095)",
4999 ),
5000 (
5001 "std::cmp::Ordering",
5002 "Type `std::cmp::Ordering` comes from the `core` crate, \
5003 but no `--other-crate-bindings` were specified for this crate",
5004 ),
5005 (
5006 "Option<i8>",
5007 "Generic types are not supported yet (b/259749095)",
5008 ),
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -08005009 (
5010 "PublicReexportOfStruct",
5011 "Not directly public type (re-exports are not supported yet - b/262052635)",
5012 ),
5013 (
5014 // This testcase is like `PublicReexportOfStruct`, but the private type and the
5015 // re-export are in another crate. When authoring this test
5016 // `core::alloc::LayoutError` was a public re-export of
5017 // `core::alloc::layout::LayoutError`:
5018 // `https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=d2b5528af9b33b25abe44cc4646d65e3`
5019 // TODO(b/258261328): Once cross-crate bindings are supported we should try
5020 // to test them via a test crate that we control (rather than testing via
5021 // implementation details of the std crate).
5022 "core::alloc::LayoutError",
5023 "Not directly public type (re-exports are not supported yet - b/262052635)",
5024 ),
Lukasz Anforowicz2eb24292023-03-14 09:38:21 -07005025 (
5026 "*const Option<i8>",
5027 "Failed to format the pointee of the pointer type `std::option::Option<i8>`: \
5028 Generic types are not supported yet (b/259749095)",
5029 ),
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005030 ];
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005031 let preamble = quote! {
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005032 #![feature(never_type)]
5033
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07005034 pub struct SomeStruct {
5035 pub x: i32,
5036 pub y: i32,
5037 }
5038
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005039 pub struct StructWithCustomDrop {
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005040 pub x: i32,
5041 pub y: i32,
5042 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005043
5044 impl Drop for StructWithCustomDrop {
5045 fn drop(&mut self) {}
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005046 }
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005047
5048 pub struct ConstGenericStruct<const N: usize> {
5049 pub arr: [u8; N],
5050 }
5051
5052 pub struct TypeGenericStruct<T = u8> {
5053 pub t: T,
5054 }
5055
5056 pub struct LifetimeGenericStruct<'a> {
5057 pub reference: &'a u8,
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005058 }
Lukasz Anforowiczf36762a2023-03-02 18:43:07 -08005059
5060 mod private_submodule {
5061 pub struct PublicStructInPrivateModule;
5062 }
5063 pub use private_submodule::PublicStructInPrivateModule
5064 as PublicReexportOfStruct;
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005065 };
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005066 test_ty(&testcases, preamble, |desc, tcx, ty, expected_msg| {
5067 let input = bindings_input_for_tests(tcx);
Lukasz Anforowicz8574c9d2023-04-13 15:11:20 -07005068 let anyhow_err = format_ty_for_cc(&input, ty, TypeLocation::Other).unwrap_err();
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005069 let actual_msg = format!("{anyhow_err:#}");
5070 assert_eq!(&actual_msg, *expected_msg, "{desc}");
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005071 });
5072 }
5073
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005074 #[test]
5075 fn test_format_ty_for_rs_successes() {
5076 // Test coverage for cases where `format_ty_for_rs` returns an `Ok(...)`.
5077 let testcases = [
5078 // ( <Rust type>, <expected Rust spelling for ..._cc_api_impl.rs> )
5079 ("bool", "bool"),
5080 ("f32", "f32"),
5081 ("f64", "f64"),
5082 ("i8", "i8"),
5083 ("i16", "i16"),
5084 ("i32", "i32"),
5085 ("i64", "i64"),
5086 ("i128", "i128"),
5087 ("isize", "isize"),
5088 ("u8", "u8"),
5089 ("u16", "u16"),
5090 ("u32", "u32"),
5091 ("u64", "u64"),
5092 ("u128", "u128"),
5093 ("usize", "usize"),
5094 ("char", "char"),
5095 ("!", "!"),
5096 ("()", "()"),
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08005097 // ADTs:
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08005098 ("SomeStruct", "::rust_out::SomeStruct"),
5099 ("SomeEnum", "::rust_out::SomeEnum"),
5100 ("SomeUnion", "::rust_out::SomeUnion"),
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08005101 // Type from another crate:
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08005102 ("std::cmp::Ordering", "::core::cmp::Ordering"),
Lukasz Anforowiczeb58a492023-01-07 08:25:48 -08005103 // `const` and `mut` pointers:
5104 ("*const i32", "*const i32"),
5105 ("*mut i32", "*mut i32"),
5106 // Pointer to an ADT:
5107 ("*mut SomeStruct", "* mut :: rust_out :: SomeStruct"),
Lukasz Anforowicz5c1b3ad2023-04-13 17:05:00 -07005108 ("extern \"C\" fn(i32) -> i32", "extern \"C\" fn(i32) -> i32"),
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005109 ];
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005110 let preamble = quote! {
5111 #![feature(never_type)]
5112
5113 pub struct SomeStruct {
5114 pub x: i32,
5115 pub y: i32,
5116 }
5117 pub enum SomeEnum {
5118 Cartesian{x: f64, y: f64},
5119 Polar{angle: f64, dist: f64},
5120 }
5121 pub union SomeUnion {
5122 pub x: i32,
5123 pub y: i32,
5124 }
5125 };
Lukasz Anforowicz5c9c00d2022-12-02 08:41:39 -08005126 test_ty(&testcases, preamble, |desc, tcx, ty, expected_tokens| {
5127 let actual_tokens = format_ty_for_rs(tcx, ty).unwrap().to_string();
5128 let expected_tokens = expected_tokens.parse::<TokenStream>().unwrap().to_string();
5129 assert_eq!(actual_tokens, expected_tokens, "{desc}");
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005130 });
5131 }
5132
5133 #[test]
5134 fn test_format_ty_for_rs_failures() {
5135 // This test provides coverage for cases where `format_ty_for_rs` returns an
5136 // `Err(...)`.
5137 let testcases = [
5138 // ( <Rust type>, <expected error message> )
5139 (
5140 "(i32, i32)", // Non-empty TyKind::Tuple
5141 "Tuples are not supported yet: (i32, i32) (b/254099023)",
5142 ),
5143 (
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005144 "&'static i32", // TyKind::Ref
5145 "The following Rust type is not supported yet: &'static i32",
5146 ),
5147 (
5148 "[i32; 42]", // TyKind::Array
5149 "The following Rust type is not supported yet: [i32; 42]",
5150 ),
5151 (
5152 "&'static [i32]", // TyKind::Slice (nested underneath TyKind::Ref)
5153 "The following Rust type is not supported yet: &'static [i32]",
5154 ),
5155 (
5156 "&'static str", // TyKind::Str (nested underneath TyKind::Ref)
5157 "The following Rust type is not supported yet: &'static str",
5158 ),
5159 (
Lukasz Anforowicz0182c5c2022-12-29 10:08:50 -08005160 "impl Eq", // TyKind::Alias
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005161 "The following Rust type is not supported yet: impl std::cmp::Eq",
5162 ),
5163 (
Lukasz Anforowicz8b475a42022-11-29 09:33:02 -08005164 "Option<i8>", // TyKind::Adt - generic + different crate
5165 "Generic types are not supported yet (b/259749095)",
5166 ),
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005167 ];
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005168 let preamble = quote! {};
5169 test_ty(&testcases, preamble, |desc, tcx, ty, expected_err| {
5170 let anyhow_err = format_ty_for_rs(tcx, ty).unwrap_err();
Lukasz Anforowicz7860d0e2022-11-15 08:47:56 -08005171 let actual_err = format!("{anyhow_err:#}");
5172 assert_eq!(&actual_err, *expected_err, "{desc}");
5173 });
5174 }
5175
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005176 fn test_ty<TestFn, Expectation>(
5177 testcases: &[(&str, Expectation)],
5178 preamble: TokenStream,
5179 test_fn: TestFn,
5180 ) where
Lukasz Anforowiczd16b6bf2022-11-22 18:35:08 -08005181 TestFn: for<'tcx> Fn(
5182 /* testcase_description: */ &str,
5183 TyCtxt<'tcx>,
5184 Ty<'tcx>,
5185 &Expectation,
5186 ) + Sync,
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005187 Expectation: Sync,
5188 {
Lukasz Anforowiczd0f0a842022-11-03 12:40:13 -07005189 for (index, (input, expected)) in testcases.iter().enumerate() {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005190 let desc = format!("test #{index}: test input: `{input}`");
5191 let input = {
5192 let ty_tokens: TokenStream = input.parse().unwrap();
5193 let input = quote! {
Lukasz Anforowicz40472722022-11-08 13:29:08 -08005194 #preamble
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005195 pub fn test_function() -> #ty_tokens { panic!("") }
5196 };
5197 input.to_string()
5198 };
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08005199 run_compiler_for_testing(input, |tcx| {
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005200 let def_id = find_def_id_by_name(tcx, "test_function");
Lukasz Anforowicz087dff72023-02-17 12:13:32 -08005201 let ty = tcx
5202 .fn_sig(def_id.to_def_id())
5203 .subst_identity()
5204 .no_bound_vars()
5205 .unwrap()
5206 .output();
Lukasz Anforowicz75894832022-11-22 18:25:28 -08005207 test_fn(&desc, tcx, ty, expected);
Lukasz Anforowicz4f9f5522022-10-17 15:03:37 -07005208 });
5209 }
5210 }
5211
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08005212 /// Tests invoking `format_item` on the item with the specified `name` from
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005213 /// the given Rust `source`. Returns the result of calling
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08005214 /// `test_function` with `format_item`'s result as an argument.
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005215 /// (`test_function` should typically `assert!` that it got the expected
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08005216 /// result from `format_item`.)
5217 fn test_format_item<F, T>(source: &str, name: &str, test_function: F) -> T
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005218 where
Lukasz Anforowicz73a2c0d2023-03-22 14:07:00 -07005219 F: FnOnce(Result<Option<ApiSnippets>, String>) -> T + Send,
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005220 T: Send,
5221 {
Lukasz Anforowicz0bef2642023-01-05 09:20:31 -08005222 run_compiler_for_testing(source, |tcx| {
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005223 let def_id = find_def_id_by_name(tcx, name);
Lukasz Anforowicze465b9e2023-02-06 15:00:25 -08005224 let result = format_item(&bindings_input_for_tests(tcx), def_id);
Lukasz Anforowicze4333062022-10-17 14:47:53 -07005225
5226 // https://docs.rs/anyhow/latest/anyhow/struct.Error.html#display-representations says:
5227 // To print causes as well [...], use the alternate selector “{:#}”.
5228 let result = result.map_err(|anyhow_err| format!("{anyhow_err:#}"));
5229
5230 test_function(result)
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005231 })
5232 }
5233
5234 /// Finds the definition id of a Rust item with the specified `name`.
5235 /// Panics if no such item is found, or if there is more than one match.
5236 fn find_def_id_by_name(tcx: TyCtxt, name: &str) -> LocalDefId {
5237 let hir_items = || tcx.hir().items().map(|item_id| tcx.hir().item(item_id));
5238 let items_with_matching_name =
5239 hir_items().filter(|item| item.ident.name.as_str() == name).collect_vec();
Lukasz Anforowiczd0f0a842022-11-03 12:40:13 -07005240 match *items_with_matching_name.as_slice() {
5241 [] => {
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005242 let found_names = hir_items()
5243 .map(|item| item.ident.name.as_str())
5244 .filter(|s| !s.is_empty())
5245 .sorted()
5246 .dedup()
5247 .map(|name| format!("`{name}`"))
Lukasz Anforowiczd0f0a842022-11-03 12:40:13 -07005248 .join(",\n");
5249 panic!("No items named `{name}`.\nInstead found:\n{found_names}");
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005250 }
Lukasz Anforowicz61cb1e32022-11-04 09:08:35 -07005251 [item] => item.owner_id.def_id,
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005252 _ => panic!("More than one item named `{name}`"),
5253 }
5254 }
5255
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005256 fn bindings_input_for_tests(tcx: TyCtxt) -> Input {
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08005257 Input {
5258 tcx,
5259 crubit_support_path: "crubit/support/for/tests".into(),
Lukasz Anforowiczcfe84552023-04-20 13:38:54 -07005260 crate_name_to_include_path: Default::default(),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08005261 _features: (),
Lukasz Anforowicza782bda2023-01-17 14:04:50 -08005262 }
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005263 }
5264
5265 /// Tests invoking `generate_bindings` on the given Rust `source`.
Lukasz Anforowicz8a833412022-10-14 14:44:13 -07005266 /// Returns the result of calling `test_function` with the generated
5267 /// bindings as an argument. (`test_function` should typically `assert!`
5268 /// that it got the expected `GeneratedBindings`.)
5269 fn test_generated_bindings<F, T>(source: &str, test_function: F) -> T
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07005270 where
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005271 F: FnOnce(Result<Output>) -> T + Send,
Lukasz Anforowicz581fd752022-09-21 11:30:15 -07005272 T: Send,
5273 {
Lukasz Anforowiczbe25c762023-01-17 12:43:52 -08005274 run_compiler_for_testing(source, |tcx| {
5275 test_function(generate_bindings(&bindings_input_for_tests(tcx)))
5276 })
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07005277 }
Lukasz Anforowiczbda1cfe2022-09-20 06:25:43 -07005278}