2020-03-31 13:44:52 +00:00
|
|
|
use crate::abi::{self, Abi, Align, FieldsShape, Size};
|
2021-08-25 18:15:09 +03:00
|
|
|
use crate::abi::{HasDataLayout, TyAbiInterface, TyAndLayout};
|
2021-01-26 11:09:06 +01:00
|
|
|
use crate::spec::{self, HasTargetSpec};
|
2022-02-01 18:44:45 +01:00
|
|
|
use rustc_span::Symbol;
|
2023-01-23 16:02:00 +00:00
|
|
|
use std::fmt;
|
2022-11-05 14:36:38 +05:30
|
|
|
use std::str::FromStr;
|
2018-04-25 16:45:29 +03:00
|
|
|
|
|
|
|
mod aarch64;
|
2018-07-18 22:01:19 -05:00
|
|
|
mod amdgpu;
|
2018-04-25 16:45:29 +03:00
|
|
|
mod arm;
|
2016-05-06 09:32:10 -04:00
|
|
|
mod avr;
|
2020-11-30 19:41:57 +00:00
|
|
|
mod bpf;
|
2023-07-13 22:19:59 +08:00
|
|
|
mod csky;
|
2018-04-25 16:45:29 +03:00
|
|
|
mod hexagon;
|
2022-09-17 18:00:34 +08:00
|
|
|
mod loongarch;
|
2021-08-25 07:45:06 +00:00
|
|
|
mod m68k;
|
2018-04-25 16:45:29 +03:00
|
|
|
mod mips;
|
|
|
|
mod mips64;
|
|
|
|
mod msp430;
|
|
|
|
mod nvptx64;
|
|
|
|
mod powerpc;
|
|
|
|
mod powerpc64;
|
2018-07-24 12:04:17 +02:00
|
|
|
mod riscv;
|
2018-04-25 16:45:29 +03:00
|
|
|
mod s390x;
|
|
|
|
mod sparc;
|
|
|
|
mod sparc64;
|
rustc: Add a new `wasm` ABI
This commit implements the idea of a new ABI for the WebAssembly target,
one called `"wasm"`. This ABI is entirely of my own invention
and has no current precedent, but I think that the addition of this ABI
might help solve a number of issues with the WebAssembly targets.
When `wasm32-unknown-unknown` was first added to Rust I naively
"implemented an abi" for the target. I then went to write `wasm-bindgen`
which accidentally relied on details of this ABI. Turns out the ABI
definition didn't match C, which is causing issues for C/Rust interop.
Currently the compiler has a "wasm32 bindgen compat" ABI which is the
original implementation I added, and it's purely there for, well,
`wasm-bindgen`.
Another issue with the WebAssembly target is that it's not clear to me
when and if the default C ABI will change to account for WebAssembly's
multi-value feature (a feature that allows functions to return multiple
values). Even if this does happen, though, it seems like the C ABI will
be guided based on the performance of WebAssembly code and will likely
not match even what the current wasm-bindgen-compat ABI is today. This
leaves a hole in Rust's expressivity in binding WebAssembly where given
a particular import type, Rust may not be able to import that signature
with an updated C ABI for multi-value.
To fix these issues I had the idea of a new ABI for WebAssembly, one
called `wasm`. The definition of this ABI is "what you write
maps straight to wasm". The goal here is that whatever you write down in
the parameter list or in the return values goes straight into the
function's signature in the WebAssembly file. This special ABI is for
intentionally matching the ABI of an imported function from the
environment or exporting a function with the right signature.
With the addition of a new ABI, this enables rustc to:
* Eventually remove the "wasm-bindgen compat hack". Once this
ABI is stable wasm-bindgen can switch to using it everywhere.
Afterwards the wasm32-unknown-unknown target can have its default ABI
updated to match C.
* Expose the ability to precisely match an ABI signature for a
WebAssembly function, regardless of what the C ABI that clang chooses
turns out to be.
* Continue to evolve the definition of the default C ABI to match what
clang does on all targets, since the purpose of that ABI will be
explicitly matching C rather than generating particular function
imports/exports.
Naturally this is implemented as an unstable feature initially, but it
would be nice for this to get stabilized (if it works) in the near-ish
future to remove the wasm32-unknown-unknown incompatibility with the C
ABI. Doing this, however, requires the feature to be on stable because
wasm-bindgen works with stable Rust.
2021-04-01 16:08:29 -07:00
|
|
|
mod wasm;
|
2018-04-25 16:45:29 +03:00
|
|
|
mod x86;
|
|
|
|
mod x86_64;
|
|
|
|
mod x86_win64;
|
|
|
|
|
2023-05-03 22:22:24 +00:00
|
|
|
#[derive(Clone, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
2018-04-25 16:45:29 +03:00
|
|
|
pub enum PassMode {
|
2019-08-10 14:38:17 +03:00
|
|
|
/// Ignore the argument.
|
2021-01-23 12:57:35 +01:00
|
|
|
///
|
|
|
|
/// The argument is either uninhabited or a ZST.
|
2019-08-10 14:38:17 +03:00
|
|
|
Ignore,
|
2018-04-25 16:45:29 +03:00
|
|
|
/// Pass the argument directly.
|
2021-01-23 12:57:35 +01:00
|
|
|
///
|
2023-09-07 16:04:22 +02:00
|
|
|
/// The argument has a layout abi of `Scalar` or `Vector`.
|
|
|
|
/// Unfortunately due to past mistakes, in rare cases on wasm, it can also be `Aggregate`.
|
|
|
|
/// This is bad since it leaks LLVM implementation details into the ABI.
|
|
|
|
/// (Also see <https://github.com/rust-lang/rust/issues/115666>.)
|
2018-04-25 16:45:29 +03:00
|
|
|
Direct(ArgAttributes),
|
|
|
|
/// Pass a pair's elements directly in two arguments.
|
2021-01-23 12:57:35 +01:00
|
|
|
///
|
|
|
|
/// The argument has a layout abi of `ScalarPair`.
|
2018-04-25 16:45:29 +03:00
|
|
|
Pair(ArgAttributes, ArgAttributes),
|
2023-09-07 22:06:37 +02:00
|
|
|
/// Pass the argument after casting it. See the `CastTarget` docs for details. The bool
|
|
|
|
/// indicates if a `Reg::i32()` dummy argument is emitted before the real argument.
|
|
|
|
Cast { pad_i32: bool, cast: Box<CastTarget> },
|
2018-04-25 16:45:29 +03:00
|
|
|
/// Pass the argument indirectly via a hidden pointer.
|
2023-09-08 08:48:41 +02:00
|
|
|
/// The `meta_attrs` value, if any, is for the metadata (vtable or length) of an unsized
|
|
|
|
/// argument. (This is the only mode that supports unsized arguments.)
|
|
|
|
/// `on_stack` defines that the value should be passed at a fixed stack offset in accordance to
|
|
|
|
/// the ABI rather than passed using a pointer. This corresponds to the `byval` LLVM argument
|
|
|
|
/// attribute (using the Rust type of this argument). `on_stack` cannot be true for unsized
|
|
|
|
/// arguments, i.e., when `meta_attrs` is `Some`.
|
|
|
|
Indirect { attrs: ArgAttributes, meta_attrs: Option<ArgAttributes>, on_stack: bool },
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2023-09-02 11:59:31 +02:00
|
|
|
impl PassMode {
|
|
|
|
/// Checks if these two `PassMode` are equal enough to be considered "the same for all
|
2023-09-07 16:48:02 +02:00
|
|
|
/// function call ABIs". However, the `Layout` can also impact ABI decisions,
|
|
|
|
/// so that needs to be compared as well!
|
2023-09-02 11:59:31 +02:00
|
|
|
pub fn eq_abi(&self, other: &Self) -> bool {
|
|
|
|
match (self, other) {
|
2023-09-07 22:06:37 +02:00
|
|
|
(PassMode::Ignore, PassMode::Ignore) => true,
|
2023-09-02 11:59:31 +02:00
|
|
|
(PassMode::Direct(a1), PassMode::Direct(a2)) => a1.eq_abi(a2),
|
|
|
|
(PassMode::Pair(a1, b1), PassMode::Pair(a2, b2)) => a1.eq_abi(a2) && b1.eq_abi(b2),
|
|
|
|
(
|
2023-09-07 22:06:37 +02:00
|
|
|
PassMode::Cast { cast: c1, pad_i32: pad1 },
|
|
|
|
PassMode::Cast { cast: c2, pad_i32: pad2 },
|
|
|
|
) => c1.eq_abi(c2) && pad1 == pad2,
|
2023-09-02 11:59:31 +02:00
|
|
|
(
|
2023-09-08 08:48:41 +02:00
|
|
|
PassMode::Indirect { attrs: a1, meta_attrs: None, on_stack: s1 },
|
|
|
|
PassMode::Indirect { attrs: a2, meta_attrs: None, on_stack: s2 },
|
2023-09-02 11:59:31 +02:00
|
|
|
) => a1.eq_abi(a2) && s1 == s2,
|
|
|
|
(
|
2023-09-08 08:48:41 +02:00
|
|
|
PassMode::Indirect { attrs: a1, meta_attrs: Some(e1), on_stack: s1 },
|
|
|
|
PassMode::Indirect { attrs: a2, meta_attrs: Some(e2), on_stack: s2 },
|
2023-09-02 11:59:31 +02:00
|
|
|
) => a1.eq_abi(a2) && e1.eq_abi(e2) && s1 == s2,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-25 16:45:29 +03:00
|
|
|
// Hack to disable non_upper_case_globals only for the bitflags! and not for the rest
|
|
|
|
// of this module
|
2019-02-08 21:00:07 +09:00
|
|
|
pub use attr_impl::ArgAttribute;
|
2018-04-25 16:45:29 +03:00
|
|
|
|
|
|
|
#[allow(non_upper_case_globals)]
|
|
|
|
#[allow(unused)]
|
|
|
|
mod attr_impl {
|
|
|
|
// The subset of llvm::Attribute needed for arguments, packed into a bitfield.
|
2019-02-08 21:00:07 +09:00
|
|
|
bitflags::bitflags! {
|
2021-08-26 21:58:34 +03:00
|
|
|
#[derive(Default, HashStable_Generic)]
|
2023-04-30 19:08:46 +02:00
|
|
|
pub struct ArgAttribute: u8 {
|
2018-04-25 16:45:29 +03:00
|
|
|
const NoAlias = 1 << 1;
|
|
|
|
const NoCapture = 1 << 2;
|
|
|
|
const NonNull = 1 << 3;
|
|
|
|
const ReadOnly = 1 << 4;
|
2021-03-18 21:50:28 +01:00
|
|
|
const InReg = 1 << 5;
|
2023-01-04 19:24:42 -05:00
|
|
|
const NoUndef = 1 << 6;
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-14 12:27:57 +01:00
|
|
|
/// Sometimes an ABI requires small integers to be extended to a full or partial register. This enum
|
2021-04-19 15:57:08 +03:00
|
|
|
/// defines if this extension should be zero-extension or sign-extension when necessary. When it is
|
|
|
|
/// not necessary to extend the argument, this enum is ignored.
|
2021-08-26 21:58:34 +03:00
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
2020-11-14 12:27:57 +01:00
|
|
|
pub enum ArgExtension {
|
|
|
|
None,
|
|
|
|
Zext,
|
|
|
|
Sext,
|
|
|
|
}
|
|
|
|
|
2018-04-25 16:45:29 +03:00
|
|
|
/// A compact representation of LLVM attributes (at least those relevant for this module)
|
|
|
|
/// that can be manipulated without interacting with LLVM's Attribute machinery.
|
2021-08-26 21:58:34 +03:00
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
2018-04-25 16:45:29 +03:00
|
|
|
pub struct ArgAttributes {
|
|
|
|
pub regular: ArgAttribute,
|
2020-11-14 12:27:57 +01:00
|
|
|
pub arg_ext: ArgExtension,
|
2019-11-25 22:45:00 +01:00
|
|
|
/// The minimum size of the pointee, guaranteed to be valid for the duration of the whole call
|
|
|
|
/// (corresponding to LLVM's dereferenceable and dereferenceable_or_null attributes).
|
2018-04-25 16:45:29 +03:00
|
|
|
pub pointee_size: Size,
|
2018-09-09 01:16:45 +03:00
|
|
|
pub pointee_align: Option<Align>,
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl ArgAttributes {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
ArgAttributes {
|
|
|
|
regular: ArgAttribute::default(),
|
2020-11-14 12:27:57 +01:00
|
|
|
arg_ext: ArgExtension::None,
|
2018-05-20 14:14:39 +02:00
|
|
|
pointee_size: Size::ZERO,
|
2018-04-25 16:45:29 +03:00
|
|
|
pointee_align: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-21 19:06:35 +01:00
|
|
|
pub fn ext(&mut self, ext: ArgExtension) -> &mut Self {
|
2020-12-31 18:02:52 +02:00
|
|
|
assert!(
|
|
|
|
self.arg_ext == ArgExtension::None || self.arg_ext == ext,
|
|
|
|
"cannot set {:?} when {:?} is already set",
|
|
|
|
ext,
|
|
|
|
self.arg_ext
|
|
|
|
);
|
2020-11-21 19:06:35 +01:00
|
|
|
self.arg_ext = ext;
|
2020-11-14 12:27:57 +01:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2018-04-25 16:45:29 +03:00
|
|
|
pub fn set(&mut self, attr: ArgAttribute) -> &mut Self {
|
2018-08-09 15:42:43 +02:00
|
|
|
self.regular |= attr;
|
2018-04-25 16:45:29 +03:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn contains(&self, attr: ArgAttribute) -> bool {
|
|
|
|
self.regular.contains(attr)
|
|
|
|
}
|
2023-09-02 11:59:31 +02:00
|
|
|
|
|
|
|
/// Checks if these two `ArgAttributes` are equal enough to be considered "the same for all
|
|
|
|
/// function call ABIs".
|
|
|
|
pub fn eq_abi(&self, other: &Self) -> bool {
|
|
|
|
// There's only one regular attribute that matters for the call ABI: InReg.
|
|
|
|
// Everything else is things like noalias, dereferenceable, nonnull, ...
|
|
|
|
// (This also applies to pointee_size, pointee_align.)
|
|
|
|
if self.regular.contains(ArgAttribute::InReg) != other.regular.contains(ArgAttribute::InReg)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
// We also compare the sign extension mode -- this could let the callee make assumptions
|
|
|
|
// about bits that conceptually were not even passed.
|
|
|
|
if self.arg_ext != other.arg_ext {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2021-08-26 21:58:34 +03:00
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
2018-04-25 16:45:29 +03:00
|
|
|
pub enum RegKind {
|
|
|
|
Integer,
|
|
|
|
Float,
|
|
|
|
Vector,
|
|
|
|
}
|
|
|
|
|
2021-08-26 21:58:34 +03:00
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
2018-04-25 16:45:29 +03:00
|
|
|
pub struct Reg {
|
|
|
|
pub kind: RegKind,
|
|
|
|
pub size: Size,
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! reg_ctor {
|
|
|
|
($name:ident, $kind:ident, $bits:expr) => {
|
|
|
|
pub fn $name() -> Reg {
|
|
|
|
Reg { kind: RegKind::$kind, size: Size::from_bits($bits) }
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Reg {
|
|
|
|
reg_ctor!(i8, Integer, 8);
|
|
|
|
reg_ctor!(i16, Integer, 16);
|
|
|
|
reg_ctor!(i32, Integer, 32);
|
|
|
|
reg_ctor!(i64, Integer, 64);
|
2020-01-21 21:52:19 +07:00
|
|
|
reg_ctor!(i128, Integer, 128);
|
2018-04-25 16:45:29 +03:00
|
|
|
|
|
|
|
reg_ctor!(f32, Float, 32);
|
|
|
|
reg_ctor!(f64, Float, 64);
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Reg {
|
2018-09-09 01:16:45 +03:00
|
|
|
pub fn align<C: HasDataLayout>(&self, cx: &C) -> Align {
|
2018-04-25 16:45:29 +03:00
|
|
|
let dl = cx.data_layout();
|
|
|
|
match self.kind {
|
|
|
|
RegKind::Integer => match self.size.bits() {
|
2018-09-09 01:16:45 +03:00
|
|
|
1 => dl.i1_align.abi,
|
|
|
|
2..=8 => dl.i8_align.abi,
|
|
|
|
9..=16 => dl.i16_align.abi,
|
|
|
|
17..=32 => dl.i32_align.abi,
|
|
|
|
33..=64 => dl.i64_align.abi,
|
|
|
|
65..=128 => dl.i128_align.abi,
|
2022-12-19 10:31:55 +01:00
|
|
|
_ => panic!("unsupported integer: {self:?}"),
|
2018-04-25 16:45:29 +03:00
|
|
|
},
|
|
|
|
RegKind::Float => match self.size.bits() {
|
2018-09-09 01:16:45 +03:00
|
|
|
32 => dl.f32_align.abi,
|
|
|
|
64 => dl.f64_align.abi,
|
2022-12-19 10:31:55 +01:00
|
|
|
_ => panic!("unsupported float: {self:?}"),
|
2018-04-25 16:45:29 +03:00
|
|
|
},
|
2018-09-09 01:16:45 +03:00
|
|
|
RegKind::Vector => dl.vector_align(self.size).abi,
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// An argument passed entirely registers with the
|
2018-11-27 02:59:49 +00:00
|
|
|
/// same kind (e.g., HFA / HVA on PPC64 and AArch64).
|
2021-08-26 21:58:34 +03:00
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
2018-04-25 16:45:29 +03:00
|
|
|
pub struct Uniform {
|
|
|
|
pub unit: Reg,
|
|
|
|
|
|
|
|
/// The total size of the argument, which can be:
|
2019-02-08 14:53:55 +01:00
|
|
|
/// * equal to `unit.size` (one scalar/vector),
|
|
|
|
/// * a multiple of `unit.size` (an array of scalar/vectors),
|
2018-04-25 16:45:29 +03:00
|
|
|
/// * if `unit.kind` is `Integer`, the last element
|
2018-11-27 02:59:49 +00:00
|
|
|
/// can be shorter, i.e., `{ i64, i64, i32 }` for
|
2019-02-08 14:53:55 +01:00
|
|
|
/// 64-bit integers with a total size of 20 bytes.
|
2018-04-25 16:45:29 +03:00
|
|
|
pub total: Size,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Reg> for Uniform {
|
|
|
|
fn from(unit: Reg) -> Uniform {
|
|
|
|
Uniform { unit, total: unit.size }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Uniform {
|
2018-09-09 01:16:45 +03:00
|
|
|
pub fn align<C: HasDataLayout>(&self, cx: &C) -> Align {
|
2018-04-25 16:45:29 +03:00
|
|
|
self.unit.align(cx)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-07 22:06:37 +02:00
|
|
|
/// Describes the type used for `PassMode::Cast`.
|
|
|
|
///
|
|
|
|
/// Passing arguments in this mode works as follows: the registers in the `prefix` (the ones that
|
|
|
|
/// are `Some`) get laid out one after the other (using `repr(C)` layout rules). Then the
|
|
|
|
/// `rest.unit` register type gets repeated often enough to cover `rest.size`. This describes the
|
|
|
|
/// actual type used for the call; the Rust type of the argument is then transmuted to this ABI type
|
|
|
|
/// (and all data in the padding between the registers is dropped).
|
2023-05-03 22:22:24 +00:00
|
|
|
#[derive(Clone, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
2018-04-25 16:45:29 +03:00
|
|
|
pub struct CastTarget {
|
2021-12-01 10:03:45 +01:00
|
|
|
pub prefix: [Option<Reg>; 8],
|
2018-04-25 16:45:29 +03:00
|
|
|
pub rest: Uniform,
|
2021-12-01 10:03:45 +01:00
|
|
|
pub attrs: ArgAttributes,
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Reg> for CastTarget {
|
|
|
|
fn from(unit: Reg) -> CastTarget {
|
|
|
|
CastTarget::from(Uniform::from(unit))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Uniform> for CastTarget {
|
|
|
|
fn from(uniform: Uniform) -> CastTarget {
|
2021-12-01 10:03:45 +01:00
|
|
|
CastTarget {
|
|
|
|
prefix: [None; 8],
|
|
|
|
rest: uniform,
|
|
|
|
attrs: ArgAttributes {
|
|
|
|
regular: ArgAttribute::default(),
|
|
|
|
arg_ext: ArgExtension::None,
|
|
|
|
pointee_size: Size::ZERO,
|
|
|
|
pointee_align: None,
|
|
|
|
},
|
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CastTarget {
|
|
|
|
pub fn pair(a: Reg, b: Reg) -> CastTarget {
|
|
|
|
CastTarget {
|
2021-12-01 10:03:45 +01:00
|
|
|
prefix: [Some(a), None, None, None, None, None, None, None],
|
2018-04-25 16:45:29 +03:00
|
|
|
rest: Uniform::from(b),
|
2021-12-01 10:03:45 +01:00
|
|
|
attrs: ArgAttributes {
|
|
|
|
regular: ArgAttribute::default(),
|
|
|
|
arg_ext: ArgExtension::None,
|
|
|
|
pointee_size: Size::ZERO,
|
|
|
|
pointee_align: None,
|
|
|
|
},
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-01 10:03:45 +01:00
|
|
|
pub fn size<C: HasDataLayout>(&self, _cx: &C) -> Size {
|
|
|
|
let mut size = self.rest.total;
|
|
|
|
for i in 0..self.prefix.iter().count() {
|
|
|
|
match self.prefix[i] {
|
2022-11-07 00:36:11 +03:30
|
|
|
Some(v) => size += v.size,
|
2021-12-01 10:03:45 +01:00
|
|
|
None => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return size;
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2018-09-09 01:16:45 +03:00
|
|
|
pub fn align<C: HasDataLayout>(&self, cx: &C) -> Align {
|
2018-04-25 16:45:29 +03:00
|
|
|
self.prefix
|
|
|
|
.iter()
|
2021-12-01 10:03:45 +01:00
|
|
|
.filter_map(|x| x.map(|reg| reg.align(cx)))
|
2018-09-09 01:16:45 +03:00
|
|
|
.fold(cx.data_layout().aggregate_align.abi.max(self.rest.align(cx)), |acc, align| {
|
2018-04-25 16:45:29 +03:00
|
|
|
acc.max(align)
|
2019-12-22 17:42:04 -05:00
|
|
|
})
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
2023-09-02 11:59:31 +02:00
|
|
|
|
|
|
|
/// Checks if these two `CastTarget` are equal enough to be considered "the same for all
|
|
|
|
/// function call ABIs".
|
|
|
|
pub fn eq_abi(&self, other: &Self) -> bool {
|
|
|
|
let CastTarget { prefix: prefix_l, rest: rest_l, attrs: attrs_l } = self;
|
|
|
|
let CastTarget { prefix: prefix_r, rest: rest_r, attrs: attrs_r } = other;
|
|
|
|
prefix_l == prefix_r && rest_l == rest_r && attrs_l.eq_abi(attrs_r)
|
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2020-01-22 02:52:14 +02:00
|
|
|
/// Return value from the `homogeneous_aggregate` test function.
|
2019-01-09 15:16:32 -05:00
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
|
|
pub enum HomogeneousAggregate {
|
|
|
|
/// Yes, all the "leaf fields" of this struct are passed in the
|
|
|
|
/// same way (specified in the `Reg` value).
|
|
|
|
Homogeneous(Reg),
|
|
|
|
|
|
|
|
/// There are no leaf fields at all.
|
|
|
|
NoData,
|
|
|
|
}
|
|
|
|
|
2020-01-22 02:52:14 +02:00
|
|
|
/// Error from the `homogeneous_aggregate` test function, indicating
|
|
|
|
/// there are distinct leaf fields passed in different ways,
|
|
|
|
/// or this is uninhabited.
|
|
|
|
#[derive(Copy, Clone, Debug)]
|
|
|
|
pub struct Heterogeneous;
|
|
|
|
|
2019-01-09 15:16:32 -05:00
|
|
|
impl HomogeneousAggregate {
|
|
|
|
/// If this is a homogeneous aggregate, returns the homogeneous
|
|
|
|
/// unit, else `None`.
|
|
|
|
pub fn unit(self) -> Option<Reg> {
|
2020-01-22 02:52:14 +02:00
|
|
|
match self {
|
|
|
|
HomogeneousAggregate::Homogeneous(reg) => Some(reg),
|
|
|
|
HomogeneousAggregate::NoData => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Try to combine two `HomogeneousAggregate`s, e.g. from two fields in
|
|
|
|
/// the same `struct`. Only succeeds if only one of them has any data,
|
|
|
|
/// or both units are identical.
|
|
|
|
fn merge(self, other: HomogeneousAggregate) -> Result<HomogeneousAggregate, Heterogeneous> {
|
|
|
|
match (self, other) {
|
|
|
|
(x, HomogeneousAggregate::NoData) | (HomogeneousAggregate::NoData, x) => Ok(x),
|
|
|
|
|
|
|
|
(HomogeneousAggregate::Homogeneous(a), HomogeneousAggregate::Homogeneous(b)) => {
|
|
|
|
if a != b {
|
|
|
|
return Err(Heterogeneous);
|
|
|
|
}
|
|
|
|
Ok(self)
|
|
|
|
}
|
|
|
|
}
|
2019-01-09 15:16:32 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-04 14:50:21 +00:00
|
|
|
impl<'a, Ty> TyAndLayout<'a, Ty> {
|
2023-11-21 10:15:59 +01:00
|
|
|
/// Returns `true` if this is an aggregate type (including a ScalarPair!)
|
2018-04-25 16:45:29 +03:00
|
|
|
fn is_aggregate(&self) -> bool {
|
|
|
|
match self.abi {
|
|
|
|
Abi::Uninhabited | Abi::Scalar(_) | Abi::Vector { .. } => false,
|
|
|
|
Abi::ScalarPair(..) | Abi::Aggregate { .. } => true,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-22 02:52:14 +02:00
|
|
|
/// Returns `Homogeneous` if this layout is an aggregate containing fields of
|
|
|
|
/// only a single type (e.g., `(u32, u32)`). Such aggregates are often
|
2019-01-09 15:16:32 -05:00
|
|
|
/// special-cased in ABIs.
|
|
|
|
///
|
2023-09-10 07:38:03 +02:00
|
|
|
/// Note: We generally ignore 1-ZST fields when computing this value (see #56877).
|
2019-01-09 15:16:32 -05:00
|
|
|
///
|
|
|
|
/// This is public so that it can be used in unit tests, but
|
|
|
|
/// should generally only be relevant to the ABI details of
|
|
|
|
/// specific targets.
|
2020-01-22 02:52:14 +02:00
|
|
|
pub fn homogeneous_aggregate<C>(&self, cx: &C) -> Result<HomogeneousAggregate, Heterogeneous>
|
2018-11-03 22:57:53 +02:00
|
|
|
where
|
2021-08-25 16:45:24 +03:00
|
|
|
Ty: TyAbiInterface<'a, C> + Copy,
|
2018-04-25 16:45:29 +03:00
|
|
|
{
|
|
|
|
match self.abi {
|
2020-01-22 02:52:14 +02:00
|
|
|
Abi::Uninhabited => Err(Heterogeneous),
|
2018-04-25 16:45:29 +03:00
|
|
|
|
|
|
|
// The primitive for this algorithm.
|
2021-08-29 11:06:55 +02:00
|
|
|
Abi::Scalar(scalar) => {
|
2022-03-03 12:02:12 +00:00
|
|
|
let kind = match scalar.primitive() {
|
2023-01-22 23:03:58 -05:00
|
|
|
abi::Int(..) | abi::Pointer(_) => RegKind::Integer,
|
2019-11-07 16:54:25 +02:00
|
|
|
abi::F32 | abi::F64 => RegKind::Float,
|
2018-04-25 16:45:29 +03:00
|
|
|
};
|
2020-01-22 02:52:14 +02:00
|
|
|
Ok(HomogeneousAggregate::Homogeneous(Reg { kind, size: self.size }))
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Abi::Vector { .. } => {
|
2019-01-09 15:16:32 -05:00
|
|
|
assert!(!self.is_zst());
|
2020-01-22 02:52:14 +02:00
|
|
|
Ok(HomogeneousAggregate::Homogeneous(Reg {
|
|
|
|
kind: RegKind::Vector,
|
|
|
|
size: self.size,
|
|
|
|
}))
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2023-11-01 23:01:53 +01:00
|
|
|
Abi::ScalarPair(..) | Abi::Aggregate { sized: true } => {
|
2020-04-16 15:15:46 +00:00
|
|
|
// Helper for computing `homogeneous_aggregate`, allowing a custom
|
2020-01-22 02:52:54 +02:00
|
|
|
// starting offset (used below for handling variants).
|
2020-01-22 02:52:14 +02:00
|
|
|
let from_fields_at =
|
|
|
|
|layout: Self,
|
|
|
|
start: Size|
|
|
|
|
-> Result<(HomogeneousAggregate, Size), Heterogeneous> {
|
|
|
|
let is_union = match layout.fields {
|
2020-04-16 15:15:46 +00:00
|
|
|
FieldsShape::Primitive => {
|
|
|
|
unreachable!("aggregates can't have `FieldsShape::Primitive`")
|
|
|
|
}
|
2020-03-31 13:44:52 +00:00
|
|
|
FieldsShape::Array { count, .. } => {
|
2020-01-22 02:52:14 +02:00
|
|
|
assert_eq!(start, Size::ZERO);
|
|
|
|
|
|
|
|
let result = if count > 0 {
|
|
|
|
layout.field(cx, 0).homogeneous_aggregate(cx)?
|
|
|
|
} else {
|
|
|
|
HomogeneousAggregate::NoData
|
|
|
|
};
|
|
|
|
return Ok((result, layout.size));
|
|
|
|
}
|
2020-03-31 13:44:52 +00:00
|
|
|
FieldsShape::Union(_) => true,
|
|
|
|
FieldsShape::Arbitrary { .. } => false,
|
2020-01-22 02:52:14 +02:00
|
|
|
};
|
2018-04-25 16:45:29 +03:00
|
|
|
|
2020-01-22 02:52:14 +02:00
|
|
|
let mut result = HomogeneousAggregate::NoData;
|
|
|
|
let mut total = start;
|
2018-04-25 16:45:29 +03:00
|
|
|
|
2020-01-22 02:52:14 +02:00
|
|
|
for i in 0..layout.fields.count() {
|
2023-09-10 07:38:03 +02:00
|
|
|
let field = layout.field(cx, i);
|
|
|
|
if field.is_1zst() {
|
|
|
|
// No data here and no impact on layout, can be ignored.
|
|
|
|
// (We might be able to also ignore all aligned ZST but that's less clear.)
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-01-22 02:52:14 +02:00
|
|
|
if !is_union && total != layout.fields.offset(i) {
|
2023-09-10 07:38:03 +02:00
|
|
|
// This field isn't just after the previous one we considered, abort.
|
2020-01-22 02:52:14 +02:00
|
|
|
return Err(Heterogeneous);
|
|
|
|
}
|
2019-01-09 15:16:32 -05:00
|
|
|
|
2020-01-22 02:52:14 +02:00
|
|
|
result = result.merge(field.homogeneous_aggregate(cx)?)?;
|
|
|
|
|
|
|
|
// Keep track of the offset (without padding).
|
|
|
|
let size = field.size;
|
|
|
|
if is_union {
|
|
|
|
total = total.max(size);
|
|
|
|
} else {
|
|
|
|
total += size;
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-22 02:52:14 +02:00
|
|
|
Ok((result, total))
|
|
|
|
};
|
|
|
|
|
|
|
|
let (mut result, mut total) = from_fields_at(*self, Size::ZERO)?;
|
2018-04-25 16:45:29 +03:00
|
|
|
|
2020-01-22 02:52:54 +02:00
|
|
|
match &self.variants {
|
|
|
|
abi::Variants::Single { .. } => {}
|
|
|
|
abi::Variants::Multiple { variants, .. } => {
|
|
|
|
// Treat enum variants like union members.
|
|
|
|
// HACK(eddyb) pretend the `enum` field (discriminant)
|
|
|
|
// is at the start of every variant (otherwise the gap
|
|
|
|
// at the start of all variants would disqualify them).
|
|
|
|
//
|
|
|
|
// NB: for all tagged `enum`s (which include all non-C-like
|
|
|
|
// `enum`s with defined FFI representation), this will
|
2020-03-06 12:13:55 +01:00
|
|
|
// match the homogeneous computation on the equivalent
|
2020-01-22 02:52:54 +02:00
|
|
|
// `struct { tag; union { variant1; ... } }` and/or
|
|
|
|
// `union { struct { tag; variant1; } ... }`
|
|
|
|
// (the offsets of variant fields should be identical
|
2020-03-06 12:13:55 +01:00
|
|
|
// between the two for either to be a homogeneous aggregate).
|
2020-01-22 02:52:54 +02:00
|
|
|
let variant_start = total;
|
|
|
|
for variant_idx in variants.indices() {
|
|
|
|
let (variant_result, variant_total) =
|
|
|
|
from_fields_at(self.for_variant(cx, variant_idx), variant_start)?;
|
|
|
|
|
|
|
|
result = result.merge(variant_result)?;
|
|
|
|
total = total.max(variant_total);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-25 16:45:29 +03:00
|
|
|
// There needs to be no padding.
|
|
|
|
if total != self.size {
|
2020-01-22 02:52:14 +02:00
|
|
|
Err(Heterogeneous)
|
2018-04-25 16:45:29 +03:00
|
|
|
} else {
|
2019-01-09 15:16:32 -05:00
|
|
|
match result {
|
2020-01-22 02:52:14 +02:00
|
|
|
HomogeneousAggregate::Homogeneous(_) => {
|
2019-01-09 15:16:32 -05:00
|
|
|
assert_ne!(total, Size::ZERO);
|
|
|
|
}
|
2020-01-22 02:52:14 +02:00
|
|
|
HomogeneousAggregate::NoData => {
|
2019-01-09 15:16:32 -05:00
|
|
|
assert_eq!(total, Size::ZERO);
|
|
|
|
}
|
|
|
|
}
|
2020-01-22 02:52:14 +02:00
|
|
|
Ok(result)
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
}
|
2023-11-01 23:01:53 +01:00
|
|
|
Abi::Aggregate { sized: false } => Err(Heterogeneous),
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Information about how to pass an argument to,
|
|
|
|
/// or return a value from, a function, under some ABI.
|
2023-01-23 16:02:00 +00:00
|
|
|
#[derive(Clone, PartialEq, Eq, Hash, HashStable_Generic)]
|
2019-10-29 16:35:26 +02:00
|
|
|
pub struct ArgAbi<'a, Ty> {
|
2020-03-04 14:50:21 +00:00
|
|
|
pub layout: TyAndLayout<'a, Ty>,
|
2018-04-25 16:45:29 +03:00
|
|
|
pub mode: PassMode,
|
|
|
|
}
|
|
|
|
|
2023-01-23 16:02:00 +00:00
|
|
|
// Needs to be a custom impl because of the bounds on the `TyAndLayout` debug impl.
|
|
|
|
impl<'a, Ty: fmt::Display> fmt::Debug for ArgAbi<'a, Ty> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
let ArgAbi { layout, mode } = self;
|
|
|
|
f.debug_struct("ArgAbi").field("layout", layout).field("mode", mode).finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-29 16:35:26 +02:00
|
|
|
impl<'a, Ty> ArgAbi<'a, Ty> {
|
2023-09-06 11:12:23 +02:00
|
|
|
/// This defines the "default ABI" for that type, that is then later adjusted in `fn_abi_adjust_for_abi`.
|
2020-12-29 20:00:57 +01:00
|
|
|
pub fn new(
|
|
|
|
cx: &impl HasDataLayout,
|
|
|
|
layout: TyAndLayout<'a, Ty>,
|
2021-08-29 11:06:55 +02:00
|
|
|
scalar_attrs: impl Fn(&TyAndLayout<'a, Ty>, abi::Scalar, Size) -> ArgAttributes,
|
2020-12-29 20:00:57 +01:00
|
|
|
) -> Self {
|
2021-08-29 11:06:55 +02:00
|
|
|
let mode = match layout.abi {
|
2020-12-29 20:00:57 +01:00
|
|
|
Abi::Uninhabited => PassMode::Ignore,
|
|
|
|
Abi::Scalar(scalar) => PassMode::Direct(scalar_attrs(&layout, scalar, Size::ZERO)),
|
|
|
|
Abi::ScalarPair(a, b) => PassMode::Pair(
|
|
|
|
scalar_attrs(&layout, a, Size::ZERO),
|
2022-03-03 12:02:12 +00:00
|
|
|
scalar_attrs(&layout, b, a.size(cx).align_to(b.align(cx).abi)),
|
2020-12-29 20:00:57 +01:00
|
|
|
),
|
|
|
|
Abi::Vector { .. } => PassMode::Direct(ArgAttributes::new()),
|
2023-11-01 23:01:53 +01:00
|
|
|
Abi::Aggregate { .. } => Self::indirect_pass_mode(&layout),
|
2020-12-29 20:00:57 +01:00
|
|
|
};
|
2022-08-25 22:19:38 +10:00
|
|
|
ArgAbi { layout, mode }
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2020-12-29 20:00:57 +01:00
|
|
|
fn indirect_pass_mode(layout: &TyAndLayout<'a, Ty>) -> PassMode {
|
2018-04-25 16:45:29 +03:00
|
|
|
let mut attrs = ArgAttributes::new();
|
|
|
|
|
|
|
|
// For non-immediate arguments the callee gets its own copy of
|
|
|
|
// the value on the stack, so there are no aliases. It's also
|
|
|
|
// program-invisible so can't possibly capture
|
2022-02-05 01:00:37 -05:00
|
|
|
attrs
|
|
|
|
.set(ArgAttribute::NoAlias)
|
|
|
|
.set(ArgAttribute::NoCapture)
|
|
|
|
.set(ArgAttribute::NonNull)
|
|
|
|
.set(ArgAttribute::NoUndef);
|
2020-12-29 20:00:57 +01:00
|
|
|
attrs.pointee_size = layout.size;
|
2022-10-31 20:38:40 -07:00
|
|
|
attrs.pointee_align = Some(layout.align.abi);
|
2018-04-25 16:45:29 +03:00
|
|
|
|
2023-09-08 08:48:41 +02:00
|
|
|
let meta_attrs = layout.is_unsized().then_some(ArgAttributes::new());
|
2018-08-03 23:32:21 +09:00
|
|
|
|
2023-09-08 08:48:41 +02:00
|
|
|
PassMode::Indirect { attrs, meta_attrs, on_stack: false }
|
2020-12-29 20:00:57 +01:00
|
|
|
}
|
|
|
|
|
2023-11-01 23:01:53 +01:00
|
|
|
/// Pass this argument directly instead. Should NOT be used!
|
|
|
|
/// Only exists because of past ABI mistakes that will take time to fix
|
|
|
|
/// (see <https://github.com/rust-lang/rust/issues/115666>).
|
|
|
|
pub fn make_direct_deprecated(&mut self) {
|
2023-11-19 16:03:07 +01:00
|
|
|
match self.mode {
|
|
|
|
PassMode::Indirect { .. } => {
|
|
|
|
self.mode = PassMode::Direct(ArgAttributes::new());
|
|
|
|
}
|
|
|
|
PassMode::Ignore | PassMode::Direct(_) | PassMode::Pair(_, _) => return, // already direct
|
|
|
|
_ => panic!("Tried to make {:?} direct", self.mode),
|
|
|
|
}
|
2023-11-01 23:01:53 +01:00
|
|
|
}
|
|
|
|
|
2020-12-29 20:00:57 +01:00
|
|
|
pub fn make_indirect(&mut self) {
|
|
|
|
match self.mode {
|
2023-11-19 16:03:07 +01:00
|
|
|
PassMode::Direct(_) | PassMode::Pair(_, _) => {
|
|
|
|
self.mode = Self::indirect_pass_mode(&self.layout);
|
|
|
|
}
|
2023-11-01 23:01:53 +01:00
|
|
|
PassMode::Indirect { attrs: _, meta_attrs: _, on_stack: false } => {
|
|
|
|
// already indirect
|
|
|
|
return;
|
|
|
|
}
|
2020-12-29 20:00:57 +01:00
|
|
|
_ => panic!("Tried to make {:?} indirect", self.mode),
|
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2022-10-31 20:38:40 -07:00
|
|
|
pub fn make_indirect_byval(&mut self, byval_align: Option<Align>) {
|
2018-04-25 16:45:29 +03:00
|
|
|
self.make_indirect();
|
|
|
|
match self.mode {
|
2023-09-08 08:48:41 +02:00
|
|
|
PassMode::Indirect { ref mut attrs, meta_attrs: _, ref mut on_stack } => {
|
2020-11-14 14:29:40 +01:00
|
|
|
*on_stack = true;
|
2022-10-31 20:38:40 -07:00
|
|
|
|
|
|
|
// Some platforms, like 32-bit x86, change the alignment of the type when passing
|
|
|
|
// `byval`. Account for that.
|
|
|
|
if let Some(byval_align) = byval_align {
|
|
|
|
// On all targets with byval align this is currently true, so let's assert it.
|
|
|
|
debug_assert!(byval_align >= Align::from_bytes(4).unwrap());
|
|
|
|
attrs.pointee_align = Some(byval_align);
|
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
_ => unreachable!(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn extend_integer_width_to(&mut self, bits: u64) {
|
|
|
|
// Only integers have signedness
|
2021-08-29 11:06:55 +02:00
|
|
|
if let Abi::Scalar(scalar) = self.layout.abi {
|
2022-03-03 12:02:12 +00:00
|
|
|
if let abi::Int(i, signed) = scalar.primitive() {
|
2018-04-25 16:45:29 +03:00
|
|
|
if i.size().bits() < bits {
|
|
|
|
if let PassMode::Direct(ref mut attrs) = self.mode {
|
2020-11-14 12:27:57 +01:00
|
|
|
if signed {
|
2020-11-21 19:06:35 +01:00
|
|
|
attrs.ext(ArgExtension::Sext)
|
2020-11-14 12:27:57 +01:00
|
|
|
} else {
|
2020-11-21 19:06:35 +01:00
|
|
|
attrs.ext(ArgExtension::Zext)
|
2020-11-14 12:27:57 +01:00
|
|
|
};
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn cast_to<T: Into<CastTarget>>(&mut self, target: T) {
|
2023-09-07 22:06:37 +02:00
|
|
|
self.mode = PassMode::Cast { cast: Box::new(target.into()), pad_i32: false };
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2022-08-25 22:19:38 +10:00
|
|
|
pub fn cast_to_and_pad_i32<T: Into<CastTarget>>(&mut self, target: T, pad_i32: bool) {
|
2023-09-07 22:06:37 +02:00
|
|
|
self.mode = PassMode::Cast { cast: Box::new(target.into()), pad_i32 };
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_indirect(&self) -> bool {
|
2021-01-09 12:00:45 -05:00
|
|
|
matches!(self.mode, PassMode::Indirect { .. })
|
2018-08-03 23:32:21 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn is_sized_indirect(&self) -> bool {
|
2023-09-08 08:48:41 +02:00
|
|
|
matches!(self.mode, PassMode::Indirect { attrs: _, meta_attrs: None, on_stack: _ })
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2018-05-29 00:12:55 +09:00
|
|
|
pub fn is_unsized_indirect(&self) -> bool {
|
2023-09-08 08:48:41 +02:00
|
|
|
matches!(self.mode, PassMode::Indirect { attrs: _, meta_attrs: Some(_), on_stack: _ })
|
2018-05-29 00:12:55 +09:00
|
|
|
}
|
|
|
|
|
2018-04-25 16:45:29 +03:00
|
|
|
pub fn is_ignore(&self) -> bool {
|
2020-10-26 21:02:48 -04:00
|
|
|
matches!(self.mode, PassMode::Ignore)
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
2023-09-07 16:48:02 +02:00
|
|
|
|
|
|
|
/// Checks if these two `ArgAbi` are equal enough to be considered "the same for all
|
|
|
|
/// function call ABIs".
|
|
|
|
pub fn eq_abi(&self, other: &Self) -> bool {
|
|
|
|
// Ideally we'd just compare the `mode`, but that is not enough -- for some modes LLVM will look
|
|
|
|
// at the type.
|
|
|
|
self.layout.eq_abi(&other.layout) && self.mode.eq_abi(&other.mode)
|
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2021-08-26 21:58:34 +03:00
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
2018-04-25 16:45:29 +03:00
|
|
|
pub enum Conv {
|
2019-11-27 13:23:30 +02:00
|
|
|
// General language calling conventions, for which every target
|
|
|
|
// should have its own backend (e.g. LLVM) support.
|
2018-04-25 16:45:29 +03:00
|
|
|
C,
|
2019-11-27 13:23:30 +02:00
|
|
|
Rust,
|
|
|
|
|
2023-08-26 17:42:59 -07:00
|
|
|
Cold,
|
|
|
|
PreserveMost,
|
|
|
|
PreserveAll,
|
2022-05-29 00:25:14 -07:00
|
|
|
|
2019-11-27 13:23:30 +02:00
|
|
|
// Target-specific calling conventions.
|
2018-04-25 16:45:29 +03:00
|
|
|
ArmAapcs,
|
2021-01-24 17:15:05 +00:00
|
|
|
CCmseNonSecureCall,
|
2018-04-25 16:45:29 +03:00
|
|
|
|
|
|
|
Msp430Intr,
|
|
|
|
|
|
|
|
PtxKernel,
|
|
|
|
|
|
|
|
X86Fastcall,
|
|
|
|
X86Intr,
|
|
|
|
X86Stdcall,
|
|
|
|
X86ThisCall,
|
|
|
|
X86VectorCall,
|
|
|
|
|
|
|
|
X86_64SysV,
|
|
|
|
X86_64Win64,
|
2018-07-01 22:42:00 -05:00
|
|
|
|
|
|
|
AmdGpuKernel,
|
2016-05-06 09:32:10 -04:00
|
|
|
AvrInterrupt,
|
|
|
|
AvrNonBlockingInterrupt,
|
feat: `riscv-interrupt-{m,s}` calling conventions
Similar to prior support added for the mips430, avr, and x86 targets
this change implements the rough equivalent of clang's
[`__attribute__((interrupt))`][clang-attr] for riscv targets, enabling
e.g.
```rust
static mut CNT: usize = 0;
pub extern "riscv-interrupt-m" fn isr_m() {
unsafe {
CNT += 1;
}
}
```
to produce highly effective assembly like:
```asm
pub extern "riscv-interrupt-m" fn isr_m() {
420003a0: 1141 addi sp,sp,-16
unsafe {
CNT += 1;
420003a2: c62a sw a0,12(sp)
420003a4: c42e sw a1,8(sp)
420003a6: 3fc80537 lui a0,0x3fc80
420003aa: 63c52583 lw a1,1596(a0) # 3fc8063c <_ZN12esp_riscv_rt3CNT17hcec3e3a214887d53E.0>
420003ae: 0585 addi a1,a1,1
420003b0: 62b52e23 sw a1,1596(a0)
}
}
420003b4: 4532 lw a0,12(sp)
420003b6: 45a2 lw a1,8(sp)
420003b8: 0141 addi sp,sp,16
420003ba: 30200073 mret
```
(disassembly via `riscv64-unknown-elf-objdump -C -S --disassemble ./esp32c3-hal/target/riscv32imc-unknown-none-elf/release/examples/gpio_interrupt`)
This outcome is superior to hand-coded interrupt routines which, lacking
visibility into any non-assembly body of the interrupt handler, have to
be very conservative and save the [entire CPU state to the stack
frame][full-frame-save]. By instead asking LLVM to only save the
registers that it uses, we defer the decision to the tool with the best
context: it can more accurately account for the cost of spills if it
knows that every additional register used is already at the cost of an
implicit spill.
At the LLVM level, this is apparently [implemented by] marking every
register as "[callee-save]," matching the semantics of an interrupt
handler nicely (it has to leave the CPU state just as it found it after
its `{m|s}ret`).
This approach is not suitable for every interrupt handler, as it makes
no attempt to e.g. save the state in a user-accessible stack frame. For
a full discussion of those challenges and tradeoffs, please refer to
[the interrupt calling conventions RFC][rfc].
Inside rustc, this implementation differs from prior art because LLVM
does not expose the "all-saved" function flavor as a calling convention
directly, instead preferring to use an attribute that allows for
differentiating between "machine-mode" and "superivsor-mode" interrupts.
Finally, some effort has been made to guide those who may not yet be
aware of the differences between machine-mode and supervisor-mode
interrupts as to why no `riscv-interrupt` calling convention is exposed
through rustc, and similarly for why `riscv-interrupt-u` makes no
appearance (as it would complicate future LLVM upgrades).
[clang-attr]: https://clang.llvm.org/docs/AttributeReference.html#interrupt-risc-v
[full-frame-save]: https://github.com/esp-rs/esp-riscv-rt/blob/9281af2ecffe13e40992917316f36920c26acaf3/src/lib.rs#L440-L469
[implemented by]: https://github.com/llvm/llvm-project/blob/b7fb2a3fec7c187d58a6d338ab512d9173bca987/llvm/lib/Target/RISCV/RISCVRegisterInfo.cpp#L61-L67
[callee-save]: https://github.com/llvm/llvm-project/blob/973f1fe7a8591c7af148e573491ab68cc15b6ecf/llvm/lib/Target/RISCV/RISCVCallingConv.td#L30-L37
[rfc]: https://github.com/rust-lang/rfcs/pull/3246
2023-05-23 15:08:23 -07:00
|
|
|
|
2023-08-26 17:42:59 -07:00
|
|
|
RiscvInterrupt { kind: RiscvInterruptKind },
|
feat: `riscv-interrupt-{m,s}` calling conventions
Similar to prior support added for the mips430, avr, and x86 targets
this change implements the rough equivalent of clang's
[`__attribute__((interrupt))`][clang-attr] for riscv targets, enabling
e.g.
```rust
static mut CNT: usize = 0;
pub extern "riscv-interrupt-m" fn isr_m() {
unsafe {
CNT += 1;
}
}
```
to produce highly effective assembly like:
```asm
pub extern "riscv-interrupt-m" fn isr_m() {
420003a0: 1141 addi sp,sp,-16
unsafe {
CNT += 1;
420003a2: c62a sw a0,12(sp)
420003a4: c42e sw a1,8(sp)
420003a6: 3fc80537 lui a0,0x3fc80
420003aa: 63c52583 lw a1,1596(a0) # 3fc8063c <_ZN12esp_riscv_rt3CNT17hcec3e3a214887d53E.0>
420003ae: 0585 addi a1,a1,1
420003b0: 62b52e23 sw a1,1596(a0)
}
}
420003b4: 4532 lw a0,12(sp)
420003b6: 45a2 lw a1,8(sp)
420003b8: 0141 addi sp,sp,16
420003ba: 30200073 mret
```
(disassembly via `riscv64-unknown-elf-objdump -C -S --disassemble ./esp32c3-hal/target/riscv32imc-unknown-none-elf/release/examples/gpio_interrupt`)
This outcome is superior to hand-coded interrupt routines which, lacking
visibility into any non-assembly body of the interrupt handler, have to
be very conservative and save the [entire CPU state to the stack
frame][full-frame-save]. By instead asking LLVM to only save the
registers that it uses, we defer the decision to the tool with the best
context: it can more accurately account for the cost of spills if it
knows that every additional register used is already at the cost of an
implicit spill.
At the LLVM level, this is apparently [implemented by] marking every
register as "[callee-save]," matching the semantics of an interrupt
handler nicely (it has to leave the CPU state just as it found it after
its `{m|s}ret`).
This approach is not suitable for every interrupt handler, as it makes
no attempt to e.g. save the state in a user-accessible stack frame. For
a full discussion of those challenges and tradeoffs, please refer to
[the interrupt calling conventions RFC][rfc].
Inside rustc, this implementation differs from prior art because LLVM
does not expose the "all-saved" function flavor as a calling convention
directly, instead preferring to use an attribute that allows for
differentiating between "machine-mode" and "superivsor-mode" interrupts.
Finally, some effort has been made to guide those who may not yet be
aware of the differences between machine-mode and supervisor-mode
interrupts as to why no `riscv-interrupt` calling convention is exposed
through rustc, and similarly for why `riscv-interrupt-u` makes no
appearance (as it would complicate future LLVM upgrades).
[clang-attr]: https://clang.llvm.org/docs/AttributeReference.html#interrupt-risc-v
[full-frame-save]: https://github.com/esp-rs/esp-riscv-rt/blob/9281af2ecffe13e40992917316f36920c26acaf3/src/lib.rs#L440-L469
[implemented by]: https://github.com/llvm/llvm-project/blob/b7fb2a3fec7c187d58a6d338ab512d9173bca987/llvm/lib/Target/RISCV/RISCVRegisterInfo.cpp#L61-L67
[callee-save]: https://github.com/llvm/llvm-project/blob/973f1fe7a8591c7af148e573491ab68cc15b6ecf/llvm/lib/Target/RISCV/RISCVCallingConv.td#L30-L37
[rfc]: https://github.com/rust-lang/rfcs/pull/3246
2023-05-23 15:08:23 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash, Debug, HashStable_Generic)]
|
|
|
|
pub enum RiscvInterruptKind {
|
|
|
|
Machine,
|
|
|
|
Supervisor,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl RiscvInterruptKind {
|
|
|
|
pub fn as_str(&self) -> &'static str {
|
|
|
|
match self {
|
|
|
|
Self::Machine => "machine",
|
|
|
|
Self::Supervisor => "supervisor",
|
|
|
|
}
|
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Metadata describing how the arguments to a native function
|
|
|
|
/// should be passed in order to respect the native ABI.
|
|
|
|
///
|
|
|
|
/// I will do my best to describe this structure, but these
|
|
|
|
/// comments are reverse-engineered and may be inaccurate. -NDM
|
2023-01-23 16:02:00 +00:00
|
|
|
#[derive(Clone, PartialEq, Eq, Hash, HashStable_Generic)]
|
2019-10-29 16:35:26 +02:00
|
|
|
pub struct FnAbi<'a, Ty> {
|
2018-04-25 16:45:29 +03:00
|
|
|
/// The LLVM types of each argument.
|
2022-08-25 19:08:04 +10:00
|
|
|
pub args: Box<[ArgAbi<'a, Ty>]>,
|
2018-04-25 16:45:29 +03:00
|
|
|
|
|
|
|
/// LLVM return type.
|
2019-10-29 16:35:26 +02:00
|
|
|
pub ret: ArgAbi<'a, Ty>,
|
2018-04-25 16:45:29 +03:00
|
|
|
|
2019-02-08 17:30:42 +00:00
|
|
|
pub c_variadic: bool,
|
2018-04-25 16:45:29 +03:00
|
|
|
|
2020-01-21 21:52:19 +07:00
|
|
|
/// The count of non-variadic arguments.
|
|
|
|
///
|
|
|
|
/// Should only be different from args.len() when c_variadic is true.
|
2020-03-06 12:13:55 +01:00
|
|
|
/// This can be used to know whether an argument is variadic or not.
|
2022-08-25 19:02:22 +10:00
|
|
|
pub fixed_count: u32,
|
2020-01-21 21:52:19 +07:00
|
|
|
|
2018-04-25 16:45:29 +03:00
|
|
|
pub conv: Conv,
|
2020-03-28 21:47:50 -04:00
|
|
|
|
|
|
|
pub can_unwind: bool,
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
2023-01-23 16:02:00 +00:00
|
|
|
// Needs to be a custom impl because of the bounds on the `TyAndLayout` debug impl.
|
|
|
|
impl<'a, Ty: fmt::Display> fmt::Debug for FnAbi<'a, Ty> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
let FnAbi { args, ret, c_variadic, fixed_count, conv, can_unwind } = self;
|
|
|
|
f.debug_struct("FnAbi")
|
|
|
|
.field("args", args)
|
|
|
|
.field("ret", ret)
|
|
|
|
.field("c_variadic", c_variadic)
|
|
|
|
.field("fixed_count", fixed_count)
|
|
|
|
.field("conv", conv)
|
|
|
|
.field("can_unwind", can_unwind)
|
|
|
|
.finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-26 19:01:55 +03:00
|
|
|
/// Error produced by attempting to adjust a `FnAbi`, for a "foreign" ABI.
|
2022-02-01 18:44:45 +01:00
|
|
|
#[derive(Copy, Clone, Debug, HashStable_Generic)]
|
2021-08-26 19:01:55 +03:00
|
|
|
pub enum AdjustForForeignAbiError {
|
|
|
|
/// Target architecture doesn't support "foreign" (i.e. non-Rust) ABIs.
|
2022-02-01 18:44:45 +01:00
|
|
|
Unsupported { arch: Symbol, abi: spec::abi::Abi },
|
2021-08-26 19:01:55 +03:00
|
|
|
}
|
|
|
|
|
2019-10-29 16:35:26 +02:00
|
|
|
impl<'a, Ty> FnAbi<'a, Ty> {
|
2021-08-26 19:01:55 +03:00
|
|
|
pub fn adjust_for_foreign_abi<C>(
|
|
|
|
&mut self,
|
|
|
|
cx: &C,
|
|
|
|
abi: spec::abi::Abi,
|
|
|
|
) -> Result<(), AdjustForForeignAbiError>
|
2018-04-25 16:45:29 +03:00
|
|
|
where
|
2021-08-25 16:45:24 +03:00
|
|
|
Ty: TyAbiInterface<'a, C> + Copy,
|
2021-08-25 18:15:09 +03:00
|
|
|
C: HasDataLayout + HasTargetSpec,
|
2018-04-25 16:45:29 +03:00
|
|
|
{
|
2021-02-23 21:38:49 +01:00
|
|
|
if abi == spec::abi::Abi::X86Interrupt {
|
|
|
|
if let Some(arg) = self.args.first_mut() {
|
2022-10-31 20:38:40 -07:00
|
|
|
// FIXME(pcwalton): This probably should use the x86 `byval` ABI...
|
|
|
|
arg.make_indirect_byval(None);
|
2021-02-23 21:38:49 +01:00
|
|
|
}
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
2018-04-25 16:45:29 +03:00
|
|
|
match &cx.target_spec().arch[..] {
|
|
|
|
"x86" => {
|
2022-07-19 10:40:26 -07:00
|
|
|
let flavor = if let spec::abi::Abi::Fastcall { .. }
|
|
|
|
| spec::abi::Abi::Vectorcall { .. } = abi
|
|
|
|
{
|
|
|
|
x86::Flavor::FastcallOrVectorcall
|
2018-04-25 16:45:29 +03:00
|
|
|
} else {
|
|
|
|
x86::Flavor::General
|
|
|
|
};
|
|
|
|
x86::compute_abi_info(cx, self, flavor);
|
|
|
|
}
|
2022-02-01 18:53:45 +01:00
|
|
|
"x86_64" => match abi {
|
|
|
|
spec::abi::Abi::SysV64 { .. } => x86_64::compute_abi_info(cx, self),
|
|
|
|
spec::abi::Abi::Win64 { .. } => x86_win64::compute_abi_info(self),
|
|
|
|
_ => {
|
|
|
|
if cx.target_spec().is_like_windows {
|
|
|
|
x86_win64::compute_abi_info(self)
|
|
|
|
} else {
|
|
|
|
x86_64::compute_abi_info(cx, self)
|
|
|
|
}
|
2019-12-22 17:42:04 -05:00
|
|
|
}
|
2022-02-01 18:53:45 +01:00
|
|
|
},
|
2022-06-06 12:40:10 -04:00
|
|
|
"aarch64" => {
|
2023-06-14 22:39:11 -04:00
|
|
|
let kind = if cx.target_spec().is_like_osx {
|
|
|
|
aarch64::AbiKind::DarwinPCS
|
|
|
|
} else if cx.target_spec().is_like_windows {
|
|
|
|
aarch64::AbiKind::Win64
|
2022-06-06 12:40:10 -04:00
|
|
|
} else {
|
2023-06-14 22:39:11 -04:00
|
|
|
aarch64::AbiKind::AAPCS
|
2022-06-06 12:40:10 -04:00
|
|
|
};
|
2023-06-14 22:39:11 -04:00
|
|
|
aarch64::compute_abi_info(cx, self, kind)
|
2022-06-06 12:40:10 -04:00
|
|
|
}
|
2018-07-18 22:01:19 -05:00
|
|
|
"amdgpu" => amdgpu::compute_abi_info(cx, self),
|
2018-04-25 16:45:29 +03:00
|
|
|
"arm" => arm::compute_abi_info(cx, self),
|
2016-05-06 09:32:10 -04:00
|
|
|
"avr" => avr::compute_abi_info(self),
|
2022-09-17 18:00:34 +08:00
|
|
|
"loongarch64" => loongarch::compute_abi_info(cx, self),
|
2021-08-25 07:45:06 +00:00
|
|
|
"m68k" => m68k::compute_abi_info(self),
|
2023-07-13 22:19:59 +08:00
|
|
|
"csky" => csky::compute_abi_info(self),
|
2023-06-28 13:35:39 +08:00
|
|
|
"mips" | "mips32r6" => mips::compute_abi_info(cx, self),
|
2023-06-28 13:24:01 +08:00
|
|
|
"mips64" | "mips64r6" => mips64::compute_abi_info(cx, self),
|
2019-09-04 20:40:18 -05:00
|
|
|
"powerpc" => powerpc::compute_abi_info(self),
|
2018-04-25 16:45:29 +03:00
|
|
|
"powerpc64" => powerpc64::compute_abi_info(cx, self),
|
|
|
|
"s390x" => s390x::compute_abi_info(cx, self),
|
|
|
|
"msp430" => msp430::compute_abi_info(self),
|
|
|
|
"sparc" => sparc::compute_abi_info(cx, self),
|
|
|
|
"sparc64" => sparc64::compute_abi_info(cx, self),
|
2022-03-07 15:09:28 +01:00
|
|
|
"nvptx64" => {
|
|
|
|
if cx.target_spec().adjust_abi(abi) == spec::abi::Abi::PtxKernel {
|
|
|
|
nvptx64::compute_ptx_kernel_abi_info(cx, self)
|
|
|
|
} else {
|
|
|
|
nvptx64::compute_abi_info(self)
|
|
|
|
}
|
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
"hexagon" => hexagon::compute_abi_info(self),
|
2020-01-21 21:52:19 +07:00
|
|
|
"riscv32" | "riscv64" => riscv::compute_abi_info(cx, self),
|
rustc: Add a new `wasm` ABI
This commit implements the idea of a new ABI for the WebAssembly target,
one called `"wasm"`. This ABI is entirely of my own invention
and has no current precedent, but I think that the addition of this ABI
might help solve a number of issues with the WebAssembly targets.
When `wasm32-unknown-unknown` was first added to Rust I naively
"implemented an abi" for the target. I then went to write `wasm-bindgen`
which accidentally relied on details of this ABI. Turns out the ABI
definition didn't match C, which is causing issues for C/Rust interop.
Currently the compiler has a "wasm32 bindgen compat" ABI which is the
original implementation I added, and it's purely there for, well,
`wasm-bindgen`.
Another issue with the WebAssembly target is that it's not clear to me
when and if the default C ABI will change to account for WebAssembly's
multi-value feature (a feature that allows functions to return multiple
values). Even if this does happen, though, it seems like the C ABI will
be guided based on the performance of WebAssembly code and will likely
not match even what the current wasm-bindgen-compat ABI is today. This
leaves a hole in Rust's expressivity in binding WebAssembly where given
a particular import type, Rust may not be able to import that signature
with an updated C ABI for multi-value.
To fix these issues I had the idea of a new ABI for WebAssembly, one
called `wasm`. The definition of this ABI is "what you write
maps straight to wasm". The goal here is that whatever you write down in
the parameter list or in the return values goes straight into the
function's signature in the WebAssembly file. This special ABI is for
intentionally matching the ABI of an imported function from the
environment or exporting a function with the right signature.
With the addition of a new ABI, this enables rustc to:
* Eventually remove the "wasm-bindgen compat hack". Once this
ABI is stable wasm-bindgen can switch to using it everywhere.
Afterwards the wasm32-unknown-unknown target can have its default ABI
updated to match C.
* Expose the ability to precisely match an ABI signature for a
WebAssembly function, regardless of what the C ABI that clang chooses
turns out to be.
* Continue to evolve the definition of the default C ABI to match what
clang does on all targets, since the purpose of that ABI will be
explicitly matching C rather than generating particular function
imports/exports.
Naturally this is implemented as an unstable feature initially, but it
would be nice for this to get stabilized (if it works) in the near-ish
future to remove the wasm32-unknown-unknown incompatibility with the C
ABI. Doing this, however, requires the feature to be on stable because
wasm-bindgen works with stable Rust.
2021-04-01 16:08:29 -07:00
|
|
|
"wasm32" | "wasm64" => {
|
|
|
|
if cx.target_spec().adjust_abi(abi) == spec::abi::Abi::Wasm {
|
|
|
|
wasm::compute_wasm_abi_info(self)
|
|
|
|
} else {
|
|
|
|
wasm::compute_c_abi_info(cx, self)
|
|
|
|
}
|
|
|
|
}
|
2021-04-20 19:03:10 +10:00
|
|
|
"bpf" => bpf::compute_abi_info(self),
|
2021-08-26 19:01:55 +03:00
|
|
|
arch => {
|
2022-02-01 18:44:45 +01:00
|
|
|
return Err(AdjustForForeignAbiError::Unsupported {
|
|
|
|
arch: Symbol::intern(arch),
|
|
|
|
abi,
|
|
|
|
});
|
2021-08-26 19:01:55 +03:00
|
|
|
}
|
2018-04-25 16:45:29 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
2022-08-25 17:55:21 +10:00
|
|
|
|
2022-11-05 14:36:38 +05:30
|
|
|
impl FromStr for Conv {
|
|
|
|
type Err = String;
|
|
|
|
|
|
|
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
|
|
|
match s {
|
|
|
|
"C" => Ok(Conv::C),
|
|
|
|
"Rust" => Ok(Conv::Rust),
|
|
|
|
"RustCold" => Ok(Conv::Rust),
|
|
|
|
"ArmAapcs" => Ok(Conv::ArmAapcs),
|
|
|
|
"CCmseNonSecureCall" => Ok(Conv::CCmseNonSecureCall),
|
|
|
|
"Msp430Intr" => Ok(Conv::Msp430Intr),
|
|
|
|
"PtxKernel" => Ok(Conv::PtxKernel),
|
|
|
|
"X86Fastcall" => Ok(Conv::X86Fastcall),
|
|
|
|
"X86Intr" => Ok(Conv::X86Intr),
|
|
|
|
"X86Stdcall" => Ok(Conv::X86Stdcall),
|
|
|
|
"X86ThisCall" => Ok(Conv::X86ThisCall),
|
|
|
|
"X86VectorCall" => Ok(Conv::X86VectorCall),
|
|
|
|
"X86_64SysV" => Ok(Conv::X86_64SysV),
|
|
|
|
"X86_64Win64" => Ok(Conv::X86_64Win64),
|
|
|
|
"AmdGpuKernel" => Ok(Conv::AmdGpuKernel),
|
|
|
|
"AvrInterrupt" => Ok(Conv::AvrInterrupt),
|
|
|
|
"AvrNonBlockingInterrupt" => Ok(Conv::AvrNonBlockingInterrupt),
|
feat: `riscv-interrupt-{m,s}` calling conventions
Similar to prior support added for the mips430, avr, and x86 targets
this change implements the rough equivalent of clang's
[`__attribute__((interrupt))`][clang-attr] for riscv targets, enabling
e.g.
```rust
static mut CNT: usize = 0;
pub extern "riscv-interrupt-m" fn isr_m() {
unsafe {
CNT += 1;
}
}
```
to produce highly effective assembly like:
```asm
pub extern "riscv-interrupt-m" fn isr_m() {
420003a0: 1141 addi sp,sp,-16
unsafe {
CNT += 1;
420003a2: c62a sw a0,12(sp)
420003a4: c42e sw a1,8(sp)
420003a6: 3fc80537 lui a0,0x3fc80
420003aa: 63c52583 lw a1,1596(a0) # 3fc8063c <_ZN12esp_riscv_rt3CNT17hcec3e3a214887d53E.0>
420003ae: 0585 addi a1,a1,1
420003b0: 62b52e23 sw a1,1596(a0)
}
}
420003b4: 4532 lw a0,12(sp)
420003b6: 45a2 lw a1,8(sp)
420003b8: 0141 addi sp,sp,16
420003ba: 30200073 mret
```
(disassembly via `riscv64-unknown-elf-objdump -C -S --disassemble ./esp32c3-hal/target/riscv32imc-unknown-none-elf/release/examples/gpio_interrupt`)
This outcome is superior to hand-coded interrupt routines which, lacking
visibility into any non-assembly body of the interrupt handler, have to
be very conservative and save the [entire CPU state to the stack
frame][full-frame-save]. By instead asking LLVM to only save the
registers that it uses, we defer the decision to the tool with the best
context: it can more accurately account for the cost of spills if it
knows that every additional register used is already at the cost of an
implicit spill.
At the LLVM level, this is apparently [implemented by] marking every
register as "[callee-save]," matching the semantics of an interrupt
handler nicely (it has to leave the CPU state just as it found it after
its `{m|s}ret`).
This approach is not suitable for every interrupt handler, as it makes
no attempt to e.g. save the state in a user-accessible stack frame. For
a full discussion of those challenges and tradeoffs, please refer to
[the interrupt calling conventions RFC][rfc].
Inside rustc, this implementation differs from prior art because LLVM
does not expose the "all-saved" function flavor as a calling convention
directly, instead preferring to use an attribute that allows for
differentiating between "machine-mode" and "superivsor-mode" interrupts.
Finally, some effort has been made to guide those who may not yet be
aware of the differences between machine-mode and supervisor-mode
interrupts as to why no `riscv-interrupt` calling convention is exposed
through rustc, and similarly for why `riscv-interrupt-u` makes no
appearance (as it would complicate future LLVM upgrades).
[clang-attr]: https://clang.llvm.org/docs/AttributeReference.html#interrupt-risc-v
[full-frame-save]: https://github.com/esp-rs/esp-riscv-rt/blob/9281af2ecffe13e40992917316f36920c26acaf3/src/lib.rs#L440-L469
[implemented by]: https://github.com/llvm/llvm-project/blob/b7fb2a3fec7c187d58a6d338ab512d9173bca987/llvm/lib/Target/RISCV/RISCVRegisterInfo.cpp#L61-L67
[callee-save]: https://github.com/llvm/llvm-project/blob/973f1fe7a8591c7af148e573491ab68cc15b6ecf/llvm/lib/Target/RISCV/RISCVCallingConv.td#L30-L37
[rfc]: https://github.com/rust-lang/rfcs/pull/3246
2023-05-23 15:08:23 -07:00
|
|
|
"RiscvInterrupt(machine)" => {
|
|
|
|
Ok(Conv::RiscvInterrupt { kind: RiscvInterruptKind::Machine })
|
|
|
|
}
|
|
|
|
"RiscvInterrupt(supervisor)" => {
|
|
|
|
Ok(Conv::RiscvInterrupt { kind: RiscvInterruptKind::Supervisor })
|
|
|
|
}
|
2023-04-10 22:02:52 +02:00
|
|
|
_ => Err(format!("'{s}' is not a valid value for entry function call convention.")),
|
2022-11-05 14:36:38 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-25 17:55:21 +10:00
|
|
|
// Some types are used a lot. Make sure they don't unintentionally get bigger.
|
|
|
|
#[cfg(all(target_arch = "x86_64", target_pointer_width = "64"))]
|
|
|
|
mod size_asserts {
|
|
|
|
use super::*;
|
|
|
|
use rustc_data_structures::static_assert_size;
|
2022-10-05 21:46:21 +02:00
|
|
|
// tidy-alphabetical-start
|
2022-08-25 22:19:38 +10:00
|
|
|
static_assert_size!(ArgAbi<'_, usize>, 56);
|
|
|
|
static_assert_size!(FnAbi<'_, usize>, 80);
|
2022-10-05 21:46:21 +02:00
|
|
|
// tidy-alphabetical-end
|
2022-08-25 17:55:21 +10:00
|
|
|
}
|