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 }), |
Googler | 7a77a80 | 2023-04-21 08:32:50 -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)"); |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 373 | } |
| 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 }), |
Googler | 7a77a80 | 2023-04-21 08:32:50 -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)"); |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 382 | } |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 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()) |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 485 | .ok_or_else(|| { |
| 486 | anyhow!( |
Lukasz Anforowicz | cfe8455 | 2023-04-20 13:38:54 -0700 | [diff] [blame] | 487 | "Type `{ty}` comes from the `{other_crate_name}` crate, \ |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 488 | but no `--other-crate-bindings` were specified for this crate" |
| 489 | ) |
| 490 | })?; |
Lukasz Anforowicz | cfe8455 | 2023-04-20 13:38:54 -0700 | [diff] [blame] | 491 | prereqs.includes.insert(include.clone()); |
| 492 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 493 | |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 494 | // Verify if definition of `ty` can be succesfully imported and bail otherwise. |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 495 | format_adt_core(input.tcx, def_id).with_context(|| { |
| 496 | format!("Failed to generate bindings for the definition of `{ty}`") |
| 497 | })?; |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 498 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 499 | CcSnippet { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 500 | tokens: FullyQualifiedName::new(input.tcx, def_id).format_for_cc()?, |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 501 | prereqs, |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 502 | } |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 503 | } |
| 504 | |
| 505 | ty::TyKind::RawPtr(ty::TypeAndMut { ty, mutbl }) => { |
| 506 | let const_qualifier = match mutbl { |
| 507 | Mutability::Mut => quote! {}, |
| 508 | Mutability::Not => quote! { const }, |
| 509 | }; |
| 510 | let CcSnippet { tokens, mut prereqs } = |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 511 | format_ty_for_cc(input, *ty, TypeLocation::Other).with_context(|| { |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 512 | format!("Failed to format the pointee of the pointer type `{ty}`") |
| 513 | })?; |
| 514 | prereqs.move_defs_to_fwd_decls(); |
| 515 | CcSnippet { prereqs, tokens: quote! { #const_qualifier #tokens * } } |
| 516 | } |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 517 | |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 518 | ty::TyKind::FnPtr(sig) => { |
| 519 | let sig = match sig.no_bound_vars() { |
| 520 | None => bail!("Generic functions are not supported yet (b/259749023)"), |
| 521 | Some(sig) => sig, |
| 522 | }; |
| 523 | check_fn_sig(&sig)?; |
| 524 | is_thunk_required(&sig).context("Function pointers can't have a thunk")?; |
| 525 | |
| 526 | // `is_thunk_required` check above implies `extern "C"` (or `"C-unwind"`). |
| 527 | // This assertion reinforces that the generated C++ code doesn't need |
| 528 | // to use calling convention attributes like `_stdcall`, etc. |
| 529 | assert!(matches!(sig.abi, rustc_target::spec::abi::Abi::C { .. })); |
| 530 | |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 531 | // C++ references are not rebindable and therefore can't be used to replicate |
| 532 | // semantics of Rust field types (or, say, element types of Rust |
| 533 | // arrays). Because of this, C++ references are only used for |
| 534 | // top-level return types and parameter types (and pointers are used |
| 535 | // in other locations). |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 536 | let ptr_or_ref_sigil = match location { |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 537 | TypeLocation::FnReturn | TypeLocation::FnParam => quote! { & }, |
| 538 | TypeLocation::Other => quote! { * }, |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 539 | }; |
| 540 | |
| 541 | let mut prereqs = CcPrerequisites::default(); |
| 542 | prereqs.includes.insert(input.support_header("internal/cxx20_backports.h")); |
| 543 | let ret_type = format_ret_ty_for_cc(input, &sig)?.into_tokens(&mut prereqs); |
| 544 | let param_types = format_param_types_for_cc(input, &sig)? |
| 545 | .into_iter() |
| 546 | .map(|snippet| snippet.into_tokens(&mut prereqs)); |
| 547 | let tokens = quote! { |
| 548 | crubit::type_identity_t< |
| 549 | #ret_type( #( #param_types ),* ) |
| 550 | > #ptr_or_ref_sigil |
| 551 | }; |
| 552 | |
| 553 | CcSnippet { tokens, prereqs } |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 554 | } |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 555 | |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 556 | // 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] | 557 | // element type of an Array, a referent of a Ref, a parameter type of an FnPtr, etc), one |
| 558 | // should also 1) propagate `CcPrerequisites::defs`, 2) cover `CcPrerequisites::defs` in |
| 559 | // `test_format_ty_for_cc...`. For ptr/ref it might be possible to use |
| 560 | // `CcPrerequisites::move_defs_to_fwd_decls`. |
Lukasz Anforowicz | 087dff7 | 2023-02-17 12:13:32 -0800 | [diff] [blame] | 561 | _ => bail!("The following Rust type is not supported yet: {ty}"), |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 562 | }) |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 563 | } |
| 564 | |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 565 | fn format_ret_ty_for_cc<'tcx>(input: &Input<'tcx>, sig: &ty::FnSig<'tcx>) -> Result<CcSnippet> { |
| 566 | format_ty_for_cc(input, sig.output(), TypeLocation::FnReturn) |
| 567 | .context("Error formatting function return type") |
| 568 | } |
| 569 | |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 570 | fn format_param_types_for_cc<'tcx>( |
| 571 | input: &Input<'tcx>, |
| 572 | sig: &ty::FnSig<'tcx>, |
| 573 | ) -> Result<Vec<CcSnippet>> { |
| 574 | sig.inputs() |
| 575 | .iter() |
| 576 | .enumerate() |
| 577 | .map(|(i, &ty)| { |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 578 | Ok(format_ty_for_cc(input, ty, TypeLocation::FnParam) |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 579 | .with_context(|| format!("Error handling parameter #{i}"))?) |
| 580 | }) |
| 581 | .collect() |
| 582 | } |
| 583 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 584 | /// Formats `ty` for Rust - to be used in `..._cc_api_impl.rs` (e.g. as a type |
| 585 | /// of a parameter in a Rust thunk). Because `..._cc_api_impl.rs` is a |
| 586 | /// distinct, separate crate, the returned `TokenStream` uses crate-qualified |
| 587 | /// names whenever necessary - for example: `target_crate::SomeStruct` rather |
| 588 | /// than just `SomeStruct`. |
Lukasz Anforowicz | 1ab5e87 | 2022-12-05 10:07:00 -0800 | [diff] [blame] | 589 | // |
| 590 | // TODO(b/259724276): This function's results should be memoized. |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 591 | fn format_ty_for_rs(tcx: TyCtxt, ty: Ty) -> Result<TokenStream> { |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 592 | Ok(match ty.kind() { |
| 593 | ty::TyKind::Bool |
| 594 | | ty::TyKind::Float(_) |
| 595 | | ty::TyKind::Char |
| 596 | | ty::TyKind::Int(_) |
| 597 | | ty::TyKind::Uint(_) |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 598 | | ty::TyKind::FnPtr(_) |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 599 | | ty::TyKind::Never => ty |
| 600 | .to_string() |
| 601 | .parse() |
| 602 | .expect("rustc_middle::ty::Ty::to_string() should produce no parsing errors"), |
| 603 | ty::TyKind::Tuple(types) => { |
| 604 | if types.len() == 0 { |
| 605 | quote! { () } |
| 606 | } else { |
| 607 | // TODO(b/254099023): Add support for tuples. |
| 608 | bail!("Tuples are not supported yet: {} (b/254099023)", ty); |
| 609 | } |
| 610 | } |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 611 | ty::TyKind::Adt(adt, substs) => { |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 612 | ensure!(substs.len() == 0, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 613 | FullyQualifiedName::new(tcx, adt.did()).format_for_rs() |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 614 | } |
| 615 | ty::TyKind::RawPtr(ty::TypeAndMut { ty, mutbl }) => { |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 616 | let qualifier = match mutbl { |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 617 | Mutability::Mut => quote! { mut }, |
| 618 | Mutability::Not => quote! { const }, |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 619 | }; |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 620 | let ty = format_ty_for_rs(tcx, *ty).with_context(|| { |
| 621 | format!("Failed to format the pointee of the pointer type `{ty}`") |
| 622 | })?; |
| 623 | quote! { * #qualifier #ty } |
| 624 | } |
Lukasz Anforowicz | 087dff7 | 2023-02-17 12:13:32 -0800 | [diff] [blame] | 625 | _ => bail!("The following Rust type is not supported yet: {ty}"), |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 626 | }) |
| 627 | } |
| 628 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 629 | #[derive(Debug, Default)] |
| 630 | struct ApiSnippets { |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 631 | /// Main API - for example: |
| 632 | /// - A C++ declaration of a function (with a doc comment), |
| 633 | /// - A C++ definition of a struct (with a doc comment). |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 634 | main_api: CcSnippet, |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 635 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 636 | /// C++ implementation details - for example: |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 637 | /// - A C++ declaration of an `extern "C"` thunk, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 638 | /// - C++ `static_assert`s about struct size, aligment, and field offsets. |
| 639 | cc_details: CcSnippet, |
| 640 | |
| 641 | /// Rust implementation details - for exmaple: |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 642 | /// - A Rust implementation of an `extern "C"` thunk, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 643 | /// - Rust `assert!`s about struct size, aligment, and field offsets. |
| 644 | rs_details: TokenStream, |
Lukasz Anforowicz | 33f4630 | 2023-02-10 15:38:05 -0800 | [diff] [blame] | 645 | } |
| 646 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 647 | impl FromIterator<ApiSnippets> for ApiSnippets { |
| 648 | fn from_iter<I: IntoIterator<Item = ApiSnippets>>(iter: I) -> Self { |
| 649 | let mut result = ApiSnippets::default(); |
| 650 | for ApiSnippets { main_api, cc_details, rs_details } in iter.into_iter() { |
| 651 | result.main_api += main_api; |
| 652 | result.cc_details += cc_details; |
| 653 | result.rs_details.extend(rs_details); |
| 654 | } |
| 655 | result |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 656 | } |
| 657 | } |
| 658 | |
Lukasz Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 659 | fn get_fn_sig<'tcx>(tcx: TyCtxt<'tcx>, fn_def_id: LocalDefId) -> Result<ty::FnSig<'tcx>> { |
| 660 | match tcx.fn_sig(fn_def_id).subst_identity().no_bound_vars() { |
| 661 | None => bail!("Generic functions are not supported yet (b/259749023)"), |
| 662 | Some(sig) => Ok(sig), |
| 663 | } |
| 664 | } |
| 665 | |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 666 | /// Formats a C++ function declaration of a thunk that wraps a Rust function |
| 667 | /// identified by `fn_def_id`. `format_thunk_impl` may panic if `fn_def_id` |
| 668 | /// doesn't identify a function. |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 669 | fn format_thunk_decl( |
| 670 | input: &Input, |
| 671 | fn_def_id: LocalDefId, |
| 672 | thunk_name: &TokenStream, |
| 673 | ) -> Result<CcSnippet> { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 674 | let tcx = input.tcx; |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 675 | |
| 676 | let mut prereqs = CcPrerequisites::default(); |
| 677 | let sig = get_fn_sig(tcx, fn_def_id)?; |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 678 | 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] | 679 | |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 680 | let mut thunk_params = { |
| 681 | let cc_types = format_param_types_for_cc(input, &sig)?; |
| 682 | sig.inputs() |
| 683 | .iter() |
| 684 | .zip(cc_types.into_iter()) |
| 685 | .map(|(&ty, cc_type)| -> Result<TokenStream> { |
| 686 | let cc_type = cc_type.into_tokens(&mut prereqs); |
| 687 | if is_c_abi_compatible_by_value(ty) { |
| 688 | Ok(quote! { #cc_type }) |
| 689 | } else { |
| 690 | // Rust thunk will move a value via memcpy - we need to `ensure` that |
| 691 | // invoking the C++ destructor (on the moved-away value) is safe. |
| 692 | // TODO(b/259749095): Support generic structs (with non-empty ParamEnv). |
| 693 | ensure!( |
| 694 | !ty.needs_drop(tcx, ty::ParamEnv::empty()), |
| 695 | "Only trivially-movable and trivially-destructible types \ |
| 696 | may be passed by value over the FFI boundary" |
| 697 | ); |
| 698 | Ok(quote! { #cc_type* }) |
| 699 | } |
| 700 | }) |
| 701 | .collect::<Result<Vec<_>>>()? |
| 702 | }; |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 703 | |
| 704 | let thunk_ret_type: TokenStream; |
| 705 | if is_c_abi_compatible_by_value(sig.output()) { |
| 706 | thunk_ret_type = main_api_ret_type.clone(); |
| 707 | } else { |
| 708 | thunk_ret_type = quote! { void }; |
| 709 | thunk_params.push(quote! { #main_api_ret_type* __ret_ptr }); |
| 710 | prereqs.includes.insert(CcInclude::utility()); |
| 711 | prereqs.includes.insert(input.support_header("internal/return_value_slot.h")); |
| 712 | }; |
| 713 | Ok(CcSnippet { |
| 714 | prereqs, |
| 715 | tokens: quote! { |
| 716 | namespace __crubit_internal { |
| 717 | extern "C" #thunk_ret_type #thunk_name ( #( #thunk_params ),* ); |
| 718 | } |
| 719 | }, |
| 720 | }) |
| 721 | } |
| 722 | |
Lukasz Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 723 | /// Formats a thunk implementation in Rust that provides an `extern "C"` ABI for |
| 724 | /// calling a Rust function identified by `fn_def_id`. `format_thunk_impl` may |
| 725 | /// panic if `fn_def_id` doesn't identify a function. |
| 726 | /// |
| 727 | /// `fully_qualified_fn_name` specifies how the thunk can identify the function |
| 728 | /// to call. Examples of valid arguments: |
| 729 | /// - `::crate_name::some_module::free_function` |
| 730 | /// - `::crate_name::some_module::SomeStruct::method` |
| 731 | /// - `<::create_name::some_module::SomeStruct as |
| 732 | /// ::core::default::Default>::default` |
| 733 | fn format_thunk_impl( |
| 734 | tcx: TyCtxt, |
| 735 | fn_def_id: LocalDefId, |
| 736 | thunk_name: &str, |
| 737 | fully_qualified_fn_name: TokenStream, |
| 738 | ) -> Result<TokenStream> { |
| 739 | let sig = get_fn_sig(tcx, fn_def_id)?; |
| 740 | let param_names_and_types: Vec<(Ident, Ty)> = { |
| 741 | let param_names = tcx.fn_arg_names(fn_def_id).iter().enumerate().map(|(i, name)| { |
| 742 | if name.as_str().is_empty() { |
| 743 | format_ident!("__param_{i}") |
| 744 | } else { |
| 745 | make_rs_ident(name.as_str()) |
| 746 | } |
| 747 | }); |
| 748 | let param_types = sig.inputs().iter().copied(); |
| 749 | param_names.zip(param_types).collect_vec() |
| 750 | }; |
| 751 | |
| 752 | let mut thunk_params = param_names_and_types |
| 753 | .iter() |
| 754 | .map(|(param_name, ty)| { |
| 755 | let rs_type = format_ty_for_rs(tcx, *ty) |
| 756 | .with_context(|| format!("Error handling parameter `{param_name}`"))?; |
| 757 | Ok(if is_c_abi_compatible_by_value(*ty) { |
| 758 | quote! { #param_name: #rs_type } |
| 759 | } else { |
| 760 | quote! { #param_name: &mut ::core::mem::MaybeUninit<#rs_type> } |
| 761 | }) |
| 762 | }) |
| 763 | .collect::<Result<Vec<_>>>()?; |
| 764 | |
| 765 | let mut thunk_ret_type = format_ty_for_rs(tcx, sig.output())?; |
| 766 | let mut thunk_body = { |
| 767 | let fn_args = param_names_and_types.iter().map(|(rs_name, ty)| { |
| 768 | if is_c_abi_compatible_by_value(*ty) { |
| 769 | quote! { #rs_name } |
| 770 | } else { |
| 771 | quote! { unsafe { #rs_name.assume_init_read() } } |
| 772 | } |
| 773 | }); |
| 774 | quote! { |
| 775 | #fully_qualified_fn_name( #( #fn_args ),* ) |
| 776 | } |
| 777 | }; |
| 778 | if !is_c_abi_compatible_by_value(sig.output()) { |
| 779 | thunk_params.push(quote! { |
| 780 | __ret_slot: &mut ::core::mem::MaybeUninit<#thunk_ret_type> |
| 781 | }); |
| 782 | thunk_ret_type = quote! { () }; |
| 783 | thunk_body = quote! { __ret_slot.write(#thunk_body); }; |
| 784 | }; |
| 785 | |
| 786 | let thunk_name = make_rs_ident(thunk_name); |
| 787 | Ok(quote! { |
| 788 | #[no_mangle] |
| 789 | extern "C" fn #thunk_name( #( #thunk_params ),* ) -> #thunk_ret_type { |
| 790 | #thunk_body |
| 791 | } |
| 792 | }) |
| 793 | } |
| 794 | |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 795 | fn get_symbol_name<'tcx>(tcx: TyCtxt<'tcx>, def_id: LocalDefId) -> Result<&'tcx str> { |
| 796 | ensure!( |
| 797 | tcx.generics_of(def_id).count() == 0, |
| 798 | "Generic functions are not supported yet (b/259749023) - caller should filter them out", |
| 799 | ); |
| 800 | |
| 801 | // Call to `mono` is ok - `generics_of` have been checked above. |
| 802 | let instance = ty::Instance::mono(tcx, def_id.to_def_id()); |
| 803 | |
| 804 | Ok(tcx.symbol_name(instance).name) |
| 805 | } |
| 806 | |
| 807 | fn get_thunk_name(symbol_name: &str) -> String { |
| 808 | format!("__crubit_thunk_{}", &escape_non_identifier_chars(symbol_name)) |
| 809 | } |
| 810 | |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 811 | fn check_fn_sig(sig: &ty::FnSig) -> Result<()> { |
| 812 | if sig.c_variadic { |
| 813 | // TODO(b/254097223): Add support for variadic functions. |
| 814 | bail!("C variadic functions are not supported (b/254097223)"); |
| 815 | } |
| 816 | |
| 817 | match sig.unsafety { |
| 818 | Unsafety::Normal => (), |
| 819 | Unsafety::Unsafe => { |
| 820 | // TODO(b/254095482): Figure out how to handle `unsafe` functions. |
| 821 | bail!("Bindings for `unsafe` functions are not fully designed yet (b/254095482)"); |
| 822 | } |
| 823 | } |
| 824 | |
| 825 | Ok(()) |
| 826 | } |
| 827 | |
| 828 | /// Returns `Ok(())` if no thunk is required. |
| 829 | /// Otherwise returns an error the describes why the thunk is needed. |
| 830 | fn is_thunk_required(sig: &ty::FnSig) -> Result<()> { |
| 831 | match sig.abi { |
| 832 | // "C" ABI is okay: Before https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a |
| 833 | // Rust panic that "escapes" a "C" ABI function leads to Undefined Behavior. This is |
| 834 | // unfortunate, but Crubit's `panics_and_exceptions.md` documents that `-Cpanic=abort` |
| 835 | // is the only supported configuration. |
| 836 | // |
| 837 | // After https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a Rust panic that |
| 838 | // tries to "escape" a "C" ABI function will terminate the program. This is okay. |
| 839 | rustc_target::spec::abi::Abi::C { unwind: false } => (), |
| 840 | |
| 841 | // "C-unwind" ABI is okay: After |
| 842 | // https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html a new "C-unwind" ABI may be |
| 843 | // used by Rust functions that want to safely propagate Rust panics through frames that |
| 844 | // may belong to another language. |
| 845 | rustc_target::spec::abi::Abi::C { unwind: true } => (), |
| 846 | |
| 847 | // All other ABIs trigger thunk generation. This covers Rust ABI functions, but also |
| 848 | // ABIs that theoretically are understood both by C++ and Rust (e.g. see |
| 849 | // `format_cc_call_conv_as_clang_attribute` in `rs_bindings_from_cc/src_code_gen.rs`). |
| 850 | _ => bail!("Calling convention other than `extern \"C\"` requires a thunk"), |
| 851 | }; |
| 852 | |
| 853 | ensure!(is_c_abi_compatible_by_value(sig.output()), "Return type requires a thunk"); |
| 854 | for (i, param_ty) in sig.inputs().iter().enumerate() { |
| 855 | ensure!(is_c_abi_compatible_by_value(*param_ty), "Type of parameter #{i} requires a thunk",); |
| 856 | } |
| 857 | |
| 858 | Ok(()) |
| 859 | } |
| 860 | |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 861 | /// Formats a function with the given `local_def_id`. |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 862 | /// |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 863 | /// Will panic if `local_def_id` |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 864 | /// - is invalid |
| 865 | /// - doesn't identify a function, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 866 | fn format_fn(input: &Input, local_def_id: LocalDefId) -> Result<ApiSnippets> { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 867 | let tcx = input.tcx; |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 868 | 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] | 869 | |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 870 | ensure!( |
| 871 | tcx.generics_of(def_id).count() == 0, |
| 872 | "Generic functions are not supported yet (b/259749023)" |
| 873 | ); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 874 | |
Lukasz Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 875 | let sig = get_fn_sig(tcx, local_def_id)?; |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 876 | check_fn_sig(&sig)?; |
| 877 | let needs_thunk = is_thunk_required(&sig).is_err(); |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 878 | let thunk_name = { |
| 879 | let symbol_name = get_symbol_name(tcx, local_def_id)?; |
| 880 | if needs_thunk { get_thunk_name(symbol_name) } else { symbol_name.to_string() } |
| 881 | }; |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 882 | |
Lukasz Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 883 | let FullyQualifiedName { krate, mod_path, name } = FullyQualifiedName::new(tcx, def_id); |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 884 | let fn_name = name.expect("Functions are assumed to always have a name"); |
| 885 | let main_api_fn_name = |
| 886 | format_cc_ident(fn_name.as_str()).context("Error formatting function name")?; |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 887 | |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 888 | let mut main_api_prereqs = CcPrerequisites::default(); |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 889 | 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] | 890 | let is_static_method = match tcx.hir().get_by_def_id(local_def_id) { |
| 891 | Node::ImplItem(impl_item) => match &impl_item.kind { |
| 892 | ImplItemKind::Fn(fn_sig, _) => match fn_sig.decl.implicit_self { |
| 893 | ImplicitSelfKind::None => true, |
| 894 | _ => bail!("`self` parameter is not supported yet"), |
| 895 | }, |
| 896 | _ => panic!("`format_fn` can only work with functions"), |
| 897 | }, |
| 898 | Node::Item(_) => false, // Free function |
| 899 | other => panic!("Unexpected HIR node kind: {other:?}"), |
| 900 | }; |
| 901 | |
| 902 | struct Param<'tcx> { |
| 903 | cc_name: TokenStream, |
| 904 | cc_type: TokenStream, |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 905 | ty: Ty<'tcx>, |
| 906 | } |
| 907 | let params = { |
| 908 | let names = tcx.fn_arg_names(def_id).iter(); |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 909 | let cc_types = format_param_types_for_cc(input, &sig)?; |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 910 | names |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 911 | .enumerate() |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 912 | .zip(sig.inputs().iter()) |
| 913 | .zip(cc_types.into_iter()) |
| 914 | .map(|(((i, name), &ty), cc_type)| { |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 915 | let cc_name = format_cc_ident(name.as_str()) |
| 916 | .unwrap_or_else(|_err| format_cc_ident(&format!("__param_{i}")).unwrap()); |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 917 | let cc_type = cc_type.into_tokens(&mut main_api_prereqs); |
| 918 | Param { cc_name, cc_type, ty } |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 919 | }) |
Lukasz Anforowicz | c6ff453 | 2023-04-13 15:16:30 -0700 | [diff] [blame] | 920 | .collect_vec() |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 921 | }; |
| 922 | let main_api_params = params |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 923 | .iter() |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 924 | .map(|Param { cc_name, cc_type, .. }| quote! { #cc_type #cc_name }) |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 925 | .collect_vec(); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 926 | |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 927 | let struct_name = match tcx.impl_of_method(def_id) { |
| 928 | Some(impl_id) => match tcx.impl_subject(impl_id) { |
| 929 | ty::ImplSubject::Inherent(ty) => match ty.kind() { |
| 930 | ty::TyKind::Adt(adt, substs) => { |
| 931 | assert_eq!(0, substs.len(), "Callers should filter out generics"); |
| 932 | Some(tcx.item_name(adt.did())) |
| 933 | } |
| 934 | _ => panic!("Non-ADT `impl`s should be filtered by caller"), |
| 935 | }, |
| 936 | ty::ImplSubject::Trait(_) => panic!("Trait methods should be filtered by caller"), |
| 937 | }, |
| 938 | None => None, |
| 939 | }; |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 940 | let needs_definition = fn_name.as_str() != thunk_name; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 941 | let main_api = { |
| 942 | let doc_comment = { |
| 943 | let doc_comment = format_doc_comment(tcx, local_def_id); |
| 944 | quote! { __NEWLINE__ #doc_comment } |
| 945 | }; |
| 946 | |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 947 | let mut prereqs = main_api_prereqs.clone(); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 948 | prereqs.move_defs_to_fwd_decls(); |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 949 | |
| 950 | let static_ = if is_static_method { |
| 951 | quote! { static } |
| 952 | } else { |
| 953 | quote! {} |
| 954 | }; |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 955 | let extern_c_or_inline = if !needs_definition { |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 956 | quote! { extern "C" } |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 957 | } else { |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 958 | quote! { inline } |
| 959 | }; |
| 960 | CcSnippet { |
| 961 | prereqs, |
| 962 | tokens: quote! { |
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 | #doc_comment |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 965 | #static_ #extern_c_or_inline |
| 966 | #main_api_ret_type #main_api_fn_name ( #( #main_api_params ),* ); |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 967 | __NEWLINE__ |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 968 | }, |
| 969 | } |
| 970 | }; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 971 | let cc_details = if !needs_definition { |
| 972 | CcSnippet::default() |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 973 | } else { |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 974 | 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] | 975 | let struct_name = match struct_name.as_ref() { |
| 976 | None => quote! {}, |
| 977 | Some(symbol) => { |
| 978 | let name = format_cc_ident(symbol.as_str()) |
| 979 | .expect("Caller of format_fn should verify struct via format_adt_core"); |
| 980 | quote! { #name :: } |
| 981 | } |
| 982 | }; |
| 983 | |
| 984 | let mut prereqs = main_api_prereqs; |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 985 | let thunk_decl = |
| 986 | 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] | 987 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 988 | let mut thunk_args = params |
| 989 | .iter() |
| 990 | .map(|Param { cc_name, ty, .. }| { |
| 991 | if is_c_abi_compatible_by_value(*ty) { |
| 992 | quote! { #cc_name } |
| 993 | } else { |
| 994 | quote! { & #cc_name } |
| 995 | } |
| 996 | }) |
| 997 | .collect_vec(); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 998 | let impl_body: TokenStream; |
| 999 | if is_c_abi_compatible_by_value(sig.output()) { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1000 | impl_body = quote! { |
| 1001 | return __crubit_internal :: #thunk_name( #( #thunk_args ),* ); |
| 1002 | }; |
| 1003 | } else { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1004 | thunk_args.push(quote! { __ret_slot.Get() }); |
| 1005 | impl_body = quote! { |
| 1006 | crubit::ReturnValueSlot<#main_api_ret_type> __ret_slot; |
| 1007 | __crubit_internal :: #thunk_name( #( #thunk_args ),* ); |
| 1008 | return std::move(__ret_slot).AssumeInitAndTakeValue(); |
| 1009 | }; |
| 1010 | prereqs.includes.insert(CcInclude::utility()); |
| 1011 | prereqs.includes.insert(input.support_header("internal/return_value_slot.h")); |
| 1012 | }; |
| 1013 | CcSnippet { |
| 1014 | prereqs, |
| 1015 | tokens: quote! { |
| 1016 | __NEWLINE__ |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 1017 | #thunk_decl |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1018 | inline #main_api_ret_type #struct_name #main_api_fn_name ( |
| 1019 | #( #main_api_params ),* ) { |
| 1020 | #impl_body |
| 1021 | } |
| 1022 | __NEWLINE__ |
| 1023 | }, |
| 1024 | } |
| 1025 | }; |
| 1026 | |
| 1027 | let rs_details = if !needs_thunk { |
| 1028 | quote! {} |
| 1029 | } else { |
Lukasz Anforowicz | 6b1ee8c | 2023-04-04 11:58:29 -0700 | [diff] [blame] | 1030 | let crate_name = make_rs_ident(krate.as_str()); |
| 1031 | let mod_path = mod_path.format_for_rs(); |
| 1032 | let fn_name = make_rs_ident(fn_name.as_str()); |
| 1033 | let struct_name = match struct_name.as_ref() { |
| 1034 | None => quote! {}, |
| 1035 | Some(symbol) => { |
| 1036 | let name = make_rs_ident(symbol.as_str()); |
| 1037 | quote! { #name :: } |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1038 | } |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 1039 | }; |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 1040 | 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] | 1041 | 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] | 1042 | }; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1043 | Ok(ApiSnippets { main_api, cc_details, rs_details }) |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1044 | } |
| 1045 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1046 | /// Represents bindings for the "core" part of an algebraic data type (an ADT - |
| 1047 | /// a struct, an enum, or a union) in a way that supports later injecting the |
| 1048 | /// other parts like so: |
| 1049 | /// |
| 1050 | /// ``` |
| 1051 | /// quote! { |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1052 | /// #keyword #alignment #name final { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1053 | /// #core |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1054 | /// #decls_of_other_parts // (e.g. struct fields, methods, etc.) |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1055 | /// } |
| 1056 | /// } |
| 1057 | /// ``` |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1058 | /// |
| 1059 | /// `keyword`, `name` are stored separately, to support formatting them as a |
| 1060 | /// forward declaration - e.g. `struct SomeStruct`. |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1061 | struct AdtCoreBindings { |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1062 | /// DefId of the ADT. |
| 1063 | def_id: DefId, |
| 1064 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1065 | /// C++ tag - e.g. `struct`, `class`, `enum`, or `union`. This isn't always |
| 1066 | /// a direct mapping from Rust (e.g. a Rust `enum` might end up being |
| 1067 | /// represented as an opaque C++ `struct`). |
| 1068 | keyword: TokenStream, |
| 1069 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1070 | /// C++ translation of the ADT identifier - e.g. `SomeStruct`. |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1071 | /// |
| 1072 | /// A _short_ name is sufficient (i.e. there is no need to use a |
| 1073 | /// namespace-qualified name), for `CcSnippet`s that are emitted into |
| 1074 | /// the same namespace as the ADT. (This seems to be all the snippets |
| 1075 | /// today.) |
| 1076 | cc_short_name: TokenStream, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1077 | |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1078 | /// Rust spelling of the ADT type - e.g. |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1079 | /// `::some_crate::some_module::SomeStruct`. |
| 1080 | rs_fully_qualified_name: TokenStream, |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1081 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1082 | /// `core` contains declarations of |
| 1083 | /// - the default constructor |
| 1084 | /// - the copy constructor |
| 1085 | /// - the move constructor |
| 1086 | /// - the copy assignment operator |
| 1087 | /// - the move assignment operator |
| 1088 | /// - the destructor |
| 1089 | core: TokenStream, |
| 1090 | |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1091 | alignment_in_bytes: u64, |
| 1092 | size_in_bytes: u64, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1093 | } |
| 1094 | |
Lukasz Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 1095 | /// Like `TyCtxt::is_directly_public`, but works not only with `LocalDefId`, but |
| 1096 | /// also with `DefId`. |
| 1097 | fn is_directly_public(tcx: TyCtxt, def_id: DefId) -> bool { |
| 1098 | match def_id.as_local() { |
| 1099 | None => { |
| 1100 | // This mimics the checks in `try_print_visible_def_path_recur` in |
| 1101 | // `compiler/rustc_middle/src/ty/print/pretty.rs`. |
| 1102 | let actual_parent = tcx.opt_parent(def_id); |
| 1103 | let visible_parent = tcx.visible_parent_map(()).get(&def_id).copied(); |
| 1104 | actual_parent == visible_parent |
| 1105 | } |
| 1106 | Some(local_def_id) => tcx.effective_visibilities(()).is_directly_public(local_def_id), |
| 1107 | } |
| 1108 | } |
| 1109 | |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1110 | fn get_layout<'tcx>(tcx: TyCtxt<'tcx>, ty: Ty<'tcx>) -> Result<Layout<'tcx>> { |
| 1111 | // TODO(b/259749095): Support non-empty set of generic parameters. |
| 1112 | let param_env = ty::ParamEnv::empty(); |
| 1113 | |
| 1114 | tcx.layout_of(param_env.and(ty)).map(|ty_and_layout| ty_and_layout.layout).map_err( |
| 1115 | |layout_err| { |
| 1116 | // Have to use `.map_err`, because `LayoutError` doesn't satisfy the |
| 1117 | // `anyhow::context::ext::StdError` trait bound. |
| 1118 | anyhow!("Error computing the layout: {layout_err}") |
| 1119 | }, |
| 1120 | ) |
| 1121 | } |
| 1122 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1123 | /// Formats the core of an algebraic data type (an ADT - a struct, an enum, or a |
| 1124 | /// union) represented by `def_id`. |
| 1125 | /// |
| 1126 | /// The "core" means things that are necessary for a succesful binding (e.g. |
| 1127 | /// inability to generate a correct C++ destructor means that the ADT cannot |
| 1128 | /// have any bindings). "core" excludes things that are A) infallible (e.g. |
| 1129 | /// struct or union fields which can always be translated into private, opaque |
| 1130 | /// blobs of bytes) or B) optional (e.g. a problematic instance method |
| 1131 | /// can just be ignored, unlike a problematic destructor). The split between |
| 1132 | /// fallible "core" and non-fallible "rest" is motivated by the need to avoid |
| 1133 | /// cycles / infinite recursion (e.g. when processing fields that refer back to |
| 1134 | /// the struct type, possible with an indirection of a pointer). |
| 1135 | /// |
| 1136 | /// `format_adt_core` is used both to 1) format bindings for the core of an ADT, |
| 1137 | /// and 2) check if formatting would have succeeded (e.g. when called from |
| 1138 | /// `format_ty`). The 2nd case is needed for ADTs defined in any crate - this |
| 1139 | /// is why the `def_id` parameter is a DefId rather than LocalDefId. |
| 1140 | // |
Lukasz Anforowicz | 1ab5e87 | 2022-12-05 10:07:00 -0800 | [diff] [blame] | 1141 | // TODO(b/259724276): This function's results should be memoized. |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1142 | fn format_adt_core(tcx: TyCtxt, def_id: DefId) -> Result<AdtCoreBindings> { |
Lukasz Anforowicz | 0c07b09 | 2023-03-03 10:34:15 -0800 | [diff] [blame] | 1143 | let ty = tcx.type_of(def_id).subst_identity(); |
Lukasz Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 1144 | assert!(ty.is_adt()); |
| 1145 | assert!(is_directly_public(tcx, def_id), "Caller should verify"); |
| 1146 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1147 | // TODO(b/259749095): Support non-empty set of generic parameters. |
| 1148 | let param_env = ty::ParamEnv::empty(); |
| 1149 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1150 | if ty.needs_drop(tcx, param_env) { |
| 1151 | // TODO(b/258251148): Support custom `Drop` impls. |
| 1152 | bail!("`Drop` trait and \"drop glue\" are not supported yet (b/258251148)"); |
| 1153 | } |
| 1154 | |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 1155 | let adt_def = ty.ty_adt_def().expect("`def_id` needs to identify an ADT"); |
| 1156 | let keyword = match adt_def.adt_kind() { |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 1157 | ty::AdtKind::Struct | ty::AdtKind::Enum => quote! { struct }, |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 1158 | ty::AdtKind::Union => quote! { union }, |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 1159 | }; |
| 1160 | |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1161 | let item_name = tcx.item_name(def_id); |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1162 | let rs_fully_qualified_name = format_ty_for_rs(tcx, ty)?; |
| 1163 | let cc_short_name = |
| 1164 | format_cc_ident(item_name.as_str()).context("Error formatting item name")?; |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1165 | |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1166 | let layout = get_layout(tcx, ty) |
| 1167 | .with_context(|| format!("Error computing the layout of #{item_name}"))?; |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1168 | let alignment_in_bytes = { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1169 | // Only the ABI-mandated alignment is considered (i.e. `AbiAndPrefAlign::pref` |
| 1170 | // is ignored), because 1) Rust's `std::mem::align_of` returns the |
| 1171 | // ABI-mandated alignment and 2) the generated C++'s `alignas(...)` |
| 1172 | // should specify the minimal/mandatory alignment. |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1173 | layout.align().abi.bytes() |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1174 | }; |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1175 | let size_in_bytes = layout.size().bytes(); |
| 1176 | 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] | 1177 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1178 | let core = quote! { |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1179 | // TODO(b/258249993): Provide `default` copy constructor and assignment operator if |
| 1180 | // the wrapped type is `Copy` on Rust side. |
| 1181 | // TODO(b/259741191): If the wrapped type implements the `Clone` trait, then we should |
| 1182 | // *consider* calling `clone` from the copy constructor and `clone_from` from the copy |
| 1183 | // assignment operator. |
| 1184 | #cc_short_name(const #cc_short_name&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1185 | |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1186 | // The generated bindings have to follow Rust move semantics: |
| 1187 | // * All Rust types are memcpy-movable (e.g. <internal link>/constructors.html says |
| 1188 | // that "Every type must be ready for it to be blindly memcopied to somewhere else |
| 1189 | // in memory") |
| 1190 | // * The only valid operation on a moved-from non-`Copy` Rust struct is to assign to |
| 1191 | // it. |
| 1192 | // |
| 1193 | // The generated C++ bindings match the required semantics because they: |
| 1194 | // * Generate trivial` C++ move constructor and move assignment operator. Per |
| 1195 | // <internal link>/cpp/language/move_constructor#Trivial_move_constructor: "A trivial move |
| 1196 | // constructor is a constructor that performs the same action as the trivial copy |
| 1197 | // constructor, that is, makes a copy of the object representation as if by |
| 1198 | // std::memmove." |
| 1199 | // * Generate trivial C++ destructor. (Types that implement `Drop` trait or require |
| 1200 | // "drop glue" are not *yet* supported - this might eventually change as part of the |
| 1201 | // work tracked under b/258251148). Per |
| 1202 | // <internal link>/cpp/language/destructor#Trivial_destructor: "A trivial destructor is a |
| 1203 | // destructor that performs no action." |
| 1204 | // |
| 1205 | // In particular, note that the following C++ code and Rust code are exactly equivalent |
| 1206 | // (except that in Rust, reuse of `y` is forbidden at compile time, whereas in C++, |
| 1207 | // it's only prohibited by convention): |
| 1208 | // * C++, assumming trivial move constructor and trivial destructor: |
| 1209 | // `auto x = std::move(y);` |
| 1210 | // * Rust, assumming non-`Copy`, no custom `Drop` or drop glue: |
| 1211 | // `let x = y;` |
| 1212 | // |
| 1213 | // TODO(b/258251148): If the ADT provides a custom `Drop` impls or requires drop glue, |
| 1214 | // then extra care should be taken to ensure the C++ destructor can handle the |
| 1215 | // moved-from object in a way that meets Rust move semantics. For example, the |
| 1216 | // generated C++ move constructor might need to assign `Default::default()` to the |
| 1217 | // moved-from object. |
| 1218 | #cc_short_name(#cc_short_name&&) = default; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1219 | |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1220 | // TODO(b/258235219): Providing assignment operators enables mutation which |
| 1221 | // may negatively interact with support for references. Therefore until we |
| 1222 | // have more confidence in our reference-handling-plans, we are deleting the |
| 1223 | // assignment operators. |
| 1224 | // |
| 1225 | // (Move assignment operator has another set of concerns and constraints - see the |
| 1226 | // comment for the move constructor above). |
| 1227 | #cc_short_name& operator=(const #cc_short_name&) = delete; |
| 1228 | #cc_short_name& operator=(#cc_short_name&&) = delete; |
Lukasz Anforowicz | 554ed65 | 2023-01-12 15:41:58 -0800 | [diff] [blame] | 1229 | |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1230 | // TODO(b/258251148): Support custom `Drop` impls and drop glue. |
| 1231 | ~#cc_short_name() = default; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1232 | }; |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1233 | Ok(AdtCoreBindings { |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1234 | def_id, |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 1235 | keyword, |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1236 | cc_short_name, |
| 1237 | rs_fully_qualified_name, |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1238 | core, |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1239 | alignment_in_bytes, |
| 1240 | size_in_bytes, |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1241 | }) |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1242 | } |
| 1243 | |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1244 | fn format_fields(input: &Input, core: &AdtCoreBindings) -> ApiSnippets { |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 1245 | let tcx = input.tcx; |
| 1246 | |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1247 | // TODO(b/259749095): Support non-empty set of generic parameters. |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1248 | let substs_ref = ty::List::empty().as_substs(); |
| 1249 | |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1250 | struct FieldTypeInfo { |
| 1251 | size: u64, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1252 | cc_type: CcSnippet, |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1253 | } |
| 1254 | struct Field { |
| 1255 | type_info: Result<FieldTypeInfo>, |
| 1256 | cc_name: TokenStream, |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 1257 | rs_name: TokenStream, |
| 1258 | is_public: bool, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1259 | index: usize, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1260 | offset: u64, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1261 | offset_of_next_field: u64, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1262 | } |
| 1263 | let ty = tcx.type_of(core.def_id).subst_identity(); |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1264 | let layout = |
| 1265 | 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] | 1266 | let fields: Vec<Field> = if ty.is_enum() || ty.is_union() { |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1267 | // Note that `#[repr(Rust)]` unions don't guarantee that all their fields |
| 1268 | // have offset 0. |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1269 | vec![Field { |
| 1270 | type_info: Err(anyhow!( |
| 1271 | "No support for bindings of individual fields of \ |
| 1272 | `union` (b/272801632) or `enum`" |
| 1273 | )), |
| 1274 | cc_name: quote! { __opaque_blob_of_bytes }, |
| 1275 | rs_name: quote! { __opaque_blob_of_bytes }, |
| 1276 | is_public: false, |
| 1277 | index: 0, |
| 1278 | offset: 0, |
| 1279 | offset_of_next_field: core.size_in_bytes, |
| 1280 | }] |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1281 | } else { |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1282 | let mut fields = ty |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1283 | .ty_adt_def() |
| 1284 | .expect("`core.def_id` needs to identify an ADT") |
| 1285 | .all_fields() |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1286 | .sorted_by_key(|f| tcx.def_span(f.did)) |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1287 | .enumerate() |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1288 | .map(|(index, field_def)| { |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1289 | let field_ty = field_def.ty(tcx, substs_ref); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1290 | let size = get_layout(tcx, field_ty).map(|layout| layout.size().bytes()); |
| 1291 | let type_info = size.and_then(|size| { |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 1292 | Ok(FieldTypeInfo { |
| 1293 | size, |
| 1294 | cc_type: format_ty_for_cc(input, field_ty, TypeLocation::Other)?, |
| 1295 | }) |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1296 | }); |
| 1297 | let name = field_def.ident(tcx); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 1298 | let cc_name = format_cc_ident(name.as_str()) |
| 1299 | .unwrap_or_else(|_err| format_ident!("__field{index}").into_token_stream()); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 1300 | let rs_name = { |
| 1301 | let name_starts_with_digit = name |
| 1302 | .as_str() |
| 1303 | .chars() |
| 1304 | .next() |
| 1305 | .expect("Empty names are unexpected (here and in general)") |
| 1306 | .is_ascii_digit(); |
| 1307 | if name_starts_with_digit { |
| 1308 | let index = Literal::usize_unsuffixed(index); |
| 1309 | quote! { #index } |
| 1310 | } else { |
| 1311 | let name = make_rs_ident(name.as_str()); |
| 1312 | quote! { #name } |
| 1313 | } |
| 1314 | }; |
| 1315 | let is_public = field_def.vis == ty::Visibility::Public; |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1316 | |
| 1317 | // `offset` and `offset_of_next_field` will be fixed by FieldsShape::Arbitrary |
| 1318 | // branch below. |
| 1319 | let offset = 0; |
| 1320 | let offset_of_next_field = 0; |
| 1321 | |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1322 | Field { |
| 1323 | type_info, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1324 | cc_name, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1325 | rs_name, |
| 1326 | is_public, |
| 1327 | index, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 1328 | offset, |
| 1329 | offset_of_next_field, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1330 | } |
| 1331 | }) |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1332 | .collect_vec(); |
| 1333 | match layout.fields() { |
| 1334 | FieldsShape::Arbitrary { offsets, .. } => { |
| 1335 | for (index, offset) in offsets.iter().enumerate() { |
| 1336 | // Documentation of `FieldsShape::Arbitrary says that the offsets are "ordered |
| 1337 | // to match the source definition order". We can coorelate them with elements |
| 1338 | // of the `fields` vector because we've explicitly `sorted_by_key` using |
| 1339 | // `def_span`. |
| 1340 | fields[index].offset = offset.bytes(); |
| 1341 | } |
Devin Jeanpierre | 9e15d0b | 2023-04-06 13:18:22 -0700 | [diff] [blame] | 1342 | // Sort by offset first; ZSTs in the same offset are sorted by source order. |
| 1343 | fields.sort_by_key(|field| (field.offset, field.index)); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1344 | let next_offsets = fields |
| 1345 | .iter() |
| 1346 | .map(|Field { offset, .. }| *offset) |
| 1347 | .skip(1) |
| 1348 | .chain(once(core.size_in_bytes)) |
| 1349 | .collect_vec(); |
| 1350 | for (field, next_offset) in fields.iter_mut().zip(next_offsets) { |
| 1351 | field.offset_of_next_field = next_offset; |
| 1352 | } |
| 1353 | fields |
| 1354 | } |
| 1355 | unexpected => panic!("Unexpected FieldsShape: {unexpected:?}"), |
| 1356 | } |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1357 | }; |
| 1358 | |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1359 | let cc_details = if fields.is_empty() { |
| 1360 | CcSnippet::default() |
| 1361 | } else { |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1362 | let adt_cc_name = &core.cc_short_name; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1363 | let cc_assertions: TokenStream = fields |
| 1364 | .iter() |
| 1365 | .map(|Field { cc_name, offset, .. }| { |
| 1366 | let offset = Literal::u64_unsuffixed(*offset); |
| 1367 | quote! { static_assert(#offset == offsetof(#adt_cc_name, #cc_name)); } |
| 1368 | }) |
| 1369 | .collect(); |
| 1370 | CcSnippet::new(quote! { |
| 1371 | inline void #adt_cc_name::__crubit_field_offset_assertions() { |
| 1372 | #cc_assertions |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 1373 | } |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1374 | }) |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1375 | }; |
| 1376 | let rs_details: TokenStream = { |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1377 | let adt_rs_name = &core.rs_fully_qualified_name; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1378 | fields |
| 1379 | .iter() |
| 1380 | .filter(|Field { is_public, .. }| *is_public) |
| 1381 | .map(|Field { rs_name, offset, .. }| { |
| 1382 | let expected_offset = Literal::u64_unsuffixed(*offset); |
| 1383 | let actual_offset = quote! { memoffset::offset_of!(#adt_rs_name, #rs_name) }; |
| 1384 | quote! { const _: () = assert!(#actual_offset == #expected_offset); } |
| 1385 | }) |
| 1386 | .collect() |
| 1387 | }; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 1388 | let main_api = { |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1389 | let assertions_method_decl = if fields.is_empty() { |
| 1390 | quote! {} |
| 1391 | } else { |
| 1392 | // We put the assertions in a method so that they can read private member |
| 1393 | // variables. |
| 1394 | quote! { inline static void __crubit_field_offset_assertions(); } |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1395 | }; |
| 1396 | |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1397 | let mut prereqs = CcPrerequisites::default(); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1398 | let fields: TokenStream = fields |
| 1399 | .into_iter() |
| 1400 | .map(|field| { |
| 1401 | let cc_name = field.cc_name; |
| 1402 | match field.type_info { |
| 1403 | Err(err) => { |
Devin Jeanpierre | 9e15d0b | 2023-04-06 13:18:22 -0700 | [diff] [blame] | 1404 | let size = field.offset_of_next_field - field.offset; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1405 | let msg = |
| 1406 | format!("Field type has been replaced with a blob of bytes: {err:#}"); |
Devin Jeanpierre | 9e15d0b | 2023-04-06 13:18:22 -0700 | [diff] [blame] | 1407 | |
| 1408 | // Empty arrays are ill-formed, but also unnecessary for padding. |
| 1409 | if size > 0 { |
| 1410 | let size = Literal::u64_unsuffixed(size); |
| 1411 | quote! { |
| 1412 | __COMMENT__ #msg |
| 1413 | unsigned char #cc_name[#size]; |
| 1414 | } |
| 1415 | } else { |
| 1416 | // TODO(b/258259459): finalize the approach here. |
| 1417 | // Possibly we should, rather than using no_unique_address, drop the |
| 1418 | // field entirely. This also requires removing the field's assertions, |
| 1419 | // added above. |
| 1420 | quote! { |
| 1421 | __COMMENT__ #msg |
| 1422 | [[no_unique_address]] struct{} #cc_name; |
| 1423 | } |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1424 | } |
| 1425 | } |
| 1426 | Ok(FieldTypeInfo { cc_type, size }) => { |
| 1427 | let padding = field.offset_of_next_field - field.offset - size; |
| 1428 | let padding = if padding == 0 { |
| 1429 | quote! {} |
| 1430 | } else { |
| 1431 | let padding = Literal::u64_unsuffixed(padding); |
| 1432 | let ident = format_ident!("__padding{}", field.index); |
| 1433 | quote! { unsigned char #ident[#padding]; } |
| 1434 | }; |
| 1435 | let cc_type = cc_type.into_tokens(&mut prereqs); |
| 1436 | quote! { #cc_type #cc_name; #padding } |
| 1437 | } |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1438 | } |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 1439 | }) |
| 1440 | .collect(); |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1441 | |
| 1442 | CcSnippet { |
| 1443 | prereqs, |
| 1444 | tokens: quote! { |
| 1445 | // TODO(b/271002281): Preserve actual field visibility. |
| 1446 | private: __NEWLINE__ |
| 1447 | #fields |
| 1448 | #assertions_method_decl |
| 1449 | }, |
| 1450 | } |
| 1451 | }; |
| 1452 | |
| 1453 | ApiSnippets { main_api, cc_details, rs_details } |
| 1454 | } |
| 1455 | |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1456 | /// Finds the `Impl` of a trait impl for `self_ty`. Returns an error if the |
| 1457 | /// impl wasn't found. |
| 1458 | /// |
| 1459 | /// `self_ty` should specify a *local* type (i.e. type defined in the crate |
| 1460 | /// being "compiled"). |
| 1461 | /// |
| 1462 | /// `trait_name` should specify the name of a `core` trait - e.g. |
| 1463 | /// [`sym::Default`](https://doc.rust-lang.org/beta/nightly-rustc/rustc_span/symbol/sym/constant.Default.html) is a valid |
| 1464 | /// argument. |
| 1465 | fn find_core_trait_impl<'tcx>( |
| 1466 | tcx: TyCtxt<'tcx>, |
| 1467 | self_ty: Ty<'tcx>, |
| 1468 | trait_name: Symbol, |
| 1469 | ) -> Result<&'tcx Impl<'tcx>> { |
| 1470 | let trait_id = tcx |
| 1471 | .get_diagnostic_item(trait_name) |
| 1472 | .expect("`find_core_trait_impl` should only be called with `core`, always-present traits"); |
| 1473 | // TODO(b/275387739): Eventually we might need to support blanket impls. |
| 1474 | let mut impls = tcx.non_blanket_impls_for_ty(trait_id, self_ty); |
| 1475 | let impl_id = impls.next(); |
| 1476 | if impl_id.is_some() { |
| 1477 | assert_eq!(None, impls.next(), "Expecting only a single trait impl"); |
| 1478 | } |
| 1479 | let impl_id = |
| 1480 | impl_id.ok_or_else(|| anyhow!("`{self_ty}` doesn't implement the `{trait_name}` trait"))?; |
| 1481 | let impl_id = impl_id.expect_local(); // Expecting that `self_ty` is a local type. |
| 1482 | match &tcx.hir().expect_item(impl_id).kind { |
| 1483 | ItemKind::Impl(impl_) => Ok(impl_), |
| 1484 | other => panic!("Unexpected `ItemKind` from `non_blanket_impls_for_ty`: {other:?}"), |
| 1485 | } |
| 1486 | } |
| 1487 | |
| 1488 | /// Formats a default constructor for an ADT if possible (i.e. if the `Default` |
| 1489 | /// trait is implemented for the ADT). Returns an error otherwise (e.g. if |
| 1490 | /// there is no `Default` impl). |
| 1491 | fn format_default_ctor(input: &Input, core: &AdtCoreBindings) -> Result<ApiSnippets> { |
| 1492 | let tcx = input.tcx; |
| 1493 | let ty = tcx.type_of(core.def_id).subst_identity(); |
| 1494 | |
| 1495 | let trait_impl = find_core_trait_impl(input.tcx, ty, sym::Default)?; |
| 1496 | assert_eq!(trait_impl.items.len(), 1, "Only the `default` method is expected"); |
| 1497 | assert_eq!(trait_impl.items[0].ident.name.as_str(), "default"); |
| 1498 | let cc_struct_name = &core.cc_short_name; |
| 1499 | let main_api = CcSnippet::new(quote! { |
| 1500 | __NEWLINE__ __COMMENT__ "Default::default" |
| 1501 | inline #cc_struct_name(); __NEWLINE__ __NEWLINE__ |
| 1502 | }); |
| 1503 | let fn_def_id = trait_impl.items[0].id.owner_id.def_id; |
| 1504 | let thunk_name = get_thunk_name(get_symbol_name(tcx, fn_def_id)?); |
| 1505 | let cc_details = { |
| 1506 | let thunk_name = format_cc_ident(&thunk_name)?; |
| 1507 | let CcSnippet { tokens: thunk_decl, prereqs } = |
| 1508 | format_thunk_decl(input, fn_def_id, &thunk_name)?; |
| 1509 | let tokens = quote! { |
| 1510 | #thunk_decl |
| 1511 | #cc_struct_name::#cc_struct_name() { |
| 1512 | __crubit_internal::#thunk_name(this); |
| 1513 | } |
| 1514 | }; |
| 1515 | CcSnippet { tokens, prereqs } |
| 1516 | }; |
| 1517 | let rs_details = { |
| 1518 | let struct_name = &core.rs_fully_qualified_name; |
| 1519 | let fully_qualified_fn_name = |
| 1520 | quote! { <#struct_name as ::core::default::Default>::default }; |
| 1521 | format_thunk_impl(tcx, fn_def_id, &thunk_name, fully_qualified_fn_name)? |
| 1522 | }; |
| 1523 | Ok(ApiSnippets { main_api, cc_details, rs_details }) |
| 1524 | } |
| 1525 | |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1526 | /// Formats an algebraic data type (an ADT - a struct, an enum, or a union) |
| 1527 | /// represented by `core`. This function is infallible - after |
| 1528 | /// `format_adt_core` returns success we have committed to emitting C++ bindings |
| 1529 | /// for the ADT. |
| 1530 | fn format_adt(input: &Input, core: &AdtCoreBindings) -> ApiSnippets { |
| 1531 | let tcx = input.tcx; |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1532 | let adt_cc_name = &core.cc_short_name; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1533 | |
| 1534 | // `format_adt` should only be called for local ADTs. |
| 1535 | let local_def_id = core.def_id.expect_local(); |
| 1536 | |
| 1537 | let ApiSnippets { |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1538 | main_api: default_ctor_main_api, |
| 1539 | cc_details: default_ctor_cc_details, |
| 1540 | rs_details: default_ctor_rs_details, |
| 1541 | } = format_default_ctor(input, core).unwrap_or_else(|err| { |
| 1542 | let msg = format!("{err:#}"); |
| 1543 | ApiSnippets { |
| 1544 | main_api: CcSnippet::new(quote! { |
| 1545 | __NEWLINE__ __COMMENT__ #msg |
| 1546 | #adt_cc_name() = delete; __NEWLINE__ |
| 1547 | }), |
| 1548 | ..Default::default() |
| 1549 | } |
| 1550 | }); |
| 1551 | |
| 1552 | let ApiSnippets { |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1553 | main_api: fields_main_api, |
| 1554 | cc_details: fields_cc_details, |
| 1555 | rs_details: fields_rs_details, |
| 1556 | } = format_fields(input, core); |
| 1557 | |
| 1558 | let ApiSnippets { |
| 1559 | main_api: impl_items_main_api, |
| 1560 | cc_details: impl_items_cc_details, |
| 1561 | rs_details: impl_items_rs_details, |
| 1562 | } = tcx |
| 1563 | .inherent_impls(core.def_id) |
| 1564 | .iter() |
| 1565 | .map(|impl_id| tcx.hir().expect_item(impl_id.expect_local())) |
| 1566 | .flat_map(|item| match &item.kind { |
| 1567 | ItemKind::Impl(impl_) => impl_.items, |
| 1568 | other => panic!("Unexpected `ItemKind` from `inherent_impls`: {other:?}"), |
| 1569 | }) |
| 1570 | .sorted_by_key(|impl_item_ref| { |
| 1571 | let def_id = impl_item_ref.id.owner_id.def_id; |
| 1572 | tcx.def_span(def_id) |
| 1573 | }) |
| 1574 | .filter_map(|impl_item_ref| { |
| 1575 | let def_id = impl_item_ref.id.owner_id.def_id; |
| 1576 | if !tcx.effective_visibilities(()).is_directly_public(def_id) { |
| 1577 | return None; |
| 1578 | } |
| 1579 | let result = match impl_item_ref.kind { |
| 1580 | AssocItemKind::Fn { .. } => format_fn(input, def_id).map(Some), |
| 1581 | other => Err(anyhow!("Unsupported `impl` item kind: {other:?}")), |
| 1582 | }; |
| 1583 | result.unwrap_or_else(|err| Some(format_unsupported_def(tcx, def_id, err))) |
| 1584 | }) |
| 1585 | .collect(); |
| 1586 | |
| 1587 | let alignment = Literal::u64_unsuffixed(core.alignment_in_bytes); |
| 1588 | let size = Literal::u64_unsuffixed(core.size_in_bytes); |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1589 | let main_api = { |
| 1590 | let cc_packed_attribute = { |
| 1591 | let has_packed_attribute = tcx |
| 1592 | .get_attrs(core.def_id, rustc_span::symbol::sym::repr) |
| 1593 | .flat_map(|attr| rustc_attr::parse_repr_attr(tcx.sess(), attr)) |
| 1594 | .any(|repr| matches!(repr, rustc_attr::ReprPacked { .. })); |
| 1595 | if has_packed_attribute { |
| 1596 | quote! { __attribute__((packed)) } |
| 1597 | } else { |
| 1598 | quote! {} |
| 1599 | } |
| 1600 | }; |
| 1601 | |
| 1602 | let doc_comment = format_doc_comment(tcx, core.def_id.expect_local()); |
| 1603 | let keyword = &core.keyword; |
| 1604 | let core = &core.core; |
| 1605 | |
| 1606 | let mut prereqs = CcPrerequisites::default(); |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1607 | 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] | 1608 | 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] | 1609 | quote! {} |
| 1610 | } else { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1611 | let tokens = impl_items_main_api.into_tokens(&mut prereqs); |
| 1612 | quote! { public: #tokens } |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1613 | }; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1614 | let fields_main_api = fields_main_api.into_tokens(&mut prereqs); |
Lukasz Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 1615 | prereqs.fwd_decls.remove(&local_def_id); |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1616 | |
| 1617 | CcSnippet { |
| 1618 | prereqs, |
| 1619 | tokens: quote! { |
| 1620 | __NEWLINE__ #doc_comment |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 1621 | #keyword alignas(#alignment) #cc_packed_attribute #adt_cc_name final { |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1622 | public: |
| 1623 | #default_ctor_main_api |
| 1624 | #core |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1625 | #impl_items_main_api |
| 1626 | #fields_main_api |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1627 | }; |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1628 | __NEWLINE__ |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 1629 | }, |
| 1630 | } |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1631 | }; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1632 | let cc_details = { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1633 | let mut prereqs = CcPrerequisites::default(); |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1634 | 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] | 1635 | 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] | 1636 | let fields_cc_details = fields_cc_details.into_tokens(&mut prereqs); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1637 | prereqs.defs.insert(local_def_id); |
| 1638 | CcSnippet { |
| 1639 | prereqs, |
| 1640 | tokens: quote! { |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 1641 | __NEWLINE__ |
| 1642 | static_assert( |
| 1643 | sizeof(#adt_cc_name) == #size, |
| 1644 | "Verify that struct layout didn't change since this header got generated"); |
| 1645 | static_assert( |
| 1646 | alignof(#adt_cc_name) == #alignment, |
| 1647 | "Verify that struct layout didn't change since this header got generated"); |
| 1648 | __NEWLINE__ |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1649 | #default_ctor_cc_details |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1650 | #impl_items_cc_details |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1651 | #fields_cc_details |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1652 | }, |
| 1653 | } |
Lukasz Anforowicz | 3b57954 | 2023-02-06 11:23:49 -0800 | [diff] [blame] | 1654 | }; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1655 | let rs_details = { |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1656 | let adt_rs_name = &core.rs_fully_qualified_name; |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1657 | quote! { |
| 1658 | const _: () = assert!(::std::mem::size_of::<#adt_rs_name>() == #size); |
| 1659 | const _: () = assert!(::std::mem::align_of::<#adt_rs_name>() == #alignment); |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 1660 | #default_ctor_rs_details |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 1661 | #impl_items_rs_details |
| 1662 | #fields_rs_details |
| 1663 | } |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1664 | }; |
| 1665 | ApiSnippets { main_api, cc_details, rs_details } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1666 | } |
| 1667 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1668 | /// Formats the forward declaration of an algebraic data type (an ADT - a |
| 1669 | /// struct, an enum, or a union), returning something like |
| 1670 | /// `quote!{ struct SomeStruct; }`. |
| 1671 | /// |
| 1672 | /// Will panic if `def_id` doesn't identify an ADT that can be successfully |
| 1673 | /// handled by `format_adt_core`. |
| 1674 | fn format_fwd_decl(tcx: TyCtxt, def_id: LocalDefId) -> TokenStream { |
| 1675 | let def_id = def_id.to_def_id(); // LocalDefId -> DefId conversion. |
| 1676 | |
| 1677 | // `format_fwd_decl` should only be called for items from |
| 1678 | // `CcPrerequisites::fwd_decls` and `fwd_decls` should only contain ADTs |
| 1679 | // that `format_adt_core` succeeds for. |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1680 | let AdtCoreBindings { keyword, cc_short_name, .. } = format_adt_core(tcx, def_id) |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1681 | .expect("`format_fwd_decl` should only be called if `format_adt_core` succeeded"); |
| 1682 | |
Lukasz Anforowicz | ce56a80 | 2023-04-04 12:18:19 -0700 | [diff] [blame] | 1683 | quote! { #keyword #cc_short_name; } |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1684 | } |
| 1685 | |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1686 | fn format_source_location(tcx: TyCtxt, local_def_id: LocalDefId) -> String { |
| 1687 | let def_span = tcx.def_span(local_def_id); |
| 1688 | let rustc_span::FileLines { file, lines } = |
| 1689 | match tcx.sess().source_map().span_to_lines(def_span) { |
| 1690 | Ok(filelines) => filelines, |
| 1691 | Err(_) => return "unknown location".to_string(), |
| 1692 | }; |
| 1693 | let file_name = file.name.prefer_local().to_string(); |
| 1694 | // Note: line_index starts at 0, while CodeSearch starts indexing at 1. |
| 1695 | let line_number = lines[0].line_index + 1; |
| 1696 | let google3_prefix = { |
| 1697 | // If rustc_span::FileName isn't a 'real' file, then it's surrounded by by angle |
Lukasz Anforowicz | 1f23391 | 2023-02-14 08:50:26 -0800 | [diff] [blame] | 1698 | // brackets, thus don't prepend "google3/" prefix. |
| 1699 | if file.name.is_real() { "google3/" } else { "" } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1700 | }; |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 1701 | format!("{google3_prefix}{file_name};l={line_number}") |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1702 | } |
| 1703 | |
| 1704 | /// Formats the doc comment (if any) associated with the item identified by |
| 1705 | /// `local_def_id`, and appends the source location at which the item is |
| 1706 | /// defined. |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 1707 | fn format_doc_comment(tcx: TyCtxt, local_def_id: LocalDefId) -> TokenStream { |
| 1708 | let hir_id = tcx.local_def_id_to_hir_id(local_def_id); |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1709 | let doc_comment = tcx |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 1710 | .hir() |
| 1711 | .attrs(hir_id) |
| 1712 | .iter() |
Lukasz Anforowicz | df363ee | 2022-12-16 14:56:38 -0800 | [diff] [blame] | 1713 | .filter_map(|attr| attr.doc_str()) |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1714 | .map(|symbol| symbol.to_string()) |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 1715 | .chain(once(format!("Generated from: {}", format_source_location(tcx, local_def_id)))) |
Googler | 34f3d57 | 2022-12-02 00:53:37 -0800 | [diff] [blame] | 1716 | .join("\n\n"); |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 1717 | quote! { __COMMENT__ #doc_comment} |
Googler | fb20427 | 2022-12-02 00:52:05 -0800 | [diff] [blame] | 1718 | } |
| 1719 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1720 | /// 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] | 1721 | /// 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] | 1722 | /// |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1723 | /// 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] | 1724 | fn format_item(input: &Input, def_id: LocalDefId) -> Result<Option<ApiSnippets>> { |
Lukasz Anforowicz | 1382f39 | 2022-12-12 17:13:23 -0800 | [diff] [blame] | 1725 | // 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] | 1726 | // `is_directly_public` below into `is_exported`. (OTOH such change *alone* is |
| 1727 | // undesirable, because it would mean exposing items from a private module. |
| 1728 | // Exposing a private module is undesirable, because it would mean that |
| 1729 | // changes of private implementation details of the crate could become |
| 1730 | // breaking changes for users of the generated C++ bindings.) |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1731 | if !input.tcx.effective_visibilities(()).is_directly_public(def_id) { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1732 | return Ok(None); |
Lukasz Anforowicz | 8b68a55 | 2022-12-12 15:07:58 -0800 | [diff] [blame] | 1733 | } |
| 1734 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1735 | match input.tcx.hir().expect_item(def_id) { |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 1736 | Item { kind: ItemKind::Struct(_, generics) | |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 1737 | ItemKind::Enum(_, generics) | |
| 1738 | ItemKind::Union(_, generics), |
| 1739 | .. } if !generics.params.is_empty() => { |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 1740 | bail!("Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 1741 | }, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1742 | Item { kind: ItemKind::Fn(..), .. } => format_fn(input, def_id).map(Some), |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 1743 | Item { kind: ItemKind::Struct(..) | ItemKind::Enum(..) | ItemKind::Union(..), .. } => |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 1744 | format_adt_core(input.tcx, def_id.to_def_id()) |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1745 | .map(|core| Some(format_adt(input, &core))), |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 1746 | Item { kind: ItemKind::Impl(_), .. } | // Handled by `format_adt` |
| 1747 | Item { kind: ItemKind::Mod(_), .. } => // Handled by `format_crate` |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1748 | Ok(None), |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1749 | Item { kind, .. } => bail!("Unsupported rustc_hir::hir::ItemKind: {}", kind.descr()), |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1750 | } |
| 1751 | } |
| 1752 | |
| 1753 | /// Formats a C++ comment explaining why no bindings have been generated for |
| 1754 | /// `local_def_id`. |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 1755 | fn format_unsupported_def( |
| 1756 | tcx: TyCtxt, |
| 1757 | local_def_id: LocalDefId, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1758 | err: anyhow::Error, |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1759 | ) -> ApiSnippets { |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 1760 | let source_loc = format_source_location(tcx, local_def_id); |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 1761 | let name = tcx.def_path_str(local_def_id.to_def_id()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1762 | |
| 1763 | // https://docs.rs/anyhow/latest/anyhow/struct.Error.html#display-representations |
| 1764 | // says: To print causes as well [...], use the alternate selector “{:#}”. |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 1765 | 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] | 1766 | let main_api = CcSnippet::new(quote! { __NEWLINE__ __NEWLINE__ __COMMENT__ #msg __NEWLINE__ }); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1767 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1768 | ApiSnippets { main_api, cc_details: CcSnippet::default(), rs_details: quote! {} } |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 1769 | } |
| 1770 | |
Lukasz Anforowicz | 2ec1312 | 2022-11-10 12:39:04 -0800 | [diff] [blame] | 1771 | /// Formats all public items from the Rust crate being compiled. |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 1772 | fn format_crate(input: &Input) -> Result<Output> { |
| 1773 | let tcx = input.tcx; |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1774 | let mut cc_details_prereqs = CcPrerequisites::default(); |
| 1775 | let mut cc_details: Vec<(LocalDefId, TokenStream)> = vec![]; |
| 1776 | let mut rs_body = TokenStream::default(); |
Lukasz Anforowicz | 991b2a5 | 2023-03-23 07:28:41 -0700 | [diff] [blame] | 1777 | let mut main_apis = HashMap::<LocalDefId, CcSnippet>::new(); |
| 1778 | let formatted_items = tcx |
Lukasz Anforowicz | 2ec1312 | 2022-11-10 12:39:04 -0800 | [diff] [blame] | 1779 | .hir() |
| 1780 | .items() |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1781 | .filter_map(|item_id| { |
Lukasz Anforowicz | 2ec1312 | 2022-11-10 12:39:04 -0800 | [diff] [blame] | 1782 | let def_id: LocalDefId = item_id.owner_id.def_id; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1783 | format_item(input, def_id) |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1784 | .unwrap_or_else(|err| Some(format_unsupported_def(tcx, def_id, err))) |
| 1785 | .map(|api_snippets| (def_id, api_snippets)) |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 1786 | }) |
Lukasz Anforowicz | 991b2a5 | 2023-03-23 07:28:41 -0700 | [diff] [blame] | 1787 | .sorted_by_key(|(def_id, _)| tcx.def_span(*def_id)); |
| 1788 | for (def_id, api_snippets) in formatted_items { |
| 1789 | let old_item = main_apis.insert(def_id, api_snippets.main_api); |
| 1790 | assert!(old_item.is_none(), "Duplicated key: {def_id:?}"); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1791 | |
Lukasz Anforowicz | 991b2a5 | 2023-03-23 07:28:41 -0700 | [diff] [blame] | 1792 | // `cc_details` don't participate in the toposort, because |
| 1793 | // `CcPrerequisites::defs` always use `main_api` as the predecessor |
| 1794 | // - `chain`ing `cc_details` after `ordered_main_apis` trivially |
| 1795 | // meets the prerequisites. |
| 1796 | cc_details.push((def_id, api_snippets.cc_details.into_tokens(&mut cc_details_prereqs))); |
| 1797 | rs_body.extend(api_snippets.rs_details); |
| 1798 | } |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 1799 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1800 | // Find the order of `main_apis` that 1) meets the requirements of |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 1801 | // `CcPrerequisites::defs` and 2) makes a best effort attempt to keep the |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1802 | // `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] | 1803 | let ordered_ids = { |
| 1804 | let toposort::TopoSortResult { ordered: ordered_ids, failed: failed_ids } = { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1805 | let nodes = main_apis.keys().copied(); |
| 1806 | let deps = main_apis.iter().flat_map(|(&successor, main_api)| { |
| 1807 | let predecessors = main_api.prereqs.defs.iter().map(|&def_id| def_id); |
Lukasz Anforowicz | 2ecb27f | 2023-01-12 15:29:51 -0800 | [diff] [blame] | 1808 | predecessors.map(move |predecessor| toposort::Dependency { predecessor, successor }) |
| 1809 | }); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1810 | toposort::toposort(nodes, deps, move |lhs_id, rhs_id| { |
| 1811 | tcx.def_span(*lhs_id).cmp(&tcx.def_span(*rhs_id)) |
| 1812 | }) |
Lukasz Anforowicz | 2ecb27f | 2023-01-12 15:29:51 -0800 | [diff] [blame] | 1813 | }; |
| 1814 | assert_eq!( |
| 1815 | 0, |
| 1816 | failed_ids.len(), |
| 1817 | "There are no known scenarios where CcPrerequisites::defs can form \ |
| 1818 | a dependency cycle. These `LocalDefId`s form an unexpected cycle: {}", |
| 1819 | failed_ids.into_iter().map(|id| format!("{:?}", id)).join(",") |
| 1820 | ); |
| 1821 | ordered_ids |
Lukasz Anforowicz | ab563af | 2022-12-15 08:09:50 -0800 | [diff] [blame] | 1822 | }; |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 1823 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1824 | // Destructure/rebuild `main_apis` (in the same order as `ordered_ids`) into |
| 1825 | // `includes`, and `ordered_cc` (mixing in `fwd_decls` and `cc_details`). |
| 1826 | let (includes, ordered_cc) = { |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1827 | let mut already_declared = HashSet::new(); |
| 1828 | let mut fwd_decls = HashSet::new(); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1829 | let mut includes = cc_details_prereqs.includes; |
| 1830 | let mut ordered_main_apis: Vec<(LocalDefId, TokenStream)> = Vec::new(); |
| 1831 | for def_id in ordered_ids.into_iter() { |
| 1832 | let CcSnippet { |
| 1833 | tokens: cc_tokens, |
| 1834 | prereqs: CcPrerequisites { |
| 1835 | includes: mut inner_includes, |
| 1836 | fwd_decls: inner_fwd_decls, |
| 1837 | .. // `defs` have already been utilized by `toposort` above |
Lukasz Anforowicz | 7f31f80 | 2022-12-16 08:24:13 -0800 | [diff] [blame] | 1838 | } |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1839 | } = main_apis.remove(&def_id).unwrap(); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1840 | |
| 1841 | fwd_decls.extend(inner_fwd_decls.difference(&already_declared).copied()); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1842 | already_declared.insert(def_id); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1843 | already_declared.extend(inner_fwd_decls.into_iter()); |
| 1844 | |
| 1845 | includes.append(&mut inner_includes); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1846 | ordered_main_apis.push((def_id, cc_tokens)); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1847 | } |
| 1848 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1849 | let fwd_decls = fwd_decls |
| 1850 | .into_iter() |
| 1851 | .sorted_by_key(|def_id| tcx.def_span(*def_id)) |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 1852 | .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] | 1853 | |
| 1854 | let ordered_cc: Vec<(NamespaceQualifier, TokenStream)> = fwd_decls |
| 1855 | .into_iter() |
| 1856 | .chain(ordered_main_apis.into_iter()) |
| 1857 | .chain(cc_details.into_iter()) |
| 1858 | .map(|(local_def_id, tokens)| { |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1859 | 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] | 1860 | (mod_path, tokens) |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1861 | }) |
| 1862 | .collect_vec(); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1863 | |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 1864 | (includes, ordered_cc) |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 1865 | }; |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 1866 | |
| 1867 | // Generate top-level elements of the C++ header file. |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1868 | let h_body = { |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1869 | // TODO(b/254690602): Decide whether using `#crate_name` as the name of the |
| 1870 | // top-level namespace is okay (e.g. investigate if this name is globally |
| 1871 | // unique + ergonomic). |
| 1872 | let crate_name = format_cc_ident(tcx.crate_name(LOCAL_CRATE).as_str())?; |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1873 | |
Lukasz Anforowicz | 54efc16 | 2022-12-16 15:58:44 -0800 | [diff] [blame] | 1874 | let includes = format_cc_includes(&includes); |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 1875 | let ordered_cc = format_namespace_bound_cc_tokens(ordered_cc); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1876 | quote! { |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1877 | #includes |
| 1878 | __NEWLINE__ __NEWLINE__ |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1879 | namespace #crate_name { |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1880 | __NEWLINE__ |
Lukasz Anforowicz | 54efc16 | 2022-12-16 15:58:44 -0800 | [diff] [blame] | 1881 | #ordered_cc |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1882 | __NEWLINE__ |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1883 | } |
Lukasz Anforowicz | a0502fb | 2023-02-13 15:33:18 -0800 | [diff] [blame] | 1884 | __NEWLINE__ |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1885 | } |
| 1886 | }; |
Lukasz Anforowicz | 7f31f80 | 2022-12-16 08:24:13 -0800 | [diff] [blame] | 1887 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 1888 | Ok(Output { h_body, rs_body }) |
Lukasz Anforowicz | ed1c4f0 | 2022-09-29 11:11:20 -0700 | [diff] [blame] | 1889 | } |
| 1890 | |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 1891 | #[cfg(test)] |
Lukasz Anforowicz | f018e4d | 2022-09-28 07:35:59 -0700 | [diff] [blame] | 1892 | pub mod tests { |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 1893 | use super::*; |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 1894 | |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1895 | use anyhow::Result; |
| 1896 | use itertools::Itertools; |
| 1897 | use proc_macro2::TokenStream; |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1898 | use quote::quote; |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 1899 | use rustc_middle::ty::{Ty, TyCtxt}; |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1900 | use rustc_span::def_id::LocalDefId; |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 1901 | |
Lukasz Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 1902 | use crate::run_compiler::tests::run_compiler_for_testing; |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 1903 | use code_gen_utils::format_cc_includes; |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 1904 | use token_stream_matchers::{ |
| 1905 | assert_cc_matches, assert_cc_not_matches, assert_rs_matches, assert_rs_not_matches, |
| 1906 | }; |
Lukasz Anforowicz | 2b38d27 | 2022-09-23 08:08:18 -0700 | [diff] [blame] | 1907 | |
Lukasz Anforowicz | 5bddf18 | 2022-09-30 16:06:59 -0700 | [diff] [blame] | 1908 | #[test] |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1909 | #[should_panic(expected = "No items named `missing_name`.\n\ |
| 1910 | Instead found:\n`bar`,\n`foo`,\n`m1`,\n`m2`,\n`std`")] |
| 1911 | fn test_find_def_id_by_name_panic_when_no_item_with_matching_name() { |
| 1912 | let test_src = r#" |
| 1913 | pub extern "C" fn foo() {} |
| 1914 | |
| 1915 | pub mod m1 { |
| 1916 | pub fn bar() {} |
| 1917 | } |
| 1918 | pub mod m2 { |
| 1919 | pub fn bar() {} |
| 1920 | } |
| 1921 | "#; |
Lukasz Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 1922 | 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] | 1923 | } |
| 1924 | |
| 1925 | #[test] |
| 1926 | #[should_panic(expected = "More than one item named `some_name`")] |
| 1927 | fn test_find_def_id_by_name_panic_when_multiple_items_with_matching_name() { |
| 1928 | let test_src = r#" |
| 1929 | pub mod m1 { |
| 1930 | pub fn some_name() {} |
| 1931 | } |
| 1932 | pub mod m2 { |
| 1933 | pub fn some_name() {} |
| 1934 | } |
| 1935 | "#; |
Lukasz Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 1936 | 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] | 1937 | } |
| 1938 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1939 | /// This test covers only a single example of a function that should get a |
| 1940 | /// C++ binding. The test focuses on verification that the output from |
| 1941 | /// `format_fn` gets propagated all the way to `GenerateBindings::new`. |
| 1942 | /// Additional coverage of how functions are formatted is provided |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1943 | /// by `test_format_item_..._fn_...` tests (which work at the `format_fn` |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1944 | /// level). |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 1945 | #[test] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 1946 | fn test_generated_bindings_fn_no_mangle_extern_c() { |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 1947 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 1948 | #[no_mangle] |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1949 | pub extern "C" fn public_function() { |
| 1950 | println!("foo"); |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 1951 | } |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 1952 | "#; |
| 1953 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1954 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1955 | assert_cc_matches!( |
| 1956 | bindings.h_body, |
| 1957 | quote! { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 1958 | extern "C" void public_function(); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1959 | } |
Lukasz Anforowicz | d9ff4ab | 2022-09-23 08:11:18 -0700 | [diff] [blame] | 1960 | ); |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1961 | |
| 1962 | // No Rust thunks should be generated in this test scenario. |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 1963 | assert_rs_not_matches!(bindings.rs_body, quote! { public_function }); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1964 | }); |
| 1965 | } |
| 1966 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1967 | /// `test_generated_bindings_fn_export_name` covers a scenario where |
| 1968 | /// `MixedSnippet::cc` is present but `MixedSnippet::rs` is empty |
| 1969 | /// (because no Rust thunks are needed). |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 1970 | #[test] |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1971 | fn test_generated_bindings_fn_export_name() { |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1972 | let test_src = r#" |
| 1973 | #[export_name = "export_name"] |
| 1974 | pub extern "C" fn public_function(x: f64, y: f64) -> f64 { x + y } |
| 1975 | "#; |
| 1976 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1977 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1978 | assert_cc_matches!( |
| 1979 | bindings.h_body, |
| 1980 | quote! { |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1981 | namespace rust_out { |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 1982 | ... |
| 1983 | inline double public_function(double x, double y); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1984 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 1985 | extern "C" double export_name(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1986 | } |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1987 | inline double public_function(double x, double y) { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 1988 | return __crubit_internal::export_name(x, y); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 1989 | } |
| 1990 | } |
| 1991 | } |
| 1992 | ); |
| 1993 | }); |
| 1994 | } |
| 1995 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 1996 | /// The `test_generated_bindings_struct` test covers only a single example |
| 1997 | /// of an ADT (struct/enum/union) that should get a C++ binding. |
| 1998 | /// Additional coverage of how items are formatted is provided by |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 1999 | /// `test_format_item_..._struct_...`, `test_format_item_..._enum_...`, |
| 2000 | /// and `test_format_item_..._union_...` tests. |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2001 | /// |
| 2002 | /// We don't want to duplicate coverage already provided by |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2003 | /// `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] | 2004 | /// * `format_crate` will actually find and process the struct |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2005 | /// (`test_format_item_...` doesn't cover this aspect - it uses a |
| 2006 | /// test-only `find_def_id_by_name` instead) |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2007 | /// * The actual shape of the bindings still looks okay at this level. |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2008 | #[test] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2009 | fn test_generated_bindings_struct() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2010 | let test_src = r#" |
| 2011 | pub struct Point { |
| 2012 | pub x: i32, |
| 2013 | pub y: i32, |
| 2014 | } |
| 2015 | "#; |
| 2016 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2017 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2018 | assert_cc_matches!( |
| 2019 | bindings.h_body, |
| 2020 | quote! { |
| 2021 | namespace rust_out { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2022 | ... |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2023 | struct alignas(4) Point final { |
| 2024 | // No point replicating test coverage of |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2025 | // `test_format_item_struct_with_fields`. |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2026 | ... |
| 2027 | }; |
| 2028 | static_assert(sizeof(Point) == 8, ...); |
| 2029 | static_assert(alignof(Point) == 4, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 2030 | ... // Other static_asserts are covered by |
| 2031 | // `test_format_item_struct_with_fields` |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2032 | } // namespace rust_out |
| 2033 | } |
| 2034 | ); |
| 2035 | assert_rs_matches!( |
| 2036 | bindings.rs_body, |
| 2037 | quote! { |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 2038 | // No point replicating test coverage of |
| 2039 | // `test_format_item_struct_with_fields`. |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 2040 | const _: () = assert!(::std::mem::size_of::<::rust_out::Point>() == 8); |
| 2041 | const _: () = assert!(::std::mem::align_of::<::rust_out::Point>() == 4); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 2042 | const _: () = assert!( memoffset::offset_of!(::rust_out::Point, x) == 0); |
| 2043 | const _: () = assert!( memoffset::offset_of!(::rust_out::Point, y) == 4); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2044 | } |
| 2045 | ); |
| 2046 | }); |
| 2047 | } |
| 2048 | |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2049 | /// The `test_generated_bindings_impl` test covers only a single example of |
| 2050 | /// a non-trait `impl`. Additional coverage of how items are formatted |
| 2051 | /// should be provided in the future by `test_format_item_...` tests. |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2052 | /// |
| 2053 | /// We don't want to duplicate coverage already provided by |
| 2054 | /// `test_format_item_static_method`, but we do want to verify that |
| 2055 | /// * `format_crate` won't process the `impl` as a standalone HIR item |
| 2056 | /// * The actual shape of the bindings still looks okay at this level. |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2057 | #[test] |
| 2058 | fn test_generated_bindings_impl() { |
| 2059 | let test_src = r#" |
| 2060 | pub struct SomeStruct(i32); |
| 2061 | |
| 2062 | impl SomeStruct { |
| 2063 | pub fn public_static_method() -> i32 { 123 } |
| 2064 | |
| 2065 | #[allow(dead_code)] |
| 2066 | fn private_static_method() -> i32 { 123 } |
| 2067 | } |
| 2068 | "#; |
| 2069 | test_generated_bindings(test_src, |bindings| { |
| 2070 | let bindings = bindings.unwrap(); |
| 2071 | assert_cc_matches!( |
| 2072 | bindings.h_body, |
| 2073 | quote! { |
| 2074 | namespace rust_out { |
| 2075 | ... |
| 2076 | struct ... SomeStruct ... { |
| 2077 | // No point replicating test coverage of |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2078 | // `test_format_item_static_method`. |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2079 | ... |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2080 | std::int32_t public_static_method(); |
| 2081 | ... |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2082 | }; |
| 2083 | ... |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2084 | std::int32_t SomeStruct::public_static_method() { |
| 2085 | ... |
| 2086 | } |
Lukasz Anforowicz | b7145e6 | 2023-03-23 09:00:37 -0700 | [diff] [blame] | 2087 | ... |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2088 | } // namespace rust_out |
| 2089 | } |
| 2090 | ); |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 2091 | assert_rs_matches!( |
| 2092 | bindings.rs_body, |
| 2093 | quote! { |
| 2094 | extern "C" fn ...() -> i32 { |
| 2095 | ::rust_out::SomeStruct::public_static_method() |
| 2096 | } |
| 2097 | } |
| 2098 | ); |
Lukasz Anforowicz | 93513ec | 2023-02-10 14:21:20 -0800 | [diff] [blame] | 2099 | }); |
| 2100 | } |
| 2101 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2102 | #[test] |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2103 | fn test_generated_bindings_includes() { |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 2104 | let test_src = r#" |
| 2105 | #[no_mangle] |
| 2106 | pub extern "C" fn public_function(i: i32, d: isize, u: u64) { |
| 2107 | dbg!(i); |
| 2108 | dbg!(d); |
| 2109 | dbg!(u); |
| 2110 | } |
| 2111 | "#; |
| 2112 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2113 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 2114 | assert_cc_matches!( |
| 2115 | bindings.h_body, |
| 2116 | quote! { |
| 2117 | __HASH_TOKEN__ include <cstdint> ... |
| 2118 | namespace ... { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2119 | ... |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 2120 | extern "C" void public_function( |
| 2121 | std::int32_t i, |
| 2122 | std::intptr_t d, |
| 2123 | std::uint64_t u); |
| 2124 | } |
| 2125 | } |
| 2126 | ); |
| 2127 | }); |
| 2128 | } |
| 2129 | |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2130 | /// Tests that `toposort` is used to reorder item bindings. |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 2131 | #[test] |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2132 | fn test_generated_bindings_prereq_defs_field_deps_require_reordering() { |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2133 | let test_src = r#" |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2134 | // In the generated bindings `Outer` needs to come *after* `Inner`. |
| 2135 | pub struct Outer(Inner); |
| 2136 | pub struct Inner(bool); |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2137 | "#; |
| 2138 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2139 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2140 | assert_cc_matches!( |
| 2141 | bindings.h_body, |
| 2142 | quote! { |
| 2143 | namespace rust_out { |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2144 | ... |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2145 | struct alignas(1) Inner final { |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 2146 | ... bool __field0; ... |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2147 | }; |
| 2148 | ... |
| 2149 | struct alignas(1) Outer final { |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 2150 | ... ::rust_out::Inner __field0; ... |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2151 | }; |
| 2152 | ... |
| 2153 | } // namespace rust_out |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2154 | } |
| 2155 | ); |
| 2156 | }); |
| 2157 | } |
| 2158 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2159 | /// Tests that a forward declaration is present when it is required to |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2160 | /// preserve the original source order. In this test the |
| 2161 | /// `CcPrerequisites::fwd_decls` dependency comes from a pointer parameter. |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2162 | #[test] |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2163 | fn test_generated_bindings_prereq_fwd_decls_for_ptr_param() { |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2164 | let test_src = r#" |
| 2165 | // To preserve original API order we need to forward declare S. |
| 2166 | pub fn f(_: *const S) {} |
| 2167 | pub struct S(bool); |
| 2168 | "#; |
| 2169 | test_generated_bindings(test_src, |bindings| { |
| 2170 | let bindings = bindings.unwrap(); |
| 2171 | assert_cc_matches!( |
| 2172 | bindings.h_body, |
| 2173 | quote! { |
| 2174 | namespace rust_out { |
| 2175 | ... |
| 2176 | // Verifing the presence of this forward declaration |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2177 | // it the essence of this test. The order of the items |
| 2178 | // below also matters. |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2179 | struct S; |
| 2180 | ... |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2181 | inline void f(const ::rust_out::S* __param_0); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2182 | ... |
| 2183 | struct alignas(...) S final { ... } |
| 2184 | ... |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2185 | inline void f(const ::rust_out::S* __param_0) { ... } |
| 2186 | ... |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2187 | } // namespace rust_out |
| 2188 | } |
| 2189 | ); |
| 2190 | }); |
| 2191 | } |
| 2192 | |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2193 | /// Tests that a forward declaration is present when it is required to |
| 2194 | /// preserve the original source order. In this test the |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2195 | /// `CcPrerequisites::fwd_decls` dependency comes from a |
| 2196 | /// function declaration that has a parameter that takes a struct by value. |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2197 | #[test] |
| 2198 | fn test_generated_bindings_prereq_fwd_decls_for_cpp_fn_decl() { |
| 2199 | let test_src = r#" |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2200 | #[no_mangle] |
| 2201 | pub extern "C" fn f(s: S) -> bool { s.0 } |
| 2202 | |
| 2203 | #[repr(C)] |
| 2204 | pub struct S(bool); |
| 2205 | "#; |
| 2206 | |
| 2207 | test_generated_bindings(test_src, |bindings| { |
| 2208 | let bindings = bindings.unwrap(); |
| 2209 | assert_cc_matches!( |
| 2210 | bindings.h_body, |
| 2211 | quote! { |
| 2212 | namespace rust_out { |
| 2213 | ... |
| 2214 | // Verifing the presence of this forward declaration |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2215 | // is the essence of this test. The order also matters: |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2216 | // 1. The fwd decl of `S` should come first, |
| 2217 | // 2. Declaration of `f` and definition of `S` should come next |
| 2218 | // (in their original order - `f` first and then `S`). |
| 2219 | struct S; |
| 2220 | ... |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2221 | // `CcPrerequisites` of `f` declaration below (the main api of `f`) should |
| 2222 | // include `S` as a `fwd_decls` edge, rather than as a `defs` edge. |
| 2223 | inline bool f(::rust_out::S s); |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2224 | ... |
| 2225 | struct alignas(...) S final { ... } |
| 2226 | ... |
| 2227 | } // namespace rust_out |
| 2228 | } |
| 2229 | ); |
| 2230 | }); |
| 2231 | } |
| 2232 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2233 | /// This test verifies that a forward declaration for a given ADT is only |
| 2234 | /// emitted once (and not once for every API item that requires the |
| 2235 | /// forward declaration as a prerequisite). |
| 2236 | #[test] |
| 2237 | fn test_generated_bindings_prereq_fwd_decls_no_duplication() { |
| 2238 | let test_src = r#" |
| 2239 | // All three functions below require a forward declaration of S. |
| 2240 | pub fn f1(_: *const S) {} |
| 2241 | pub fn f2(_: *const S) {} |
| 2242 | pub fn f3(_: *const S) {} |
| 2243 | |
| 2244 | pub struct S(bool); |
| 2245 | |
| 2246 | // This function also includes S in its CcPrerequisites::fwd_decls |
| 2247 | // (although here it is not required, because the definition of S |
| 2248 | // is already available above). |
| 2249 | pub fn f4(_: *const S) {} |
| 2250 | "#; |
| 2251 | test_generated_bindings(test_src, |bindings| { |
| 2252 | let bindings = bindings.unwrap().h_body.to_string(); |
| 2253 | |
| 2254 | // Only a single forward declaration is expected. |
| 2255 | assert_eq!(1, bindings.matches("struct S ;").count(), "bindings = {bindings}"); |
| 2256 | }); |
| 2257 | } |
| 2258 | |
| 2259 | /// This test verifies that forward declarations are emitted in a |
| 2260 | /// deterministic order. The particular order doesn't matter _that_ |
| 2261 | /// much, but it definitely shouldn't change every time |
| 2262 | /// `cc_bindings_from_rs` is invoked again. The current order preserves |
| 2263 | /// the original source order of the Rust API items. |
| 2264 | #[test] |
| 2265 | fn test_generated_bindings_prereq_fwd_decls_deterministic_order() { |
| 2266 | let test_src = r#" |
| 2267 | // To try to mix things up, the bindings for the functions below |
| 2268 | // will *ask* for forward declarations in a different order: |
| 2269 | // * Different from the order in which the forward declarations |
| 2270 | // are expected to be *emitted* (the original source order). |
| 2271 | // * Different from alphabetical order. |
| 2272 | pub fn f1(_: *const b::S3) {} |
| 2273 | pub fn f2(_: *const a::S2) {} |
| 2274 | pub fn f3(_: *const a::S1) {} |
| 2275 | |
| 2276 | pub mod a { |
| 2277 | pub struct S1(bool); |
| 2278 | pub struct S2(bool); |
| 2279 | } |
| 2280 | |
| 2281 | pub mod b { |
| 2282 | pub struct S3(bool); |
| 2283 | } |
| 2284 | "#; |
| 2285 | test_generated_bindings(test_src, |bindings| { |
| 2286 | let bindings = bindings.unwrap(); |
| 2287 | assert_cc_matches!( |
| 2288 | bindings.h_body, |
| 2289 | quote! { |
| 2290 | namespace rust_out { |
| 2291 | ... |
| 2292 | // Verifying that we get the same order in each test |
| 2293 | // run is the essence of this test. |
| 2294 | namespace a { |
| 2295 | struct S1; |
| 2296 | struct S2; |
| 2297 | } |
| 2298 | namespace b { |
| 2299 | struct S3; |
| 2300 | } |
| 2301 | ... |
| 2302 | inline void f1 ... |
| 2303 | inline void f2 ... |
| 2304 | inline void f3 ... |
| 2305 | |
| 2306 | namespace a { ... |
| 2307 | struct alignas(...) S1 final { ... } ... |
| 2308 | struct alignas(...) S2 final { ... } ... |
| 2309 | } ... |
| 2310 | namespace b { ... |
| 2311 | struct alignas(...) S3 final { ... } ... |
| 2312 | } ... |
| 2313 | } // namespace rust_out |
| 2314 | } |
| 2315 | ); |
| 2316 | }); |
| 2317 | } |
| 2318 | |
| 2319 | /// This test verifies that forward declarations are not emitted if they are |
| 2320 | /// not needed (e.g. if bindings the given `struct` or other ADT have |
| 2321 | /// already been defined earlier). In particular, we don't want to emit |
| 2322 | /// forward declarations for *all* `structs` (regardless if they are |
| 2323 | /// needed or not). |
| 2324 | #[test] |
Lukasz Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 2325 | 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] | 2326 | let test_src = r#" |
| 2327 | pub struct S(bool); |
| 2328 | |
| 2329 | // S is already defined above - no need for forward declaration in C++. |
| 2330 | pub fn f(_s: *const S) {} |
| 2331 | "#; |
| 2332 | test_generated_bindings(test_src, |bindings| { |
| 2333 | let bindings = bindings.unwrap(); |
| 2334 | assert_cc_not_matches!(bindings.h_body, quote! { struct S; }); |
Lukasz Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 2335 | assert_cc_matches!(bindings.h_body, quote! { void f(const ::rust_out::S* _s); }); |
| 2336 | }); |
| 2337 | } |
| 2338 | |
| 2339 | /// This test verifies that a method declaration doesn't ask for a forward |
| 2340 | /// declaration to the struct. |
| 2341 | #[test] |
| 2342 | fn test_generated_bindings_prereq_fwd_decls_not_needed_inside_struct_definition() { |
| 2343 | let test_src = r#" |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2344 | #![allow(dead_code)] |
| 2345 | |
| 2346 | pub struct S { |
| 2347 | // This shouldn't require a fwd decl of S. |
| 2348 | field: *const S, |
| 2349 | } |
Lukasz Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 2350 | |
| 2351 | impl S { |
| 2352 | // This shouldn't require a fwd decl of S. |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2353 | pub fn create() -> S { Self{ field: std::ptr::null() } } |
Lukasz Anforowicz | 64b04ba | 2023-02-10 17:19:05 -0800 | [diff] [blame] | 2354 | } |
| 2355 | "#; |
| 2356 | test_generated_bindings(test_src, |bindings| { |
| 2357 | let bindings = bindings.unwrap(); |
| 2358 | assert_cc_not_matches!(bindings.h_body, quote! { struct S; }); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2359 | assert_cc_matches!( |
| 2360 | bindings.h_body, |
| 2361 | quote! { |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 2362 | static inline ::rust_out::S create(); ... |
| 2363 | const ::rust_out::S* field; ... |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 2364 | } |
| 2365 | ); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 2366 | }); |
| 2367 | } |
| 2368 | |
Lukasz Anforowicz | 816cbaa | 2022-12-07 09:31:30 -0800 | [diff] [blame] | 2369 | #[test] |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2370 | fn test_generated_bindings_module_basics() { |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2371 | let test_src = r#" |
| 2372 | pub mod some_module { |
| 2373 | pub fn some_func() {} |
| 2374 | } |
| 2375 | "#; |
| 2376 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2377 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2378 | assert_cc_matches!( |
| 2379 | bindings.h_body, |
| 2380 | quote! { |
| 2381 | namespace rust_out { |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2382 | namespace some_module { |
| 2383 | ... |
| 2384 | inline void some_func() { ... } |
| 2385 | ... |
| 2386 | } // namespace some_module |
| 2387 | } // namespace rust_out |
| 2388 | } |
| 2389 | ); |
| 2390 | assert_rs_matches!( |
| 2391 | bindings.rs_body, |
| 2392 | quote! { |
| 2393 | #[no_mangle] |
| 2394 | extern "C" |
| 2395 | fn ...() -> () { |
| 2396 | ::rust_out::some_module::some_func() |
| 2397 | } |
| 2398 | } |
| 2399 | ); |
| 2400 | }); |
| 2401 | } |
| 2402 | |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2403 | #[test] |
| 2404 | fn test_generated_bindings_module_name_is_cpp_reserved_keyword() { |
| 2405 | let test_src = r#" |
| 2406 | pub mod working_module { |
| 2407 | pub fn working_module_f1() {} |
| 2408 | pub fn working_module_f2() {} |
| 2409 | } |
| 2410 | pub mod reinterpret_cast { |
| 2411 | pub fn broken_module_f1() {} |
| 2412 | pub fn broken_module_f2() {} |
| 2413 | } |
| 2414 | "#; |
| 2415 | test_generated_bindings(test_src, |bindings| { |
| 2416 | let bindings = bindings.unwrap(); |
| 2417 | |
| 2418 | // Items in the broken module should be replaced with a comment explaining the |
| 2419 | // problem. |
| 2420 | let broken_module_msg = "Failed to format namespace name `reinterpret_cast`: \ |
| 2421 | `reinterpret_cast` is a C++ reserved keyword \ |
| 2422 | and can't be used as a C++ identifier"; |
| 2423 | assert_cc_not_matches!(bindings.h_body, quote! { namespace reinterpret_cast }); |
| 2424 | assert_cc_not_matches!(bindings.h_body, quote! { broken_module_f1 }); |
| 2425 | assert_cc_not_matches!(bindings.h_body, quote! { broken_module_f2 }); |
| 2426 | |
| 2427 | // Items in the other module should still go through. |
| 2428 | assert_cc_matches!( |
| 2429 | bindings.h_body, |
| 2430 | quote! { |
| 2431 | namespace rust_out { |
| 2432 | namespace working_module { |
| 2433 | ... |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2434 | inline void working_module_f1(); |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2435 | ... |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2436 | inline void working_module_f2(); |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2437 | ... |
| 2438 | } // namespace some_module |
| 2439 | |
| 2440 | __COMMENT__ #broken_module_msg |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2441 | ... |
Lukasz Anforowicz | e1da537 | 2023-01-03 12:31:14 -0800 | [diff] [blame] | 2442 | } // namespace rust_out |
| 2443 | } |
| 2444 | ); |
| 2445 | }); |
| 2446 | } |
| 2447 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2448 | /// `test_generated_bindings_non_pub_items` verifies that non-public items |
| 2449 | /// are not present/propagated into the generated bindings. |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2450 | #[test] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2451 | fn test_generated_bindings_non_pub_items() { |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2452 | let test_src = r#" |
Lukasz Anforowicz | 88bde9b | 2022-10-14 14:48:07 -0700 | [diff] [blame] | 2453 | #![allow(dead_code)] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2454 | |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2455 | extern "C" fn private_function() { |
| 2456 | println!("foo"); |
| 2457 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2458 | |
| 2459 | struct PrivateStruct { |
| 2460 | x: i32, |
| 2461 | y: i32, |
| 2462 | } |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2463 | |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 2464 | pub struct PublicStruct(i32); |
| 2465 | |
| 2466 | impl PublicStruct { |
| 2467 | fn private_method() {} |
| 2468 | } |
| 2469 | |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2470 | pub mod public_module { |
| 2471 | fn priv_func_in_pub_module() {} |
| 2472 | } |
| 2473 | |
| 2474 | mod private_module { |
| 2475 | pub fn pub_func_in_priv_module() { priv_func_in_priv_module() } |
| 2476 | fn priv_func_in_priv_module() {} |
| 2477 | } |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2478 | "#; |
| 2479 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2480 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2481 | assert_cc_not_matches!(bindings.h_body, quote! { private_function }); |
Lukasz Anforowicz | e7a2500 | 2022-11-10 06:21:42 -0800 | [diff] [blame] | 2482 | assert_rs_not_matches!(bindings.rs_body, quote! { private_function }); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2483 | assert_cc_not_matches!(bindings.h_body, quote! { PrivateStruct }); |
| 2484 | assert_rs_not_matches!(bindings.rs_body, quote! { PrivateStruct }); |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 2485 | assert_cc_not_matches!(bindings.h_body, quote! { private_method }); |
| 2486 | assert_rs_not_matches!(bindings.rs_body, quote! { private_method }); |
Lukasz Anforowicz | a577d82 | 2022-12-12 15:00:46 -0800 | [diff] [blame] | 2487 | assert_cc_not_matches!(bindings.h_body, quote! { priv_func_in_priv_module }); |
| 2488 | assert_rs_not_matches!(bindings.rs_body, quote! { priv_func_in_priv_module }); |
| 2489 | assert_cc_not_matches!(bindings.h_body, quote! { priv_func_in_pub_module }); |
| 2490 | assert_rs_not_matches!(bindings.rs_body, quote! { priv_func_in_pub_module }); |
Lukasz Anforowicz | 1382f39 | 2022-12-12 17:13:23 -0800 | [diff] [blame] | 2491 | assert_cc_not_matches!(bindings.h_body, quote! { private_module }); |
| 2492 | assert_rs_not_matches!(bindings.rs_body, quote! { private_module }); |
| 2493 | assert_cc_not_matches!(bindings.h_body, quote! { pub_func_in_priv_module }); |
| 2494 | assert_rs_not_matches!(bindings.rs_body, quote! { pub_func_in_priv_module }); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2495 | }); |
| 2496 | } |
| 2497 | |
| 2498 | #[test] |
Lukasz Anforowicz | 4aa1ff9 | 2022-10-10 11:22:22 -0700 | [diff] [blame] | 2499 | fn test_generated_bindings_top_level_items() { |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2500 | let test_src = "pub fn public_function() {}"; |
| 2501 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2502 | let bindings = bindings.unwrap(); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 2503 | 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] | 2504 | rust_out"; |
| 2505 | assert_cc_matches!( |
| 2506 | bindings.h_body, |
| 2507 | quote! { |
| 2508 | __COMMENT__ #expected_comment_txt |
Lukasz Anforowicz | 4aa1ff9 | 2022-10-10 11:22:22 -0700 | [diff] [blame] | 2509 | ... |
Lukasz Anforowicz | 31b29cd | 2022-10-10 11:33:41 -0700 | [diff] [blame] | 2510 | __HASH_TOKEN__ pragma once |
| 2511 | ... |
Lukasz Anforowicz | 4aa1ff9 | 2022-10-10 11:22:22 -0700 | [diff] [blame] | 2512 | namespace rust_out { |
| 2513 | ... |
| 2514 | } |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2515 | } |
| 2516 | ); |
Lukasz Anforowicz | e7a2500 | 2022-11-10 06:21:42 -0800 | [diff] [blame] | 2517 | assert_cc_matches!( |
| 2518 | bindings.rs_body, |
| 2519 | quote! { |
| 2520 | __COMMENT__ #expected_comment_txt |
| 2521 | } |
| 2522 | ); |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2523 | }) |
| 2524 | } |
| 2525 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2526 | /// The `test_generated_bindings_unsupported_item` test verifies how `Err` |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2527 | /// from `format_item` is formatted as a C++ comment (in `format_crate` |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2528 | /// and `format_unsupported_def`): |
| 2529 | /// - This test covers only a single example of an unsupported item. |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2530 | /// Additional coverage is provided by `test_format_item_unsupported_...` |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2531 | /// tests. |
| 2532 | /// - This test somewhat arbitrarily chooses an example of an unsupported |
| 2533 | /// item, trying to pick one that 1) will never be supported (b/254104998 |
| 2534 | /// has some extra notes about APIs named after reserved C++ keywords) and |
| 2535 | /// 2) tests that the full error chain is included in the message. |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2536 | #[test] |
| 2537 | fn test_generated_bindings_unsupported_item() { |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2538 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2539 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2540 | pub extern "C" fn reinterpret_cast() {} |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2541 | "#; |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2542 | test_generated_bindings(test_src, |bindings| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2543 | let bindings = bindings.unwrap(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2544 | let expected_comment_txt = "Error generating bindings for `reinterpret_cast` \ |
Googler | 785e6b4 | 2023-01-23 12:11:36 -0800 | [diff] [blame] | 2545 | defined at <crubit_unittests.rs>;l=3: \ |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2546 | Error formatting function name: \ |
| 2547 | `reinterpret_cast` is a C++ reserved keyword \ |
| 2548 | and can't be used as a C++ identifier"; |
Lukasz Anforowicz | 2f2510a | 2022-09-29 13:06:44 -0700 | [diff] [blame] | 2549 | assert_cc_matches!( |
| 2550 | bindings.h_body, |
| 2551 | quote! { |
| 2552 | __COMMENT__ #expected_comment_txt |
| 2553 | } |
| 2554 | ); |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 2555 | }) |
| 2556 | } |
| 2557 | |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 2558 | #[test] |
Lukasz Anforowicz | ce17f3f | 2023-02-27 11:32:14 -0800 | [diff] [blame] | 2559 | fn test_generated_bindings_reimports() { |
| 2560 | let test_src = r#" |
| 2561 | #![allow(dead_code)] |
| 2562 | #![allow(unused_imports)] |
| 2563 | mod private_submodule1 { |
| 2564 | pub fn subfunction1() {} |
| 2565 | pub fn subfunction2() {} |
| 2566 | pub fn subfunction3() {} |
| 2567 | } |
| 2568 | mod private_submodule2 { |
| 2569 | pub fn subfunction8() {} |
| 2570 | pub fn subfunction9() {} |
| 2571 | } |
| 2572 | |
| 2573 | // Public re-import. |
| 2574 | pub use private_submodule1::subfunction1; |
| 2575 | |
| 2576 | // Private re-import. |
| 2577 | use private_submodule1::subfunction2; |
| 2578 | |
| 2579 | // Re-import that renames. |
| 2580 | pub use private_submodule1::subfunction3 as public_function3; |
| 2581 | |
| 2582 | // Re-import of multiple items via glob. |
| 2583 | pub use private_submodule2::*; |
| 2584 | "#; |
| 2585 | test_generated_bindings(test_src, |bindings| { |
| 2586 | let bindings = bindings.unwrap(); |
| 2587 | |
| 2588 | let failures = vec![(1, 15), (3, 21), (4, 24)]; |
| 2589 | for (use_number, line_number) in failures.into_iter() { |
| 2590 | let expected_comment_txt = format!( |
| 2591 | "Error generating bindings for `{{use#{use_number}}}` defined at \ |
| 2592 | <crubit_unittests.rs>;l={line_number}: \ |
| 2593 | Unsupported rustc_hir::hir::ItemKind: `use` import" |
| 2594 | ); |
| 2595 | assert_cc_matches!( |
| 2596 | bindings.h_body, |
| 2597 | quote! { |
| 2598 | __COMMENT__ #expected_comment_txt |
| 2599 | } |
| 2600 | ); |
| 2601 | } |
| 2602 | }); |
| 2603 | } |
| 2604 | |
| 2605 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2606 | 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] | 2607 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2608 | #[no_mangle] |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 2609 | pub extern "C" fn public_function() {} |
| 2610 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2611 | test_format_item(test_src, "public_function", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2612 | let result = result.unwrap().unwrap(); |
| 2613 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2614 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2615 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2616 | main_api.tokens, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2617 | quote! { |
| 2618 | extern "C" void public_function(); |
| 2619 | } |
| 2620 | ); |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2621 | |
| 2622 | // Sufficient to just re-declare the Rust API in C++. |
| 2623 | // (i.e. there is no need to have a C++-side definition of `public_function`). |
| 2624 | assert!(result.cc_details.tokens.is_empty()); |
| 2625 | |
| 2626 | // There is no need to have a separate thunk for an `extern "C"` function. |
| 2627 | assert!(result.rs_details.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2628 | }); |
| 2629 | } |
| 2630 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2631 | /// 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] | 2632 | /// similar to the |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2633 | /// `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] | 2634 | /// except that the return type is explicitly spelled out. There is no |
| 2635 | /// difference in `ty::FnSig` so our code behaves exactly the same, but the |
| 2636 | /// test has been planned based on earlier, hir-focused approach and having |
| 2637 | /// this extra test coverage shouldn't hurt. (`hir::FnSig` |
| 2638 | /// and `hir::FnRetTy` _would_ see a difference between the two tests, even |
| 2639 | /// though there is no different in the current `bindings.rs` code). |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2640 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2641 | fn test_format_item_fn_explicit_unit_return_type() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2642 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2643 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2644 | pub extern "C" fn explicit_unit_return_type() -> () {} |
| 2645 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2646 | test_format_item(test_src, "explicit_unit_return_type", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2647 | let result = result.unwrap().unwrap(); |
| 2648 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2649 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2650 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2651 | main_api.tokens, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2652 | quote! { |
| 2653 | extern "C" void explicit_unit_return_type(); |
| 2654 | } |
| 2655 | ); |
| 2656 | }); |
| 2657 | } |
| 2658 | |
| 2659 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2660 | fn test_format_item_fn_never_return_type() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2661 | let test_src = r#" |
| 2662 | #[no_mangle] |
| 2663 | pub extern "C" fn never_returning_function() -> ! { |
| 2664 | panic!("This function panics and therefore never returns"); |
| 2665 | } |
| 2666 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2667 | test_format_item(test_src, "never_returning_function", |result| { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2668 | // TODO(b/254507801): The function should be annotated with the `[[noreturn]]` |
| 2669 | // attribute. |
| 2670 | // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more |
| 2671 | // details). |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2672 | let result = result.unwrap().unwrap(); |
| 2673 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2674 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2675 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2676 | main_api.tokens, |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2677 | quote! { |
| 2678 | extern "C" void never_returning_function(); |
| 2679 | } |
| 2680 | ); |
| 2681 | }) |
| 2682 | } |
| 2683 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2684 | /// `test_format_item_fn_mangling` checks that bindings can be generated for |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2685 | /// `extern "C"` functions that do *not* have `#[no_mangle]` attribute. The |
| 2686 | /// test elides away the mangled name in the `assert_cc_matches` checks |
| 2687 | /// below, but end-to-end test coverage should eventually be provided by |
| 2688 | /// `test/functions` (see b/262904507). |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 2689 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2690 | fn test_format_item_fn_mangling() { |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2691 | let test_src = r#" |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2692 | 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] | 2693 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2694 | test_format_item(test_src, "public_function", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2695 | let result = result.unwrap().unwrap(); |
| 2696 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2697 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2698 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2699 | main_api.tokens, |
| 2700 | quote! { |
| 2701 | inline double public_function(double x, double y); |
| 2702 | } |
| 2703 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2704 | assert!(result.rs_details.is_empty()); |
| 2705 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2706 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2707 | result.cc_details.tokens, |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2708 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2709 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 2710 | extern "C" double ...(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2711 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2712 | ... |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2713 | inline double public_function(double x, double y) { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 2714 | return __crubit_internal::...(x, y); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2715 | } |
| 2716 | } |
| 2717 | ); |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2718 | }); |
| 2719 | } |
| 2720 | |
| 2721 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2722 | fn test_format_item_fn_export_name() { |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2723 | let test_src = r#" |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2724 | #[export_name = "export_name"] |
| 2725 | 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] | 2726 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2727 | test_format_item(test_src, "public_function", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2728 | let result = result.unwrap().unwrap(); |
| 2729 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2730 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2731 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2732 | main_api.tokens, |
| 2733 | quote! { |
| 2734 | inline double public_function(double x, double y); |
| 2735 | } |
| 2736 | ); |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2737 | |
| 2738 | // 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] | 2739 | assert!(result.rs_details.is_empty()); |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2740 | |
| 2741 | // We generate a C++-side definition of `public_function` so that we |
| 2742 | // can call a differently-named (but same-signature) `export_name` function. |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2743 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2744 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2745 | result.cc_details.tokens, |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2746 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2747 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 2748 | extern "C" double export_name(double, double); |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2749 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2750 | ... |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2751 | inline double public_function(double x, double y) { |
| 2752 | return __crubit_internal::export_name(x, y); |
| 2753 | } |
Lukasz Anforowicz | f04bc52 | 2022-11-04 09:19:51 -0700 | [diff] [blame] | 2754 | } |
| 2755 | ); |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2756 | }); |
| 2757 | } |
| 2758 | |
| 2759 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2760 | fn test_format_item_unsupported_fn_unsafe() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2761 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2762 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2763 | pub unsafe extern "C" fn foo() {} |
| 2764 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2765 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 2766 | let err = result.unwrap_err(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2767 | assert_eq!( |
| 2768 | err, |
| 2769 | "Bindings for `unsafe` functions \ |
| 2770 | are not fully designed yet (b/254095482)" |
| 2771 | ); |
| 2772 | }); |
| 2773 | } |
| 2774 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2775 | /// `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] | 2776 | /// generated. |
| 2777 | /// |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2778 | /// Right now the `const` qualifier is ignored, but one can imagine that in |
| 2779 | /// the (very) long-term future such functions (including their bodies) |
| 2780 | /// could be translated into C++ `consteval` functions. |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2781 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2782 | fn test_format_item_fn_const() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2783 | let test_src = r#" |
| 2784 | pub const fn foo(i: i32) -> i32 { i * 42 } |
| 2785 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2786 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2787 | // TODO(b/254095787): Update test expectations below once `const fn` from Rust |
| 2788 | // is translated into a `consteval` C++ function. |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2789 | let result = result.unwrap().unwrap(); |
| 2790 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2791 | assert!(!main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2792 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2793 | main_api.tokens, |
| 2794 | quote! { |
| 2795 | inline std::int32_t foo(std::int32_t i); |
| 2796 | } |
| 2797 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2798 | assert!(!result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2799 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2800 | result.cc_details.tokens, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2801 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2802 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 2803 | extern "C" std::int32_t ...( std::int32_t); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 2804 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2805 | ... |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2806 | inline std::int32_t foo(std::int32_t i) { |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 2807 | return __crubit_internal::...(i); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2808 | } |
| 2809 | } |
| 2810 | ); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2811 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2812 | result.rs_details, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2813 | quote! { |
| 2814 | #[no_mangle] |
| 2815 | extern "C" |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 2816 | fn ...(i: i32) -> i32 { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 2817 | ::rust_out::foo(i) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 2818 | } |
| 2819 | } |
| 2820 | ); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2821 | }); |
| 2822 | } |
| 2823 | |
| 2824 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2825 | fn test_format_item_fn_with_c_unwind_abi() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2826 | // See also https://rust-lang.github.io/rfcs/2945-c-unwind-abi.html |
| 2827 | let test_src = r#" |
| 2828 | #![feature(c_unwind)] |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2829 | |
| 2830 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2831 | pub extern "C-unwind" fn may_throw() {} |
| 2832 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2833 | test_format_item(test_src, "may_throw", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2834 | let result = result.unwrap().unwrap(); |
| 2835 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2836 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2837 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2838 | main_api.tokens, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2839 | quote! { |
| 2840 | extern "C" void may_throw(); |
| 2841 | } |
| 2842 | ); |
| 2843 | }); |
| 2844 | } |
| 2845 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2846 | /// This test mainly verifies that `format_item` correctly propagates |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2847 | /// `CcPrerequisites` of parameter types and return type. |
| 2848 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2849 | fn test_format_item_fn_cc_prerequisites_if_cpp_definition_needed() { |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2850 | let test_src = r#" |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2851 | pub fn foo(_i: i32) -> S { panic!("foo") } |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2852 | pub struct S(i32); |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2853 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2854 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2855 | let result = result.unwrap().unwrap(); |
| 2856 | let main_api = &result.main_api; |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2857 | |
| 2858 | // Minimal coverage, just to double-check that the test setup works. |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2859 | // |
| 2860 | // Note that this is a definition, and therefore `S` should be defined |
| 2861 | // earlier (not just forward declared). |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2862 | 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] | 2863 | 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] | 2864 | |
| 2865 | // Main checks: `CcPrerequisites::includes`. |
| 2866 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2867 | format_cc_includes(&main_api.prereqs.includes), |
| 2868 | quote! { include <cstdint> } |
| 2869 | ); |
| 2870 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2871 | format_cc_includes(&result.cc_details.prereqs.includes), |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2872 | quote! { include <cstdint> } |
| 2873 | ); |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2874 | |
| 2875 | // Main checks: `CcPrerequisites::defs` and `CcPrerequisites::fwd_decls`. |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2876 | // |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2877 | // 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] | 2878 | // expose `tcx` to the verification function (and therefore calling |
| 2879 | // `find_def_id_by_name` is not easily possible). |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 2880 | // |
| 2881 | // Note that `main_api` and `impl_details` have different expectations. |
| 2882 | assert_eq!(0, main_api.prereqs.defs.len()); |
| 2883 | assert_eq!(1, main_api.prereqs.fwd_decls.len()); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2884 | assert_eq!(1, result.cc_details.prereqs.defs.len()); |
| 2885 | assert_eq!(0, result.cc_details.prereqs.fwd_decls.len()); |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2886 | }); |
| 2887 | } |
| 2888 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2889 | /// This test verifies that `format_item` uses `CcPrerequisites::fwd_decls` |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2890 | /// rather than `CcPrerequisites::defs` for function declarations in the |
| 2891 | /// `main_api`. |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2892 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2893 | fn test_format_item_fn_cc_prerequisites_if_only_cpp_declaration_needed() { |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2894 | let test_src = r#" |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2895 | #[no_mangle] |
| 2896 | pub extern "C" fn foo(s: S) -> bool { s.0 } |
| 2897 | |
| 2898 | #[repr(C)] |
| 2899 | pub struct S(bool); |
| 2900 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2901 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2902 | let result = result.unwrap().unwrap(); |
| 2903 | let main_api = &result.main_api; |
Lukasz Anforowicz | 7a7b81f | 2023-01-12 15:50:46 -0800 | [diff] [blame] | 2904 | |
| 2905 | // Minimal coverage, just to double-check that the test setup works. |
| 2906 | // |
| 2907 | // Note that this is only a function *declaration* (not a function definition - |
| 2908 | // there is no function body), and therefore `S` just needs to be |
| 2909 | // forward-declared earlier. |
Lukasz Anforowicz | 10b1a29 | 2023-04-03 16:19:08 -0700 | [diff] [blame] | 2910 | 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] | 2911 | |
| 2912 | // Main checks: `CcPrerequisites::defs` and `CcPrerequisites::fwd_decls`. |
| 2913 | // |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2914 | // 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] | 2915 | // expose `tcx` to the verification function (and therefore calling |
| 2916 | // `find_def_id_by_name` is not easily possible). |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2917 | assert_eq!(0, main_api.prereqs.defs.len()); |
| 2918 | assert_eq!(1, main_api.prereqs.fwd_decls.len()); |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 2919 | }); |
| 2920 | } |
| 2921 | |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2922 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2923 | fn test_format_item_fn_with_type_aliased_return_type() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2924 | // Type aliases disappear at the `rustc_middle::ty::Ty` level and therefore in |
| 2925 | // the short-term the generated bindings also ignore type aliases. |
| 2926 | // |
| 2927 | // TODO(b/254096006): Consider preserving `type` aliases when generating |
| 2928 | // bindings. |
| 2929 | let test_src = r#" |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 2930 | type MyTypeAlias = f64; |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2931 | |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 2932 | #[no_mangle] |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 2933 | pub extern "C" fn type_aliased_return() -> MyTypeAlias { 42.0 } |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2934 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2935 | test_format_item(test_src, "type_aliased_return", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 2936 | let result = result.unwrap().unwrap(); |
| 2937 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2938 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2939 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2940 | main_api.tokens, |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2941 | quote! { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 2942 | extern "C" double type_aliased_return(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 2943 | } |
| 2944 | ); |
| 2945 | }); |
| 2946 | } |
| 2947 | |
| 2948 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2949 | fn test_format_item_fn_with_doc_comment_with_unmangled_name() { |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2950 | let test_src = r#" |
| 2951 | /// Outer line doc. |
| 2952 | /** Outer block doc that spans lines. |
| 2953 | */ |
| 2954 | #[doc = "Doc comment via doc attribute."] |
| 2955 | #[no_mangle] |
| 2956 | pub extern "C" fn fn_with_doc_comment_with_unmangled_name() {} |
| 2957 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2958 | 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] | 2959 | let result = result.unwrap().unwrap(); |
| 2960 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2961 | assert!(main_api.prereqs.is_empty()); |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2962 | let doc_comments = [ |
| 2963 | " Outer line doc.", |
Googler | 34f3d57 | 2022-12-02 00:53:37 -0800 | [diff] [blame] | 2964 | "", |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2965 | " Outer block doc that spans lines.", |
| 2966 | " ", |
Googler | 34f3d57 | 2022-12-02 00:53:37 -0800 | [diff] [blame] | 2967 | "", |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2968 | "Doc comment via doc attribute.", |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2969 | "", |
| 2970 | "Generated from: <crubit_unittests.rs>;l=7", |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2971 | ] |
| 2972 | .join("\n"); |
| 2973 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2974 | main_api.tokens, |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2975 | quote! { |
| 2976 | __COMMENT__ #doc_comments |
| 2977 | extern "C" void fn_with_doc_comment_with_unmangled_name(); |
| 2978 | } |
| 2979 | ); |
| 2980 | }); |
| 2981 | } |
| 2982 | |
| 2983 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2984 | fn test_format_item_fn_with_inner_doc_comment_with_unmangled_name() { |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 2985 | let test_src = r#" |
| 2986 | /// Outer doc comment. |
| 2987 | #[no_mangle] |
| 2988 | pub extern "C" fn fn_with_inner_doc_comment_with_unmangled_name() { |
| 2989 | //! Inner doc comment. |
| 2990 | } |
| 2991 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 2992 | 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] | 2993 | let result = result.unwrap().unwrap(); |
| 2994 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 2995 | assert!(main_api.prereqs.is_empty()); |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 2996 | let doc_comments = [ |
| 2997 | " Outer doc comment.", |
| 2998 | " Inner doc comment.", |
| 2999 | "Generated from: <crubit_unittests.rs>;l=4", |
| 3000 | ] |
| 3001 | .join("\n\n"); |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3002 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3003 | main_api.tokens, |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3004 | quote! { |
| 3005 | __COMMENT__ #doc_comments |
| 3006 | extern "C" void fn_with_inner_doc_comment_with_unmangled_name(); |
| 3007 | } |
| 3008 | ); |
| 3009 | }); |
| 3010 | } |
| 3011 | |
| 3012 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3013 | fn test_format_item_fn_with_doc_comment_with_mangled_name() { |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3014 | let test_src = r#" |
| 3015 | /// Doc comment of a function with mangled name. |
| 3016 | pub extern "C" fn fn_with_doc_comment_with_mangled_name() {} |
| 3017 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3018 | 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] | 3019 | let result = result.unwrap().unwrap(); |
| 3020 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3021 | assert!(main_api.prereqs.is_empty()); |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3022 | let comment = " Doc comment of a function with mangled name.\n\n\ |
| 3023 | Generated from: <crubit_unittests.rs>;l=3"; |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3024 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3025 | main_api.tokens, |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3026 | quote! { |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3027 | __COMMENT__ #comment |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3028 | inline void fn_with_doc_comment_with_mangled_name(); |
Googler | 624580b | 2022-12-01 01:23:42 -0800 | [diff] [blame] | 3029 | } |
| 3030 | ); |
| 3031 | }); |
| 3032 | } |
| 3033 | |
| 3034 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3035 | fn test_format_item_unsupported_fn_name_is_reserved_cpp_keyword() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3036 | let test_src = r#" |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 3037 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3038 | pub extern "C" fn reinterpret_cast() -> () {} |
| 3039 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3040 | test_format_item(test_src, "reinterpret_cast", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3041 | let err = result.unwrap_err(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3042 | assert_eq!( |
| 3043 | err, |
| 3044 | "Error formatting function name: \ |
| 3045 | `reinterpret_cast` is a C++ reserved keyword \ |
| 3046 | and can't be used as a C++ identifier" |
| 3047 | ); |
| 3048 | }); |
| 3049 | } |
| 3050 | |
| 3051 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3052 | fn test_format_item_unsupported_fn_ret_type() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3053 | let test_src = r#" |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 3054 | pub fn foo() -> (i32, i32) { (123, 456) } |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3055 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3056 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3057 | let err = result.unwrap_err(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3058 | assert_eq!( |
| 3059 | err, |
| 3060 | "Error formatting function return type: \ |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 3061 | Tuples are not supported yet: (i32, i32) (b/254099023)" |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3062 | ); |
| 3063 | }); |
| 3064 | } |
| 3065 | |
| 3066 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3067 | fn test_format_item_unsupported_fn_with_late_bound_lifetimes() { |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3068 | // TODO(b/258235219): Expect success after adding support for references. |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3069 | let test_src = r#" |
| 3070 | pub fn foo(arg: &i32) -> &i32 { arg } |
| 3071 | |
| 3072 | // Lifetime inference translates the above into: |
| 3073 | // pub fn foo<'a>(arg: &'a i32) -> &'a i32 { ... } |
| 3074 | // leaving 'a lifetime late-bound (it is bound with a lifetime |
| 3075 | // taken from each of the callsites). In other words, we can't |
| 3076 | // just call `no_bound_vars` on this `FnSig`'s `Binder`. |
| 3077 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3078 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3079 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3080 | assert_eq!(err, "Generic functions are not supported yet (b/259749023)"); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3081 | }); |
| 3082 | } |
| 3083 | |
| 3084 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3085 | fn test_format_item_unsupported_generic_fn() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3086 | let test_src = r#" |
| 3087 | use std::default::Default; |
| 3088 | use std::fmt::Display; |
| 3089 | pub fn generic_function<T: Default + Display>() { |
| 3090 | println!("{}", T::default()); |
| 3091 | } |
| 3092 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3093 | test_format_item(test_src, "generic_function", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3094 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3095 | assert_eq!(err, "Generic functions are not supported yet (b/259749023)"); |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 3096 | }); |
| 3097 | } |
| 3098 | |
| 3099 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3100 | fn test_format_item_unsupported_generic_struct() { |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3101 | let test_src = r#" |
| 3102 | pub struct Point<T> { |
| 3103 | pub x: T, |
| 3104 | pub y: T, |
| 3105 | } |
| 3106 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3107 | test_format_item(test_src, "Point", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3108 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3109 | assert_eq!(err, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3110 | }); |
| 3111 | } |
| 3112 | |
| 3113 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3114 | fn test_format_item_unsupported_generic_enum() { |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3115 | let test_src = r#" |
| 3116 | pub enum Point<T> { |
| 3117 | Cartesian{x: T, y: T}, |
| 3118 | Polar{angle: T, dist: T}, |
| 3119 | } |
| 3120 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3121 | test_format_item(test_src, "Point", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3122 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3123 | assert_eq!(err, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3124 | }); |
| 3125 | } |
| 3126 | |
| 3127 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3128 | fn test_format_item_unsupported_generic_union() { |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3129 | let test_src = r#" |
| 3130 | pub union SomeUnion<T> { |
| 3131 | pub x: std::mem::ManuallyDrop<T>, |
| 3132 | pub y: i32, |
| 3133 | } |
| 3134 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3135 | test_format_item(test_src, "SomeUnion", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3136 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3137 | assert_eq!(err, "Generic types are not supported yet (b/259749095)"); |
Lukasz Anforowicz | 27914f5 | 2022-11-08 10:55:03 -0800 | [diff] [blame] | 3138 | }); |
| 3139 | } |
| 3140 | |
| 3141 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3142 | fn test_format_item_unsupported_fn_async() { |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 3143 | let test_src = r#" |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3144 | pub async fn async_function() {} |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 3145 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3146 | test_format_item(test_src, "async_function", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3147 | let err = result.unwrap_err(); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3148 | assert_eq!( |
| 3149 | err, |
| 3150 | "Error formatting function return type: \ |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3151 | The following Rust type is not supported yet: \ |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3152 | impl std::future::Future<Output = ()>" |
| 3153 | ); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3154 | }); |
| 3155 | } |
| 3156 | |
| 3157 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3158 | fn test_format_item_fn_rust_abi() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3159 | let test_src = r#" |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3160 | pub fn add(x: f64, y: f64) -> f64 { x * y } |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3161 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3162 | test_format_item(test_src, "add", |result| { |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3163 | // TODO(b/261074843): Re-add thunk name verification once we are using stable |
| 3164 | // name mangling (which may be coming in Q1 2023). (This might mean |
| 3165 | // reverting cl/492333432 + manual review and tweaks.) |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3166 | let result = result.unwrap().unwrap(); |
| 3167 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3168 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3169 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3170 | main_api.tokens, |
| 3171 | quote! { |
| 3172 | inline double add(double x, double y); |
| 3173 | } |
| 3174 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3175 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3176 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3177 | result.cc_details.tokens, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3178 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3179 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3180 | extern "C" double ...(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3181 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3182 | ... |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3183 | inline double add(double x, double y) { |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 3184 | return __crubit_internal::...(x, y); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3185 | } |
| 3186 | } |
| 3187 | ); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3188 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3189 | result.rs_details, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3190 | quote! { |
| 3191 | #[no_mangle] |
| 3192 | extern "C" |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 3193 | fn ...(x: f64, y: f64) -> f64 { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3194 | ::rust_out::add(x, y) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3195 | } |
| 3196 | } |
| 3197 | ); |
| 3198 | }); |
| 3199 | } |
| 3200 | |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3201 | #[test] |
| 3202 | fn test_format_item_fn_rust_abi_with_param_taking_struct_by_value() { |
| 3203 | let test_src = r#" |
| 3204 | pub struct S(i32); |
| 3205 | pub fn into_i32(s: S) -> i32 { s.0 } |
| 3206 | "#; |
| 3207 | test_format_item(test_src, "into_i32", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3208 | let result = result.unwrap().unwrap(); |
| 3209 | let main_api = &result.main_api; |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3210 | assert_cc_matches!( |
| 3211 | main_api.tokens, |
| 3212 | quote! { |
| 3213 | inline std::int32_t into_i32(::rust_out::S s); |
| 3214 | } |
| 3215 | ); |
| 3216 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3217 | result.cc_details.tokens, |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3218 | quote! { |
| 3219 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3220 | extern "C" std::int32_t ...(::rust_out::S*); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3221 | } |
| 3222 | ... |
| 3223 | inline std::int32_t into_i32(::rust_out::S s) { |
Lukasz Anforowicz | d082f35 | 2023-03-09 17:46:11 -0800 | [diff] [blame] | 3224 | return __crubit_internal::...(&s); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3225 | } |
| 3226 | } |
| 3227 | ); |
| 3228 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3229 | result.rs_details, |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3230 | quote! { |
| 3231 | #[no_mangle] |
| 3232 | extern "C" |
Lukasz Anforowicz | d082f35 | 2023-03-09 17:46:11 -0800 | [diff] [blame] | 3233 | fn ...(s: &mut ::core::mem::MaybeUninit<::rust_out::S>) -> i32 { |
| 3234 | ::rust_out::into_i32(unsafe { s.assume_init_read() }) |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3235 | } |
| 3236 | } |
| 3237 | ); |
| 3238 | }); |
| 3239 | } |
| 3240 | |
| 3241 | #[test] |
| 3242 | fn test_format_item_fn_rust_abi_returning_struct_by_value() { |
| 3243 | let test_src = r#" |
| 3244 | pub struct S(i32); |
| 3245 | pub fn create(i: i32) -> S { S(i) } |
| 3246 | "#; |
| 3247 | test_format_item(test_src, "create", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3248 | let result = result.unwrap().unwrap(); |
| 3249 | let main_api = &result.main_api; |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3250 | assert_cc_matches!( |
| 3251 | main_api.tokens, |
| 3252 | quote! { |
| 3253 | inline ::rust_out::S create(std::int32_t i); |
| 3254 | } |
| 3255 | ); |
| 3256 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3257 | result.cc_details.tokens, |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3258 | quote! { |
| 3259 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3260 | extern "C" void ...(std::int32_t, ::rust_out::S* __ret_ptr); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3261 | } |
| 3262 | ... |
| 3263 | inline ::rust_out::S create(std::int32_t i) { |
Lukasz Anforowicz | a3b7db0 | 2023-03-09 17:34:05 -0800 | [diff] [blame] | 3264 | crubit::ReturnValueSlot<::rust_out::S> __ret_slot; |
| 3265 | __crubit_internal::...(i, __ret_slot.Get()); |
| 3266 | return std::move(__ret_slot).AssumeInitAndTakeValue(); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3267 | } |
| 3268 | } |
| 3269 | ); |
| 3270 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3271 | result.rs_details, |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3272 | quote! { |
| 3273 | #[no_mangle] |
| 3274 | extern "C" |
Lukasz Anforowicz | a3b7db0 | 2023-03-09 17:34:05 -0800 | [diff] [blame] | 3275 | fn ...( |
| 3276 | i: i32, |
| 3277 | __ret_slot: &mut ::core::mem::MaybeUninit<::rust_out::S> |
| 3278 | ) -> () { |
| 3279 | __ret_slot.write(::rust_out::create(i)); |
Lukasz Anforowicz | 5227499 | 2023-03-08 12:29:28 -0800 | [diff] [blame] | 3280 | } |
| 3281 | } |
| 3282 | ); |
| 3283 | }); |
| 3284 | } |
| 3285 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3286 | /// `test_format_item_fn_rust_abi` tests a function call that is not a |
| 3287 | /// C-ABI, and is not the default Rust ABI. It can't use `"stdcall"`, |
| 3288 | /// because it is not supported on the targets where Crubit's tests run. |
| 3289 | /// So, it ended up using `"vectorcall"`. |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3290 | /// |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3291 | /// This test almost entirely replicates `test_format_item_fn_rust_abi`, |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3292 | /// except for the `extern "vectorcall"` part in the `test_src` test |
| 3293 | /// input. |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3294 | /// |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3295 | /// This test verifies the current behavior that gives reasonable and |
| 3296 | /// functional FFI bindings. OTOH, in the future we may decide to avoid |
| 3297 | /// having the extra thunk for cases where the given non-C-ABI function |
| 3298 | /// call convention is supported by both C++ and Rust |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3299 | /// (see also `format_cc_call_conv_as_clang_attribute` in |
| 3300 | /// `rs_bindings_from_cc/src_code_gen.rs`) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3301 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3302 | fn test_format_item_fn_vectorcall_abi() { |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3303 | let test_src = r#" |
| 3304 | #![feature(abi_vectorcall)] |
| 3305 | pub extern "vectorcall" fn add(x: f64, y: f64) -> f64 { x * y } |
| 3306 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3307 | test_format_item(test_src, "add", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3308 | let result = result.unwrap().unwrap(); |
| 3309 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3310 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3311 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3312 | main_api.tokens, |
| 3313 | quote! { |
| 3314 | inline double add(double x, double y); |
| 3315 | } |
| 3316 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3317 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3318 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3319 | result.cc_details.tokens, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3320 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3321 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3322 | extern "C" double ...(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3323 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3324 | ... |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3325 | inline double add(double x, double y) { |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 3326 | return __crubit_internal::...(x, y); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3327 | } |
| 3328 | } |
| 3329 | ); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3330 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3331 | result.rs_details, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3332 | quote! { |
| 3333 | #[no_mangle] |
| 3334 | extern "C" |
Lukasz Anforowicz | b4beb39 | 2022-12-01 16:49:11 -0800 | [diff] [blame] | 3335 | fn ...(x: f64, y: f64) -> f64 { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3336 | ::rust_out::add(x, y) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3337 | } |
| 3338 | } |
| 3339 | ); |
| 3340 | }); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3341 | } |
| 3342 | |
| 3343 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3344 | fn test_format_item_unsupported_fn_variadic() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3345 | let test_src = r#" |
| 3346 | #![feature(c_variadic)] |
Lukasz Anforowicz | 7123f21 | 2022-10-20 08:51:04 -0700 | [diff] [blame] | 3347 | |
| 3348 | #[no_mangle] |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3349 | pub unsafe extern "C" fn variadic_function(_fmt: *const u8, ...) {} |
| 3350 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3351 | test_format_item(test_src, "variadic_function", |result| { |
Lukasz Anforowicz | 13794df | 2022-10-21 07:56:34 -0700 | [diff] [blame] | 3352 | // TODO(b/254097223): Add support for variadic functions. |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3353 | let err = result.unwrap_err(); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3354 | assert_eq!(err, "C variadic functions are not supported (b/254097223)"); |
| 3355 | }); |
| 3356 | } |
| 3357 | |
| 3358 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3359 | fn test_format_item_fn_params() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3360 | let test_src = r#" |
| 3361 | #[allow(unused_variables)] |
| 3362 | #[no_mangle] |
| 3363 | pub extern "C" fn foo(b: bool, f: f64) {} |
| 3364 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3365 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3366 | let result = result.unwrap().unwrap(); |
| 3367 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3368 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3369 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3370 | main_api.tokens, |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3371 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3372 | ... |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3373 | extern "C" void foo(bool b, double f); |
| 3374 | } |
| 3375 | ); |
| 3376 | }); |
| 3377 | } |
| 3378 | |
| 3379 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3380 | fn test_format_item_fn_param_name_reserved_keyword() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3381 | let test_src = r#" |
| 3382 | #[allow(unused_variables)] |
| 3383 | #[no_mangle] |
| 3384 | pub extern "C" fn some_function(reinterpret_cast: f64) {} |
| 3385 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3386 | test_format_item(test_src, "some_function", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3387 | let result = result.unwrap().unwrap(); |
| 3388 | let main_api = &result.main_api; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3389 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3390 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3391 | main_api.tokens, |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3392 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3393 | ... |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3394 | extern "C" void some_function(double __param_0); |
| 3395 | } |
| 3396 | ); |
| 3397 | }); |
| 3398 | } |
| 3399 | |
| 3400 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3401 | fn test_format_item_fn_with_multiple_anonymous_parameter_names() { |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3402 | let test_src = r#" |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3403 | pub fn foo(_: f64, _: f64) {} |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3404 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3405 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3406 | let result = result.unwrap().unwrap(); |
| 3407 | let main_api = &result.main_api; |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3408 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3409 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3410 | main_api.tokens, |
| 3411 | quote! { |
| 3412 | inline void foo(double __param_0, double __param_1); |
| 3413 | } |
| 3414 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3415 | assert!(result.cc_details.prereqs.is_empty()); |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3416 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3417 | result.cc_details.tokens, |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3418 | quote! { |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3419 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3420 | extern "C" void ...(double, double); |
Lukasz Anforowicz | bad9963 | 2022-11-18 16:39:13 -0800 | [diff] [blame] | 3421 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3422 | ... |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3423 | inline void foo(double __param_0, double __param_1) { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3424 | return __crubit_internal::...(__param_0, __param_1); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3425 | } |
| 3426 | } |
| 3427 | ); |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3428 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3429 | result.rs_details, |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3430 | quote! { |
| 3431 | #[no_mangle] |
| 3432 | extern "C" fn ...(__param_0: f64, __param_1: f64) -> () { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3433 | ::rust_out::foo(__param_0, __param_1) |
Lukasz Anforowicz | 2f849cf | 2022-11-17 15:52:39 -0800 | [diff] [blame] | 3434 | } |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3435 | } |
| 3436 | ); |
| 3437 | }); |
| 3438 | } |
| 3439 | |
Lukasz Anforowicz | c51aeb1 | 2022-11-07 10:56:18 -0800 | [diff] [blame] | 3440 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3441 | fn test_format_item_fn_with_destructuring_parameter_name() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3442 | let test_src = r#" |
| 3443 | pub struct S { |
| 3444 | pub f1: i32, |
| 3445 | pub f2: i32, |
| 3446 | } |
| 3447 | |
| 3448 | // This test mostly focuses on the weird parameter "name" below. |
| 3449 | // See also |
| 3450 | // https://doc.rust-lang.org/reference/items/functions.html#function-parameters |
| 3451 | // which points out that function parameters are just irrefutable patterns. |
| 3452 | pub fn func(S{f1, f2}: S) -> i32 { f1 + f2 } |
| 3453 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3454 | test_format_item(test_src, "func", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3455 | let result = result.unwrap().unwrap(); |
| 3456 | let main_api = &result.main_api; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3457 | assert_cc_matches!( |
Lukasz Anforowicz | 6753d40 | 2023-02-10 16:41:23 -0800 | [diff] [blame] | 3458 | main_api.tokens, |
| 3459 | quote! { |
| 3460 | inline std::int32_t func(::rust_out::S __param_0); |
| 3461 | } |
| 3462 | ); |
| 3463 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3464 | result.cc_details.tokens, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3465 | quote! { |
| 3466 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3467 | extern "C" std::int32_t ...(::rust_out::S*); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3468 | } |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3469 | ... |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3470 | inline std::int32_t func(::rust_out::S __param_0) { |
Lukasz Anforowicz | d082f35 | 2023-03-09 17:46:11 -0800 | [diff] [blame] | 3471 | return __crubit_internal::...(&__param_0); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3472 | } |
| 3473 | } |
| 3474 | ); |
| 3475 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3476 | result.rs_details, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3477 | quote! { |
| 3478 | #[no_mangle] |
Lukasz Anforowicz | d082f35 | 2023-03-09 17:46:11 -0800 | [diff] [blame] | 3479 | extern "C" fn ...( |
| 3480 | __param_0: &mut ::core::mem::MaybeUninit<::rust_out::S> |
| 3481 | ) -> i32 { |
| 3482 | ::rust_out::func(unsafe {__param_0.assume_init_read() }) |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3483 | } |
| 3484 | } |
| 3485 | ); |
| 3486 | }); |
| 3487 | } |
| 3488 | |
| 3489 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3490 | fn test_format_item_unsupported_fn_param_type() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3491 | let test_src = r#" |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 3492 | pub fn foo(_param: (i32, i32)) {} |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3493 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3494 | test_format_item(test_src, "foo", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3495 | let err = result.unwrap_err(); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3496 | assert_eq!( |
| 3497 | err, |
| 3498 | "Error handling parameter #0: \ |
| 3499 | Tuples are not supported yet: (i32, i32) (b/254099023)" |
| 3500 | ); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3501 | }); |
| 3502 | } |
| 3503 | |
| 3504 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3505 | fn test_format_item_unsupported_fn_param_type_unit() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3506 | let test_src = r#" |
| 3507 | #[no_mangle] |
| 3508 | pub fn fn_with_params(_param: ()) {} |
| 3509 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3510 | test_format_item(test_src, "fn_with_params", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3511 | let err = result.unwrap_err(); |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 3512 | assert_eq!( |
| 3513 | err, |
| 3514 | "Error handling parameter #0: \ |
| 3515 | `()` / `void` is only supported as a return type (b/254507801)" |
| 3516 | ); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3517 | }); |
| 3518 | } |
| 3519 | |
| 3520 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3521 | fn test_format_item_unsupported_fn_param_type_never() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3522 | let test_src = r#" |
| 3523 | #![feature(never_type)] |
| 3524 | |
| 3525 | #[no_mangle] |
| 3526 | pub extern "C" fn fn_with_params(_param: !) {} |
| 3527 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3528 | test_format_item(test_src, "fn_with_params", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 3529 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3530 | assert_eq!( |
| 3531 | err, |
Lukasz Anforowicz | a691cf5 | 2023-03-08 12:24:33 -0800 | [diff] [blame] | 3532 | "Error handling parameter #0: \ |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 3533 | The never type `!` is only supported as a return type (b/254507801)" |
| 3534 | ); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 3535 | }); |
| 3536 | } |
| 3537 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3538 | /// This is a test for a regular struct - a struct with named fields. |
| 3539 | /// https://doc.rust-lang.org/reference/items/structs.html refers to this kind of struct as |
| 3540 | /// `StructStruct` or "nominal struct type". |
| 3541 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3542 | fn test_format_item_struct_with_fields() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3543 | let test_src = r#" |
| 3544 | pub struct SomeStruct { |
| 3545 | pub x: i32, |
| 3546 | pub y: i32, |
| 3547 | } |
| 3548 | |
| 3549 | const _: () = assert!(std::mem::size_of::<SomeStruct>() == 8); |
| 3550 | const _: () = assert!(std::mem::align_of::<SomeStruct>() == 4); |
| 3551 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3552 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3553 | let result = result.unwrap().unwrap(); |
| 3554 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3555 | assert!(!main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3556 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3557 | main_api.tokens, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3558 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3559 | ... |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3560 | struct alignas(4) SomeStruct final { |
| 3561 | public: |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 3562 | __COMMENT__ "`SomeStruct` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3563 | SomeStruct() = delete; |
| 3564 | |
| 3565 | // In this test there is no `Copy` implementation / derive. |
| 3566 | SomeStruct(const SomeStruct&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3567 | |
| 3568 | // All Rust types are trivially-movable. |
| 3569 | SomeStruct(SomeStruct&&) = default; |
Lukasz Anforowicz | 554ed65 | 2023-01-12 15:41:58 -0800 | [diff] [blame] | 3570 | |
| 3571 | // Assignment operators are disabled for now. |
| 3572 | SomeStruct& operator=(const SomeStruct&) = delete; |
| 3573 | SomeStruct& operator=(SomeStruct&&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3574 | |
| 3575 | // In this test there is no custom `Drop`, so C++ can also |
| 3576 | // just use the `default` destructor. |
| 3577 | ~SomeStruct() = default; |
| 3578 | private: |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3579 | ... std::int32_t x; |
| 3580 | ... std::int32_t y; |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3581 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3582 | }; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3583 | } |
| 3584 | ); |
| 3585 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3586 | result.cc_details.tokens, |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3587 | quote! { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3588 | static_assert(sizeof(SomeStruct) == 8, ...); |
| 3589 | static_assert(alignof(SomeStruct) == 4, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3590 | inline void SomeStruct::__crubit_field_offset_assertions() { |
| 3591 | static_assert(0 == offsetof(SomeStruct, x)); |
| 3592 | static_assert(4 == offsetof(SomeStruct, y)); |
| 3593 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3594 | } |
| 3595 | ); |
| 3596 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3597 | result.rs_details, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3598 | quote! { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3599 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8); |
| 3600 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3601 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, x) == 0); |
| 3602 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, y) == 4); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3603 | } |
| 3604 | ); |
| 3605 | }); |
| 3606 | } |
| 3607 | |
| 3608 | /// This is a test for `TupleStruct` or "tuple struct" - for more details |
| 3609 | /// please refer to https://doc.rust-lang.org/reference/items/structs.html |
| 3610 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3611 | fn test_format_item_struct_with_tuple() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3612 | let test_src = r#" |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3613 | pub struct TupleStruct(pub i32, pub i32); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3614 | const _: () = assert!(std::mem::size_of::<TupleStruct>() == 8); |
| 3615 | const _: () = assert!(std::mem::align_of::<TupleStruct>() == 4); |
| 3616 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 3617 | test_format_item(test_src, "TupleStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3618 | let result = result.unwrap().unwrap(); |
| 3619 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3620 | assert!(!main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3621 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3622 | main_api.tokens, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3623 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 3624 | ... |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3625 | struct alignas(4) TupleStruct final { |
| 3626 | public: |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 3627 | __COMMENT__ "`TupleStruct` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3628 | TupleStruct() = delete; |
| 3629 | |
| 3630 | // In this test there is no `Copy` implementation / derive. |
| 3631 | TupleStruct(const TupleStruct&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3632 | |
| 3633 | // All Rust types are trivially-movable. |
| 3634 | TupleStruct(TupleStruct&&) = default; |
Lukasz Anforowicz | 554ed65 | 2023-01-12 15:41:58 -0800 | [diff] [blame] | 3635 | |
| 3636 | // Assignment operators are disabled for now. |
| 3637 | TupleStruct& operator=(const TupleStruct&) = delete; |
| 3638 | TupleStruct& operator=(TupleStruct&&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3639 | |
| 3640 | // In this test there is no custom `Drop`, so C++ can also |
| 3641 | // just use the `default` destructor. |
| 3642 | ~TupleStruct() = default; |
| 3643 | private: |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3644 | ... std::int32_t __field0; |
| 3645 | ... std::int32_t __field1; |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3646 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3647 | }; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3648 | } |
| 3649 | ); |
| 3650 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3651 | result.cc_details.tokens, |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 3652 | quote! { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3653 | static_assert(sizeof(TupleStruct) == 8, ...); |
| 3654 | static_assert(alignof(TupleStruct) == 4, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3655 | inline void TupleStruct::__crubit_field_offset_assertions() { |
| 3656 | static_assert(0 == offsetof(TupleStruct, __field0)); |
| 3657 | static_assert(4 == offsetof(TupleStruct, __field1)); |
| 3658 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3659 | } |
| 3660 | ); |
| 3661 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3662 | result.rs_details, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3663 | quote! { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 3664 | const _: () = assert!(::std::mem::size_of::<::rust_out::TupleStruct>() == 8); |
| 3665 | const _: () = assert!(::std::mem::align_of::<::rust_out::TupleStruct>() == 4); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3666 | const _: () = assert!( memoffset::offset_of!(::rust_out::TupleStruct, 0) == 0); |
| 3667 | const _: () = assert!( memoffset::offset_of!(::rust_out::TupleStruct, 1) == 4); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3668 | } |
| 3669 | ); |
| 3670 | }); |
| 3671 | } |
| 3672 | |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3673 | /// This test the scenario where Rust lays out field in a different order |
| 3674 | /// than the source order. |
| 3675 | #[test] |
| 3676 | fn test_format_item_struct_with_reordered_field_offsets() { |
| 3677 | let test_src = r#" |
| 3678 | pub struct SomeStruct { |
| 3679 | pub field1: i16, |
| 3680 | pub field2: i32, |
| 3681 | pub field3: i16, |
| 3682 | } |
| 3683 | |
| 3684 | const _: () = assert!(std::mem::size_of::<SomeStruct>() == 8); |
| 3685 | const _: () = assert!(std::mem::align_of::<SomeStruct>() == 4); |
| 3686 | "#; |
| 3687 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3688 | let result = result.unwrap().unwrap(); |
| 3689 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3690 | assert!(!main_api.prereqs.is_empty()); |
| 3691 | assert_cc_matches!( |
| 3692 | main_api.tokens, |
| 3693 | quote! { |
| 3694 | ... |
| 3695 | struct alignas(4) SomeStruct final { |
| 3696 | ... |
| 3697 | private: |
| 3698 | // The particular order below is not guaranteed, |
| 3699 | // so we may need to adjust this test assertion |
| 3700 | // (if Rust changes how it lays out the fields). |
| 3701 | ... std::int32_t field2; |
| 3702 | ... std::int16_t field1; |
| 3703 | ... std::int16_t field3; |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3704 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3705 | }; |
| 3706 | } |
| 3707 | ); |
| 3708 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3709 | result.cc_details.tokens, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3710 | quote! { |
| 3711 | static_assert(sizeof(SomeStruct) == 8, ...); |
| 3712 | static_assert(alignof(SomeStruct) == 4, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3713 | inline void SomeStruct::__crubit_field_offset_assertions() { |
| 3714 | static_assert(0 == offsetof(SomeStruct, field2)); |
| 3715 | static_assert(4 == offsetof(SomeStruct, field1)); |
| 3716 | static_assert(6 == offsetof(SomeStruct, field3)); |
| 3717 | } |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3718 | } |
| 3719 | ); |
| 3720 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3721 | result.rs_details, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3722 | quote! { |
| 3723 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8); |
| 3724 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3725 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field2) |
| 3726 | == 0); |
| 3727 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field1) |
| 3728 | == 4); |
| 3729 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field3) |
| 3730 | == 6); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3731 | } |
| 3732 | ); |
| 3733 | }); |
| 3734 | } |
| 3735 | |
| 3736 | #[test] |
| 3737 | fn test_format_item_struct_with_packed_layout() { |
| 3738 | let test_src = r#" |
| 3739 | #[repr(packed(1))] |
| 3740 | pub struct SomeStruct { |
| 3741 | pub field1: u16, |
| 3742 | pub field2: u32, |
| 3743 | } |
| 3744 | const _: () = assert!(::std::mem::size_of::<SomeStruct>() == 6); |
| 3745 | const _: () = assert!(::std::mem::align_of::<SomeStruct>() == 1); |
| 3746 | "#; |
| 3747 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3748 | let result = result.unwrap().unwrap(); |
| 3749 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3750 | assert!(!main_api.prereqs.is_empty()); |
| 3751 | assert_cc_matches!( |
| 3752 | main_api.tokens, |
| 3753 | quote! { |
| 3754 | ... |
| 3755 | struct alignas(1) __attribute__((packed)) SomeStruct final { |
| 3756 | ... |
| 3757 | std::uint16_t field1; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3758 | std::uint32_t field2; |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3759 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3760 | }; |
| 3761 | } |
| 3762 | ); |
| 3763 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3764 | result.cc_details.tokens, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3765 | quote! { |
| 3766 | static_assert(sizeof(SomeStruct) == 6, ...); |
| 3767 | static_assert(alignof(SomeStruct) == 1, ...); |
Lukasz Anforowicz | e3eb1cf | 2023-03-14 09:56:00 -0700 | [diff] [blame] | 3768 | inline void SomeStruct::__crubit_field_offset_assertions() { |
| 3769 | static_assert(0 == offsetof(SomeStruct, field1)); |
| 3770 | static_assert(2 == offsetof(SomeStruct, field2)); |
| 3771 | } |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3772 | } |
| 3773 | ); |
| 3774 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3775 | result.rs_details, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3776 | quote! { |
| 3777 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 6); |
| 3778 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 1); |
Lukasz Anforowicz | cf60f52 | 2023-03-14 10:03:55 -0700 | [diff] [blame] | 3779 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field1) |
| 3780 | == 0); |
| 3781 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, field2) |
| 3782 | == 2); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 3783 | } |
| 3784 | ); |
| 3785 | }); |
| 3786 | } |
| 3787 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 3788 | #[test] |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 3789 | fn test_format_item_struct_with_explicit_padding_in_generated_code() { |
| 3790 | let test_src = r#" |
| 3791 | pub struct SomeStruct { |
| 3792 | pub f1: u8, |
| 3793 | pub f2: u32, |
| 3794 | } |
| 3795 | const _: () = assert!(::std::mem::size_of::<SomeStruct>() == 8); |
| 3796 | const _: () = assert!(::std::mem::align_of::<SomeStruct>() == 4); |
| 3797 | "#; |
| 3798 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3799 | let result = result.unwrap().unwrap(); |
| 3800 | let main_api = &result.main_api; |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 3801 | assert!(!main_api.prereqs.is_empty()); |
| 3802 | assert_cc_matches!( |
| 3803 | main_api.tokens, |
| 3804 | quote! { |
| 3805 | ... |
| 3806 | struct alignas(4) SomeStruct final { |
| 3807 | ... |
| 3808 | std::uint32_t f2; |
| 3809 | std::uint8_t f1; |
| 3810 | unsigned char __padding0[3]; |
| 3811 | inline static void __crubit_field_offset_assertions(); |
| 3812 | }; |
| 3813 | } |
| 3814 | ); |
| 3815 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3816 | result.cc_details.tokens, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 3817 | quote! { |
| 3818 | static_assert(sizeof(SomeStruct) == 8, ...); |
| 3819 | static_assert(alignof(SomeStruct) == 4, ...); |
| 3820 | inline void SomeStruct::__crubit_field_offset_assertions() { |
| 3821 | static_assert(0 == offsetof(SomeStruct, f2)); |
| 3822 | static_assert(4 == offsetof(SomeStruct, f1)); |
| 3823 | } |
| 3824 | } |
| 3825 | ); |
| 3826 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3827 | result.rs_details, |
Lukasz Anforowicz | 735bfb8 | 2023-03-15 16:18:44 -0700 | [diff] [blame] | 3828 | quote! { |
| 3829 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 8); |
| 3830 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4); |
| 3831 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, f2) == 0); |
| 3832 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, f1) == 4); |
| 3833 | } |
| 3834 | ); |
| 3835 | }); |
| 3836 | } |
| 3837 | |
| 3838 | #[test] |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3839 | fn test_format_item_static_method() { |
| 3840 | let test_src = r#" |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3841 | /// No-op `f32` placeholder is used, because ZSTs are not supported |
| 3842 | /// (b/258259459). |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3843 | pub struct Math(f32); |
| 3844 | |
| 3845 | impl Math { |
| 3846 | pub fn add_i32(x: f32, y: f32) -> f32 { |
| 3847 | x + y |
| 3848 | } |
| 3849 | } |
| 3850 | "#; |
| 3851 | test_format_item(test_src, "Math", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3852 | let result = result.unwrap().unwrap(); |
| 3853 | let main_api = &result.main_api; |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3854 | assert!(main_api.prereqs.is_empty()); |
| 3855 | assert_cc_matches!( |
| 3856 | main_api.tokens, |
| 3857 | quote! { |
| 3858 | ... |
| 3859 | struct ... Math final { |
| 3860 | ... |
| 3861 | public: |
| 3862 | ... |
| 3863 | static inline float add_i32(float x, float y); |
| 3864 | ... |
| 3865 | }; |
| 3866 | } |
| 3867 | ); |
| 3868 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3869 | result.cc_details.tokens, |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3870 | quote! { |
| 3871 | namespace __crubit_internal { |
Lukasz Anforowicz | 73edf15 | 2023-04-04 12:05:00 -0700 | [diff] [blame] | 3872 | extern "C" float ... (float, float); |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3873 | } |
| 3874 | inline float Math::add_i32(float x, float y) { |
| 3875 | return __crubit_internal::...(x, y); |
| 3876 | } |
| 3877 | } |
| 3878 | ); |
| 3879 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3880 | result.rs_details, |
Lukasz Anforowicz | 43f06c6 | 2023-02-10 17:14:19 -0800 | [diff] [blame] | 3881 | quote! { |
| 3882 | #[no_mangle] |
| 3883 | extern "C" fn ...(x: f32, y: f32) -> f32 { |
| 3884 | ::rust_out::Math::add_i32(x, y) |
| 3885 | } |
| 3886 | } |
| 3887 | ); |
| 3888 | }); |
| 3889 | } |
| 3890 | |
| 3891 | #[test] |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3892 | fn test_format_item_static_method_with_generic_type_parameters() { |
| 3893 | let test_src = r#" |
| 3894 | /// No-op `f32` placeholder is used, because ZSTs are not supported |
| 3895 | /// (b/258259459). |
| 3896 | pub struct SomeStruct(f32); |
| 3897 | |
| 3898 | impl SomeStruct { |
| 3899 | // To make this testcase distinct / non-overlapping wrt |
| 3900 | // test_format_item_static_method_with_generic_lifetime_parameters |
| 3901 | // `t` is taken by value below. |
| 3902 | pub fn generic_method<T: Clone>(t: T) -> T { |
| 3903 | t.clone() |
| 3904 | } |
| 3905 | } |
| 3906 | "#; |
| 3907 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3908 | let result = result.unwrap().unwrap(); |
| 3909 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3910 | assert!(main_api.prereqs.is_empty()); |
| 3911 | let unsupported_msg = "Error generating bindings for `SomeStruct::generic_method` \ |
| 3912 | defined at <crubit_unittests.rs>;l=10: \ |
| 3913 | Generic functions are not supported yet (b/259749023)"; |
| 3914 | assert_cc_matches!( |
| 3915 | main_api.tokens, |
| 3916 | quote! { |
| 3917 | ... |
| 3918 | struct ... SomeStruct final { |
| 3919 | ... |
| 3920 | __COMMENT__ #unsupported_msg |
| 3921 | ... |
| 3922 | }; |
| 3923 | ... |
| 3924 | } |
| 3925 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3926 | assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::generic_method },); |
| 3927 | assert_rs_not_matches!(result.rs_details, quote! { generic_method },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3928 | }); |
| 3929 | } |
| 3930 | |
| 3931 | #[test] |
| 3932 | fn test_format_item_static_method_with_generic_lifetime_parameters() { |
| 3933 | let test_src = r#" |
| 3934 | /// No-op `f32` placeholder is used, because ZSTs are not supported |
| 3935 | /// (b/258259459). |
| 3936 | pub struct SomeStruct(f32); |
| 3937 | |
| 3938 | impl SomeStruct { |
| 3939 | pub fn fn_taking_reference<'a>(x: &'a i32) -> i32 { *x } |
| 3940 | } |
| 3941 | "#; |
| 3942 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3943 | let result = result.unwrap().unwrap(); |
| 3944 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3945 | assert!(main_api.prereqs.is_empty()); |
| 3946 | let unsupported_msg = "Error generating bindings for `SomeStruct::fn_taking_reference` \ |
| 3947 | defined at <crubit_unittests.rs>;l=7: \ |
| 3948 | Generic functions are not supported yet (b/259749023)"; |
| 3949 | assert_cc_matches!( |
| 3950 | main_api.tokens, |
| 3951 | quote! { |
| 3952 | ... |
| 3953 | struct ... SomeStruct final { |
| 3954 | ... |
| 3955 | __COMMENT__ #unsupported_msg |
| 3956 | ... |
| 3957 | }; |
| 3958 | ... |
| 3959 | } |
| 3960 | ); |
| 3961 | assert_cc_not_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3962 | result.cc_details.tokens, |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3963 | quote! { SomeStruct::fn_taking_reference }, |
| 3964 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3965 | assert_rs_not_matches!(result.rs_details, quote! { fn_taking_reference },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3966 | }); |
| 3967 | } |
| 3968 | |
| 3969 | #[test] |
| 3970 | fn test_format_item_method_taking_self_by_value() { |
| 3971 | let test_src = r#" |
| 3972 | pub struct SomeStruct(f32); |
| 3973 | |
| 3974 | impl SomeStruct { |
| 3975 | pub fn into_f32(self) -> f32 { |
| 3976 | self.0 |
| 3977 | } |
| 3978 | } |
| 3979 | "#; |
| 3980 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3981 | let result = result.unwrap().unwrap(); |
| 3982 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 3983 | assert!(main_api.prereqs.is_empty()); |
| 3984 | let unsupported_msg = "Error generating bindings for `SomeStruct::into_f32` \ |
| 3985 | defined at <crubit_unittests.rs>;l=5: \ |
| 3986 | `self` parameter is not supported yet"; |
| 3987 | assert_cc_matches!( |
| 3988 | main_api.tokens, |
| 3989 | quote! { |
| 3990 | ... |
| 3991 | struct ... SomeStruct final { |
| 3992 | ... |
| 3993 | __COMMENT__ #unsupported_msg |
| 3994 | ... |
| 3995 | }; |
| 3996 | ... |
| 3997 | } |
| 3998 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 3999 | assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::into_f32 },); |
| 4000 | assert_rs_not_matches!(result.rs_details, quote! { into_f32 },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4001 | }); |
| 4002 | } |
| 4003 | |
| 4004 | #[test] |
| 4005 | fn test_format_item_method_taking_self_by_const_ref() { |
| 4006 | let test_src = r#" |
| 4007 | pub struct SomeStruct(f32); |
| 4008 | |
| 4009 | impl SomeStruct { |
| 4010 | pub fn get_f32(&self) -> f32 { |
| 4011 | self.0 |
| 4012 | } |
| 4013 | } |
| 4014 | "#; |
| 4015 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4016 | let result = result.unwrap().unwrap(); |
| 4017 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4018 | assert!(main_api.prereqs.is_empty()); |
| 4019 | let unsupported_msg = "Error generating bindings for `SomeStruct::get_f32` \ |
| 4020 | defined at <crubit_unittests.rs>;l=5: \ |
| 4021 | Generic functions are not supported yet (b/259749023)"; |
| 4022 | assert_cc_matches!( |
| 4023 | main_api.tokens, |
| 4024 | quote! { |
| 4025 | ... |
| 4026 | struct ... SomeStruct final { |
| 4027 | ... |
| 4028 | __COMMENT__ #unsupported_msg |
| 4029 | ... |
| 4030 | }; |
| 4031 | ... |
| 4032 | } |
| 4033 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4034 | assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::get_f32 },); |
| 4035 | assert_rs_not_matches!(result.rs_details, quote! { get_f32 },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4036 | }); |
| 4037 | } |
| 4038 | |
| 4039 | #[test] |
| 4040 | fn test_format_item_method_taking_self_by_mutable_ref() { |
| 4041 | let test_src = r#" |
| 4042 | pub struct SomeStruct(f32); |
| 4043 | |
| 4044 | impl SomeStruct { |
| 4045 | pub fn set_f32(&mut self, new_value: f32) { |
| 4046 | self.0 = new_value; |
| 4047 | } |
| 4048 | } |
| 4049 | "#; |
| 4050 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4051 | let result = result.unwrap().unwrap(); |
| 4052 | let main_api = &result.main_api; |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4053 | assert!(main_api.prereqs.is_empty()); |
| 4054 | let unsupported_msg = "Error generating bindings for `SomeStruct::set_f32` \ |
| 4055 | defined at <crubit_unittests.rs>;l=5: \ |
| 4056 | Generic functions are not supported yet (b/259749023)"; |
| 4057 | assert_cc_matches!( |
| 4058 | main_api.tokens, |
| 4059 | quote! { |
| 4060 | ... |
| 4061 | struct ... SomeStruct final { |
| 4062 | ... |
| 4063 | __COMMENT__ #unsupported_msg |
| 4064 | ... |
| 4065 | }; |
| 4066 | ... |
| 4067 | } |
| 4068 | ); |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4069 | assert_cc_not_matches!(result.cc_details.tokens, quote! { SomeStruct::set_f32 },); |
| 4070 | assert_rs_not_matches!(result.rs_details, quote! { set_f32 },); |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4071 | }); |
| 4072 | } |
| 4073 | |
| 4074 | #[test] |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 4075 | fn test_format_item_struct_with_default_constructor() { |
| 4076 | let test_src = r#" |
| 4077 | #[derive(Default)] |
| 4078 | pub struct Point(i32, i32); |
| 4079 | "#; |
| 4080 | test_format_item(test_src, "Point", |result| { |
| 4081 | let result = result.unwrap().unwrap(); |
| 4082 | let main_api = &result.main_api; |
| 4083 | assert_cc_matches!( |
| 4084 | main_api.tokens, |
| 4085 | quote! { |
| 4086 | ... |
| 4087 | struct ... Point final { |
| 4088 | ... |
| 4089 | public: |
| 4090 | __COMMENT__ "Default::default" |
| 4091 | inline Point(); |
| 4092 | ... |
| 4093 | }; |
| 4094 | } |
| 4095 | ); |
| 4096 | assert_cc_matches!( |
| 4097 | result.cc_details.tokens, |
| 4098 | quote! { |
| 4099 | namespace __crubit_internal { |
| 4100 | extern "C" void ...(::rust_out::Point* __ret_ptr); |
| 4101 | } |
| 4102 | Point::Point() { |
| 4103 | ...(this); |
| 4104 | } |
| 4105 | } |
| 4106 | ); |
| 4107 | assert_rs_matches!( |
| 4108 | result.rs_details, |
| 4109 | quote! { |
| 4110 | #[no_mangle] |
| 4111 | extern "C" fn ...( |
| 4112 | __ret_slot: &mut ::core::mem::MaybeUninit<::rust_out::Point> |
| 4113 | ) -> () { |
| 4114 | __ret_slot.write(<::rust_out::Point as ::core::default::Default>::default()); |
| 4115 | } |
| 4116 | } |
| 4117 | ); |
| 4118 | }); |
| 4119 | } |
| 4120 | |
| 4121 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4122 | fn test_format_item_unsupported_struct_with_name_that_is_reserved_keyword() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4123 | let test_src = r#" |
| 4124 | #[allow(non_camel_case_types)] |
| 4125 | pub struct reinterpret_cast { |
| 4126 | pub x: i32, |
| 4127 | pub y: i32, |
| 4128 | } |
| 4129 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4130 | test_format_item(test_src, "reinterpret_cast", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4131 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4132 | assert_eq!( |
| 4133 | err, |
| 4134 | "Error formatting item name: \ |
| 4135 | `reinterpret_cast` is a C++ reserved keyword \ |
| 4136 | and can't be used as a C++ identifier" |
| 4137 | ); |
| 4138 | }); |
| 4139 | } |
| 4140 | |
| 4141 | #[test] |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4142 | fn test_format_item_struct_with_unsupported_field_type() { |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4143 | let test_src = r#" |
| 4144 | pub struct SomeStruct { |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4145 | pub successful_field: i32, |
| 4146 | pub unsupported_field: Option<[i32; 3]>, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4147 | } |
| 4148 | "#; |
| 4149 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4150 | let result = result.unwrap().unwrap(); |
| 4151 | let main_api = &result.main_api; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4152 | 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] | 4153 | Generic types are not supported yet (b/259749095)"; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4154 | assert_cc_matches!( |
| 4155 | main_api.tokens, |
| 4156 | quote! { |
| 4157 | ... |
| 4158 | struct ... SomeStruct final { |
| 4159 | ... |
| 4160 | private: |
| 4161 | __COMMENT__ #broken_field_msg |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4162 | unsigned char unsupported_field[16]; |
| 4163 | std::int32_t successful_field; |
| 4164 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4165 | }; |
| 4166 | ... |
| 4167 | } |
| 4168 | ); |
| 4169 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4170 | result.cc_details.tokens, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4171 | quote! { |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4172 | static_assert(sizeof(SomeStruct) == 20, ...); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4173 | static_assert(alignof(SomeStruct) == 4, ...); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4174 | inline void SomeStruct::__crubit_field_offset_assertions() { |
| 4175 | static_assert(0 == offsetof(SomeStruct, unsupported_field)); |
| 4176 | static_assert(16 == offsetof(SomeStruct, successful_field)); |
| 4177 | } |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4178 | } |
| 4179 | ); |
| 4180 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4181 | result.rs_details, |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4182 | quote! { |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4183 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 20); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4184 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4); |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4185 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, |
| 4186 | unsupported_field) == 0); |
| 4187 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, |
| 4188 | successful_field) == 16); |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4189 | } |
| 4190 | ); |
| 4191 | }); |
| 4192 | } |
| 4193 | |
| 4194 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4195 | fn test_format_item_unsupported_struct_with_custom_drop_impl() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4196 | let test_src = r#" |
| 4197 | pub struct StructWithCustomDropImpl { |
| 4198 | pub x: i32, |
| 4199 | pub y: i32, |
| 4200 | } |
| 4201 | |
| 4202 | impl Drop for StructWithCustomDropImpl { |
| 4203 | fn drop(&mut self) {} |
| 4204 | } |
| 4205 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4206 | test_format_item(test_src, "StructWithCustomDropImpl", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4207 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4208 | assert_eq!(err, "`Drop` trait and \"drop glue\" are not supported yet (b/258251148)"); |
| 4209 | }); |
| 4210 | } |
| 4211 | |
| 4212 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4213 | fn test_format_item_unsupported_struct_with_custom_drop_glue() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4214 | let test_src = r#" |
| 4215 | #![allow(dead_code)] |
| 4216 | |
| 4217 | // `i32` is present to avoid hitting the ZST checks related to (b/258259459) |
| 4218 | struct StructWithCustomDropImpl(i32); |
| 4219 | |
| 4220 | impl Drop for StructWithCustomDropImpl { |
| 4221 | fn drop(&mut self) { |
| 4222 | println!("dropping!"); |
| 4223 | } |
| 4224 | } |
| 4225 | |
| 4226 | pub struct StructRequiringCustomDropGlue { |
| 4227 | field: StructWithCustomDropImpl, |
| 4228 | } |
| 4229 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4230 | test_format_item(test_src, "StructRequiringCustomDropGlue", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4231 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4232 | assert_eq!(err, "`Drop` trait and \"drop glue\" are not supported yet (b/258251148)"); |
| 4233 | }); |
| 4234 | } |
| 4235 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4236 | /// This test covers how ZSTs (zero-sized-types) are handled. |
| 4237 | /// https://doc.rust-lang.org/reference/items/structs.html refers to this kind of struct as a |
| 4238 | /// "unit-like struct". |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4239 | #[test] |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4240 | fn test_format_item_unsupported_struct_zero_sized_type_with_no_fields() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4241 | let test_src = r#" |
| 4242 | pub struct ZeroSizedType1; |
| 4243 | pub struct ZeroSizedType2(); |
| 4244 | pub struct ZeroSizedType3{} |
| 4245 | "#; |
| 4246 | for name in ["ZeroSizedType1", "ZeroSizedType2", "ZeroSizedType3"] { |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4247 | test_format_item(test_src, name, |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4248 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4249 | assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)"); |
| 4250 | }); |
| 4251 | } |
| 4252 | } |
| 4253 | |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4254 | #[test] |
| 4255 | fn test_format_item_unsupported_struct_with_only_zero_sized_type_fields() { |
| 4256 | let test_src = r#" |
| 4257 | pub struct ZeroSizedType; |
| 4258 | pub struct SomeStruct { |
| 4259 | pub zst1: ZeroSizedType, |
| 4260 | pub zst2: ZeroSizedType, |
| 4261 | } |
| 4262 | "#; |
| 4263 | test_format_item(test_src, "SomeStruct", |result| { |
| 4264 | let err = result.unwrap_err(); |
| 4265 | assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)",); |
| 4266 | }); |
| 4267 | } |
| 4268 | |
Devin Jeanpierre | 9e15d0b | 2023-04-06 13:18:22 -0700 | [diff] [blame] | 4269 | #[test] |
| 4270 | fn test_format_item_unsupported_struct_with_some_zero_sized_type_fields() { |
| 4271 | let test_src = r#" |
| 4272 | pub struct ZeroSizedType; |
| 4273 | pub struct SomeStruct { |
| 4274 | pub zst1: ZeroSizedType, |
| 4275 | pub zst2: ZeroSizedType, |
| 4276 | pub successful_field: i32, |
| 4277 | } |
| 4278 | "#; |
| 4279 | test_format_item(test_src, "SomeStruct", |result| { |
| 4280 | let result = result.unwrap().unwrap(); |
| 4281 | let main_api = &result.main_api; |
| 4282 | let broken_field_msg = "Field type has been replaced with a blob of bytes: \ |
| 4283 | Failed to generate bindings for the definition of `ZeroSizedType`: \ |
| 4284 | Zero-sized types (ZSTs) are not supported (b/258259459)"; |
| 4285 | assert_cc_matches!( |
| 4286 | main_api.tokens, |
| 4287 | quote! { |
| 4288 | ... |
| 4289 | struct ... SomeStruct final { |
| 4290 | ... |
| 4291 | private: |
| 4292 | __COMMENT__ #broken_field_msg |
| 4293 | [[no_unique_address]] struct{} zst1; |
| 4294 | __COMMENT__ #broken_field_msg |
| 4295 | [[no_unique_address]] struct{} zst2; |
| 4296 | std::int32_t successful_field; |
| 4297 | inline static void __crubit_field_offset_assertions(); |
| 4298 | }; |
| 4299 | ... |
| 4300 | } |
| 4301 | ); |
| 4302 | assert_cc_matches!( |
| 4303 | result.cc_details.tokens, |
| 4304 | quote! { |
| 4305 | static_assert(sizeof(SomeStruct) == 4, ...); |
| 4306 | static_assert(alignof(SomeStruct) == 4, ...); |
| 4307 | inline void SomeStruct::__crubit_field_offset_assertions() { |
| 4308 | static_assert(0 == offsetof(SomeStruct, zst1)); |
| 4309 | static_assert(0 == offsetof(SomeStruct, zst2)); |
| 4310 | static_assert(0 == offsetof(SomeStruct, successful_field)); |
| 4311 | } |
| 4312 | } |
| 4313 | ); |
| 4314 | assert_rs_matches!( |
| 4315 | result.rs_details, |
| 4316 | quote! { |
| 4317 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeStruct>() == 4); |
| 4318 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeStruct>() == 4); |
| 4319 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, |
| 4320 | zst1) == 0); |
| 4321 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, |
| 4322 | zst2) == 0); |
| 4323 | const _: () = assert!( memoffset::offset_of!(::rust_out::SomeStruct, |
| 4324 | successful_field) == 0); |
| 4325 | } |
| 4326 | ); |
| 4327 | }); |
| 4328 | } |
| 4329 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4330 | /// This is a test for an enum that only has `EnumItemDiscriminant` items |
| 4331 | /// (and doesn't have `EnumItemTuple` or `EnumItemStruct` items). See |
| 4332 | /// also https://doc.rust-lang.org/reference/items/enumerations.html |
| 4333 | #[test] |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4334 | fn test_format_item_enum_with_only_discriminant_items() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4335 | let test_src = r#" |
| 4336 | pub enum SomeEnum { |
| 4337 | Red, |
| 4338 | Green = 123, |
| 4339 | Blue, |
| 4340 | } |
| 4341 | |
| 4342 | const _: () = assert!(std::mem::size_of::<SomeEnum>() == 1); |
| 4343 | const _: () = assert!(std::mem::align_of::<SomeEnum>() == 1); |
| 4344 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4345 | test_format_item(test_src, "SomeEnum", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4346 | let result = result.unwrap().unwrap(); |
| 4347 | let main_api = &result.main_api; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4348 | 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] | 4349 | No support for bindings of individual fields of \ |
| 4350 | `union` (b/272801632) or `enum`"; |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4351 | assert!(main_api.prereqs.is_empty()); |
| 4352 | assert_cc_matches!( |
| 4353 | main_api.tokens, |
| 4354 | quote! { |
| 4355 | ... |
| 4356 | struct alignas(1) SomeEnum final { |
| 4357 | public: |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 4358 | __COMMENT__ "`SomeEnum` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4359 | SomeEnum() = delete; |
| 4360 | |
| 4361 | // In this test there is no `Copy` implementation / derive. |
| 4362 | SomeEnum(const SomeEnum&) = delete; |
| 4363 | |
| 4364 | // All Rust types are trivially-movable. |
| 4365 | SomeEnum(SomeEnum&&) = default; |
| 4366 | |
| 4367 | // Assignment operators are disabled for now. |
| 4368 | SomeEnum& operator=(const SomeEnum&) = delete; |
| 4369 | SomeEnum& operator=(SomeEnum&&) = delete; |
| 4370 | |
| 4371 | // In this test there is no custom `Drop`, so C++ can also |
| 4372 | // just use the `default` destructor. |
| 4373 | ~SomeEnum() = default; |
| 4374 | private: |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4375 | __COMMENT__ #no_fields_msg |
| 4376 | unsigned char __opaque_blob_of_bytes[1]; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4377 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4378 | }; |
| 4379 | } |
| 4380 | ); |
| 4381 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4382 | result.cc_details.tokens, |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4383 | quote! { |
| 4384 | static_assert(sizeof(SomeEnum) == 1, ...); |
| 4385 | static_assert(alignof(SomeEnum) == 1, ...); |
| 4386 | } |
| 4387 | ); |
| 4388 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4389 | result.rs_details, |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4390 | quote! { |
| 4391 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeEnum>() == 1); |
| 4392 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeEnum>() == 1); |
| 4393 | } |
| 4394 | ); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4395 | }); |
| 4396 | } |
| 4397 | |
| 4398 | /// This is a test for an enum that has `EnumItemTuple` and `EnumItemStruct` |
| 4399 | /// items. See also https://doc.rust-lang.org/reference/items/enumerations.html |
| 4400 | #[test] |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4401 | fn test_format_item_enum_with_tuple_and_struct_items() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4402 | let test_src = r#" |
| 4403 | pub enum Point { |
| 4404 | Cartesian(f32, f32), |
| 4405 | Polar{ dist: f32, angle: f32 }, |
| 4406 | } |
| 4407 | |
| 4408 | const _: () = assert!(std::mem::size_of::<Point>() == 12); |
| 4409 | const _: () = assert!(std::mem::align_of::<Point>() == 4); |
| 4410 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4411 | test_format_item(test_src, "Point", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4412 | let result = result.unwrap().unwrap(); |
| 4413 | let main_api = &result.main_api; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4414 | 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] | 4415 | No support for bindings of individual fields of \ |
| 4416 | `union` (b/272801632) or `enum`"; |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4417 | assert!(main_api.prereqs.is_empty()); |
| 4418 | assert_cc_matches!( |
| 4419 | main_api.tokens, |
| 4420 | quote! { |
| 4421 | ... |
| 4422 | struct alignas(4) Point final { |
| 4423 | public: |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 4424 | __COMMENT__ "`Point` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4425 | Point() = delete; |
| 4426 | |
| 4427 | // In this test there is no `Copy` implementation / derive. |
| 4428 | Point(const Point&) = delete; |
| 4429 | |
| 4430 | // All Rust types are trivially-movable. |
| 4431 | Point(Point&&) = default; |
| 4432 | |
| 4433 | // Assignment operators are disabled for now. |
| 4434 | Point& operator=(const Point&) = delete; |
| 4435 | Point& operator=(Point&&) = delete; |
| 4436 | |
| 4437 | // In this test there is no custom `Drop`, so C++ can also |
| 4438 | // just use the `default` destructor. |
| 4439 | ~Point() = default; |
| 4440 | private: |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4441 | __COMMENT__ #no_fields_msg |
| 4442 | unsigned char __opaque_blob_of_bytes[12]; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4443 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4444 | }; |
| 4445 | } |
| 4446 | ); |
| 4447 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4448 | result.cc_details.tokens, |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4449 | quote! { |
| 4450 | static_assert(sizeof(Point) == 12, ...); |
| 4451 | static_assert(alignof(Point) == 4, ...); |
| 4452 | } |
| 4453 | ); |
| 4454 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4455 | result.rs_details, |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4456 | quote! { |
| 4457 | const _: () = assert!(::std::mem::size_of::<::rust_out::Point>() == 12); |
| 4458 | const _: () = assert!(::std::mem::align_of::<::rust_out::Point>() == 4); |
| 4459 | } |
| 4460 | ); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4461 | }); |
| 4462 | } |
| 4463 | |
| 4464 | /// This test covers how zero-variant enums are handled. See also |
| 4465 | /// https://doc.rust-lang.org/reference/items/enumerations.html#zero-variant-enums |
| 4466 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4467 | fn test_format_item_unsupported_enum_zero_variants() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4468 | let test_src = r#" |
| 4469 | pub enum ZeroVariantEnum {} |
| 4470 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4471 | test_format_item(test_src, "ZeroVariantEnum", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4472 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4473 | assert_eq!(err, "Zero-sized types (ZSTs) are not supported (b/258259459)"); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4474 | }); |
| 4475 | } |
| 4476 | |
| 4477 | /// This is a test for a `union`. See also |
| 4478 | /// https://doc.rust-lang.org/reference/items/unions.html |
| 4479 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4480 | fn test_format_item_union() { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4481 | let test_src = r#" |
| 4482 | pub union SomeUnion { |
| 4483 | pub i: i32, |
| 4484 | pub f: f64, |
| 4485 | } |
| 4486 | |
| 4487 | const _: () = assert!(std::mem::size_of::<SomeUnion>() == 8); |
| 4488 | const _: () = assert!(std::mem::align_of::<SomeUnion>() == 8); |
| 4489 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4490 | test_format_item(test_src, "SomeUnion", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4491 | let result = result.unwrap().unwrap(); |
| 4492 | let main_api = &result.main_api; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4493 | 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] | 4494 | No support for bindings of individual fields of \ |
| 4495 | `union` (b/272801632) or `enum`"; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4496 | assert!(main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4497 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4498 | main_api.tokens, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4499 | quote! { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4500 | ... |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 4501 | union alignas(8) SomeUnion final { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4502 | public: |
Lukasz Anforowicz | 9924c43 | 2023-04-04 12:51:22 -0700 | [diff] [blame] | 4503 | __COMMENT__ "`SomeUnion` doesn't implement the `Default` trait" |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4504 | SomeUnion() = delete; |
| 4505 | |
| 4506 | // In this test there is no `Copy` implementation / derive. |
| 4507 | SomeUnion(const SomeUnion&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4508 | |
| 4509 | // All Rust types are trivially-movable. |
| 4510 | SomeUnion(SomeUnion&&) = default; |
Lukasz Anforowicz | 554ed65 | 2023-01-12 15:41:58 -0800 | [diff] [blame] | 4511 | |
| 4512 | // Assignment operators are disabled for now. |
| 4513 | SomeUnion& operator=(const SomeUnion&) = delete; |
| 4514 | SomeUnion& operator=(SomeUnion&&) = delete; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4515 | |
| 4516 | // In this test there is no custom `Drop`, so C++ can also |
| 4517 | // just use the `default` destructor. |
| 4518 | ~SomeUnion() = default; |
| 4519 | private: |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4520 | __COMMENT__ #no_fields_msg |
| 4521 | unsigned char __opaque_blob_of_bytes[8]; |
Lukasz Anforowicz | 672c515 | 2023-03-15 16:23:00 -0700 | [diff] [blame] | 4522 | inline static void __crubit_field_offset_assertions(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4523 | }; |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4524 | } |
| 4525 | ); |
| 4526 | assert_cc_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4527 | result.cc_details.tokens, |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4528 | quote! { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4529 | static_assert(sizeof(SomeUnion) == 8, ...); |
| 4530 | static_assert(alignof(SomeUnion) == 8, ...); |
| 4531 | } |
| 4532 | ); |
| 4533 | assert_rs_matches!( |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4534 | result.rs_details, |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4535 | quote! { |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 4536 | const _: () = assert!(::std::mem::size_of::<::rust_out::SomeUnion>() == 8); |
| 4537 | const _: () = assert!(::std::mem::align_of::<::rust_out::SomeUnion>() == 8); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4538 | } |
| 4539 | ); |
| 4540 | }); |
| 4541 | } |
| 4542 | |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 4543 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4544 | fn test_format_item_doc_comments_union() { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4545 | let test_src = r#" |
| 4546 | /// Doc for some union. |
| 4547 | pub union SomeUnionWithDocs { |
| 4548 | /// Doc for a field in a union. |
| 4549 | pub i: i32, |
| 4550 | pub f: f64 |
| 4551 | } |
| 4552 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4553 | test_format_item(test_src, "SomeUnionWithDocs", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4554 | let result = result.unwrap().unwrap(); |
| 4555 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4556 | let comment = " Doc for some union.\n\n\ |
| 4557 | Generated from: <crubit_unittests.rs>;l=3"; |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4558 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4559 | main_api.tokens, |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4560 | quote! { |
| 4561 | __COMMENT__ #comment |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 4562 | union ... SomeUnionWithDocs final { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4563 | ... |
| 4564 | } |
| 4565 | ... |
| 4566 | } |
| 4567 | ); |
| 4568 | }); |
| 4569 | } |
| 4570 | |
| 4571 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4572 | fn test_format_item_doc_comments_enum() { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4573 | let test_src = r#" |
| 4574 | /** Doc for some enum. */ |
| 4575 | pub enum SomeEnumWithDocs { |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4576 | Kind1(i32), |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4577 | } |
| 4578 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4579 | test_format_item(test_src, "SomeEnumWithDocs", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4580 | let result = result.unwrap().unwrap(); |
| 4581 | let main_api = &result.main_api; |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4582 | let comment = " Doc for some enum. \n\n\ |
| 4583 | Generated from: <crubit_unittests.rs>;l=3"; |
| 4584 | assert_cc_matches!( |
| 4585 | main_api.tokens, |
| 4586 | quote! { |
| 4587 | __COMMENT__ #comment |
| 4588 | struct ... SomeEnumWithDocs final { |
| 4589 | ... |
| 4590 | } |
| 4591 | ... |
| 4592 | } |
| 4593 | ); |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4594 | }); |
| 4595 | } |
| 4596 | |
| 4597 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4598 | fn test_format_item_doc_comments_struct() { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4599 | let test_src = r#" |
| 4600 | #![allow(dead_code)] |
| 4601 | #[doc = "Doc for some struct."] |
| 4602 | pub struct SomeStructWithDocs { |
Lukasz Anforowicz | cc7a76b | 2023-02-28 14:19:42 -0800 | [diff] [blame] | 4603 | #[doc = "Doc for first field."] |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4604 | some_field : i32, |
| 4605 | } |
| 4606 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4607 | test_format_item(test_src, "SomeStructWithDocs", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4608 | let result = result.unwrap().unwrap(); |
| 4609 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4610 | let comment = "Doc for some struct.\n\n\ |
| 4611 | Generated from: <crubit_unittests.rs>;l=4"; |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4612 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4613 | main_api.tokens, |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4614 | quote! { |
| 4615 | __COMMENT__ #comment |
| 4616 | struct ... SomeStructWithDocs final { |
| 4617 | ... |
| 4618 | } |
| 4619 | ... |
| 4620 | } |
| 4621 | ); |
| 4622 | }); |
| 4623 | } |
| 4624 | |
| 4625 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4626 | fn test_format_item_doc_comments_tuple_struct() { |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4627 | let test_src = r#" |
| 4628 | /// Doc for some tuple struct. |
| 4629 | pub struct SomeTupleStructWithDocs(i32); |
| 4630 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4631 | test_format_item(test_src, "SomeTupleStructWithDocs", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4632 | let result = result.unwrap().unwrap(); |
| 4633 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4634 | let comment = " Doc for some tuple struct.\n\n\ |
| 4635 | Generated from: <crubit_unittests.rs>;l=3"; |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4636 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4637 | main_api.tokens, |
Googler | 04329a1 | 2022-12-02 00:56:33 -0800 | [diff] [blame] | 4638 | quote! { |
| 4639 | __COMMENT__ #comment |
| 4640 | struct ... SomeTupleStructWithDocs final { |
| 4641 | ... |
| 4642 | } |
| 4643 | ... |
| 4644 | }, |
| 4645 | ); |
| 4646 | }); |
| 4647 | } |
| 4648 | |
| 4649 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4650 | fn test_format_item_source_loc_macro_rules() { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4651 | let test_src = r#" |
| 4652 | macro_rules! some_tuple_struct_macro_for_testing_source_loc { |
| 4653 | () => { |
| 4654 | /// Some doc on SomeTupleStructMacroForTesingSourceLoc. |
| 4655 | pub struct SomeTupleStructMacroForTesingSourceLoc(i32); |
| 4656 | }; |
| 4657 | } |
| 4658 | |
| 4659 | some_tuple_struct_macro_for_testing_source_loc!(); |
| 4660 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4661 | test_format_item(test_src, "SomeTupleStructMacroForTesingSourceLoc", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4662 | let result = result.unwrap().unwrap(); |
| 4663 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4664 | let source_loc_comment = " Some doc on SomeTupleStructMacroForTesingSourceLoc.\n\n\ |
| 4665 | Generated from: <crubit_unittests.rs>;l=5"; |
| 4666 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4667 | main_api.tokens, |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4668 | quote! { |
| 4669 | __COMMENT__ #source_loc_comment |
| 4670 | struct ... SomeTupleStructMacroForTesingSourceLoc final { |
| 4671 | ... |
| 4672 | } |
| 4673 | ... |
| 4674 | }, |
| 4675 | ); |
| 4676 | }); |
| 4677 | } |
| 4678 | |
| 4679 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4680 | fn test_format_item_source_loc_with_no_doc_comment() { |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4681 | let test_src = r#" |
| 4682 | pub struct SomeTupleStructWithNoDocComment(i32); |
| 4683 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4684 | test_format_item(test_src, "SomeTupleStructWithNoDocComment", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4685 | let result = result.unwrap().unwrap(); |
| 4686 | let main_api = &result.main_api; |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4687 | let comment = "Generated from: <crubit_unittests.rs>;l=2"; |
| 4688 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4689 | main_api.tokens, |
Googler | 47fd957 | 2023-01-20 09:57:32 -0800 | [diff] [blame] | 4690 | quote! { |
| 4691 | __COMMENT__ #comment |
| 4692 | struct ... SomeTupleStructWithNoDocComment final { |
| 4693 | ... |
| 4694 | } |
| 4695 | ... |
| 4696 | }, |
| 4697 | ); |
| 4698 | }); |
| 4699 | } |
| 4700 | |
| 4701 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4702 | fn test_format_item_unsupported_static_value() { |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 4703 | let test_src = r#" |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4704 | #[no_mangle] |
| 4705 | pub static STATIC_VALUE: i32 = 42; |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 4706 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4707 | test_format_item(test_src, "STATIC_VALUE", |result| { |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4708 | let err = result.unwrap_err(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4709 | assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: static item"); |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 4710 | }); |
| 4711 | } |
| 4712 | |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4713 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4714 | fn test_format_item_unsupported_const_value() { |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4715 | let test_src = r#" |
| 4716 | pub const CONST_VALUE: i32 = 42; |
| 4717 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4718 | test_format_item(test_src, "CONST_VALUE", |result| { |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4719 | let err = result.unwrap_err(); |
| 4720 | assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: constant item"); |
| 4721 | }); |
| 4722 | } |
| 4723 | |
| 4724 | #[test] |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4725 | fn test_format_item_unsupported_type_alias() { |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4726 | let test_src = r#" |
| 4727 | pub type TypeAlias = i32; |
| 4728 | "#; |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 4729 | test_format_item(test_src, "TypeAlias", |result| { |
Lukasz Anforowicz | 08b8ae1 | 2023-02-06 10:51:09 -0800 | [diff] [blame] | 4730 | // TODO(b/254096006): Add support for type alias definitions. |
| 4731 | let err = result.unwrap_err(); |
| 4732 | assert_eq!(err, "Unsupported rustc_hir::hir::ItemKind: type alias"); |
| 4733 | }); |
| 4734 | } |
| 4735 | |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 4736 | #[test] |
| 4737 | fn test_format_item_unsupported_impl_item_const_value() { |
| 4738 | let test_src = r#" |
| 4739 | pub struct SomeStruct(i32); |
| 4740 | |
| 4741 | impl SomeStruct { |
| 4742 | pub const CONST_VALUE: i32 = 42; |
| 4743 | } |
| 4744 | "#; |
| 4745 | test_format_item(test_src, "SomeStruct", |result| { |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 4746 | let result = result.unwrap().unwrap(); |
| 4747 | let main_api = &result.main_api; |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 4748 | assert!(!main_api.prereqs.is_empty()); |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 4749 | let unsupported_msg = "Error generating bindings for `SomeStruct::CONST_VALUE` \ |
| 4750 | defined at <crubit_unittests.rs>;l=5: \ |
Lukasz Anforowicz | 56d94fc | 2023-02-17 12:23:20 -0800 | [diff] [blame] | 4751 | Unsupported `impl` item kind: Const"; |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 4752 | assert_cc_matches!( |
Lukasz Anforowicz | 0b8a02f | 2023-02-10 15:55:40 -0800 | [diff] [blame] | 4753 | main_api.tokens, |
Lukasz Anforowicz | 1422976 | 2023-02-10 15:28:33 -0800 | [diff] [blame] | 4754 | quote! { |
| 4755 | ... |
| 4756 | struct alignas(4) SomeStruct final { |
| 4757 | ... |
| 4758 | __COMMENT__ #unsupported_msg |
| 4759 | ... |
| 4760 | }; |
| 4761 | ... |
| 4762 | } |
| 4763 | ); |
| 4764 | }); |
| 4765 | } |
| 4766 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 4767 | /// `test_format_ret_ty_for_cc_successes` provides test coverage for cases |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 4768 | /// where `format_ty_for_cc` takes `TypeLocation::FnReturn` and returns |
| 4769 | /// an `Ok(...)`. Additional testcases are covered by |
| 4770 | /// `test_format_ty_for_cc_successes`. |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4771 | #[test] |
Lukasz Anforowicz | 8a68f50 | 2022-11-15 08:43:43 -0800 | [diff] [blame] | 4772 | fn test_format_ret_ty_for_cc_successes() { |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4773 | let testcases = [ |
| 4774 | // ( <Rust type>, <expected C++ type> ) |
| 4775 | ("bool", "bool"), // TyKind::Bool |
| 4776 | ("()", "void"), |
| 4777 | // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more |
| 4778 | // details). |
| 4779 | ("!", "void"), |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 4780 | ( |
| 4781 | "extern \"C\" fn (f32, f32) -> f32", |
| 4782 | "crubit :: type_identity_t < float (float , float) > &", |
| 4783 | ), |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4784 | ]; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4785 | test_ty(&testcases, quote! {}, |desc, tcx, ty, expected| { |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 4786 | let actual = { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 4787 | let input = bindings_input_for_tests(tcx); |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 4788 | let cc_snippet = format_ty_for_cc(&input, ty, TypeLocation::FnReturn).unwrap(); |
Lukasz Anforowicz | 3744e50 | 2022-12-02 08:40:38 -0800 | [diff] [blame] | 4789 | cc_snippet.tokens.to_string() |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 4790 | }; |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4791 | let expected = expected.parse::<TokenStream>().unwrap().to_string(); |
| 4792 | assert_eq!(actual, expected, "{desc}"); |
| 4793 | }); |
| 4794 | } |
| 4795 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4796 | /// `test_format_ty_for_cc_successes` provides test coverage for cases where |
| 4797 | /// `format_ty_for_cc` returns an `Ok(...)`. |
| 4798 | /// |
| 4799 | /// Note that using `std::int8_t` (instead of `::std::int8_t`) has been an |
| 4800 | /// explicit decision. The "Google C++ Style Guide" suggests to "avoid |
| 4801 | /// nested namespaces that match well-known top-level namespaces" and "in |
| 4802 | /// particular, [...] not create any nested std namespaces.". It |
| 4803 | /// seems desirable if the generated bindings conform to this aspect of the |
| 4804 | /// style guide, because it makes things easier for *users* of these |
| 4805 | /// bindings. |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4806 | #[test] |
Lukasz Anforowicz | 8a68f50 | 2022-11-15 08:43:43 -0800 | [diff] [blame] | 4807 | fn test_format_ty_for_cc_successes() { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4808 | let testcases = [ |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4809 | // ( <Rust type>, (<expected C++ type>, |
| 4810 | // <expected #include>, |
| 4811 | // <expected prereq def>, |
| 4812 | // <expected prereq fwd decl>) ) |
| 4813 | ("bool", ("bool", "", "", "")), |
| 4814 | ("f32", ("float", "", "", "")), |
| 4815 | ("f64", ("double", "", "", "")), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 4816 | ("i8", ("std::int8_t", "<cstdint>", "", "")), |
| 4817 | ("i16", ("std::int16_t", "<cstdint>", "", "")), |
| 4818 | ("i32", ("std::int32_t", "<cstdint>", "", "")), |
| 4819 | ("i64", ("std::int64_t", "<cstdint>", "", "")), |
| 4820 | ("isize", ("std::intptr_t", "<cstdint>", "", "")), |
| 4821 | ("u8", ("std::uint8_t", "<cstdint>", "", "")), |
| 4822 | ("u16", ("std::uint16_t", "<cstdint>", "", "")), |
| 4823 | ("u32", ("std::uint32_t", "<cstdint>", "", "")), |
| 4824 | ("u64", ("std::uint64_t", "<cstdint>", "", "")), |
| 4825 | ("usize", ("std::uintptr_t", "<cstdint>", "", "")), |
Lukasz Anforowicz | ec0b64e | 2023-02-17 14:31:12 -0800 | [diff] [blame] | 4826 | ("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] | 4827 | ("SomeStruct", ("::rust_out::SomeStruct", "", "SomeStruct", "")), |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4828 | ("SomeEnum", ("::rust_out::SomeEnum", "", "SomeEnum", "")), |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4829 | ("SomeUnion", ("::rust_out::SomeUnion", "", "SomeUnion", "")), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 4830 | ("*const i32", ("const std::int32_t*", "<cstdint>", "", "")), |
| 4831 | ("*mut i32", ("std::int32_t*", "<cstdint>", "", "")), |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4832 | // `SomeStruct` is a `fwd_decls` prerequisite (not `defs` prerequisite): |
| 4833 | ("*mut SomeStruct", ("::rust_out::SomeStruct*", "", "", "SomeStruct")), |
| 4834 | // Testing propagation of deeper/nested `fwd_decls`: |
| 4835 | ("*mut *mut SomeStruct", (":: rust_out :: SomeStruct * *", "", "", "SomeStruct")), |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 4836 | ( |
| 4837 | "extern \"C\" fn (f32, f32) -> f32", |
| 4838 | ( |
| 4839 | "crubit :: type_identity_t < float (float , float) > *", |
| 4840 | "\"crubit/support/for/tests/internal/cxx20_backports.h\"", |
| 4841 | "", |
| 4842 | "", |
| 4843 | ), |
| 4844 | ), |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4845 | // Extra parens/sugar are expected to be ignored: |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4846 | ("(bool)", ("bool", "", "", "")), |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4847 | ]; |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 4848 | let preamble = quote! { |
| 4849 | #![allow(unused_parens)] |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4850 | |
| 4851 | pub struct SomeStruct { |
| 4852 | pub x: i32, |
| 4853 | pub y: i32, |
| 4854 | } |
Lukasz Anforowicz | 32ad956 | 2023-03-10 14:48:25 -0800 | [diff] [blame] | 4855 | pub enum SomeEnum { |
| 4856 | Cartesian{x: f64, y: f64}, |
| 4857 | Polar{angle: f64, dist: f64}, |
| 4858 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4859 | pub union SomeUnion { |
| 4860 | pub x: i32, |
| 4861 | pub y: i32, |
| 4862 | } |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 4863 | }; |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 4864 | test_ty( |
| 4865 | &testcases, |
| 4866 | preamble, |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4867 | |desc, tcx, ty, |
| 4868 | (expected_tokens, expected_include, expected_prereq_def, expected_prereq_fwd_decl)| { |
| 4869 | let (actual_tokens, actual_prereqs) = { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 4870 | let input = bindings_input_for_tests(tcx); |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 4871 | let s = format_ty_for_cc(&input, ty, TypeLocation::Other).unwrap(); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4872 | (s.tokens.to_string(), s.prereqs) |
| 4873 | }; |
| 4874 | let (actual_includes, actual_prereq_defs, actual_prereq_fwd_decls) = |
| 4875 | (actual_prereqs.includes, actual_prereqs.defs, actual_prereqs.fwd_decls); |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 4876 | |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 4877 | let expected_tokens = expected_tokens.parse::<TokenStream>().unwrap().to_string(); |
| 4878 | assert_eq!(actual_tokens, expected_tokens, "{desc}"); |
Lukasz Anforowicz | ed17d05 | 2022-11-02 12:07:28 -0700 | [diff] [blame] | 4879 | |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4880 | if expected_include.is_empty() { |
| 4881 | assert!(actual_includes.is_empty()); |
| 4882 | } else { |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 4883 | let expected_include: TokenStream = expected_include.parse().unwrap(); |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4884 | assert_cc_matches!( |
| 4885 | format_cc_includes(&actual_includes), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 4886 | quote! { __HASH_TOKEN__ include #expected_include } |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4887 | ); |
| 4888 | } |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 4889 | |
| 4890 | if expected_prereq_def.is_empty() { |
| 4891 | assert!(actual_prereq_defs.is_empty()); |
| 4892 | } else { |
| 4893 | let expected_def_id = find_def_id_by_name(tcx, expected_prereq_def); |
| 4894 | assert_eq!(1, actual_prereq_defs.len()); |
| 4895 | assert_eq!(expected_def_id, actual_prereq_defs.into_iter().next().unwrap()); |
| 4896 | } |
Lukasz Anforowicz | f4501b4 | 2023-01-07 09:41:29 -0800 | [diff] [blame] | 4897 | |
| 4898 | if expected_prereq_fwd_decl.is_empty() { |
| 4899 | assert!(actual_prereq_fwd_decls.is_empty()); |
| 4900 | } else { |
| 4901 | let expected_def_id = find_def_id_by_name(tcx, expected_prereq_fwd_decl); |
| 4902 | assert_eq!(1, actual_prereq_fwd_decls.len()); |
| 4903 | assert_eq!(expected_def_id, |
| 4904 | actual_prereq_fwd_decls.into_iter().next().unwrap()); |
| 4905 | } |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 4906 | }, |
| 4907 | ); |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4908 | } |
| 4909 | |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4910 | /// `test_format_ty_for_cc_failures` provides test coverage for cases where |
| 4911 | /// `format_ty_for_cc` returns an `Err(...)`. |
| 4912 | /// |
| 4913 | /// It seems okay to have no test coverage for now for the following types |
| 4914 | /// (which should never be encountered when generating bindings and where |
| 4915 | /// `format_ty_for_cc` should panic): |
| 4916 | /// - TyKind::Closure |
| 4917 | /// - TyKind::Error |
| 4918 | /// - TyKind::FnDef |
| 4919 | /// - TyKind::Infer |
| 4920 | /// |
Lukasz Anforowicz | 0182c5c | 2022-12-29 10:08:50 -0800 | [diff] [blame] | 4921 | /// TODO(lukasza): Add test coverage (here and in the "for_rs" flavours) |
| 4922 | /// for: |
Lukasz Anforowicz | 4c19ad9 | 2022-12-16 15:23:14 -0800 | [diff] [blame] | 4923 | /// - TyKind::Bound |
| 4924 | /// - TyKind::Dynamic (`dyn Eq`) |
| 4925 | /// - TyKind::Foreign (`extern type T`) |
| 4926 | /// - https://doc.rust-lang.org/beta/unstable-book/language-features/generators.html: |
| 4927 | /// TyKind::Generator, TyKind::GeneratorWitness |
| 4928 | /// - TyKind::Param |
| 4929 | /// - TyKind::Placeholder |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4930 | #[test] |
Lukasz Anforowicz | 8a68f50 | 2022-11-15 08:43:43 -0800 | [diff] [blame] | 4931 | fn test_format_ty_for_cc_failures() { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4932 | let testcases = [ |
| 4933 | // ( <Rust type>, <expected error message> ) |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4934 | ( |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4935 | "()", // Empty TyKind::Tuple |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 4936 | "`()` / `void` is only supported as a return type (b/254507801)", |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4937 | ), |
| 4938 | ( |
| 4939 | // TODO(b/254507801): Expect `crubit::Never` instead (see the bug for more |
| 4940 | // details). |
| 4941 | "!", // TyKind::Never |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 4942 | "The never type `!` is only supported as a return type (b/254507801)", |
Lukasz Anforowicz | 903fc63 | 2022-10-25 08:55:33 -0700 | [diff] [blame] | 4943 | ), |
| 4944 | ( |
| 4945 | "(i32, i32)", // Non-empty TyKind::Tuple |
Lukasz Anforowicz | 13794df | 2022-10-21 07:56:34 -0700 | [diff] [blame] | 4946 | "Tuples are not supported yet: (i32, i32) (b/254099023)", |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4947 | ), |
| 4948 | ( |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4949 | "&'static i32", // TyKind::Ref |
| 4950 | "The following Rust type is not supported yet: &'static i32", |
| 4951 | ), |
| 4952 | ( |
| 4953 | "[i32; 42]", // TyKind::Array |
| 4954 | "The following Rust type is not supported yet: [i32; 42]", |
| 4955 | ), |
| 4956 | ( |
| 4957 | "&'static [i32]", // TyKind::Slice (nested underneath TyKind::Ref) |
| 4958 | "The following Rust type is not supported yet: &'static [i32]", |
| 4959 | ), |
| 4960 | ( |
| 4961 | "&'static str", // TyKind::Str (nested underneath TyKind::Ref) |
| 4962 | "The following Rust type is not supported yet: &'static str", |
| 4963 | ), |
| 4964 | ( |
Lukasz Anforowicz | 0182c5c | 2022-12-29 10:08:50 -0800 | [diff] [blame] | 4965 | "impl Eq", // TyKind::Alias |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4966 | "The following Rust type is not supported yet: impl std::cmp::Eq", |
| 4967 | ), |
| 4968 | ( |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 4969 | "fn(i32) -> i32", // TyKind::FnPtr (default ABI = "Rust") |
| 4970 | "Function pointers can't have a thunk: \ |
| 4971 | Calling convention other than `extern \"C\"` requires a thunk", |
| 4972 | ), |
| 4973 | ( |
| 4974 | "extern \"C\" fn (SomeStruct, f32) -> f32", |
| 4975 | "Function pointers can't have a thunk: Type of parameter #0 requires a thunk", |
| 4976 | ), |
| 4977 | ( |
| 4978 | "extern \"C\" fn (f32, f32) -> SomeStruct", |
| 4979 | "Function pointers can't have a thunk: Return type requires a thunk", |
| 4980 | ), |
| 4981 | ( |
| 4982 | "unsafe fn(i32) -> i32", |
| 4983 | "Bindings for `unsafe` functions are not fully designed yet (b/254095482)", |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 4984 | ), |
| 4985 | // TODO(b/254094650): Consider mapping this to Clang's (and GCC's) `__int128` |
| 4986 | // or to `absl::in128`. |
| 4987 | ("i128", "C++ doesn't have a standard equivalent of `i128` (b/254094650)"), |
| 4988 | ("u128", "C++ doesn't have a standard equivalent of `u128` (b/254094650)"), |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4989 | ( |
| 4990 | "StructWithCustomDrop", |
| 4991 | "Failed to generate bindings for the definition of `StructWithCustomDrop`: \ |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 4992 | `Drop` trait and \"drop glue\" are not supported yet (b/258251148)", |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4993 | ), |
Devin Jeanpierre | 81aec50 | 2023-04-04 15:33:39 -0700 | [diff] [blame] | 4994 | ("ConstGenericStruct<42>", "Generic types are not supported yet (b/259749095)"), |
| 4995 | ("TypeGenericStruct<u8>", "Generic types are not supported yet (b/259749095)"), |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 4996 | ( |
| 4997 | // This double-checks that TyKind::Adt(..., substs) are present |
| 4998 | // even if the type parameter argument is not explicitly specified |
| 4999 | // (here it comes from the default: `...Struct<T = u8>`). |
| 5000 | "TypeGenericStruct", |
| 5001 | "Generic types are not supported yet (b/259749095)", |
| 5002 | ), |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 5003 | ("LifetimeGenericStruct<'static>", "Generic types are not supported yet (b/259749095)"), |
Lukasz Anforowicz | cfe8455 | 2023-04-20 13:38:54 -0700 | [diff] [blame] | 5004 | ( |
| 5005 | "std::cmp::Ordering", |
| 5006 | "Type `std::cmp::Ordering` comes from the `core` crate, \ |
| 5007 | but no `--other-crate-bindings` were specified for this crate", |
| 5008 | ), |
Googler | 7a77a80 | 2023-04-21 08:32:50 -0700 | [diff] [blame^] | 5009 | ("Option<i8>", "Generic types are not supported yet (b/259749095)"), |
Lukasz Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 5010 | ( |
| 5011 | "PublicReexportOfStruct", |
| 5012 | "Not directly public type (re-exports are not supported yet - b/262052635)", |
| 5013 | ), |
| 5014 | ( |
| 5015 | // This testcase is like `PublicReexportOfStruct`, but the private type and the |
| 5016 | // re-export are in another crate. When authoring this test |
| 5017 | // `core::alloc::LayoutError` was a public re-export of |
| 5018 | // `core::alloc::layout::LayoutError`: |
| 5019 | // `https://play.rust-lang.org/?version=stable&mode=debug&edition=2021&gist=d2b5528af9b33b25abe44cc4646d65e3` |
| 5020 | // TODO(b/258261328): Once cross-crate bindings are supported we should try |
| 5021 | // to test them via a test crate that we control (rather than testing via |
| 5022 | // implementation details of the std crate). |
| 5023 | "core::alloc::LayoutError", |
| 5024 | "Not directly public type (re-exports are not supported yet - b/262052635)", |
| 5025 | ), |
Lukasz Anforowicz | 2eb2429 | 2023-03-14 09:38:21 -0700 | [diff] [blame] | 5026 | ( |
| 5027 | "*const Option<i8>", |
| 5028 | "Failed to format the pointee of the pointer type `std::option::Option<i8>`: \ |
| 5029 | Generic types are not supported yet (b/259749095)", |
| 5030 | ), |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5031 | ]; |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5032 | let preamble = quote! { |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5033 | #![feature(never_type)] |
| 5034 | |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 5035 | pub struct SomeStruct { |
| 5036 | pub x: i32, |
| 5037 | pub y: i32, |
| 5038 | } |
| 5039 | |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5040 | pub struct StructWithCustomDrop { |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5041 | pub x: i32, |
| 5042 | pub y: i32, |
| 5043 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5044 | |
| 5045 | impl Drop for StructWithCustomDrop { |
| 5046 | fn drop(&mut self) {} |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5047 | } |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5048 | |
| 5049 | pub struct ConstGenericStruct<const N: usize> { |
| 5050 | pub arr: [u8; N], |
| 5051 | } |
| 5052 | |
| 5053 | pub struct TypeGenericStruct<T = u8> { |
| 5054 | pub t: T, |
| 5055 | } |
| 5056 | |
| 5057 | pub struct LifetimeGenericStruct<'a> { |
| 5058 | pub reference: &'a u8, |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5059 | } |
Lukasz Anforowicz | f36762a | 2023-03-02 18:43:07 -0800 | [diff] [blame] | 5060 | |
| 5061 | mod private_submodule { |
| 5062 | pub struct PublicStructInPrivateModule; |
| 5063 | } |
| 5064 | pub use private_submodule::PublicStructInPrivateModule |
| 5065 | as PublicReexportOfStruct; |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5066 | }; |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5067 | test_ty(&testcases, preamble, |desc, tcx, ty, expected_msg| { |
| 5068 | let input = bindings_input_for_tests(tcx); |
Lukasz Anforowicz | 8574c9d | 2023-04-13 15:11:20 -0700 | [diff] [blame] | 5069 | 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] | 5070 | let actual_msg = format!("{anyhow_err:#}"); |
| 5071 | assert_eq!(&actual_msg, *expected_msg, "{desc}"); |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5072 | }); |
| 5073 | } |
| 5074 | |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5075 | #[test] |
| 5076 | fn test_format_ty_for_rs_successes() { |
| 5077 | // Test coverage for cases where `format_ty_for_rs` returns an `Ok(...)`. |
| 5078 | let testcases = [ |
| 5079 | // ( <Rust type>, <expected Rust spelling for ..._cc_api_impl.rs> ) |
| 5080 | ("bool", "bool"), |
| 5081 | ("f32", "f32"), |
| 5082 | ("f64", "f64"), |
| 5083 | ("i8", "i8"), |
| 5084 | ("i16", "i16"), |
| 5085 | ("i32", "i32"), |
| 5086 | ("i64", "i64"), |
| 5087 | ("i128", "i128"), |
| 5088 | ("isize", "isize"), |
| 5089 | ("u8", "u8"), |
| 5090 | ("u16", "u16"), |
| 5091 | ("u32", "u32"), |
| 5092 | ("u64", "u64"), |
| 5093 | ("u128", "u128"), |
| 5094 | ("usize", "usize"), |
| 5095 | ("char", "char"), |
| 5096 | ("!", "!"), |
| 5097 | ("()", "()"), |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 5098 | // ADTs: |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 5099 | ("SomeStruct", "::rust_out::SomeStruct"), |
| 5100 | ("SomeEnum", "::rust_out::SomeEnum"), |
| 5101 | ("SomeUnion", "::rust_out::SomeUnion"), |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 5102 | // Type from another crate: |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 5103 | ("std::cmp::Ordering", "::core::cmp::Ordering"), |
Lukasz Anforowicz | eb58a49 | 2023-01-07 08:25:48 -0800 | [diff] [blame] | 5104 | // `const` and `mut` pointers: |
| 5105 | ("*const i32", "*const i32"), |
| 5106 | ("*mut i32", "*mut i32"), |
| 5107 | // Pointer to an ADT: |
| 5108 | ("*mut SomeStruct", "* mut :: rust_out :: SomeStruct"), |
Lukasz Anforowicz | 5c1b3ad | 2023-04-13 17:05:00 -0700 | [diff] [blame] | 5109 | ("extern \"C\" fn(i32) -> i32", "extern \"C\" fn(i32) -> i32"), |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5110 | ]; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5111 | let preamble = quote! { |
| 5112 | #![feature(never_type)] |
| 5113 | |
| 5114 | pub struct SomeStruct { |
| 5115 | pub x: i32, |
| 5116 | pub y: i32, |
| 5117 | } |
| 5118 | pub enum SomeEnum { |
| 5119 | Cartesian{x: f64, y: f64}, |
| 5120 | Polar{angle: f64, dist: f64}, |
| 5121 | } |
| 5122 | pub union SomeUnion { |
| 5123 | pub x: i32, |
| 5124 | pub y: i32, |
| 5125 | } |
| 5126 | }; |
Lukasz Anforowicz | 5c9c00d | 2022-12-02 08:41:39 -0800 | [diff] [blame] | 5127 | test_ty(&testcases, preamble, |desc, tcx, ty, expected_tokens| { |
| 5128 | let actual_tokens = format_ty_for_rs(tcx, ty).unwrap().to_string(); |
| 5129 | let expected_tokens = expected_tokens.parse::<TokenStream>().unwrap().to_string(); |
| 5130 | assert_eq!(actual_tokens, expected_tokens, "{desc}"); |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5131 | }); |
| 5132 | } |
| 5133 | |
| 5134 | #[test] |
| 5135 | fn test_format_ty_for_rs_failures() { |
| 5136 | // This test provides coverage for cases where `format_ty_for_rs` returns an |
| 5137 | // `Err(...)`. |
| 5138 | let testcases = [ |
| 5139 | // ( <Rust type>, <expected error message> ) |
| 5140 | ( |
| 5141 | "(i32, i32)", // Non-empty TyKind::Tuple |
| 5142 | "Tuples are not supported yet: (i32, i32) (b/254099023)", |
| 5143 | ), |
| 5144 | ( |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5145 | "&'static i32", // TyKind::Ref |
| 5146 | "The following Rust type is not supported yet: &'static i32", |
| 5147 | ), |
| 5148 | ( |
| 5149 | "[i32; 42]", // TyKind::Array |
| 5150 | "The following Rust type is not supported yet: [i32; 42]", |
| 5151 | ), |
| 5152 | ( |
| 5153 | "&'static [i32]", // TyKind::Slice (nested underneath TyKind::Ref) |
| 5154 | "The following Rust type is not supported yet: &'static [i32]", |
| 5155 | ), |
| 5156 | ( |
| 5157 | "&'static str", // TyKind::Str (nested underneath TyKind::Ref) |
| 5158 | "The following Rust type is not supported yet: &'static str", |
| 5159 | ), |
| 5160 | ( |
Lukasz Anforowicz | 0182c5c | 2022-12-29 10:08:50 -0800 | [diff] [blame] | 5161 | "impl Eq", // TyKind::Alias |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5162 | "The following Rust type is not supported yet: impl std::cmp::Eq", |
| 5163 | ), |
| 5164 | ( |
Lukasz Anforowicz | 8b475a4 | 2022-11-29 09:33:02 -0800 | [diff] [blame] | 5165 | "Option<i8>", // TyKind::Adt - generic + different crate |
| 5166 | "Generic types are not supported yet (b/259749095)", |
| 5167 | ), |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5168 | ]; |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5169 | let preamble = quote! {}; |
| 5170 | test_ty(&testcases, preamble, |desc, tcx, ty, expected_err| { |
| 5171 | let anyhow_err = format_ty_for_rs(tcx, ty).unwrap_err(); |
Lukasz Anforowicz | 7860d0e | 2022-11-15 08:47:56 -0800 | [diff] [blame] | 5172 | let actual_err = format!("{anyhow_err:#}"); |
| 5173 | assert_eq!(&actual_err, *expected_err, "{desc}"); |
| 5174 | }); |
| 5175 | } |
| 5176 | |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5177 | fn test_ty<TestFn, Expectation>( |
| 5178 | testcases: &[(&str, Expectation)], |
| 5179 | preamble: TokenStream, |
| 5180 | test_fn: TestFn, |
| 5181 | ) where |
Lukasz Anforowicz | d16b6bf | 2022-11-22 18:35:08 -0800 | [diff] [blame] | 5182 | TestFn: for<'tcx> Fn( |
| 5183 | /* testcase_description: */ &str, |
| 5184 | TyCtxt<'tcx>, |
| 5185 | Ty<'tcx>, |
| 5186 | &Expectation, |
| 5187 | ) + Sync, |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5188 | Expectation: Sync, |
| 5189 | { |
Lukasz Anforowicz | d0f0a84 | 2022-11-03 12:40:13 -0700 | [diff] [blame] | 5190 | for (index, (input, expected)) in testcases.iter().enumerate() { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5191 | let desc = format!("test #{index}: test input: `{input}`"); |
| 5192 | let input = { |
| 5193 | let ty_tokens: TokenStream = input.parse().unwrap(); |
| 5194 | let input = quote! { |
Lukasz Anforowicz | 4047272 | 2022-11-08 13:29:08 -0800 | [diff] [blame] | 5195 | #preamble |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5196 | pub fn test_function() -> #ty_tokens { panic!("") } |
| 5197 | }; |
| 5198 | input.to_string() |
| 5199 | }; |
Lukasz Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 5200 | run_compiler_for_testing(input, |tcx| { |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5201 | let def_id = find_def_id_by_name(tcx, "test_function"); |
Lukasz Anforowicz | 087dff7 | 2023-02-17 12:13:32 -0800 | [diff] [blame] | 5202 | let ty = tcx |
| 5203 | .fn_sig(def_id.to_def_id()) |
| 5204 | .subst_identity() |
| 5205 | .no_bound_vars() |
| 5206 | .unwrap() |
| 5207 | .output(); |
Lukasz Anforowicz | 7589483 | 2022-11-22 18:25:28 -0800 | [diff] [blame] | 5208 | test_fn(&desc, tcx, ty, expected); |
Lukasz Anforowicz | 4f9f552 | 2022-10-17 15:03:37 -0700 | [diff] [blame] | 5209 | }); |
| 5210 | } |
| 5211 | } |
| 5212 | |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 5213 | /// Tests invoking `format_item` on the item with the specified `name` from |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5214 | /// the given Rust `source`. Returns the result of calling |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 5215 | /// `test_function` with `format_item`'s result as an argument. |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5216 | /// (`test_function` should typically `assert!` that it got the expected |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 5217 | /// result from `format_item`.) |
| 5218 | 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] | 5219 | where |
Lukasz Anforowicz | 73a2c0d | 2023-03-22 14:07:00 -0700 | [diff] [blame] | 5220 | F: FnOnce(Result<Option<ApiSnippets>, String>) -> T + Send, |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5221 | T: Send, |
| 5222 | { |
Lukasz Anforowicz | 0bef264 | 2023-01-05 09:20:31 -0800 | [diff] [blame] | 5223 | run_compiler_for_testing(source, |tcx| { |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5224 | let def_id = find_def_id_by_name(tcx, name); |
Lukasz Anforowicz | e465b9e | 2023-02-06 15:00:25 -0800 | [diff] [blame] | 5225 | let result = format_item(&bindings_input_for_tests(tcx), def_id); |
Lukasz Anforowicz | e433306 | 2022-10-17 14:47:53 -0700 | [diff] [blame] | 5226 | |
| 5227 | // https://docs.rs/anyhow/latest/anyhow/struct.Error.html#display-representations says: |
| 5228 | // To print causes as well [...], use the alternate selector “{:#}”. |
| 5229 | let result = result.map_err(|anyhow_err| format!("{anyhow_err:#}")); |
| 5230 | |
| 5231 | test_function(result) |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5232 | }) |
| 5233 | } |
| 5234 | |
| 5235 | /// Finds the definition id of a Rust item with the specified `name`. |
| 5236 | /// Panics if no such item is found, or if there is more than one match. |
| 5237 | fn find_def_id_by_name(tcx: TyCtxt, name: &str) -> LocalDefId { |
| 5238 | let hir_items = || tcx.hir().items().map(|item_id| tcx.hir().item(item_id)); |
| 5239 | let items_with_matching_name = |
| 5240 | hir_items().filter(|item| item.ident.name.as_str() == name).collect_vec(); |
Lukasz Anforowicz | d0f0a84 | 2022-11-03 12:40:13 -0700 | [diff] [blame] | 5241 | match *items_with_matching_name.as_slice() { |
| 5242 | [] => { |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5243 | let found_names = hir_items() |
| 5244 | .map(|item| item.ident.name.as_str()) |
| 5245 | .filter(|s| !s.is_empty()) |
| 5246 | .sorted() |
| 5247 | .dedup() |
| 5248 | .map(|name| format!("`{name}`")) |
Lukasz Anforowicz | d0f0a84 | 2022-11-03 12:40:13 -0700 | [diff] [blame] | 5249 | .join(",\n"); |
| 5250 | panic!("No items named `{name}`.\nInstead found:\n{found_names}"); |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5251 | } |
Lukasz Anforowicz | 61cb1e3 | 2022-11-04 09:08:35 -0700 | [diff] [blame] | 5252 | [item] => item.owner_id.def_id, |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5253 | _ => panic!("More than one item named `{name}`"), |
| 5254 | } |
| 5255 | } |
| 5256 | |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5257 | fn bindings_input_for_tests(tcx: TyCtxt) -> Input { |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 5258 | Input { |
| 5259 | tcx, |
| 5260 | crubit_support_path: "crubit/support/for/tests".into(), |
Lukasz Anforowicz | cfe8455 | 2023-04-20 13:38:54 -0700 | [diff] [blame] | 5261 | crate_name_to_include_path: Default::default(), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 5262 | _features: (), |
Lukasz Anforowicz | a782bda | 2023-01-17 14:04:50 -0800 | [diff] [blame] | 5263 | } |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5264 | } |
| 5265 | |
| 5266 | /// Tests invoking `generate_bindings` on the given Rust `source`. |
Lukasz Anforowicz | 8a83341 | 2022-10-14 14:44:13 -0700 | [diff] [blame] | 5267 | /// Returns the result of calling `test_function` with the generated |
| 5268 | /// bindings as an argument. (`test_function` should typically `assert!` |
| 5269 | /// that it got the expected `GeneratedBindings`.) |
| 5270 | fn test_generated_bindings<F, T>(source: &str, test_function: F) -> T |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 5271 | where |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5272 | F: FnOnce(Result<Output>) -> T + Send, |
Lukasz Anforowicz | 581fd75 | 2022-09-21 11:30:15 -0700 | [diff] [blame] | 5273 | T: Send, |
| 5274 | { |
Lukasz Anforowicz | be25c76 | 2023-01-17 12:43:52 -0800 | [diff] [blame] | 5275 | run_compiler_for_testing(source, |tcx| { |
| 5276 | test_function(generate_bindings(&bindings_input_for_tests(tcx))) |
| 5277 | }) |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 5278 | } |
Lukasz Anforowicz | bda1cfe | 2022-09-20 06:25:43 -0700 | [diff] [blame] | 5279 | } |