Skip to content

Commit b8bd968

Browse files
committed
Add #[rustc_pass_indirectly_in_non_rustic_abis]
1 parent c0bb3b9 commit b8bd968

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

41 files changed

+431
-56
lines changed

compiler/rustc_abi/src/layout/ty.rs

Lines changed: 20 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -172,6 +172,8 @@ pub trait TyAbiInterface<'a, C>: Sized + std::fmt::Debug {
172172
fn is_tuple(this: TyAndLayout<'a, Self>) -> bool;
173173
fn is_unit(this: TyAndLayout<'a, Self>) -> bool;
174174
fn is_transparent(this: TyAndLayout<'a, Self>) -> bool;
175+
/// See [`TyAndLayout::pass_indirectly_in_non_rustic_abis`] for details.
176+
fn is_pass_indirectly_in_non_rustic_abis_flag_set(this: TyAndLayout<'a, Self>) -> bool;
175177
}
176178

177179
impl<'a, Ty> TyAndLayout<'a, Ty> {
@@ -269,6 +271,24 @@ impl<'a, Ty> TyAndLayout<'a, Ty> {
269271
Ty::is_transparent(self)
270272
}
271273

274+
/// If this method returns `true`, then this type should always have a `PassMode` of
275+
/// `Indirect { on_stack: false, .. }` when being used as the argument type of a function with a
276+
/// non-Rustic ABI (this is true for structs annotated with the
277+
/// `#[rustc_pass_indirectly_in_non_rustic_abis]` attribute).
278+
///
279+
/// This function handles transparent types automatically.
280+
pub fn pass_indirectly_in_non_rustic_abis<C>(mut self, cx: &C) -> bool
281+
where
282+
Ty: TyAbiInterface<'a, C> + Copy,
283+
{
284+
while self.is_transparent()
285+
&& let Some((_, field)) = self.non_1zst_field(cx)
286+
{
287+
self = field;
288+
}
289+
Ty::is_pass_indirectly_in_non_rustic_abis_flag_set(self)
290+
}
291+
272292
/// Finds the one field that is not a 1-ZST.
273293
/// Returns `None` if there are multiple non-1-ZST fields or only 1-ZST-fields.
274294
pub fn non_1zst_field<C>(&self, cx: &C) -> Option<(FieldIdx, Self)>

compiler/rustc_abi/src/lib.rs

Lines changed: 9 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -86,14 +86,17 @@ bitflags! {
8686
const IS_C = 1 << 0;
8787
const IS_SIMD = 1 << 1;
8888
const IS_TRANSPARENT = 1 << 2;
89-
// Internal only for now. If true, don't reorder fields.
90-
// On its own it does not prevent ABI optimizations.
89+
/// Internal only for now. If true, don't reorder fields.
90+
/// On its own it does not prevent ABI optimizations.
9191
const IS_LINEAR = 1 << 3;
92-
// If true, the type's crate has opted into layout randomization.
93-
// Other flags can still inhibit reordering and thus randomization.
94-
// The seed stored in `ReprOptions.field_shuffle_seed`.
92+
/// If true, the type's crate has opted into layout randomization.
93+
/// Other flags can still inhibit reordering and thus randomization.
94+
/// The seed stored in `ReprOptions.field_shuffle_seed`.
9595
const RANDOMIZE_LAYOUT = 1 << 4;
96-
// Any of these flags being set prevent field reordering optimisation.
96+
/// If true, the type is always passed indirectly by non-Rustic ABIs.
97+
/// See [`TyAndLayout::pass_indirectly_in_non_rustic_abis`] for details.
98+
const PASS_INDIRECTLY_IN_NON_RUSTIC_ABIS = 1 << 5;
99+
/// Any of these flags being set prevent field reordering optimisation.
97100
const FIELD_ORDER_UNOPTIMIZABLE = ReprFlags::IS_C.bits()
98101
| ReprFlags::IS_SIMD.bits()
99102
| ReprFlags::IS_LINEAR.bits();

compiler/rustc_attr_parsing/src/attributes/codegen_attrs.rs

Lines changed: 9 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -585,3 +585,12 @@ impl<S: Stage> SingleAttributeParser<S> for SanitizeParser {
585585
Some(AttributeKind::Sanitize { on_set, off_set, span: cx.attr_span })
586586
}
587587
}
588+
589+
pub(crate) struct RustcPassIndirectlyInNonRusticAbisParser;
590+
591+
impl<S: Stage> NoArgsAttributeParser<S> for RustcPassIndirectlyInNonRusticAbisParser {
592+
const PATH: &[Symbol] = &[sym::rustc_pass_indirectly_in_non_rustic_abis];
593+
const ON_DUPLICATE: OnDuplicate<S> = OnDuplicate::Error;
594+
const ALLOWED_TARGETS: AllowedTargets = AllowedTargets::AllowList(&[Allow(Target::Struct)]);
595+
const CREATE: fn(Span) -> AttributeKind = AttributeKind::RustcPassIndirectlyInNonRusticAbis;
596+
}

compiler/rustc_attr_parsing/src/context.rs

Lines changed: 3 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -20,8 +20,8 @@ use crate::attributes::allow_unstable::{
2020
use crate::attributes::body::CoroutineParser;
2121
use crate::attributes::codegen_attrs::{
2222
ColdParser, CoverageParser, ExportNameParser, ForceTargetFeatureParser, NakedParser,
23-
NoMangleParser, OptimizeParser, SanitizeParser, TargetFeatureParser, TrackCallerParser,
24-
UsedParser,
23+
NoMangleParser, OptimizeParser, RustcPassIndirectlyInNonRusticAbisParser, SanitizeParser,
24+
TargetFeatureParser, TrackCallerParser, UsedParser,
2525
};
2626
use crate::attributes::confusables::ConfusablesParser;
2727
use crate::attributes::crate_level::CrateNameParser;
@@ -224,6 +224,7 @@ attribute_parsers!(
224224
Single<WithoutArgs<ProcMacroAttributeParser>>,
225225
Single<WithoutArgs<ProcMacroParser>>,
226226
Single<WithoutArgs<PubTransparentParser>>,
227+
Single<WithoutArgs<RustcPassIndirectlyInNonRusticAbisParser>>,
227228
Single<WithoutArgs<SpecializationTraitParser>>,
228229
Single<WithoutArgs<StdInternalSymbolParser>>,
229230
Single<WithoutArgs<TrackCallerParser>>,

compiler/rustc_feature/src/builtin_attrs.rs

Lines changed: 6 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -651,6 +651,12 @@ pub static BUILTIN_ATTRIBUTES: &[BuiltinAttribute] = &[
651651
template!(Word, "https://doc.rust-lang.org/reference/attributes/codegen.html#the-naked-attribute"),
652652
WarnFollowing, EncodeCrossCrate::No
653653
),
654+
// See `TyAndLayout::pass_indirectly_in_non_rustic_abis` for details.
655+
rustc_attr!(
656+
rustc_pass_indirectly_in_non_rustic_abis, Normal, template!(Word), ErrorFollowing,
657+
EncodeCrossCrate::No,
658+
"types marked with `#[rustc_pass_indirectly_in_non_rustic_abis]` are always passed indirectly by non-Rustic abis."
659+
),
654660

655661
// Limits:
656662
ungated!(

compiler/rustc_hir/src/attrs/data_structures.rs

Lines changed: 3 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -626,6 +626,9 @@ pub enum AttributeKind {
626626
/// Represents `#[rustc_object_lifetime_default]`.
627627
RustcObjectLifetimeDefault,
628628

629+
/// Represents `#[rustc_pass_indirectly_in_non_rustic_abis]`
630+
RustcPassIndirectlyInNonRusticAbis(Span),
631+
629632
/// Represents `#[sanitize]`
630633
///
631634
/// the on set and off set are distjoint since there's a third option: unset.

compiler/rustc_hir/src/attrs/encode_cross_crate.rs

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,7 @@ impl AttributeKind {
8080
RustcLayoutScalarValidRangeEnd(..) => Yes,
8181
RustcLayoutScalarValidRangeStart(..) => Yes,
8282
RustcObjectLifetimeDefault => No,
83+
RustcPassIndirectlyInNonRusticAbis(..) => No,
8384
Sanitize { .. } => No,
8485
ShouldPanic { .. } => No,
8586
SkipDuringMethodDispatch { .. } => No,

compiler/rustc_middle/src/ty/layout.rs

Lines changed: 6 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ use std::{cmp, fmt};
33

44
use rustc_abi::{
55
AddressSpace, Align, ExternAbi, FieldIdx, FieldsShape, HasDataLayout, LayoutData, PointeeInfo,
6-
PointerKind, Primitive, ReprOptions, Scalar, Size, TagEncoding, TargetDataLayout,
6+
PointerKind, Primitive, ReprFlags, ReprOptions, Scalar, Size, TagEncoding, TargetDataLayout,
77
TyAbiInterface, VariantIdx, Variants,
88
};
99
use rustc_error_messages::DiagMessage;
@@ -1138,6 +1138,11 @@ where
11381138
fn is_transparent(this: TyAndLayout<'tcx>) -> bool {
11391139
matches!(this.ty.kind(), ty::Adt(def, _) if def.repr().transparent())
11401140
}
1141+
1142+
/// See [`TyAndLayout::pass_indirectly_in_non_rustic_abis`] for details.
1143+
fn is_pass_indirectly_in_non_rustic_abis_flag_set(this: TyAndLayout<'tcx>) -> bool {
1144+
matches!(this.ty.kind(), ty::Adt(def, _) if def.repr().flags.contains(ReprFlags::PASS_INDIRECTLY_IN_NON_RUSTIC_ABIS))
1145+
}
11411146
}
11421147

11431148
/// Calculates whether a function's ABI can unwind or not.

compiler/rustc_middle/src/ty/mod.rs

Lines changed: 8 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1530,6 +1530,14 @@ impl<'tcx> TyCtxt<'tcx> {
15301530
flags.insert(ReprFlags::IS_LINEAR);
15311531
}
15321532

1533+
// See `TyAndLayout::pass_indirectly_in_non_rustic_abis` for details.
1534+
if find_attr!(
1535+
self.get_all_attrs(did),
1536+
AttributeKind::RustcPassIndirectlyInNonRusticAbis(..)
1537+
) {
1538+
flags.insert(ReprFlags::PASS_INDIRECTLY_IN_NON_RUSTIC_ABIS);
1539+
}
1540+
15331541
ReprOptions { int: size, align: max_align, pack: min_pack, flags, field_shuffle_seed }
15341542
}
15351543

compiler/rustc_passes/src/check_attr.rs

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -270,6 +270,7 @@ impl<'tcx> CheckAttrVisitor<'tcx> {
270270
| AttributeKind::Linkage(..)
271271
| AttributeKind::MustUse { .. }
272272
| AttributeKind::CrateName { .. }
273+
| AttributeKind::RustcPassIndirectlyInNonRusticAbis(..)
273274
) => { /* do nothing */ }
274275
Attribute::Unparsed(attr_item) => {
275276
style = Some(attr_item.style);

0 commit comments

Comments
 (0)