Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 1 | // 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5 | use anyhow::{anyhow, bail, ensure, Context, Result}; |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 6 | use code_gen_utils::{ |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 7 | escape_non_identifier_chars, format_cc_ident, format_cc_includes, |
| 8 | format_namespace_bound_cc_tokens, make_rs_ident, CcInclude, NamespaceQualifier, |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 9 | }; |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 10 | use itertools::Itertools; |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 11 | use proc_macro2::{Ident, Literal, TokenStream}; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 12 | use quote::{format_ident, quote, ToTokens}; |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 13 | use rustc_hir::{ |
| 14 | AssocItemKind, Impl, ImplItemKind, ImplicitSelfKind, Item, ItemKind, Node, Unsafety, |
| 15 | }; |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 16 | use rustc_middle::dep_graph::DepContext; |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 17 | use rustc_middle::mir::Mutability; |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 18 | use rustc_middle::ty::{self, Ty, TyCtxt}; // See <internal link>/ty.html#import-conventions |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 19 | use rustc_span::def_id::{DefId, LocalDefId, LOCAL_CRATE}; |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 20 | use rustc_span::symbol::{sym, Symbol}; |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 21 | use rustc_target::abi::{Abi, FieldsShape, Integer, Layout, Primitive, Scalar}; |
Lukasz Anforowicz | 24160d5 | 2022-10-19 06:45:45 -0700 | [diff] [blame] | 22 | use rustc_target::spec::PanicStrategy; |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 23 | use std::collections::{BTreeSet, HashMap, HashSet}; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 24 | use std::iter::once; |
Lukasz Anforowicz | 7f31f80 | 2022-12-16 08:24:13 -0800 | [diff] [blame] | 25 | use std::ops::AddAssign; |
Lukasz Anforowicz | 8e5042c | 2023-01-03 11:19:07 -0800 | [diff] [blame] | 26 | use std::rc::Rc; |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 27 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 28 | pub struct Input<'tcx> { |
| 29 | /// Compilation context for the crate that the bindings should be generated |
| 30 | /// for. |
| 31 | pub tcx: TyCtxt<'tcx>, |
| 32 | |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 33 | /// 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 Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 37 | |
Lukasz Anforowicz | cfe8455 | 2023-04-20 13:38:54 -0700 | [diff] [blame^] | 38 | /// 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 Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 45 | // TODO(b/262878759): Provide a set of enabled/disabled Crubit features. |
| 46 | pub _features: (), |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 47 | } |
| 48 | |
Lukasz Anforowicz | a3b7db0 | 2023-03-09 17:34:05 -0800 | [diff] [blame] | 49 | impl<'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 Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 60 | pub struct Output { |
Lukasz Anforowicz | 2b38d27 | 2022-09-23 08:08:18 -0700 | [diff] [blame] | 61 | pub h_body: TokenStream, |
Lukasz Anforowicz | e7a2500 | 2022-11-10 06:21:42 -0800 | [diff] [blame] | 62 | pub rs_body: TokenStream, |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 63 | } |
| 64 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 65 | pub 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 Anforowicz | 24160d5 | 2022-10-19 06:45:45 -0700 | [diff] [blame] | 70 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 71 | 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 Anforowicz | d9ff4ab | 2022-09-23 08:11:18 -0700 | [diff] [blame] | 79 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 80 | 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 Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 85 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 86 | let h_body = quote! { |
| 87 | #top_comment |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 88 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 89 | // TODO(b/251445877): Replace `#pragma once` with include guards. |
| 90 | __HASH_TOKEN__ pragma once __NEWLINE__ |
| 91 | __NEWLINE__ |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 92 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 93 | #h_body |
| 94 | }; |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 95 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 96 | let rs_body = quote! { |
| 97 | #top_comment |
Lukasz Anforowicz | d16b6bf | 2022-11-22 18:35:08 -0800 | [diff] [blame] | 98 | |
Lukasz Anforowicz | 843d161 | 2023-03-02 16:52:53 -0800 | [diff] [blame] | 99 | // `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 Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 102 | #![allow(improper_ctypes_definitions)] __NEWLINE__ |
| 103 | __NEWLINE__ |
Lukasz Anforowicz | d16b6bf | 2022-11-22 18:35:08 -0800 | [diff] [blame] | 104 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 105 | #rs_body |
| 106 | }; |
Lukasz Anforowicz | e7a2500 | 2022-11-10 06:21:42 -0800 | [diff] [blame] | 107 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 108 | Ok(Output { h_body, rs_body }) |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 109 | } |
| 110 | |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 111 | #[derive(Clone, Debug, Default)] |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 112 | struct 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 Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 116 | includes: BTreeSet<CcInclude>, |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 117 | |
| 118 | /// Set of local definitions that a `CcSnippet` depends on. For example if |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 119 | /// `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 Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 125 | 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 Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 134 | fwd_decls: HashSet<LocalDefId>, |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 135 | } |
| 136 | |
| 137 | impl CcPrerequisites { |
| 138 | #[cfg(test)] |
| 139 | fn is_empty(&self) -> bool { |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 140 | let &Self { ref includes, ref defs, ref fwd_decls } = self; |
| 141 | includes.is_empty() && defs.is_empty() && fwd_decls.is_empty() |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 142 | } |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 143 | |
| 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 Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 153 | } |
| 154 | |
| 155 | impl AddAssign for CcPrerequisites { |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 156 | fn add_assign(&mut self, rhs: Self) { |
| 157 | let Self { mut includes, defs, fwd_decls } = rhs; |
| 158 | |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 159 | // `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 Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 164 | // not documented at |
| 165 | // https://doc.rust-lang.org/std/collections/struct.BTreeSet.html#method.append |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 166 | self.includes.append(&mut includes); |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 167 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 168 | self.defs.extend(defs); |
| 169 | self.fwd_decls.extend(fwd_decls); |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 170 | } |
| 171 | } |
| 172 | |
| 173 | #[derive(Debug, Default)] |
| 174 | struct CcSnippet { |
| 175 | tokens: TokenStream, |
| 176 | prereqs: CcPrerequisites, |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 177 | } |
| 178 | |
| 179 | impl CcSnippet { |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 180 | /// Consumes `self` and returns its `tokens`, while preserving |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 181 | /// its `prereqs` into `prereqs_accumulator`. |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 182 | fn into_tokens(self, prereqs_accumulator: &mut CcPrerequisites) -> TokenStream { |
| 183 | let Self { tokens, prereqs } = self; |
| 184 | *prereqs_accumulator += prereqs; |
| 185 | tokens |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 186 | } |
Lukasz Anforowicz | a2f1cae | 2022-12-15 10:42:25 -0800 | [diff] [blame] | 187 | |
| 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 Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 199 | } |
| 200 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 201 | impl 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 Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 207 | /// Represents the fully qualified name of a Rust item (e.g. of a `struct` or a |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 208 | /// function). |
| 209 | struct 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 Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 216 | /// The path may contain multiple modules - e.g. `foo::bar::baz`. |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 217 | mod_path: NamespaceQualifier, |
| 218 | |
| 219 | /// Name of the item. |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 220 | /// 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 Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 224 | } |
| 225 | |
| 226 | impl FullyQualifiedName { |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 227 | /// Computes a `FullyQualifiedName` for `def_id`. |
| 228 | /// |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 229 | /// May panic if `def_id` is an invalid id. |
Lukasz Anforowicz | 1ab5e87 | 2022-12-05 10:07:00 -0800 | [diff] [blame] | 230 | // TODO(b/259724276): This function's results should be memoized. |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 231 | fn new(tcx: TyCtxt, def_id: DefId) -> Self { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 232 | 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 Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 236 | let name = name.data.get_opt_name(); |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 237 | |
Lukasz Anforowicz | 8e5042c | 2023-01-03 11:19:07 -0800 | [diff] [blame] | 238 | let mod_path = NamespaceQualifier::new( |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 239 | full_path |
| 240 | .into_iter() |
| 241 | .filter_map(|p| p.data.get_opt_name()) |
| 242 | .map(|s| Rc::<str>::from(s.as_str())), |
Lukasz Anforowicz | 8e5042c | 2023-01-03 11:19:07 -0800 | [diff] [blame] | 243 | ); |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 244 | |
| 245 | Self { krate, mod_path, name } |
| 246 | } |
| 247 | |
| 248 | fn format_for_cc(&self) -> Result<TokenStream> { |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 249 | let name = |
| 250 | self.name.as_ref().expect("`format_for_cc` can't be called on name-less item kinds"); |
| 251 | |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 252 | let top_level_ns = format_cc_ident(self.krate.as_str())?; |
| 253 | let ns_path = self.mod_path.format_for_cc()?; |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 254 | let name = format_cc_ident(name.as_str())?; |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 255 | Ok(quote! { :: #top_level_ns :: #ns_path #name }) |
| 256 | } |
| 257 | |
| 258 | fn format_for_rs(&self) -> TokenStream { |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 259 | let name = |
| 260 | self.name.as_ref().expect("`format_for_cc` can't be called on name-less item kinds"); |
| 261 | |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 262 | let krate = make_rs_ident(self.krate.as_str()); |
| 263 | let mod_path = self.mod_path.format_for_rs(); |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 264 | let name = make_rs_ident(name.as_str()); |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 265 | quote! { :: #krate :: #mod_path #name } |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 266 | } |
| 267 | } |
| 268 | |
Lukasz Anforowicz | a3b7db0 | 2023-03-09 17:34:05 -0800 | [diff] [blame] | 269 | /// 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). |
| 271 | fn 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 Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 327 | /// Location where a type is used. |
| 328 | enum 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 Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 337 | /// 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 Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 349 | Other, |
| 350 | } |
| 351 | |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 352 | /// Formats `ty` into a `CcSnippet` that represents how the type should be |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 353 | /// spelled in a C++ declaration of a function parameter or field. |
Lukasz Anforowicz | 1ab5e87 | 2022-12-05 10:07:00 -0800 | [diff] [blame] | 354 | // |
| 355 | // TODO(b/259724276): This function's results should be memoized. |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 356 | fn format_ty_for_cc<'tcx>( |
| 357 | input: &Input<'tcx>, |
| 358 | ty: Ty<'tcx>, |
| 359 | location: TypeLocation, |
| 360 | ) -> Result<CcSnippet> { |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 361 | fn cstdint(tokens: TokenStream) -> CcSnippet { |
Lukasz Anforowicz | a2f1cae | 2022-12-15 10:42:25 -0800 | [diff] [blame] | 362 | CcSnippet::with_include(tokens, CcInclude::cstdint()) |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 363 | } |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 364 | fn keyword(tokens: TokenStream) -> CcSnippet { |
Lukasz Anforowicz | a2f1cae | 2022-12-15 10:42:25 -0800 | [diff] [blame] | 365 | CcSnippet::new(tokens) |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 366 | } |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 367 | Ok(match ty.kind() { |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 368 | ty::TyKind::Never => match location { |
| 369 | TypeLocation::FnReturn => keyword(quote! { void }), |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 370 | _ => { |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 371 | // TODO(b/254507801): Maybe translate into `crubit::Never`? |
| 372 | bail!("The never type `!` is only supported as a return type (b/254507801)"); |
| 373 | }, |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 374 | } |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 375 | ty::TyKind::Tuple(types) => { |
| 376 | if types.len() == 0 { |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 377 | match location { |
| 378 | TypeLocation::FnReturn => keyword(quote! { void }), |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 379 | _ => { |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 380 | // TODO(b/254507801): Maybe translate into `crubit::Unit`? |
| 381 | bail!("`()` / `void` is only supported as a return type (b/254507801)"); |
| 382 | }, |
| 383 | } |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 384 | } else { |
Lukasz Anforowicz | 13794df | 2022-10-21 07:56:34 -0700 | [diff] [blame] | 385 | // TODO(b/254099023): Add support for tuples. |
| 386 | bail!("Tuples are not supported yet: {} (b/254099023)", ty); |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 387 | } |
| 388 | } |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 389 | |
Lukasz Anforowicz | e3c3947 | 2023-01-12 15:07:38 -0800 | [diff] [blame] | 390 | // 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 Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 394 | 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 Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 405 | // ABI compatibility and other details are described in the doc comments in |
Lukasz Anforowicz | ec0b64e | 2023-02-17 14:31:12 -0800 | [diff] [blame] | 406 | // `crubit/support/rs_std/rs_char.h` and `crubit/support/rs_std/char_test.cc` (search for |
Lukasz Anforowicz | d71686c | 2023-02-17 14:29:55 -0800 | [diff] [blame] | 407 | // "Layout tests"). |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 408 | ty::TyKind::Char => { |
Lukasz Anforowicz | b06e081 | 2023-03-02 15:54:32 -0800 | [diff] [blame] | 409 | // Asserting that the target architecture meets the assumption from Crubit's |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 410 | // `rust_builtin_type_abi_assumptions.md` - we assume that Rust's `char` has the |
| 411 | // same ABI as `u32`. |
Lukasz Anforowicz | b06e081 | 2023-03-02 15:54:32 -0800 | [diff] [blame] | 412 | 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 Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 420 | assert!(matches!( |
| 421 | layout.abi(), |
| 422 | Abi::Scalar(Scalar::Initialized { |
| 423 | value: Primitive::Int(Integer::I32, /* signedness = */ false), |
| 424 | .. |
| 425 | }) |
| 426 | )); |
Lukasz Anforowicz | b06e081 | 2023-03-02 15:54:32 -0800 | [diff] [blame] | 427 | |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 428 | CcSnippet::with_include( |
Lukasz Anforowicz | ec0b64e | 2023-02-17 14:31:12 -0800 | [diff] [blame] | 429 | quote! { rs_std::rs_char }, |
Lukasz Anforowicz | a3b7db0 | 2023-03-09 17:34:05 -0800 | [diff] [blame] | 430 | input.support_header("rs_std/rs_char.h"), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 431 | ) |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 432 | } |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 433 | |
| 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 Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 444 | 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 Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 452 | |
| 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 Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 456 | ty::TyKind::Int(ty::IntTy::Isize) => cstdint(quote! { std::intptr_t }), |
| 457 | ty::TyKind::Uint(ty::UintTy::Usize) => cstdint(quote! { std::uintptr_t }), |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 458 | |
| 459 | ty::TyKind::Int(ty::IntTy::I128) | ty::TyKind::Uint(ty::UintTy::U128) => { |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 460 | // 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 Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 464 | // 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 469 | ty::TyKind::Adt(adt, substs) => { |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 470 | ensure!(substs.len() == 0, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 471 | ensure!( |
| 472 | is_directly_public(input.tcx, adt.did()), |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 473 | "Not directly public type (re-exports are not supported yet - b/262052635)" |
| 474 | ); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 475 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 476 | let def_id = adt.did(); |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 477 | let mut prereqs = CcPrerequisites::default(); |
| 478 | if def_id.krate == LOCAL_CRATE { |
| 479 | prereqs.defs.insert(def_id.expect_local()); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 480 | } else { |
Lukasz Anforowicz | cfe8455 | 2023-04-20 13:38:54 -0700 | [diff] [blame^] | 481 | 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 490 | |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 491 | // Verify if definition of `ty` can be succesfully imported and bail otherwise. |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 492 | format_adt_core(input.tcx, def_id).with_context(|| { |
| 493 | format!("Failed to generate bindings for the definition of `{ty}`") |
| 494 | })?; |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 495 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 496 | CcSnippet { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 497 | tokens: FullyQualifiedName::new(input.tcx, def_id).format_for_cc()?, |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 498 | prereqs, |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 499 | } |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 500 | } |
| 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 Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 508 | format_ty_for_cc(input, *ty, TypeLocation::Other).with_context(|| { |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 509 | 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 Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 514 | |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 515 | 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 Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 552 | // TODO(b/260268230, b/260729464): When recursively processing nested types (e.g. an |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 553 | // 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 Anforowicz | 087dff7 | 2023-02-17 12:13:32 -0800 | [diff] [blame] | 557 | _ => bail!("The following Rust type is not supported yet: {ty}"), |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 558 | }) |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 559 | } |
| 560 | |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 561 | fn 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 Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 566 | fn 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 Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 574 | Ok(format_ty_for_cc(input, ty, TypeLocation::FnParam) |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 575 | .with_context(|| format!("Error handling parameter #{i}"))?) |
| 576 | }) |
| 577 | .collect() |
| 578 | } |
| 579 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 580 | /// 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 Anforowicz | 1ab5e87 | 2022-12-05 10:07:00 -0800 | [diff] [blame] | 585 | // |
| 586 | // TODO(b/259724276): This function's results should be memoized. |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 587 | fn format_ty_for_rs(tcx: TyCtxt, ty: Ty) -> Result<TokenStream> { |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 588 | 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 Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 594 | | ty::TyKind::FnPtr(_) |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 595 | | 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 Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 607 | ty::TyKind::Adt(adt, substs) => { |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 608 | ensure!(substs.len() == 0, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 609 | FullyQualifiedName::new(tcx, adt.did()).format_for_rs() |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 610 | } |
| 611 | ty::TyKind::RawPtr(ty::TypeAndMut { ty, mutbl }) => { |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 612 | let qualifier = match mutbl { |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 613 | Mutability::Mut => quote! { mut }, |
| 614 | Mutability::Not => quote! { const }, |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 615 | }; |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 616 | 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 Anforowicz | 087dff7 | 2023-02-17 12:13:32 -0800 | [diff] [blame] | 621 | _ => bail!("The following Rust type is not supported yet: {ty}"), |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 622 | }) |
| 623 | } |
| 624 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 625 | #[derive(Debug, Default)] |
| 626 | struct ApiSnippets { |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 627 | /// 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 630 | main_api: CcSnippet, |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 631 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 632 | /// C++ implementation details - for example: |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 633 | /// - A C++ declaration of an `extern "C"` thunk, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 634 | /// - C++ `static_assert`s about struct size, aligment, and field offsets. |
| 635 | cc_details: CcSnippet, |
| 636 | |
| 637 | /// Rust implementation details - for exmaple: |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 638 | /// - A Rust implementation of an `extern "C"` thunk, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 639 | /// - Rust `assert!`s about struct size, aligment, and field offsets. |
| 640 | rs_details: TokenStream, |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 641 | } |
| 642 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 643 | impl 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 Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 652 | } |
| 653 | } |
| 654 | |
Lukasz Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 655 | fn 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 Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 662 | /// 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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 665 | fn format_thunk_decl( |
| 666 | input: &Input, |
| 667 | fn_def_id: LocalDefId, |
| 668 | thunk_name: &TokenStream, |
| 669 | ) -> Result<CcSnippet> { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 670 | let tcx = input.tcx; |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 671 | |
| 672 | let mut prereqs = CcPrerequisites::default(); |
| 673 | let sig = get_fn_sig(tcx, fn_def_id)?; |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 674 | let main_api_ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut prereqs); |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 675 | |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 676 | 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 Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 699 | |
| 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 Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 719 | /// 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` |
| 729 | fn 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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 791 | fn 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 | |
| 803 | fn get_thunk_name(symbol_name: &str) -> String { |
| 804 | format!("__crubit_thunk_{}", &escape_non_identifier_chars(symbol_name)) |
| 805 | } |
| 806 | |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 807 | fn 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. |
| 826 | fn 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 | |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 857 | /// Formats a function with the given `local_def_id`. |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 858 | /// |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 859 | /// Will panic if `local_def_id` |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 860 | /// - is invalid |
| 861 | /// - doesn't identify a function, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 862 | fn format_fn(input: &Input, local_def_id: LocalDefId) -> Result<ApiSnippets> { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 863 | let tcx = input.tcx; |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 864 | let def_id: DefId = local_def_id.to_def_id(); // Convert LocalDefId to DefId. |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 865 | |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 866 | ensure!( |
| 867 | tcx.generics_of(def_id).count() == 0, |
| 868 | "Generic functions are not supported yet (b/259749023)" |
| 869 | ); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 870 | |
Lukasz Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 871 | let sig = get_fn_sig(tcx, local_def_id)?; |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 872 | check_fn_sig(&sig)?; |
| 873 | let needs_thunk = is_thunk_required(&sig).is_err(); |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 874 | 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 | }; |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 878 | |
Lukasz Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 879 | let FullyQualifiedName { krate, mod_path, name } = FullyQualifiedName::new(tcx, def_id); |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 880 | 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 Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 883 | |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 884 | let mut main_api_prereqs = CcPrerequisites::default(); |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 885 | let main_api_ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut main_api_prereqs); |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 886 | 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 Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 901 | ty: Ty<'tcx>, |
| 902 | } |
| 903 | let params = { |
| 904 | let names = tcx.fn_arg_names(def_id).iter(); |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 905 | let cc_types = format_param_types_for_cc(input, &sig)?; |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 906 | names |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 907 | .enumerate() |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 908 | .zip(sig.inputs().iter()) |
| 909 | .zip(cc_types.into_iter()) |
| 910 | .map(|(((i, name), &ty), cc_type)| { |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 911 | let cc_name = format_cc_ident(name.as_str()) |
| 912 | .unwrap_or_else(|_err| format_cc_ident(&format!("__param_{i}")).unwrap()); |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 913 | let cc_type = cc_type.into_tokens(&mut main_api_prereqs); |
| 914 | Param { cc_name, cc_type, ty } |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 915 | }) |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 916 | .collect_vec() |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 917 | }; |
| 918 | let main_api_params = params |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 919 | .iter() |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 920 | .map(|Param { cc_name, cc_type, .. }| quote! { #cc_type #cc_name }) |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 921 | .collect_vec(); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 922 | |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 923 | 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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 936 | let needs_definition = fn_name.as_str() != thunk_name; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 937 | 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 Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 943 | let mut prereqs = main_api_prereqs.clone(); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 944 | prereqs.move_defs_to_fwd_decls(); |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 945 | |
| 946 | let static_ = if is_static_method { |
| 947 | quote! { static } |
| 948 | } else { |
| 949 | quote! {} |
| 950 | }; |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 951 | let extern_c_or_inline = if !needs_definition { |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 952 | quote! { extern "C" } |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 953 | } else { |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 954 | quote! { inline } |
| 955 | }; |
| 956 | CcSnippet { |
| 957 | prereqs, |
| 958 | tokens: quote! { |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 959 | __NEWLINE__ |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 960 | #doc_comment |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 961 | #static_ #extern_c_or_inline |
| 962 | #main_api_ret_type #main_api_fn_name ( #( #main_api_params ),* ); |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 963 | __NEWLINE__ |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 964 | }, |
| 965 | } |
| 966 | }; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 967 | let cc_details = if !needs_definition { |
| 968 | CcSnippet::default() |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 969 | } else { |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 970 | let thunk_name = format_cc_ident(&thunk_name).context("Error formatting thunk name")?; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 971 | 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 Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 981 | let thunk_decl = |
| 982 | format_thunk_decl(input, local_def_id, &thunk_name)?.into_tokens(&mut prereqs); |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 983 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 984 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 994 | let impl_body: TokenStream; |
| 995 | if is_c_abi_compatible_by_value(sig.output()) { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 996 | impl_body = quote! { |
| 997 | return __crubit_internal :: #thunk_name( #( #thunk_args ),* ); |
| 998 | }; |
| 999 | } else { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1000 | 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 Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 1013 | #thunk_decl |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1014 | 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 Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 1026 | 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 Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1034 | } |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 1035 | }; |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 1036 | let fully_qualified_fn_name = quote! { :: #crate_name :: #mod_path #struct_name #fn_name }; |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1037 | format_thunk_impl(tcx, local_def_id, &thunk_name, fully_qualified_fn_name)? |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1038 | }; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1039 | Ok(ApiSnippets { main_api, cc_details, rs_details }) |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1040 | } |
| 1041 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1042 | /// 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 Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1048 | /// #keyword #alignment #name final { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1049 | /// #core |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1050 | /// #decls_of_other_parts // (e.g. struct fields, methods, etc.) |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1051 | /// } |
| 1052 | /// } |
| 1053 | /// ``` |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1054 | /// |
| 1055 | /// `keyword`, `name` are stored separately, to support formatting them as a |
| 1056 | /// forward declaration - e.g. `struct SomeStruct`. |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1057 | struct AdtCoreBindings { |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1058 | /// DefId of the ADT. |
| 1059 | def_id: DefId, |
| 1060 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1061 | /// 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 Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1066 | /// C++ translation of the ADT identifier - e.g. `SomeStruct`. |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1067 | /// |
| 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1073 | |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1074 | /// Rust spelling of the ADT type - e.g. |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1075 | /// `::some_crate::some_module::SomeStruct`. |
| 1076 | rs_fully_qualified_name: TokenStream, |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1077 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1078 | /// `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 Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1087 | alignment_in_bytes: u64, |
| 1088 | size_in_bytes: u64, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1089 | } |
| 1090 | |
Lukasz Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 1091 | /// Like `TyCtxt::is_directly_public`, but works not only with `LocalDefId`, but |
| 1092 | /// also with `DefId`. |
| 1093 | fn 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 Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1106 | fn 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1119 | /// 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 Anforowicz | 1ab5e87 | 2022-12-05 10:07:00 -0800 | [diff] [blame] | 1137 | // TODO(b/259724276): This function's results should be memoized. |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1138 | fn format_adt_core(tcx: TyCtxt, def_id: DefId) -> Result<AdtCoreBindings> { |
Lukasz Anforowicz | 0c07b09 | 2023-03-03 10:34:15 -0800 | [diff] [blame] | 1139 | let ty = tcx.type_of(def_id).subst_identity(); |
Lukasz Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 1140 | assert!(ty.is_adt()); |
| 1141 | assert!(is_directly_public(tcx, def_id), "Caller should verify"); |
| 1142 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1143 | // TODO(b/259749095): Support non-empty set of generic parameters. |
| 1144 | let param_env = ty::ParamEnv::empty(); |
| 1145 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1146 | 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 Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 1151 | let adt_def = ty.ty_adt_def().expect("`def_id` needs to identify an ADT"); |
| 1152 | let keyword = match adt_def.adt_kind() { |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 1153 | ty::AdtKind::Struct | ty::AdtKind::Enum => quote! { struct }, |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 1154 | ty::AdtKind::Union => quote! { union }, |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 1155 | }; |
| 1156 | |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1157 | let item_name = tcx.item_name(def_id); |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1158 | 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 Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1161 | |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1162 | let layout = get_layout(tcx, ty) |
| 1163 | .with_context(|| format!("Error computing the layout of #{item_name}"))?; |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1164 | let alignment_in_bytes = { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1165 | // 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 Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1169 | layout.align().abi.bytes() |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1170 | }; |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1171 | let size_in_bytes = layout.size().bytes(); |
| 1172 | ensure!(size_in_bytes != 0, "Zero-sized types (ZSTs) are not supported (b/258259459)"); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1173 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1174 | let core = quote! { |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1175 | // 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1181 | |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1182 | // 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1215 | |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1216 | // 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 Anforowicz | 554ed65 | 2023-01-12 15:41:58 -0800 | [diff] [blame] | 1225 | |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1226 | // TODO(b/258251148): Support custom `Drop` impls and drop glue. |
| 1227 | ~#cc_short_name() = default; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1228 | }; |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1229 | Ok(AdtCoreBindings { |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1230 | def_id, |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 1231 | keyword, |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1232 | cc_short_name, |
| 1233 | rs_fully_qualified_name, |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1234 | core, |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1235 | alignment_in_bytes, |
| 1236 | size_in_bytes, |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1237 | }) |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1238 | } |
| 1239 | |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1240 | fn format_fields(input: &Input, core: &AdtCoreBindings) -> ApiSnippets { |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 1241 | let tcx = input.tcx; |
| 1242 | |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1243 | // TODO(b/259749095): Support non-empty set of generic parameters. |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1244 | let substs_ref = ty::List::empty().as_substs(); |
| 1245 | |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1246 | struct FieldTypeInfo { |
| 1247 | size: u64, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1248 | cc_type: CcSnippet, |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1249 | } |
| 1250 | struct Field { |
| 1251 | type_info: Result<FieldTypeInfo>, |
| 1252 | cc_name: TokenStream, |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 1253 | rs_name: TokenStream, |
| 1254 | is_public: bool, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1255 | index: usize, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1256 | offset: u64, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1257 | offset_of_next_field: u64, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1258 | } |
| 1259 | let ty = tcx.type_of(core.def_id).subst_identity(); |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1260 | let layout = |
| 1261 | get_layout(tcx, ty).expect("Layout should be already verified by `format_adt_core`"); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1262 | let fields: Vec<Field> = if ty.is_enum() || ty.is_union() { |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1263 | // Note that `#[repr(Rust)]` unions don't guarantee that all their fields |
| 1264 | // have offset 0. |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1265 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1277 | } else { |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1278 | let mut fields = ty |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1279 | .ty_adt_def() |
| 1280 | .expect("`core.def_id` needs to identify an ADT") |
| 1281 | .all_fields() |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1282 | .sorted_by_key(|f| tcx.def_span(f.did)) |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1283 | .enumerate() |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1284 | .map(|(index, field_def)| { |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1285 | let field_ty = field_def.ty(tcx, substs_ref); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1286 | let size = get_layout(tcx, field_ty).map(|layout| layout.size().bytes()); |
| 1287 | let type_info = size.and_then(|size| { |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 1288 | Ok(FieldTypeInfo { size, cc_type: format_ty_for_cc(input, field_ty, TypeLocation::Other)? }) |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1289 | }); |
| 1290 | let name = field_def.ident(tcx); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 1291 | let cc_name = format_cc_ident(name.as_str()) |
| 1292 | .unwrap_or_else(|_err| format_ident!("__field{index}").into_token_stream()); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 1293 | 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 Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1309 | |
| 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 Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1315 | Field { |
| 1316 | type_info, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1317 | cc_name, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1318 | rs_name, |
| 1319 | is_public, |
| 1320 | index, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1321 | offset, |
| 1322 | offset_of_next_field, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1323 | } |
| 1324 | }) |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1325 | .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 Jeanpierre | 9e15d0b | 2023-04-06 13:18:22 -0700 | [diff] [blame] | 1335 | // 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 Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1337 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1350 | }; |
| 1351 | |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1352 | let cc_details = if fields.is_empty() { |
| 1353 | CcSnippet::default() |
| 1354 | } else { |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1355 | let adt_cc_name = &core.cc_short_name; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1356 | 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 Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 1366 | } |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1367 | }) |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1368 | }; |
| 1369 | let rs_details: TokenStream = { |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1370 | let adt_rs_name = &core.rs_fully_qualified_name; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1371 | 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 Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 1381 | let main_api = { |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1382 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1388 | }; |
| 1389 | |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1390 | let mut prereqs = CcPrerequisites::default(); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1391 | 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 Jeanpierre | 9e15d0b | 2023-04-06 13:18:22 -0700 | [diff] [blame] | 1397 | let size = field.offset_of_next_field - field.offset; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1398 | let msg = |
| 1399 | format!("Field type has been replaced with a blob of bytes: {err:#}"); |
Devin Jeanpierre | 9e15d0b | 2023-04-06 13:18:22 -0700 | [diff] [blame] | 1400 | |
| 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 Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1417 | } |
| 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1431 | } |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1432 | }) |
| 1433 | .collect(); |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1434 | |
| 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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1449 | /// 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. |
| 1458 | fn 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). |
| 1484 | fn 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 Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1519 | /// 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. |
| 1523 | fn format_adt(input: &Input, core: &AdtCoreBindings) -> ApiSnippets { |
| 1524 | let tcx = input.tcx; |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1525 | let adt_cc_name = &core.cc_short_name; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1526 | |
| 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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1531 | 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 Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1546 | 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 Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1582 | 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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1600 | let default_ctor_main_api = default_ctor_main_api.into_tokens(&mut prereqs); |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1601 | let impl_items_main_api = if impl_items_main_api.tokens.is_empty() { |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1602 | quote! {} |
| 1603 | } else { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1604 | let tokens = impl_items_main_api.into_tokens(&mut prereqs); |
| 1605 | quote! { public: #tokens } |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1606 | }; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1607 | let fields_main_api = fields_main_api.into_tokens(&mut prereqs); |
Lukasz Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 1608 | prereqs.fwd_decls.remove(&local_def_id); |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1609 | |
| 1610 | CcSnippet { |
| 1611 | prereqs, |
| 1612 | tokens: quote! { |
| 1613 | __NEWLINE__ #doc_comment |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 1614 | #keyword alignas(#alignment) #cc_packed_attribute #adt_cc_name final { |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1615 | public: |
| 1616 | #default_ctor_main_api |
| 1617 | #core |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1618 | #impl_items_main_api |
| 1619 | #fields_main_api |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1620 | }; |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1621 | __NEWLINE__ |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1622 | }, |
| 1623 | } |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1624 | }; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1625 | let cc_details = { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1626 | let mut prereqs = CcPrerequisites::default(); |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1627 | let default_ctor_cc_details = default_ctor_cc_details.into_tokens(&mut prereqs); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1628 | let impl_items_cc_details = impl_items_cc_details.into_tokens(&mut prereqs); |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1629 | let fields_cc_details = fields_cc_details.into_tokens(&mut prereqs); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1630 | prereqs.defs.insert(local_def_id); |
| 1631 | CcSnippet { |
| 1632 | prereqs, |
| 1633 | tokens: quote! { |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 1634 | __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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1642 | #default_ctor_cc_details |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1643 | #impl_items_cc_details |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1644 | #fields_cc_details |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1645 | }, |
| 1646 | } |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1647 | }; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1648 | let rs_details = { |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1649 | let adt_rs_name = &core.rs_fully_qualified_name; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1650 | quote! { |
| 1651 | const _: () = assert!(::std::mem::size_of::<#adt_rs_name>() == #size); |
| 1652 | const _: () = assert!(::std::mem::align_of::<#adt_rs_name>() == #alignment); |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1653 | #default_ctor_rs_details |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1654 | #impl_items_rs_details |
| 1655 | #fields_rs_details |
| 1656 | } |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1657 | }; |
| 1658 | ApiSnippets { main_api, cc_details, rs_details } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1659 | } |
| 1660 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1661 | /// 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`. |
| 1667 | fn 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 Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1673 | let AdtCoreBindings { keyword, cc_short_name, .. } = format_adt_core(tcx, def_id) |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1674 | .expect("`format_fwd_decl` should only be called if `format_adt_core` succeeded"); |
| 1675 | |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1676 | quote! { #keyword #cc_short_name; } |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1677 | } |
| 1678 | |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1679 | fn 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 Anforowicz | 1f23391 | 2023-02-14 08:50:26 -0800 | [diff] [blame] | 1691 | // brackets, thus don't prepend "google3/" prefix. |
| 1692 | if file.name.is_real() { "google3/" } else { "" } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1693 | }; |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 1694 | format!("{google3_prefix}{file_name};l={line_number}") |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1695 | } |
| 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. |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 1700 | fn format_doc_comment(tcx: TyCtxt, local_def_id: LocalDefId) -> TokenStream { |
| 1701 | let hir_id = tcx.local_def_id_to_hir_id(local_def_id); |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1702 | let doc_comment = tcx |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 1703 | .hir() |
| 1704 | .attrs(hir_id) |
| 1705 | .iter() |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 1706 | .filter_map(|attr| attr.doc_str()) |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1707 | .map(|symbol| symbol.to_string()) |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 1708 | .chain(once(format!("Generated from: {}", format_source_location(tcx, local_def_id)))) |
Googler | 34f3d57 | 2022-12-02 00:53:37 -0800 | [diff] [blame] | 1709 | .join("\n\n"); |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1710 | quote! { __COMMENT__ #doc_comment} |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 1711 | } |
| 1712 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1713 | /// Formats a HIR item idenfied by `def_id`. Returns `None` if the item |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1714 | /// can be ignored. Returns an `Err` if the definition couldn't be formatted. |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1715 | /// |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1716 | /// Will panic if `def_id` is invalid (i.e. doesn't identify a HIR item). |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1717 | fn format_item(input: &Input, def_id: LocalDefId) -> Result<Option<ApiSnippets>> { |
Lukasz Anforowicz | 1382f39 | 2022-12-12 17:13:23 -0800 | [diff] [blame] | 1718 | // TODO(b/262052635): When adding support for re-exports we may need to change |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 1719 | // `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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1724 | if !input.tcx.effective_visibilities(()).is_directly_public(def_id) { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1725 | return Ok(None); |
Lukasz Anforowicz | 8b68a55 | 2022-12-12 15:07:58 -0800 | [diff] [blame] | 1726 | } |
| 1727 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1728 | match input.tcx.hir().expect_item(def_id) { |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 1729 | Item { kind: ItemKind::Struct(_, generics) | |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 1730 | ItemKind::Enum(_, generics) | |
| 1731 | ItemKind::Union(_, generics), |
| 1732 | .. } if !generics.params.is_empty() => { |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 1733 | bail!("Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 1734 | }, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1735 | Item { kind: ItemKind::Fn(..), .. } => format_fn(input, def_id).map(Some), |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 1736 | Item { kind: ItemKind::Struct(..) | ItemKind::Enum(..) | ItemKind::Union(..), .. } => |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1737 | format_adt_core(input.tcx, def_id.to_def_id()) |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1738 | .map(|core| Some(format_adt(input, &core))), |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 1739 | Item { kind: ItemKind::Impl(_), .. } | // Handled by `format_adt` |
| 1740 | Item { kind: ItemKind::Mod(_), .. } => // Handled by `format_crate` |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1741 | Ok(None), |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1742 | Item { kind, .. } => bail!("Unsupported rustc_hir::hir::ItemKind: {}", kind.descr()), |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1743 | } |
| 1744 | } |
| 1745 | |
| 1746 | /// Formats a C++ comment explaining why no bindings have been generated for |
| 1747 | /// `local_def_id`. |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 1748 | fn format_unsupported_def( |
| 1749 | tcx: TyCtxt, |
| 1750 | local_def_id: LocalDefId, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1751 | err: anyhow::Error, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1752 | ) -> ApiSnippets { |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 1753 | let source_loc = format_source_location(tcx, local_def_id); |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 1754 | let name = tcx.def_path_str(local_def_id.to_def_id()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1755 | |
| 1756 | // https://docs.rs/anyhow/latest/anyhow/struct.Error.html#display-representations |
| 1757 | // says: To print causes as well [...], use the alternate selector “{:#}”. |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 1758 | let msg = format!("Error generating bindings for `{name}` defined at {source_loc}: {err:#}"); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1759 | let main_api = CcSnippet::new(quote! { __NEWLINE__ __NEWLINE__ __COMMENT__ #msg __NEWLINE__ }); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1760 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1761 | ApiSnippets { main_api, cc_details: CcSnippet::default(), rs_details: quote! {} } |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 1762 | } |
| 1763 | |
Lukasz Anforowicz | 2ec1312 | 2022-11-10 12:39:04 -0800 | [diff] [blame] | 1764 | /// Formats all public items from the Rust crate being compiled. |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 1765 | fn format_crate(input: &Input) -> Result<Output> { |
| 1766 | let tcx = input.tcx; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1767 | let mut cc_details_prereqs = CcPrerequisites::default(); |
| 1768 | let mut cc_details: Vec<(LocalDefId, TokenStream)> = vec![]; |
| 1769 | let mut rs_body = TokenStream::default(); |
Lukasz Anforowicz | 991b2a5 | 2023-03-23 07:28:41 -0700 | [diff] [blame] | 1770 | let mut main_apis = HashMap::<LocalDefId, CcSnippet>::new(); |
| 1771 | let formatted_items = tcx |
Lukasz Anforowicz | 2ec1312 | 2022-11-10 12:39:04 -0800 | [diff] [blame] | 1772 | .hir() |
| 1773 | .items() |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1774 | .filter_map(|item_id| { |
Lukasz Anforowicz | 2ec1312 | 2022-11-10 12:39:04 -0800 | [diff] [blame] | 1775 | let def_id: LocalDefId = item_id.owner_id.def_id; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1776 | format_item(input, def_id) |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1777 | .unwrap_or_else(|err| Some(format_unsupported_def(tcx, def_id, err))) |
| 1778 | .map(|api_snippets| (def_id, api_snippets)) |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 1779 | }) |
Lukasz Anforowicz | 991b2a5 | 2023-03-23 07:28:41 -0700 | [diff] [blame] | 1780 | .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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1784 | |
Lukasz Anforowicz | 991b2a5 | 2023-03-23 07:28:41 -0700 | [diff] [blame] | 1785 | // `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 Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 1792 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1793 | // Find the order of `main_apis` that 1) meets the requirements of |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 1794 | // `CcPrerequisites::defs` and 2) makes a best effort attempt to keep the |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1795 | // `main_apis` in the same order as the source order of the Rust APIs. |
Lukasz Anforowicz | 2ecb27f | 2023-01-12 15:29:51 -0800 | [diff] [blame] | 1796 | let ordered_ids = { |
| 1797 | let toposort::TopoSortResult { ordered: ordered_ids, failed: failed_ids } = { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1798 | 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 Anforowicz | 2ecb27f | 2023-01-12 15:29:51 -0800 | [diff] [blame] | 1801 | predecessors.map(move |predecessor| toposort::Dependency { predecessor, successor }) |
| 1802 | }); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1803 | toposort::toposort(nodes, deps, move |lhs_id, rhs_id| { |
| 1804 | tcx.def_span(*lhs_id).cmp(&tcx.def_span(*rhs_id)) |
| 1805 | }) |
Lukasz Anforowicz | 2ecb27f | 2023-01-12 15:29:51 -0800 | [diff] [blame] | 1806 | }; |
| 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 Anforowicz | ab563af | 2022-12-15 08:09:50 -0800 | [diff] [blame] | 1815 | }; |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 1816 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1817 | // 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 Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1820 | let mut already_declared = HashSet::new(); |
| 1821 | let mut fwd_decls = HashSet::new(); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1822 | 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 Anforowicz | 7f31f80 | 2022-12-16 08:24:13 -0800 | [diff] [blame] | 1831 | } |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1832 | } = main_apis.remove(&def_id).unwrap(); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1833 | |
| 1834 | fwd_decls.extend(inner_fwd_decls.difference(&already_declared).copied()); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1835 | already_declared.insert(def_id); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1836 | already_declared.extend(inner_fwd_decls.into_iter()); |
| 1837 | |
| 1838 | includes.append(&mut inner_includes); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1839 | ordered_main_apis.push((def_id, cc_tokens)); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1840 | } |
| 1841 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1842 | let fwd_decls = fwd_decls |
| 1843 | .into_iter() |
| 1844 | .sorted_by_key(|def_id| tcx.def_span(*def_id)) |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 1845 | .map(|local_def_id| (local_def_id, format_fwd_decl(tcx, local_def_id))); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1846 | |
| 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 Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1852 | let mod_path = FullyQualifiedName::new(tcx, local_def_id.to_def_id()).mod_path; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1853 | (mod_path, tokens) |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1854 | }) |
| 1855 | .collect_vec(); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1856 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1857 | (includes, ordered_cc) |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1858 | }; |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 1859 | |
| 1860 | // Generate top-level elements of the C++ header file. |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1861 | let h_body = { |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1862 | // 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 Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1866 | |
Lukasz Anforowicz | 54efc16 | 2022-12-16 15:58:44 -0800 | [diff] [blame] | 1867 | let includes = format_cc_includes(&includes); |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 1868 | let ordered_cc = format_namespace_bound_cc_tokens(ordered_cc); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1869 | quote! { |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1870 | #includes |
| 1871 | __NEWLINE__ __NEWLINE__ |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1872 | namespace #crate_name { |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1873 | __NEWLINE__ |
Lukasz Anforowicz | 54efc16 | 2022-12-16 15:58:44 -0800 | [diff] [blame] | 1874 | #ordered_cc |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1875 | __NEWLINE__ |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1876 | } |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1877 | __NEWLINE__ |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1878 | } |
| 1879 | }; |
Lukasz Anforowicz | 7f31f80 | 2022-12-16 08:24:13 -0800 | [diff] [blame] | 1880 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 1881 | Ok(Output { h_body, rs_body }) |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 1882 | } |
| 1883 | |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 1884 | #[cfg(test)] |
Lukasz Anforowicz | f018e4d | 2022-09-28 07:35:59 -0700 | [diff] [blame] | 1885 | pub mod tests { |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 1886 | use super::*; |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 1887 | |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1888 | use anyhow::Result; |
| 1889 | use itertools::Itertools; |
| 1890 | use proc_macro2::TokenStream; |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1891 | use quote::quote; |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 1892 | use rustc_middle::ty::{Ty, TyCtxt}; |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1893 | use rustc_span::def_id::LocalDefId; |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 1894 | |
Lukasz Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 1895 | use crate::run_compiler::tests::run_compiler_for_testing; |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 1896 | use code_gen_utils::format_cc_includes; |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 1897 | use token_stream_matchers::{ |
| 1898 | assert_cc_matches, assert_cc_not_matches, assert_rs_matches, assert_rs_not_matches, |
| 1899 | }; |
Lukasz Anforowicz | 2b38d27 | 2022-09-23 08:08:18 -0700 | [diff] [blame] | 1900 | |
Lukasz Anforowicz | 5bddf18 | 2022-09-30 16:06:59 -0700 | [diff] [blame] | 1901 | #[test] |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1902 | #[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 Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 1915 | run_compiler_for_testing(test_src, |tcx| find_def_id_by_name(tcx, "missing_name")); |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1916 | } |
| 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 Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 1929 | run_compiler_for_testing(test_src, |tcx| find_def_id_by_name(tcx, "some_name")); |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1930 | } |
| 1931 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1932 | /// 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1936 | /// by `test_format_item_..._fn_...` tests (which work at the `format_fn` |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1937 | /// level). |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1938 | #[test] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1939 | fn test_generated_bindings_fn_no_mangle_extern_c() { |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 1940 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 1941 | #[no_mangle] |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1942 | pub extern "C" fn public_function() { |
| 1943 | println!("foo"); |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 1944 | } |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 1945 | "#; |
| 1946 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1947 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1948 | assert_cc_matches!( |
| 1949 | bindings.h_body, |
| 1950 | quote! { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1951 | extern "C" void public_function(); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1952 | } |
Lukasz Anforowicz | d9ff4ab | 2022-09-23 08:11:18 -0700 | [diff] [blame] | 1953 | ); |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1954 | |
| 1955 | // No Rust thunks should be generated in this test scenario. |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 1956 | assert_rs_not_matches!(bindings.rs_body, quote! { public_function }); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1957 | }); |
| 1958 | } |
| 1959 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1960 | /// `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 Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1963 | #[test] |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1964 | fn test_generated_bindings_fn_export_name() { |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1965 | 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 Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1970 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1971 | assert_cc_matches!( |
| 1972 | bindings.h_body, |
| 1973 | quote! { |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1974 | namespace rust_out { |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 1975 | ... |
| 1976 | inline double public_function(double x, double y); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1977 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 1978 | extern "C" double export_name(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1979 | } |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1980 | inline double public_function(double x, double y) { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1981 | return __crubit_internal::export_name(x, y); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1982 | } |
| 1983 | } |
| 1984 | } |
| 1985 | ); |
| 1986 | }); |
| 1987 | } |
| 1988 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1989 | /// 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1992 | /// `test_format_item_..._struct_...`, `test_format_item_..._enum_...`, |
| 1993 | /// and `test_format_item_..._union_...` tests. |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1994 | /// |
| 1995 | /// We don't want to duplicate coverage already provided by |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1996 | /// `test_format_item_struct_with_fields`, but we do want to verify that |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1997 | /// * `format_crate` will actually find and process the struct |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1998 | /// (`test_format_item_...` doesn't cover this aspect - it uses a |
| 1999 | /// test-only `find_def_id_by_name` instead) |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2000 | /// * The actual shape of the bindings still looks okay at this level. |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2001 | #[test] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2002 | fn test_generated_bindings_struct() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2003 | 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 Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2010 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2011 | assert_cc_matches!( |
| 2012 | bindings.h_body, |
| 2013 | quote! { |
| 2014 | namespace rust_out { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2015 | ... |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2016 | struct alignas(4) Point final { |
| 2017 | // No point replicating test coverage of |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2018 | // `test_format_item_struct_with_fields`. |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2019 | ... |
| 2020 | }; |
| 2021 | static_assert(sizeof(Point) == 8, ...); |
| 2022 | static_assert(alignof(Point) == 4, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 2023 | ... // Other static_asserts are covered by |
| 2024 | // `test_format_item_struct_with_fields` |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2025 | } // namespace rust_out |
| 2026 | } |
| 2027 | ); |
| 2028 | assert_rs_matches!( |
| 2029 | bindings.rs_body, |
| 2030 | quote! { |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 2031 | // No point replicating test coverage of |
| 2032 | // `test_format_item_struct_with_fields`. |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 2033 | const _: () = assert!(::std::mem::size_of::<::rust_out::Point>() == 8); |
| 2034 | const _: () = assert!(::std::mem::align_of::<::rust_out::Point>() == 4); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 2035 | const _: () = assert!( memoffset::offset_of!(::rust_out::Point, x) == 0); |
| 2036 | const _: () = assert!( memoffset::offset_of!(::rust_out::Point, y) == 4); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2037 | } |
| 2038 | ); |
| 2039 | }); |
| 2040 | } |
| 2041 | |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2042 | /// 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 Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2045 | /// |
| 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 Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2050 | #[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 Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2071 | // `test_format_item_static_method`. |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2072 | ... |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2073 | std::int32_t public_static_method(); |
| 2074 | ... |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2075 | }; |
| 2076 | ... |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2077 | std::int32_t SomeStruct::public_static_method() { |
| 2078 | ... |
| 2079 | } |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 2080 | ... |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2081 | } // namespace rust_out |
| 2082 | } |
| 2083 | ); |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2084 | 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 Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2092 | }); |
| 2093 | } |
| 2094 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2095 | #[test] |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2096 | fn test_generated_bindings_includes() { |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 2097 | 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 Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2106 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 2107 | assert_cc_matches!( |
| 2108 | bindings.h_body, |
| 2109 | quote! { |
| 2110 | __HASH_TOKEN__ include <cstdint> ... |
| 2111 | namespace ... { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2112 | ... |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 2113 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2123 | /// Tests that `toposort` is used to reorder item bindings. |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 2124 | #[test] |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2125 | fn test_generated_bindings_prereq_defs_field_deps_require_reordering() { |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2126 | let test_src = r#" |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2127 | // In the generated bindings `Outer` needs to come *after* `Inner`. |
| 2128 | pub struct Outer(Inner); |
| 2129 | pub struct Inner(bool); |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2130 | "#; |
| 2131 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2132 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2133 | assert_cc_matches!( |
| 2134 | bindings.h_body, |
| 2135 | quote! { |
| 2136 | namespace rust_out { |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2137 | ... |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2138 | struct alignas(1) Inner final { |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 2139 | ... bool __field0; ... |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2140 | }; |
| 2141 | ... |
| 2142 | struct alignas(1) Outer final { |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 2143 | ... ::rust_out::Inner __field0; ... |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2144 | }; |
| 2145 | ... |
| 2146 | } // namespace rust_out |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2147 | } |
| 2148 | ); |
| 2149 | }); |
| 2150 | } |
| 2151 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2152 | /// Tests that a forward declaration is present when it is required to |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2153 | /// preserve the original source order. In this test the |
| 2154 | /// `CcPrerequisites::fwd_decls` dependency comes from a pointer parameter. |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2155 | #[test] |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2156 | fn test_generated_bindings_prereq_fwd_decls_for_ptr_param() { |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2157 | 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 Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2170 | // it the essence of this test. The order of the items |
| 2171 | // below also matters. |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2172 | struct S; |
| 2173 | ... |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2174 | inline void f(const ::rust_out::S* __param_0); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2175 | ... |
| 2176 | struct alignas(...) S final { ... } |
| 2177 | ... |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2178 | inline void f(const ::rust_out::S* __param_0) { ... } |
| 2179 | ... |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2180 | } // namespace rust_out |
| 2181 | } |
| 2182 | ); |
| 2183 | }); |
| 2184 | } |
| 2185 | |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2186 | /// Tests that a forward declaration is present when it is required to |
| 2187 | /// preserve the original source order. In this test the |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2188 | /// `CcPrerequisites::fwd_decls` dependency comes from a |
| 2189 | /// function declaration that has a parameter that takes a struct by value. |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2190 | #[test] |
| 2191 | fn test_generated_bindings_prereq_fwd_decls_for_cpp_fn_decl() { |
| 2192 | let test_src = r#" |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2193 | #[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 Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2208 | // is the essence of this test. The order also matters: |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2209 | // 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 Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2214 | // `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 Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2217 | ... |
| 2218 | struct alignas(...) S final { ... } |
| 2219 | ... |
| 2220 | } // namespace rust_out |
| 2221 | } |
| 2222 | ); |
| 2223 | }); |
| 2224 | } |
| 2225 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2226 | /// 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 Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 2318 | fn test_generated_bindings_prereq_fwd_decls_not_needed_because_of_initial_order() { |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2319 | 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 Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 2328 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2337 | #![allow(dead_code)] |
| 2338 | |
| 2339 | pub struct S { |
| 2340 | // This shouldn't require a fwd decl of S. |
| 2341 | field: *const S, |
| 2342 | } |
Lukasz Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 2343 | |
| 2344 | impl S { |
| 2345 | // This shouldn't require a fwd decl of S. |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2346 | pub fn create() -> S { Self{ field: std::ptr::null() } } |
Lukasz Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 2347 | } |
| 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2352 | assert_cc_matches!( |
| 2353 | bindings.h_body, |
| 2354 | quote! { |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 2355 | static inline ::rust_out::S create(); ... |
| 2356 | const ::rust_out::S* field; ... |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2357 | } |
| 2358 | ); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2359 | }); |
| 2360 | } |
| 2361 | |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2362 | #[test] |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2363 | fn test_generated_bindings_module_basics() { |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2364 | let test_src = r#" |
| 2365 | pub mod some_module { |
| 2366 | pub fn some_func() {} |
| 2367 | } |
| 2368 | "#; |
| 2369 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2370 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2371 | assert_cc_matches!( |
| 2372 | bindings.h_body, |
| 2373 | quote! { |
| 2374 | namespace rust_out { |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2375 | 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 Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2396 | #[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 Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2427 | inline void working_module_f1(); |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2428 | ... |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2429 | inline void working_module_f2(); |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2430 | ... |
| 2431 | } // namespace some_module |
| 2432 | |
| 2433 | __COMMENT__ #broken_module_msg |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2434 | ... |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2435 | } // namespace rust_out |
| 2436 | } |
| 2437 | ); |
| 2438 | }); |
| 2439 | } |
| 2440 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2441 | /// `test_generated_bindings_non_pub_items` verifies that non-public items |
| 2442 | /// are not present/propagated into the generated bindings. |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2443 | #[test] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2444 | fn test_generated_bindings_non_pub_items() { |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2445 | let test_src = r#" |
Lukasz Anforowicz | 88bde9b | 2022-10-14 14:48:07 -0700 | [diff] [blame] | 2446 | #![allow(dead_code)] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2447 | |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2448 | extern "C" fn private_function() { |
| 2449 | println!("foo"); |
| 2450 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2451 | |
| 2452 | struct PrivateStruct { |
| 2453 | x: i32, |
| 2454 | y: i32, |
| 2455 | } |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2456 | |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 2457 | pub struct PublicStruct(i32); |
| 2458 | |
| 2459 | impl PublicStruct { |
| 2460 | fn private_method() {} |
| 2461 | } |
| 2462 | |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2463 | 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 Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2471 | "#; |
| 2472 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2473 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2474 | assert_cc_not_matches!(bindings.h_body, quote! { private_function }); |
Lukasz Anforowicz | e7a2500 | 2022-11-10 06:21:42 -0800 | [diff] [blame] | 2475 | assert_rs_not_matches!(bindings.rs_body, quote! { private_function }); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2476 | assert_cc_not_matches!(bindings.h_body, quote! { PrivateStruct }); |
| 2477 | assert_rs_not_matches!(bindings.rs_body, quote! { PrivateStruct }); |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 2478 | assert_cc_not_matches!(bindings.h_body, quote! { private_method }); |
| 2479 | assert_rs_not_matches!(bindings.rs_body, quote! { private_method }); |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2480 | 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 Anforowicz | 1382f39 | 2022-12-12 17:13:23 -0800 | [diff] [blame] | 2484 | 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 Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2488 | }); |
| 2489 | } |
| 2490 | |
| 2491 | #[test] |
Lukasz Anforowicz | 4aa1ff9 | 2022-10-10 11:22:22 -0700 | [diff] [blame] | 2492 | fn test_generated_bindings_top_level_items() { |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2493 | let test_src = "pub fn public_function() {}"; |
| 2494 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2495 | let bindings = bindings.unwrap(); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 2496 | let expected_comment_txt = "Automatically @generated C++ bindings for the following Rust crate:\n\ |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2497 | rust_out"; |
| 2498 | assert_cc_matches!( |
| 2499 | bindings.h_body, |
| 2500 | quote! { |
| 2501 | __COMMENT__ #expected_comment_txt |
Lukasz Anforowicz | 4aa1ff9 | 2022-10-10 11:22:22 -0700 | [diff] [blame] | 2502 | ... |
Lukasz Anforowicz | 31b29cd | 2022-10-10 11:33:41 -0700 | [diff] [blame] | 2503 | __HASH_TOKEN__ pragma once |
| 2504 | ... |
Lukasz Anforowicz | 4aa1ff9 | 2022-10-10 11:22:22 -0700 | [diff] [blame] | 2505 | namespace rust_out { |
| 2506 | ... |
| 2507 | } |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2508 | } |
| 2509 | ); |
Lukasz Anforowicz | e7a2500 | 2022-11-10 06:21:42 -0800 | [diff] [blame] | 2510 | assert_cc_matches!( |
| 2511 | bindings.rs_body, |
| 2512 | quote! { |
| 2513 | __COMMENT__ #expected_comment_txt |
| 2514 | } |
| 2515 | ); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2516 | }) |
| 2517 | } |
| 2518 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2519 | /// The `test_generated_bindings_unsupported_item` test verifies how `Err` |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2520 | /// from `format_item` is formatted as a C++ comment (in `format_crate` |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2521 | /// and `format_unsupported_def`): |
| 2522 | /// - This test covers only a single example of an unsupported item. |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2523 | /// Additional coverage is provided by `test_format_item_unsupported_...` |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2524 | /// 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 Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2529 | #[test] |
| 2530 | fn test_generated_bindings_unsupported_item() { |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2531 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2532 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2533 | pub extern "C" fn reinterpret_cast() {} |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2534 | "#; |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2535 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2536 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2537 | let expected_comment_txt = "Error generating bindings for `reinterpret_cast` \ |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 2538 | defined at <crubit_unittests.rs>;l=3: \ |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2539 | Error formatting function name: \ |
| 2540 | `reinterpret_cast` is a C++ reserved keyword \ |
| 2541 | and can't be used as a C++ identifier"; |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2542 | assert_cc_matches!( |
| 2543 | bindings.h_body, |
| 2544 | quote! { |
| 2545 | __COMMENT__ #expected_comment_txt |
| 2546 | } |
| 2547 | ); |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 2548 | }) |
| 2549 | } |
| 2550 | |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 2551 | #[test] |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 2552 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2599 | fn test_format_item_fn_extern_c_no_mangle_no_params_no_return_type() { |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 2600 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2601 | #[no_mangle] |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 2602 | pub extern "C" fn public_function() {} |
| 2603 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2604 | test_format_item(test_src, "public_function", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2605 | let result = result.unwrap().unwrap(); |
| 2606 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2607 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2608 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2609 | main_api.tokens, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2610 | quote! { |
| 2611 | extern "C" void public_function(); |
| 2612 | } |
| 2613 | ); |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2614 | |
| 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 Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2621 | }); |
| 2622 | } |
| 2623 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2624 | /// The `test_format_item_fn_explicit_unit_return_type` test below is very |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2625 | /// similar to the |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2626 | /// `test_format_item_fn_extern_c_no_mangle_no_params_no_return_type` above, |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2627 | /// 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 Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2633 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2634 | fn test_format_item_fn_explicit_unit_return_type() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2635 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2636 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2637 | pub extern "C" fn explicit_unit_return_type() -> () {} |
| 2638 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2639 | test_format_item(test_src, "explicit_unit_return_type", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2640 | let result = result.unwrap().unwrap(); |
| 2641 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2642 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2643 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2644 | main_api.tokens, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2645 | quote! { |
| 2646 | extern "C" void explicit_unit_return_type(); |
| 2647 | } |
| 2648 | ); |
| 2649 | }); |
| 2650 | } |
| 2651 | |
| 2652 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2653 | fn test_format_item_fn_never_return_type() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2654 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2660 | test_format_item(test_src, "never_returning_function", |result| { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2661 | // 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2665 | let result = result.unwrap().unwrap(); |
| 2666 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2667 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2668 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2669 | main_api.tokens, |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2670 | quote! { |
| 2671 | extern "C" void never_returning_function(); |
| 2672 | } |
| 2673 | ); |
| 2674 | }) |
| 2675 | } |
| 2676 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2677 | /// `test_format_item_fn_mangling` checks that bindings can be generated for |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2678 | /// `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 Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2682 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2683 | fn test_format_item_fn_mangling() { |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2684 | let test_src = r#" |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2685 | pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y } |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2686 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2687 | test_format_item(test_src, "public_function", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2688 | let result = result.unwrap().unwrap(); |
| 2689 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2690 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2691 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2692 | main_api.tokens, |
| 2693 | quote! { |
| 2694 | inline double public_function(double x, double y); |
| 2695 | } |
| 2696 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2697 | assert!(result.rs_details.is_empty()); |
| 2698 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2699 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2700 | result.cc_details.tokens, |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2701 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2702 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 2703 | extern "C" double ...(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2704 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2705 | ... |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2706 | inline double public_function(double x, double y) { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2707 | return __crubit_internal::...(x, y); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2708 | } |
| 2709 | } |
| 2710 | ); |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2711 | }); |
| 2712 | } |
| 2713 | |
| 2714 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2715 | fn test_format_item_fn_export_name() { |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2716 | let test_src = r#" |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2717 | #[export_name = "export_name"] |
| 2718 | pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y } |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2719 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2720 | test_format_item(test_src, "public_function", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2721 | let result = result.unwrap().unwrap(); |
| 2722 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2723 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2724 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2725 | main_api.tokens, |
| 2726 | quote! { |
| 2727 | inline double public_function(double x, double y); |
| 2728 | } |
| 2729 | ); |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2730 | |
| 2731 | // There is no need to have a separate thunk for an `extern "C"` function. |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2732 | assert!(result.rs_details.is_empty()); |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2733 | |
| 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2736 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2737 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2738 | result.cc_details.tokens, |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2739 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2740 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 2741 | extern "C" double export_name(double, double); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2742 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2743 | ... |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2744 | inline double public_function(double x, double y) { |
| 2745 | return __crubit_internal::export_name(x, y); |
| 2746 | } |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2747 | } |
| 2748 | ); |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2749 | }); |
| 2750 | } |
| 2751 | |
| 2752 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2753 | fn test_format_item_unsupported_fn_unsafe() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2754 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2755 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2756 | pub unsafe extern "C" fn foo() {} |
| 2757 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2758 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2759 | let err = result.unwrap_err(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2760 | assert_eq!( |
| 2761 | err, |
| 2762 | "Bindings for `unsafe` functions \ |
| 2763 | are not fully designed yet (b/254095482)" |
| 2764 | ); |
| 2765 | }); |
| 2766 | } |
| 2767 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2768 | /// `test_format_item_fn_const` tests how bindings for an `const fn` are |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2769 | /// generated. |
| 2770 | /// |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2771 | /// 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 Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2774 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2775 | fn test_format_item_fn_const() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2776 | let test_src = r#" |
| 2777 | pub const fn foo(i: i32) -> i32 { i * 42 } |
| 2778 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2779 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2780 | // TODO(b/254095787): Update test expectations below once `const fn` from Rust |
| 2781 | // is translated into a `consteval` C++ function. |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2782 | let result = result.unwrap().unwrap(); |
| 2783 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2784 | assert!(!main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2785 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2786 | main_api.tokens, |
| 2787 | quote! { |
| 2788 | inline std::int32_t foo(std::int32_t i); |
| 2789 | } |
| 2790 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2791 | assert!(!result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2792 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2793 | result.cc_details.tokens, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2794 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2795 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 2796 | extern "C" std::int32_t ...( std::int32_t); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2797 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2798 | ... |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2799 | inline std::int32_t foo(std::int32_t i) { |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 2800 | return __crubit_internal::...(i); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2801 | } |
| 2802 | } |
| 2803 | ); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2804 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2805 | result.rs_details, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2806 | quote! { |
| 2807 | #[no_mangle] |
| 2808 | extern "C" |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 2809 | fn ...(i: i32) -> i32 { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 2810 | ::rust_out::foo(i) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2811 | } |
| 2812 | } |
| 2813 | ); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2814 | }); |
| 2815 | } |
| 2816 | |
| 2817 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2818 | fn test_format_item_fn_with_c_unwind_abi() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2819 | // See also https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html |
| 2820 | let test_src = r#" |
| 2821 | #![feature(c_unwind)] |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2822 | |
| 2823 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2824 | pub extern "C-unwind" fn may_throw() {} |
| 2825 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2826 | test_format_item(test_src, "may_throw", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2827 | let result = result.unwrap().unwrap(); |
| 2828 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2829 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2830 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2831 | main_api.tokens, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2832 | quote! { |
| 2833 | extern "C" void may_throw(); |
| 2834 | } |
| 2835 | ); |
| 2836 | }); |
| 2837 | } |
| 2838 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2839 | /// This test mainly verifies that `format_item` correctly propagates |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2840 | /// `CcPrerequisites` of parameter types and return type. |
| 2841 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2842 | fn test_format_item_fn_cc_prerequisites_if_cpp_definition_needed() { |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2843 | let test_src = r#" |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2844 | pub fn foo(_i: i32) -> S { panic!("foo") } |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2845 | pub struct S(i32); |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2846 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2847 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2848 | let result = result.unwrap().unwrap(); |
| 2849 | let main_api = &result.main_api; |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2850 | |
| 2851 | // Minimal coverage, just to double-check that the test setup works. |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2852 | // |
| 2853 | // Note that this is a definition, and therefore `S` should be defined |
| 2854 | // earlier (not just forward declared). |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2855 | assert_cc_matches!(main_api.tokens, quote! { S foo(std::int32_t _i);}); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2856 | assert_cc_matches!(result.cc_details.tokens, quote! { S foo(std::int32_t _i) { ... }}); |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2857 | |
| 2858 | // Main checks: `CcPrerequisites::includes`. |
| 2859 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2860 | format_cc_includes(&main_api.prereqs.includes), |
| 2861 | quote! { include <cstdint> } |
| 2862 | ); |
| 2863 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2864 | format_cc_includes(&result.cc_details.prereqs.includes), |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2865 | quote! { include <cstdint> } |
| 2866 | ); |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2867 | |
| 2868 | // Main checks: `CcPrerequisites::defs` and `CcPrerequisites::fwd_decls`. |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2869 | // |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2870 | // Verifying the actual def_id is tricky, because `test_format_item` doesn't |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2871 | // expose `tcx` to the verification function (and therefore calling |
| 2872 | // `find_def_id_by_name` is not easily possible). |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2873 | // |
| 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2877 | assert_eq!(1, result.cc_details.prereqs.defs.len()); |
| 2878 | assert_eq!(0, result.cc_details.prereqs.fwd_decls.len()); |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2879 | }); |
| 2880 | } |
| 2881 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2882 | /// This test verifies that `format_item` uses `CcPrerequisites::fwd_decls` |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2883 | /// rather than `CcPrerequisites::defs` for function declarations in the |
| 2884 | /// `main_api`. |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2885 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2886 | fn test_format_item_fn_cc_prerequisites_if_only_cpp_declaration_needed() { |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2887 | let test_src = r#" |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2888 | #[no_mangle] |
| 2889 | pub extern "C" fn foo(s: S) -> bool { s.0 } |
| 2890 | |
| 2891 | #[repr(C)] |
| 2892 | pub struct S(bool); |
| 2893 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2894 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2895 | let result = result.unwrap().unwrap(); |
| 2896 | let main_api = &result.main_api; |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2897 | |
| 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 Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2903 | assert_cc_matches!(main_api.tokens, quote! { inline bool foo(::rust_out::S s); }); |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2904 | |
| 2905 | // Main checks: `CcPrerequisites::defs` and `CcPrerequisites::fwd_decls`. |
| 2906 | // |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2907 | // Verifying the actual def_id is tricky, because `test_format_item` doesn't |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2908 | // expose `tcx` to the verification function (and therefore calling |
| 2909 | // `find_def_id_by_name` is not easily possible). |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2910 | assert_eq!(0, main_api.prereqs.defs.len()); |
| 2911 | assert_eq!(1, main_api.prereqs.fwd_decls.len()); |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2912 | }); |
| 2913 | } |
| 2914 | |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2915 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2916 | fn test_format_item_fn_with_type_aliased_return_type() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2917 | // 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 Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 2923 | type MyTypeAlias = f64; |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2924 | |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2925 | #[no_mangle] |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 2926 | pub extern "C" fn type_aliased_return() -> MyTypeAlias { 42.0 } |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2927 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2928 | test_format_item(test_src, "type_aliased_return", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2929 | let result = result.unwrap().unwrap(); |
| 2930 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2931 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2932 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2933 | main_api.tokens, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2934 | quote! { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 2935 | extern "C" double type_aliased_return(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2936 | } |
| 2937 | ); |
| 2938 | }); |
| 2939 | } |
| 2940 | |
| 2941 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2942 | fn test_format_item_fn_with_doc_comment_with_unmangled_name() { |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2943 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2951 | test_format_item(test_src, "fn_with_doc_comment_with_unmangled_name", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2952 | let result = result.unwrap().unwrap(); |
| 2953 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2954 | assert!(main_api.prereqs.is_empty()); |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2955 | let doc_comments = [ |
| 2956 | " Outer line doc.", |
Googler | 34f3d57 | 2022-12-02 00:53:37 -0800 | [diff] [blame] | 2957 | "", |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2958 | " Outer block doc that spans lines.", |
| 2959 | " ", |
Googler | 34f3d57 | 2022-12-02 00:53:37 -0800 | [diff] [blame] | 2960 | "", |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2961 | "Doc comment via doc attribute.", |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2962 | "", |
| 2963 | "Generated from: <crubit_unittests.rs>;l=7", |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2964 | ] |
| 2965 | .join("\n"); |
| 2966 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2967 | main_api.tokens, |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2968 | quote! { |
| 2969 | __COMMENT__ #doc_comments |
| 2970 | extern "C" void fn_with_doc_comment_with_unmangled_name(); |
| 2971 | } |
| 2972 | ); |
| 2973 | }); |
| 2974 | } |
| 2975 | |
| 2976 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2977 | fn test_format_item_fn_with_inner_doc_comment_with_unmangled_name() { |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2978 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2985 | test_format_item(test_src, "fn_with_inner_doc_comment_with_unmangled_name", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2986 | let result = result.unwrap().unwrap(); |
| 2987 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2988 | assert!(main_api.prereqs.is_empty()); |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2989 | let doc_comments = [ |
| 2990 | " Outer doc comment.", |
| 2991 | " Inner doc comment.", |
| 2992 | "Generated from: <crubit_unittests.rs>;l=4", |
| 2993 | ] |
| 2994 | .join("\n\n"); |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2995 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2996 | main_api.tokens, |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2997 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3006 | fn test_format_item_fn_with_doc_comment_with_mangled_name() { |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3007 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3011 | test_format_item(test_src, "fn_with_doc_comment_with_mangled_name", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3012 | let result = result.unwrap().unwrap(); |
| 3013 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3014 | assert!(main_api.prereqs.is_empty()); |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3015 | let comment = " Doc comment of a function with mangled name.\n\n\ |
| 3016 | Generated from: <crubit_unittests.rs>;l=3"; |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3017 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3018 | main_api.tokens, |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3019 | quote! { |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3020 | __COMMENT__ #comment |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3021 | inline void fn_with_doc_comment_with_mangled_name(); |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3022 | } |
| 3023 | ); |
| 3024 | }); |
| 3025 | } |
| 3026 | |
| 3027 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3028 | fn test_format_item_unsupported_fn_name_is_reserved_cpp_keyword() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3029 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 3030 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3031 | pub extern "C" fn reinterpret_cast() -> () {} |
| 3032 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3033 | test_format_item(test_src, "reinterpret_cast", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3034 | let err = result.unwrap_err(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3035 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3045 | fn test_format_item_unsupported_fn_ret_type() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3046 | let test_src = r#" |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 3047 | pub fn foo() -> (i32, i32) { (123, 456) } |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3048 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3049 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3050 | let err = result.unwrap_err(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3051 | assert_eq!( |
| 3052 | err, |
| 3053 | "Error formatting function return type: \ |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 3054 | Tuples are not supported yet: (i32, i32) (b/254099023)" |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3055 | ); |
| 3056 | }); |
| 3057 | } |
| 3058 | |
| 3059 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3060 | fn test_format_item_unsupported_fn_with_late_bound_lifetimes() { |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3061 | // TODO(b/258235219): Expect success after adding support for references. |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3062 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3071 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3072 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3073 | assert_eq!(err, "Generic functions are not supported yet (b/259749023)"); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3074 | }); |
| 3075 | } |
| 3076 | |
| 3077 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3078 | fn test_format_item_unsupported_generic_fn() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3079 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3086 | test_format_item(test_src, "generic_function", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3087 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3088 | assert_eq!(err, "Generic functions are not supported yet (b/259749023)"); |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 3089 | }); |
| 3090 | } |
| 3091 | |
| 3092 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3093 | fn test_format_item_unsupported_generic_struct() { |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3094 | let test_src = r#" |
| 3095 | pub struct Point<T> { |
| 3096 | pub x: T, |
| 3097 | pub y: T, |
| 3098 | } |
| 3099 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3100 | test_format_item(test_src, "Point", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3101 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3102 | assert_eq!(err, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3103 | }); |
| 3104 | } |
| 3105 | |
| 3106 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3107 | fn test_format_item_unsupported_generic_enum() { |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3108 | let test_src = r#" |
| 3109 | pub enum Point<T> { |
| 3110 | Cartesian{x: T, y: T}, |
| 3111 | Polar{angle: T, dist: T}, |
| 3112 | } |
| 3113 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3114 | test_format_item(test_src, "Point", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3115 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3116 | assert_eq!(err, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3117 | }); |
| 3118 | } |
| 3119 | |
| 3120 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3121 | fn test_format_item_unsupported_generic_union() { |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3122 | let test_src = r#" |
| 3123 | pub union SomeUnion<T> { |
| 3124 | pub x: std::mem::ManuallyDrop<T>, |
| 3125 | pub y: i32, |
| 3126 | } |
| 3127 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3128 | test_format_item(test_src, "SomeUnion", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3129 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3130 | assert_eq!(err, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3131 | }); |
| 3132 | } |
| 3133 | |
| 3134 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3135 | fn test_format_item_unsupported_fn_async() { |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 3136 | let test_src = r#" |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3137 | pub async fn async_function() {} |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 3138 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3139 | test_format_item(test_src, "async_function", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3140 | let err = result.unwrap_err(); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3141 | assert_eq!( |
| 3142 | err, |
| 3143 | "Error formatting function return type: \ |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3144 | The following Rust type is not supported yet: \ |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3145 | impl std::future::Future<Output = ()>" |
| 3146 | ); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3147 | }); |
| 3148 | } |
| 3149 | |
| 3150 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3151 | fn test_format_item_fn_rust_abi() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3152 | let test_src = r#" |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3153 | pub fn add(x: f64, y: f64) -> f64 { x * y } |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3154 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3155 | test_format_item(test_src, "add", |result| { |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3156 | // 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3159 | let result = result.unwrap().unwrap(); |
| 3160 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3161 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3162 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3163 | main_api.tokens, |
| 3164 | quote! { |
| 3165 | inline double add(double x, double y); |
| 3166 | } |
| 3167 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3168 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3169 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3170 | result.cc_details.tokens, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3171 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3172 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3173 | extern "C" double ...(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3174 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3175 | ... |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3176 | inline double add(double x, double y) { |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 3177 | return __crubit_internal::...(x, y); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3178 | } |
| 3179 | } |
| 3180 | ); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3181 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3182 | result.rs_details, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3183 | quote! { |
| 3184 | #[no_mangle] |
| 3185 | extern "C" |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 3186 | fn ...(x: f64, y: f64) -> f64 { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3187 | ::rust_out::add(x, y) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3188 | } |
| 3189 | } |
| 3190 | ); |
| 3191 | }); |
| 3192 | } |
| 3193 | |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3194 | #[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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3201 | let result = result.unwrap().unwrap(); |
| 3202 | let main_api = &result.main_api; |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3203 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3210 | result.cc_details.tokens, |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3211 | quote! { |
| 3212 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3213 | extern "C" std::int32_t ...(::rust_out::S*); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3214 | } |
| 3215 | ... |
| 3216 | inline std::int32_t into_i32(::rust_out::S s) { |
Lukasz Anforowicz | d082f35 | 2023-03-09 17:46:11 -0800 | [diff] [blame] | 3217 | return __crubit_internal::...(&s); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3218 | } |
| 3219 | } |
| 3220 | ); |
| 3221 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3222 | result.rs_details, |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3223 | quote! { |
| 3224 | #[no_mangle] |
| 3225 | extern "C" |
Lukasz Anforowicz | d082f35 | 2023-03-09 17:46:11 -0800 | [diff] [blame] | 3226 | fn ...(s: &mut ::core::mem::MaybeUninit<::rust_out::S>) -> i32 { |
| 3227 | ::rust_out::into_i32(unsafe { s.assume_init_read() }) |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3228 | } |
| 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3241 | let result = result.unwrap().unwrap(); |
| 3242 | let main_api = &result.main_api; |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3243 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3250 | result.cc_details.tokens, |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3251 | quote! { |
| 3252 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3253 | extern "C" void ...(std::int32_t, ::rust_out::S* __ret_ptr); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3254 | } |
| 3255 | ... |
| 3256 | inline ::rust_out::S create(std::int32_t i) { |
Lukasz Anforowicz | a3b7db0 | 2023-03-09 17:34:05 -0800 | [diff] [blame] | 3257 | crubit::ReturnValueSlot<::rust_out::S> __ret_slot; |
| 3258 | __crubit_internal::...(i, __ret_slot.Get()); |
| 3259 | return std::move(__ret_slot).AssumeInitAndTakeValue(); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3260 | } |
| 3261 | } |
| 3262 | ); |
| 3263 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3264 | result.rs_details, |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3265 | quote! { |
| 3266 | #[no_mangle] |
| 3267 | extern "C" |
Lukasz Anforowicz | a3b7db0 | 2023-03-09 17:34:05 -0800 | [diff] [blame] | 3268 | fn ...( |
| 3269 | i: i32, |
| 3270 | __ret_slot: &mut ::core::mem::MaybeUninit<::rust_out::S> |
| 3271 | ) -> () { |
| 3272 | __ret_slot.write(::rust_out::create(i)); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3273 | } |
| 3274 | } |
| 3275 | ); |
| 3276 | }); |
| 3277 | } |
| 3278 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3279 | /// `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 Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3283 | /// |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3284 | /// This test almost entirely replicates `test_format_item_fn_rust_abi`, |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3285 | /// except for the `extern "vectorcall"` part in the `test_src` test |
| 3286 | /// input. |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3287 | /// |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3288 | /// 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 Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3292 | /// (see also `format_cc_call_conv_as_clang_attribute` in |
| 3293 | /// `rs_bindings_from_cc/src_code_gen.rs`) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3294 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3295 | fn test_format_item_fn_vectorcall_abi() { |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3296 | let test_src = r#" |
| 3297 | #![feature(abi_vectorcall)] |
| 3298 | pub extern "vectorcall" fn add(x: f64, y: f64) -> f64 { x * y } |
| 3299 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3300 | test_format_item(test_src, "add", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3301 | let result = result.unwrap().unwrap(); |
| 3302 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3303 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3304 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3305 | main_api.tokens, |
| 3306 | quote! { |
| 3307 | inline double add(double x, double y); |
| 3308 | } |
| 3309 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3310 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3311 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3312 | result.cc_details.tokens, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3313 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3314 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3315 | extern "C" double ...(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3316 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3317 | ... |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3318 | inline double add(double x, double y) { |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 3319 | return __crubit_internal::...(x, y); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3320 | } |
| 3321 | } |
| 3322 | ); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3323 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3324 | result.rs_details, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3325 | quote! { |
| 3326 | #[no_mangle] |
| 3327 | extern "C" |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 3328 | fn ...(x: f64, y: f64) -> f64 { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3329 | ::rust_out::add(x, y) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3330 | } |
| 3331 | } |
| 3332 | ); |
| 3333 | }); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3334 | } |
| 3335 | |
| 3336 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3337 | fn test_format_item_unsupported_fn_variadic() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3338 | let test_src = r#" |
| 3339 | #![feature(c_variadic)] |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 3340 | |
| 3341 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3342 | pub unsafe extern "C" fn variadic_function(_fmt: *const u8, ...) {} |
| 3343 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3344 | test_format_item(test_src, "variadic_function", |result| { |
Lukasz Anforowicz | 13794df | 2022-10-21 07:56:34 -0700 | [diff] [blame] | 3345 | // TODO(b/254097223): Add support for variadic functions. |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3346 | let err = result.unwrap_err(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3347 | assert_eq!(err, "C variadic functions are not supported (b/254097223)"); |
| 3348 | }); |
| 3349 | } |
| 3350 | |
| 3351 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3352 | fn test_format_item_fn_params() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3353 | let test_src = r#" |
| 3354 | #[allow(unused_variables)] |
| 3355 | #[no_mangle] |
| 3356 | pub extern "C" fn foo(b: bool, f: f64) {} |
| 3357 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3358 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3359 | let result = result.unwrap().unwrap(); |
| 3360 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3361 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3362 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3363 | main_api.tokens, |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3364 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3365 | ... |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3366 | extern "C" void foo(bool b, double f); |
| 3367 | } |
| 3368 | ); |
| 3369 | }); |
| 3370 | } |
| 3371 | |
| 3372 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3373 | fn test_format_item_fn_param_name_reserved_keyword() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3374 | let test_src = r#" |
| 3375 | #[allow(unused_variables)] |
| 3376 | #[no_mangle] |
| 3377 | pub extern "C" fn some_function(reinterpret_cast: f64) {} |
| 3378 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3379 | test_format_item(test_src, "some_function", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3380 | let result = result.unwrap().unwrap(); |
| 3381 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3382 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3383 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3384 | main_api.tokens, |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3385 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3386 | ... |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3387 | extern "C" void some_function(double __param_0); |
| 3388 | } |
| 3389 | ); |
| 3390 | }); |
| 3391 | } |
| 3392 | |
| 3393 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3394 | fn test_format_item_fn_with_multiple_anonymous_parameter_names() { |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3395 | let test_src = r#" |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3396 | pub fn foo(_: f64, _: f64) {} |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3397 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3398 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3399 | let result = result.unwrap().unwrap(); |
| 3400 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3401 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3402 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3403 | main_api.tokens, |
| 3404 | quote! { |
| 3405 | inline void foo(double __param_0, double __param_1); |
| 3406 | } |
| 3407 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3408 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3409 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3410 | result.cc_details.tokens, |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3411 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3412 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3413 | extern "C" void ...(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3414 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3415 | ... |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3416 | inline void foo(double __param_0, double __param_1) { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3417 | return __crubit_internal::...(__param_0, __param_1); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3418 | } |
| 3419 | } |
| 3420 | ); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3421 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3422 | result.rs_details, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3423 | quote! { |
| 3424 | #[no_mangle] |
| 3425 | extern "C" fn ...(__param_0: f64, __param_1: f64) -> () { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3426 | ::rust_out::foo(__param_0, __param_1) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3427 | } |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3428 | } |
| 3429 | ); |
| 3430 | }); |
| 3431 | } |
| 3432 | |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3433 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3434 | fn test_format_item_fn_with_destructuring_parameter_name() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3435 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3447 | test_format_item(test_src, "func", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3448 | let result = result.unwrap().unwrap(); |
| 3449 | let main_api = &result.main_api; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3450 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3451 | main_api.tokens, |
| 3452 | quote! { |
| 3453 | inline std::int32_t func(::rust_out::S __param_0); |
| 3454 | } |
| 3455 | ); |
| 3456 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3457 | result.cc_details.tokens, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3458 | quote! { |
| 3459 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3460 | extern "C" std::int32_t ...(::rust_out::S*); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3461 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3462 | ... |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3463 | inline std::int32_t func(::rust_out::S __param_0) { |
Lukasz Anforowicz | d082f35 | 2023-03-09 17:46:11 -0800 | [diff] [blame] | 3464 | return __crubit_internal::...(&__param_0); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3465 | } |
| 3466 | } |
| 3467 | ); |
| 3468 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3469 | result.rs_details, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3470 | quote! { |
| 3471 | #[no_mangle] |
Lukasz Anforowicz | d082f35 | 2023-03-09 17:46:11 -0800 | [diff] [blame] | 3472 | 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3476 | } |
| 3477 | } |
| 3478 | ); |
| 3479 | }); |
| 3480 | } |
| 3481 | |
| 3482 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3483 | fn test_format_item_unsupported_fn_param_type() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3484 | let test_src = r#" |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 3485 | pub fn foo(_param: (i32, i32)) {} |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3486 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3487 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3488 | let err = result.unwrap_err(); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3489 | assert_eq!( |
| 3490 | err, |
| 3491 | "Error handling parameter #0: \ |
| 3492 | Tuples are not supported yet: (i32, i32) (b/254099023)" |
| 3493 | ); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3494 | }); |
| 3495 | } |
| 3496 | |
| 3497 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3498 | fn test_format_item_unsupported_fn_param_type_unit() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3499 | let test_src = r#" |
| 3500 | #[no_mangle] |
| 3501 | pub fn fn_with_params(_param: ()) {} |
| 3502 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3503 | test_format_item(test_src, "fn_with_params", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3504 | let err = result.unwrap_err(); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3505 | assert_eq!( |
| 3506 | err, |
| 3507 | "Error handling parameter #0: \ |
| 3508 | `()` / `void` is only supported as a return type (b/254507801)" |
| 3509 | ); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3510 | }); |
| 3511 | } |
| 3512 | |
| 3513 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3514 | fn test_format_item_unsupported_fn_param_type_never() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3515 | let test_src = r#" |
| 3516 | #![feature(never_type)] |
| 3517 | |
| 3518 | #[no_mangle] |
| 3519 | pub extern "C" fn fn_with_params(_param: !) {} |
| 3520 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3521 | test_format_item(test_src, "fn_with_params", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3522 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3523 | assert_eq!( |
| 3524 | err, |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 3525 | "Error handling parameter #0: \ |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3526 | The never type `!` is only supported as a return type (b/254507801)" |
| 3527 | ); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3528 | }); |
| 3529 | } |
| 3530 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3531 | /// 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3535 | fn test_format_item_struct_with_fields() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3536 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3545 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3546 | let result = result.unwrap().unwrap(); |
| 3547 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3548 | assert!(!main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3549 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3550 | main_api.tokens, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3551 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3552 | ... |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3553 | struct alignas(4) SomeStruct final { |
| 3554 | public: |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 3555 | __COMMENT__ "`SomeStruct` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3556 | SomeStruct() = delete; |
| 3557 | |
| 3558 | // In this test there is no `Copy` implementation / derive. |
| 3559 | SomeStruct(const SomeStruct&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3560 | |
| 3561 | // All Rust types are trivially-movable. |
| 3562 | SomeStruct(SomeStruct&&) = default; |
Lukasz Anforowicz | 554ed65 | 2023-01-12 15:41:58 -0800 | [diff] [blame] | 3563 | |
| 3564 | // Assignment operators are disabled for now. |
| 3565 | SomeStruct& operator=(const SomeStruct&) = delete; |
| 3566 | SomeStruct& operator=(SomeStruct&&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3567 | |
| 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3572 | ... std::int32_t x; |
| 3573 | ... std::int32_t y; |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3574 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3575 | }; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3576 | } |
| 3577 | ); |
| 3578 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3579 | result.cc_details.tokens, |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3580 | quote! { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3581 | static_assert(sizeof(SomeStruct) == 8, ...); |
| 3582 | static_assert(alignof(SomeStruct) == 4, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3583 | inline void SomeStruct::__crubit_field_offset_assertions() { |
| 3584 | static_assert(0 == offsetof(SomeStruct, x)); |
| 3585 | static_assert(4 == offsetof(SomeStruct, y)); |
| 3586 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3587 | } |
| 3588 | ); |
| 3589 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3590 | result.rs_details, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3591 | quote! { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3592 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8); |
| 3593 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3594 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, x) == 0); |
| 3595 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, y) == 4); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3596 | } |
| 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3604 | fn test_format_item_struct_with_tuple() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3605 | let test_src = r#" |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3606 | pub struct TupleStruct(pub i32, pub i32); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3607 | const _: () = assert!(std::mem::size_of::<TupleStruct>() == 8); |
| 3608 | const _: () = assert!(std::mem::align_of::<TupleStruct>() == 4); |
| 3609 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3610 | test_format_item(test_src, "TupleStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3611 | let result = result.unwrap().unwrap(); |
| 3612 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3613 | assert!(!main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3614 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3615 | main_api.tokens, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3616 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3617 | ... |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3618 | struct alignas(4) TupleStruct final { |
| 3619 | public: |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 3620 | __COMMENT__ "`TupleStruct` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3621 | TupleStruct() = delete; |
| 3622 | |
| 3623 | // In this test there is no `Copy` implementation / derive. |
| 3624 | TupleStruct(const TupleStruct&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3625 | |
| 3626 | // All Rust types are trivially-movable. |
| 3627 | TupleStruct(TupleStruct&&) = default; |
Lukasz Anforowicz | 554ed65 | 2023-01-12 15:41:58 -0800 | [diff] [blame] | 3628 | |
| 3629 | // Assignment operators are disabled for now. |
| 3630 | TupleStruct& operator=(const TupleStruct&) = delete; |
| 3631 | TupleStruct& operator=(TupleStruct&&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3632 | |
| 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3637 | ... std::int32_t __field0; |
| 3638 | ... std::int32_t __field1; |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3639 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3640 | }; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3641 | } |
| 3642 | ); |
| 3643 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3644 | result.cc_details.tokens, |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3645 | quote! { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3646 | static_assert(sizeof(TupleStruct) == 8, ...); |
| 3647 | static_assert(alignof(TupleStruct) == 4, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3648 | inline void TupleStruct::__crubit_field_offset_assertions() { |
| 3649 | static_assert(0 == offsetof(TupleStruct, __field0)); |
| 3650 | static_assert(4 == offsetof(TupleStruct, __field1)); |
| 3651 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3652 | } |
| 3653 | ); |
| 3654 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3655 | result.rs_details, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3656 | quote! { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3657 | const _: () = assert!(::std::mem::size_of::<::rust_out::TupleStruct>() == 8); |
| 3658 | const _: () = assert!(::std::mem::align_of::<::rust_out::TupleStruct>() == 4); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3659 | const _: () = assert!( memoffset::offset_of!(::rust_out::TupleStruct, 0) == 0); |
| 3660 | const _: () = assert!( memoffset::offset_of!(::rust_out::TupleStruct, 1) == 4); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3661 | } |
| 3662 | ); |
| 3663 | }); |
| 3664 | } |
| 3665 | |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3666 | /// 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3681 | let result = result.unwrap().unwrap(); |
| 3682 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3683 | 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 Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3697 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3698 | }; |
| 3699 | } |
| 3700 | ); |
| 3701 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3702 | result.cc_details.tokens, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3703 | quote! { |
| 3704 | static_assert(sizeof(SomeStruct) == 8, ...); |
| 3705 | static_assert(alignof(SomeStruct) == 4, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3706 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3711 | } |
| 3712 | ); |
| 3713 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3714 | result.rs_details, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3715 | quote! { |
| 3716 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8); |
| 3717 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3718 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3724 | } |
| 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3741 | let result = result.unwrap().unwrap(); |
| 3742 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3743 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3751 | std::uint32_t field2; |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3752 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3753 | }; |
| 3754 | } |
| 3755 | ); |
| 3756 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3757 | result.cc_details.tokens, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3758 | quote! { |
| 3759 | static_assert(sizeof(SomeStruct) == 6, ...); |
| 3760 | static_assert(alignof(SomeStruct) == 1, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3761 | inline void SomeStruct::__crubit_field_offset_assertions() { |
| 3762 | static_assert(0 == offsetof(SomeStruct, field1)); |
| 3763 | static_assert(2 == offsetof(SomeStruct, field2)); |
| 3764 | } |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3765 | } |
| 3766 | ); |
| 3767 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3768 | result.rs_details, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3769 | quote! { |
| 3770 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 6); |
| 3771 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 1); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3772 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field1) |
| 3773 | == 0); |
| 3774 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field2) |
| 3775 | == 2); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3776 | } |
| 3777 | ); |
| 3778 | }); |
| 3779 | } |
| 3780 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3781 | #[test] |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 3782 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3792 | let result = result.unwrap().unwrap(); |
| 3793 | let main_api = &result.main_api; |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 3794 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3809 | result.cc_details.tokens, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 3810 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3820 | result.rs_details, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 3821 | 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 Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3832 | fn test_format_item_static_method() { |
| 3833 | let test_src = r#" |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3834 | /// No-op `f32` placeholder is used, because ZSTs are not supported |
| 3835 | /// (b/258259459). |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3836 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3845 | let result = result.unwrap().unwrap(); |
| 3846 | let main_api = &result.main_api; |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3847 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3862 | result.cc_details.tokens, |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3863 | quote! { |
| 3864 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3865 | extern "C" float ... (float, float); |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3866 | } |
| 3867 | inline float Math::add_i32(float x, float y) { |
| 3868 | return __crubit_internal::...(x, y); |
| 3869 | } |
| 3870 | } |
| 3871 | ); |
| 3872 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3873 | result.rs_details, |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3874 | 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 Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3885 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3901 | let result = result.unwrap().unwrap(); |
| 3902 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3903 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3919 | assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::generic_method },); |
| 3920 | assert_rs_not_matches!(result.rs_details, quote! { generic_method },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3921 | }); |
| 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3936 | let result = result.unwrap().unwrap(); |
| 3937 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3938 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3955 | result.cc_details.tokens, |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3956 | quote! { SomeStruct::fn_taking_reference }, |
| 3957 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3958 | assert_rs_not_matches!(result.rs_details, quote! { fn_taking_reference },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3959 | }); |
| 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3974 | let result = result.unwrap().unwrap(); |
| 3975 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3976 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3992 | assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::into_f32 },); |
| 3993 | assert_rs_not_matches!(result.rs_details, quote! { into_f32 },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3994 | }); |
| 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4009 | let result = result.unwrap().unwrap(); |
| 4010 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4011 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4027 | assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::get_f32 },); |
| 4028 | assert_rs_not_matches!(result.rs_details, quote! { get_f32 },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4029 | }); |
| 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4044 | let result = result.unwrap().unwrap(); |
| 4045 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4046 | 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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4062 | assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::set_f32 },); |
| 4063 | assert_rs_not_matches!(result.rs_details, quote! { set_f32 },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4064 | }); |
| 4065 | } |
| 4066 | |
| 4067 | #[test] |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 4068 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4115 | fn test_format_item_unsupported_struct_with_name_that_is_reserved_keyword() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4116 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4123 | test_format_item(test_src, "reinterpret_cast", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4124 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4125 | 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 Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4135 | fn test_format_item_struct_with_unsupported_field_type() { |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4136 | let test_src = r#" |
| 4137 | pub struct SomeStruct { |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4138 | pub successful_field: i32, |
| 4139 | pub unsupported_field: Option<[i32; 3]>, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4140 | } |
| 4141 | "#; |
| 4142 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4143 | let result = result.unwrap().unwrap(); |
| 4144 | let main_api = &result.main_api; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4145 | let broken_field_msg = "Field type has been replaced with a blob of bytes: \ |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4146 | Generic types are not supported yet (b/259749095)"; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4147 | assert_cc_matches!( |
| 4148 | main_api.tokens, |
| 4149 | quote! { |
| 4150 | ... |
| 4151 | struct ... SomeStruct final { |
| 4152 | ... |
| 4153 | private: |
| 4154 | __COMMENT__ #broken_field_msg |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4155 | unsigned char unsupported_field[16]; |
| 4156 | std::int32_t successful_field; |
| 4157 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4158 | }; |
| 4159 | ... |
| 4160 | } |
| 4161 | ); |
| 4162 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4163 | result.cc_details.tokens, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4164 | quote! { |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4165 | static_assert(sizeof(SomeStruct) == 20, ...); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4166 | static_assert(alignof(SomeStruct) == 4, ...); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4167 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4171 | } |
| 4172 | ); |
| 4173 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4174 | result.rs_details, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4175 | quote! { |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4176 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 20); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4177 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4178 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4182 | } |
| 4183 | ); |
| 4184 | }); |
| 4185 | } |
| 4186 | |
| 4187 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4188 | fn test_format_item_unsupported_struct_with_custom_drop_impl() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4189 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4199 | test_format_item(test_src, "StructWithCustomDropImpl", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4200 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4201 | assert_eq!(err, "`Drop` trait and \"drop glue\" are not supported yet (b/258251148)"); |
| 4202 | }); |
| 4203 | } |
| 4204 | |
| 4205 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4206 | fn test_format_item_unsupported_struct_with_custom_drop_glue() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4207 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4223 | test_format_item(test_src, "StructRequiringCustomDropGlue", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4224 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4225 | assert_eq!(err, "`Drop` trait and \"drop glue\" are not supported yet (b/258251148)"); |
| 4226 | }); |
| 4227 | } |
| 4228 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4229 | /// 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4232 | #[test] |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4233 | fn test_format_item_unsupported_struct_zero_sized_type_with_no_fields() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4234 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4240 | test_format_item(test_src, name, |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4241 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4242 | assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)"); |
| 4243 | }); |
| 4244 | } |
| 4245 | } |
| 4246 | |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4247 | #[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 Jeanpierre | 9e15d0b | 2023-04-06 13:18:22 -0700 | [diff] [blame] | 4262 | #[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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4323 | /// 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 Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4327 | fn test_format_item_enum_with_only_discriminant_items() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4328 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4338 | test_format_item(test_src, "SomeEnum", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4339 | let result = result.unwrap().unwrap(); |
| 4340 | let main_api = &result.main_api; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4341 | let no_fields_msg = "Field type has been replaced with a blob of bytes: \ |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4342 | No support for bindings of individual fields of \ |
| 4343 | `union` (b/272801632) or `enum`"; |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4344 | 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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 4351 | __COMMENT__ "`SomeEnum` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4352 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4368 | __COMMENT__ #no_fields_msg |
| 4369 | unsigned char __opaque_blob_of_bytes[1]; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4370 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4371 | }; |
| 4372 | } |
| 4373 | ); |
| 4374 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4375 | result.cc_details.tokens, |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4376 | quote! { |
| 4377 | static_assert(sizeof(SomeEnum) == 1, ...); |
| 4378 | static_assert(alignof(SomeEnum) == 1, ...); |
| 4379 | } |
| 4380 | ); |
| 4381 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4382 | result.rs_details, |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4383 | 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4388 | }); |
| 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 Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4394 | fn test_format_item_enum_with_tuple_and_struct_items() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4395 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4404 | test_format_item(test_src, "Point", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4405 | let result = result.unwrap().unwrap(); |
| 4406 | let main_api = &result.main_api; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4407 | let no_fields_msg = "Field type has been replaced with a blob of bytes: \ |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4408 | No support for bindings of individual fields of \ |
| 4409 | `union` (b/272801632) or `enum`"; |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4410 | 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 Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 4417 | __COMMENT__ "`Point` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4418 | 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4434 | __COMMENT__ #no_fields_msg |
| 4435 | unsigned char __opaque_blob_of_bytes[12]; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4436 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4437 | }; |
| 4438 | } |
| 4439 | ); |
| 4440 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4441 | result.cc_details.tokens, |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4442 | quote! { |
| 4443 | static_assert(sizeof(Point) == 12, ...); |
| 4444 | static_assert(alignof(Point) == 4, ...); |
| 4445 | } |
| 4446 | ); |
| 4447 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4448 | result.rs_details, |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4449 | 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4454 | }); |
| 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4460 | fn test_format_item_unsupported_enum_zero_variants() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4461 | let test_src = r#" |
| 4462 | pub enum ZeroVariantEnum {} |
| 4463 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4464 | test_format_item(test_src, "ZeroVariantEnum", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4465 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4466 | assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)"); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4467 | }); |
| 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4473 | fn test_format_item_union() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4474 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4483 | test_format_item(test_src, "SomeUnion", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4484 | let result = result.unwrap().unwrap(); |
| 4485 | let main_api = &result.main_api; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4486 | let no_fields_msg = "Field type has been replaced with a blob of bytes: \ |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4487 | No support for bindings of individual fields of \ |
| 4488 | `union` (b/272801632) or `enum`"; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4489 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4490 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4491 | main_api.tokens, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4492 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4493 | ... |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 4494 | union alignas(8) SomeUnion final { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4495 | public: |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 4496 | __COMMENT__ "`SomeUnion` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4497 | SomeUnion() = delete; |
| 4498 | |
| 4499 | // In this test there is no `Copy` implementation / derive. |
| 4500 | SomeUnion(const SomeUnion&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4501 | |
| 4502 | // All Rust types are trivially-movable. |
| 4503 | SomeUnion(SomeUnion&&) = default; |
Lukasz Anforowicz | 554ed65 | 2023-01-12 15:41:58 -0800 | [diff] [blame] | 4504 | |
| 4505 | // Assignment operators are disabled for now. |
| 4506 | SomeUnion& operator=(const SomeUnion&) = delete; |
| 4507 | SomeUnion& operator=(SomeUnion&&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4508 | |
| 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4513 | __COMMENT__ #no_fields_msg |
| 4514 | unsigned char __opaque_blob_of_bytes[8]; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4515 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4516 | }; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4517 | } |
| 4518 | ); |
| 4519 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4520 | result.cc_details.tokens, |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4521 | quote! { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4522 | static_assert(sizeof(SomeUnion) == 8, ...); |
| 4523 | static_assert(alignof(SomeUnion) == 8, ...); |
| 4524 | } |
| 4525 | ); |
| 4526 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4527 | result.rs_details, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4528 | quote! { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 4529 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeUnion>() == 8); |
| 4530 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeUnion>() == 8); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4531 | } |
| 4532 | ); |
| 4533 | }); |
| 4534 | } |
| 4535 | |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 4536 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4537 | fn test_format_item_doc_comments_union() { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4538 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4546 | test_format_item(test_src, "SomeUnionWithDocs", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4547 | let result = result.unwrap().unwrap(); |
| 4548 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4549 | let comment = " Doc for some union.\n\n\ |
| 4550 | Generated from: <crubit_unittests.rs>;l=3"; |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4551 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4552 | main_api.tokens, |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4553 | quote! { |
| 4554 | __COMMENT__ #comment |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 4555 | union ... SomeUnionWithDocs final { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4556 | ... |
| 4557 | } |
| 4558 | ... |
| 4559 | } |
| 4560 | ); |
| 4561 | }); |
| 4562 | } |
| 4563 | |
| 4564 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4565 | fn test_format_item_doc_comments_enum() { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4566 | let test_src = r#" |
| 4567 | /** Doc for some enum. */ |
| 4568 | pub enum SomeEnumWithDocs { |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4569 | Kind1(i32), |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4570 | } |
| 4571 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4572 | test_format_item(test_src, "SomeEnumWithDocs", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4573 | let result = result.unwrap().unwrap(); |
| 4574 | let main_api = &result.main_api; |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4575 | 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 | ); |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4587 | }); |
| 4588 | } |
| 4589 | |
| 4590 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4591 | fn test_format_item_doc_comments_struct() { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4592 | let test_src = r#" |
| 4593 | #![allow(dead_code)] |
| 4594 | #[doc = "Doc for some struct."] |
| 4595 | pub struct SomeStructWithDocs { |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 4596 | #[doc = "Doc for first field."] |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4597 | some_field : i32, |
| 4598 | } |
| 4599 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4600 | test_format_item(test_src, "SomeStructWithDocs", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4601 | let result = result.unwrap().unwrap(); |
| 4602 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4603 | let comment = "Doc for some struct.\n\n\ |
| 4604 | Generated from: <crubit_unittests.rs>;l=4"; |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4605 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4606 | main_api.tokens, |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4607 | quote! { |
| 4608 | __COMMENT__ #comment |
| 4609 | struct ... SomeStructWithDocs final { |
| 4610 | ... |
| 4611 | } |
| 4612 | ... |
| 4613 | } |
| 4614 | ); |
| 4615 | }); |
| 4616 | } |
| 4617 | |
| 4618 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4619 | fn test_format_item_doc_comments_tuple_struct() { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4620 | let test_src = r#" |
| 4621 | /// Doc for some tuple struct. |
| 4622 | pub struct SomeTupleStructWithDocs(i32); |
| 4623 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4624 | test_format_item(test_src, "SomeTupleStructWithDocs", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4625 | let result = result.unwrap().unwrap(); |
| 4626 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4627 | let comment = " Doc for some tuple struct.\n\n\ |
| 4628 | Generated from: <crubit_unittests.rs>;l=3"; |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4629 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4630 | main_api.tokens, |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4631 | quote! { |
| 4632 | __COMMENT__ #comment |
| 4633 | struct ... SomeTupleStructWithDocs final { |
| 4634 | ... |
| 4635 | } |
| 4636 | ... |
| 4637 | }, |
| 4638 | ); |
| 4639 | }); |
| 4640 | } |
| 4641 | |
| 4642 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4643 | fn test_format_item_source_loc_macro_rules() { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4644 | 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 Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4654 | test_format_item(test_src, "SomeTupleStructMacroForTesingSourceLoc", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4655 | let result = result.unwrap().unwrap(); |
| 4656 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4657 | let source_loc_comment = " Some doc on SomeTupleStructMacroForTesingSourceLoc.\n\n\ |
| 4658 | Generated from: <crubit_unittests.rs>;l=5"; |
| 4659 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4660 | main_api.tokens, |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4661 | quote! { |
| 4662 | __COMMENT__ #source_loc_comment |
| 4663 | struct ... SomeTupleStructMacroForTesingSourceLoc final { |
| 4664 | ... |
| 4665 | } |
| 4666 | ... |
| 4667 | }, |
| 4668 | ); |
| 4669 | }); |
| 4670 | } |
| 4671 | |
| 4672 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4673 | fn test_format_item_source_loc_with_no_doc_comment() { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4674 | let test_src = r#" |
| 4675 | pub struct SomeTupleStructWithNoDocComment(i32); |
| 4676 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4677 | test_format_item(test_src, "SomeTupleStructWithNoDocComment", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4678 | let result = result.unwrap().unwrap(); |
| 4679 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4680 | let comment = "Generated from: <crubit_unittests.rs>;l=2"; |
| 4681 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4682 | main_api.tokens, |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4683 | quote! { |
| 4684 | __COMMENT__ #comment |
| 4685 | struct ... SomeTupleStructWithNoDocComment final { |
| 4686 | ... |
| 4687 | } |
| 4688 | ... |
| 4689 | }, |
| 4690 | ); |
| 4691 | }); |
| 4692 | } |
| 4693 | |
| 4694 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4695 | fn test_format_item_unsupported_static_value() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 4696 | let test_src = r#" |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4697 | #[no_mangle] |
| 4698 | pub static STATIC_VALUE: i32 = 42; |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 4699 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4700 | test_format_item(test_src, "STATIC_VALUE", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4701 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4702 | assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: static item"); |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 4703 | }); |
| 4704 | } |
| 4705 | |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4706 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4707 | fn test_format_item_unsupported_const_value() { |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4708 | let test_src = r#" |
| 4709 | pub const CONST_VALUE: i32 = 42; |
| 4710 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4711 | test_format_item(test_src, "CONST_VALUE", |result| { |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4712 | let err = result.unwrap_err(); |
| 4713 | assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: constant item"); |
| 4714 | }); |
| 4715 | } |
| 4716 | |
| 4717 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4718 | fn test_format_item_unsupported_type_alias() { |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4719 | let test_src = r#" |
| 4720 | pub type TypeAlias = i32; |
| 4721 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4722 | test_format_item(test_src, "TypeAlias", |result| { |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4723 | // 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 Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 4729 | #[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 Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4739 | let result = result.unwrap().unwrap(); |
| 4740 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4741 | assert!(!main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 4742 | let unsupported_msg = "Error generating bindings for `SomeStruct::CONST_VALUE` \ |
| 4743 | defined at <crubit_unittests.rs>;l=5: \ |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4744 | Unsupported `impl` item kind: Const"; |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 4745 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4746 | main_api.tokens, |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 4747 | quote! { |
| 4748 | ... |
| 4749 | struct alignas(4) SomeStruct final { |
| 4750 | ... |
| 4751 | __COMMENT__ #unsupported_msg |
| 4752 | ... |
| 4753 | }; |
| 4754 | ... |
| 4755 | } |
| 4756 | ); |
| 4757 | }); |
| 4758 | } |
| 4759 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 4760 | /// `test_format_ret_ty_for_cc_successes` provides test coverage for cases |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 4761 | /// 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 Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4764 | #[test] |
Lukasz Anforowicz | 8a68f50 | 2022-11-15 08:43:43 -0800 | [diff] [blame] | 4765 | fn test_format_ret_ty_for_cc_successes() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4766 | 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 Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 4773 | ( |
| 4774 | "extern \"C\" fn (f32, f32) -> f32", |
| 4775 | "crubit :: type_identity_t < float (float , float) > &", |
| 4776 | ), |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4777 | ]; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4778 | test_ty(&testcases, quote! {}, |desc, tcx, ty, expected| { |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 4779 | let actual = { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 4780 | let input = bindings_input_for_tests(tcx); |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 4781 | let cc_snippet = format_ty_for_cc(&input, ty, TypeLocation::FnReturn).unwrap(); |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 4782 | cc_snippet.tokens.to_string() |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 4783 | }; |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4784 | let expected = expected.parse::<TokenStream>().unwrap().to_string(); |
| 4785 | assert_eq!(actual, expected, "{desc}"); |
| 4786 | }); |
| 4787 | } |
| 4788 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4789 | /// `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 Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4799 | #[test] |
Lukasz Anforowicz | 8a68f50 | 2022-11-15 08:43:43 -0800 | [diff] [blame] | 4800 | fn test_format_ty_for_cc_successes() { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4801 | let testcases = [ |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4802 | // ( <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 Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 4809 | ("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 Anforowicz | ec0b64e | 2023-02-17 14:31:12 -0800 | [diff] [blame] | 4819 | ("char", ("rs_std::rs_char", "\"crubit/support/for/tests/rs_std/rs_char.h\"", "", "")), |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4820 | ("SomeStruct", ("::rust_out::SomeStruct", "", "SomeStruct", "")), |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4821 | ("SomeEnum", ("::rust_out::SomeEnum", "", "SomeEnum", "")), |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4822 | ("SomeUnion", ("::rust_out::SomeUnion", "", "SomeUnion", "")), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 4823 | ("*const i32", ("const std::int32_t*", "<cstdint>", "", "")), |
| 4824 | ("*mut i32", ("std::int32_t*", "<cstdint>", "", "")), |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4825 | // `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 Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 4829 | ( |
| 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 Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4838 | // Extra parens/sugar are expected to be ignored: |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4839 | ("(bool)", ("bool", "", "", "")), |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4840 | ]; |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 4841 | let preamble = quote! { |
| 4842 | #![allow(unused_parens)] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4843 | |
| 4844 | pub struct SomeStruct { |
| 4845 | pub x: i32, |
| 4846 | pub y: i32, |
| 4847 | } |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4848 | pub enum SomeEnum { |
| 4849 | Cartesian{x: f64, y: f64}, |
| 4850 | Polar{angle: f64, dist: f64}, |
| 4851 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4852 | pub union SomeUnion { |
| 4853 | pub x: i32, |
| 4854 | pub y: i32, |
| 4855 | } |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 4856 | }; |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 4857 | test_ty( |
| 4858 | &testcases, |
| 4859 | preamble, |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4860 | |desc, tcx, ty, |
| 4861 | (expected_tokens, expected_include, expected_prereq_def, expected_prereq_fwd_decl)| { |
| 4862 | let (actual_tokens, actual_prereqs) = { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 4863 | let input = bindings_input_for_tests(tcx); |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 4864 | let s = format_ty_for_cc(&input, ty, TypeLocation::Other).unwrap(); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4865 | (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 Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 4869 | |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 4870 | let expected_tokens = expected_tokens.parse::<TokenStream>().unwrap().to_string(); |
| 4871 | assert_eq!(actual_tokens, expected_tokens, "{desc}"); |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 4872 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4873 | if expected_include.is_empty() { |
| 4874 | assert!(actual_includes.is_empty()); |
| 4875 | } else { |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 4876 | let expected_include: TokenStream = expected_include.parse().unwrap(); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4877 | assert_cc_matches!( |
| 4878 | format_cc_includes(&actual_includes), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 4879 | quote! { __HASH_TOKEN__ include #expected_include } |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4880 | ); |
| 4881 | } |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 4882 | |
| 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 Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4890 | |
| 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 Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 4899 | }, |
| 4900 | ); |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4901 | } |
| 4902 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4903 | /// `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 Anforowicz | 0182c5c | 2022-12-29 10:08:50 -0800 | [diff] [blame] | 4914 | /// TODO(lukasza): Add test coverage (here and in the "for_rs" flavours) |
| 4915 | /// for: |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4916 | /// - 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 Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4923 | #[test] |
Lukasz Anforowicz | 8a68f50 | 2022-11-15 08:43:43 -0800 | [diff] [blame] | 4924 | fn test_format_ty_for_cc_failures() { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4925 | let testcases = [ |
| 4926 | // ( <Rust type>, <expected error message> ) |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4927 | ( |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4928 | "()", // Empty TyKind::Tuple |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 4929 | "`()` / `void` is only supported as a return type (b/254507801)", |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4930 | ), |
| 4931 | ( |
| 4932 | // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more |
| 4933 | // details). |
| 4934 | "!", // TyKind::Never |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 4935 | "The never type `!` is only supported as a return type (b/254507801)", |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4936 | ), |
| 4937 | ( |
| 4938 | "(i32, i32)", // Non-empty TyKind::Tuple |
Lukasz Anforowicz | 13794df | 2022-10-21 07:56:34 -0700 | [diff] [blame] | 4939 | "Tuples are not supported yet: (i32, i32) (b/254099023)", |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4940 | ), |
| 4941 | ( |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4942 | "&'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 Anforowicz | 0182c5c | 2022-12-29 10:08:50 -0800 | [diff] [blame] | 4958 | "impl Eq", // TyKind::Alias |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4959 | "The following Rust type is not supported yet: impl std::cmp::Eq", |
| 4960 | ), |
| 4961 | ( |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 4962 | "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 Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4977 | ), |
| 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 Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4982 | ( |
| 4983 | "StructWithCustomDrop", |
| 4984 | "Failed to generate bindings for the definition of `StructWithCustomDrop`: \ |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 4985 | `Drop` trait and \"drop glue\" are not supported yet (b/258251148)", |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4986 | ), |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 4987 | ("ConstGenericStruct<42>", "Generic types are not supported yet (b/259749095)"), |
| 4988 | ("TypeGenericStruct<u8>", "Generic types are not supported yet (b/259749095)"), |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4989 | ( |
| 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 Anforowicz | cfe8455 | 2023-04-20 13:38:54 -0700 | [diff] [blame^] | 4996 | ( |
| 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 Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 5009 | ( |
| 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 Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 5025 | ( |
| 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 Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5030 | ]; |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5031 | let preamble = quote! { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5032 | #![feature(never_type)] |
| 5033 | |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 5034 | pub struct SomeStruct { |
| 5035 | pub x: i32, |
| 5036 | pub y: i32, |
| 5037 | } |
| 5038 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5039 | pub struct StructWithCustomDrop { |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5040 | pub x: i32, |
| 5041 | pub y: i32, |
| 5042 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5043 | |
| 5044 | impl Drop for StructWithCustomDrop { |
| 5045 | fn drop(&mut self) {} |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5046 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5047 | |
| 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 Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5058 | } |
Lukasz Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 5059 | |
| 5060 | mod private_submodule { |
| 5061 | pub struct PublicStructInPrivateModule; |
| 5062 | } |
| 5063 | pub use private_submodule::PublicStructInPrivateModule |
| 5064 | as PublicReexportOfStruct; |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5065 | }; |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5066 | test_ty(&testcases, preamble, |desc, tcx, ty, expected_msg| { |
| 5067 | let input = bindings_input_for_tests(tcx); |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 5068 | let anyhow_err = format_ty_for_cc(&input, ty, TypeLocation::Other).unwrap_err(); |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5069 | let actual_msg = format!("{anyhow_err:#}"); |
| 5070 | assert_eq!(&actual_msg, *expected_msg, "{desc}"); |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5071 | }); |
| 5072 | } |
| 5073 | |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5074 | #[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 Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 5097 | // ADTs: |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 5098 | ("SomeStruct", "::rust_out::SomeStruct"), |
| 5099 | ("SomeEnum", "::rust_out::SomeEnum"), |
| 5100 | ("SomeUnion", "::rust_out::SomeUnion"), |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 5101 | // Type from another crate: |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 5102 | ("std::cmp::Ordering", "::core::cmp::Ordering"), |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 5103 | // `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 Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 5108 | ("extern \"C\" fn(i32) -> i32", "extern \"C\" fn(i32) -> i32"), |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5109 | ]; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5110 | 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 Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 5126 | 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 Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5130 | }); |
| 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 Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5144 | "&'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 Anforowicz | 0182c5c | 2022-12-29 10:08:50 -0800 | [diff] [blame] | 5160 | "impl Eq", // TyKind::Alias |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5161 | "The following Rust type is not supported yet: impl std::cmp::Eq", |
| 5162 | ), |
| 5163 | ( |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 5164 | "Option<i8>", // TyKind::Adt - generic + different crate |
| 5165 | "Generic types are not supported yet (b/259749095)", |
| 5166 | ), |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5167 | ]; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5168 | 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 Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5171 | let actual_err = format!("{anyhow_err:#}"); |
| 5172 | assert_eq!(&actual_err, *expected_err, "{desc}"); |
| 5173 | }); |
| 5174 | } |
| 5175 | |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5176 | fn test_ty<TestFn, Expectation>( |
| 5177 | testcases: &[(&str, Expectation)], |
| 5178 | preamble: TokenStream, |
| 5179 | test_fn: TestFn, |
| 5180 | ) where |
Lukasz Anforowicz | d16b6bf | 2022-11-22 18:35:08 -0800 | [diff] [blame] | 5181 | TestFn: for<'tcx> Fn( |
| 5182 | /* testcase_description: */ &str, |
| 5183 | TyCtxt<'tcx>, |
| 5184 | Ty<'tcx>, |
| 5185 | &Expectation, |
| 5186 | ) + Sync, |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5187 | Expectation: Sync, |
| 5188 | { |
Lukasz Anforowicz | d0f0a84 | 2022-11-03 12:40:13 -0700 | [diff] [blame] | 5189 | for (index, (input, expected)) in testcases.iter().enumerate() { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5190 | let desc = format!("test #{index}: test input: `{input}`"); |
| 5191 | let input = { |
| 5192 | let ty_tokens: TokenStream = input.parse().unwrap(); |
| 5193 | let input = quote! { |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5194 | #preamble |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5195 | pub fn test_function() -> #ty_tokens { panic!("") } |
| 5196 | }; |
| 5197 | input.to_string() |
| 5198 | }; |
Lukasz Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 5199 | run_compiler_for_testing(input, |tcx| { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5200 | let def_id = find_def_id_by_name(tcx, "test_function"); |
Lukasz Anforowicz | 087dff7 | 2023-02-17 12:13:32 -0800 | [diff] [blame] | 5201 | let ty = tcx |
| 5202 | .fn_sig(def_id.to_def_id()) |
| 5203 | .subst_identity() |
| 5204 | .no_bound_vars() |
| 5205 | .unwrap() |
| 5206 | .output(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5207 | test_fn(&desc, tcx, ty, expected); |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5208 | }); |
| 5209 | } |
| 5210 | } |
| 5211 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 5212 | /// Tests invoking `format_item` on the item with the specified `name` from |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5213 | /// the given Rust `source`. Returns the result of calling |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 5214 | /// `test_function` with `format_item`'s result as an argument. |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5215 | /// (`test_function` should typically `assert!` that it got the expected |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 5216 | /// result from `format_item`.) |
| 5217 | fn test_format_item<F, T>(source: &str, name: &str, test_function: F) -> T |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5218 | where |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 5219 | F: FnOnce(Result<Option<ApiSnippets>, String>) -> T + Send, |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5220 | T: Send, |
| 5221 | { |
Lukasz Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 5222 | run_compiler_for_testing(source, |tcx| { |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5223 | let def_id = find_def_id_by_name(tcx, name); |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 5224 | let result = format_item(&bindings_input_for_tests(tcx), def_id); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 5225 | |
| 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 Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5231 | }) |
| 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 Anforowicz | d0f0a84 | 2022-11-03 12:40:13 -0700 | [diff] [blame] | 5240 | match *items_with_matching_name.as_slice() { |
| 5241 | [] => { |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5242 | 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 Anforowicz | d0f0a84 | 2022-11-03 12:40:13 -0700 | [diff] [blame] | 5248 | .join(",\n"); |
| 5249 | panic!("No items named `{name}`.\nInstead found:\n{found_names}"); |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5250 | } |
Lukasz Anforowicz | 61cb1e3 | 2022-11-04 09:08:35 -0700 | [diff] [blame] | 5251 | [item] => item.owner_id.def_id, |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5252 | _ => panic!("More than one item named `{name}`"), |
| 5253 | } |
| 5254 | } |
| 5255 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5256 | fn bindings_input_for_tests(tcx: TyCtxt) -> Input { |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 5257 | Input { |
| 5258 | tcx, |
| 5259 | crubit_support_path: "crubit/support/for/tests".into(), |
Lukasz Anforowicz | cfe8455 | 2023-04-20 13:38:54 -0700 | [diff] [blame^] | 5260 | crate_name_to_include_path: Default::default(), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 5261 | _features: (), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 5262 | } |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5263 | } |
| 5264 | |
| 5265 | /// Tests invoking `generate_bindings` on the given Rust `source`. |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5266 | /// 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 Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 5270 | where |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5271 | F: FnOnce(Result<Output>) -> T + Send, |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 5272 | T: Send, |
| 5273 | { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5274 | run_compiler_for_testing(source, |tcx| { |
| 5275 | test_function(generate_bindings(&bindings_input_for_tests(tcx))) |
| 5276 | }) |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 5277 | } |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 5278 | } |