Skip to content

Commit b1613eb

Browse files
committed
codegen: panic when trying to compute size/align of extern type
1 parent 8b1ba11 commit b1613eb

File tree

7 files changed

+54
-25
lines changed

7 files changed

+54
-25
lines changed

compiler/rustc_codegen_ssa/src/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -51,10 +51,10 @@ pub mod codegen_attrs;
5151
pub mod common;
5252
pub mod debuginfo;
5353
pub mod errors;
54-
pub mod glue;
5554
pub mod meth;
5655
pub mod mir;
5756
pub mod mono_item;
57+
pub mod size_of_val;
5858
pub mod target_features;
5959
pub mod traits;
6060

compiler/rustc_codegen_ssa/src/mir/intrinsic.rs

+9-13
Original file line numberDiff line numberDiff line change
@@ -4,8 +4,8 @@ use super::FunctionCx;
44
use crate::common::IntPredicate;
55
use crate::errors;
66
use crate::errors::InvalidMonomorphization;
7-
use crate::glue;
87
use crate::meth;
8+
use crate::size_of_val;
99
use crate::traits::*;
1010
use crate::MemFlags;
1111

@@ -88,21 +88,17 @@ impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
8888
sym::va_end => bx.va_end(args[0].immediate()),
8989
sym::size_of_val => {
9090
let tp_ty = fn_args.type_at(0);
91-
if let OperandValue::Pair(_, meta) = args[0].val {
92-
let (llsize, _) = glue::size_and_align_of_dst(bx, tp_ty, Some(meta));
93-
llsize
94-
} else {
95-
bx.const_usize(bx.layout_of(tp_ty).size.bytes())
96-
}
91+
let meta =
92+
if let OperandValue::Pair(_, meta) = args[0].val { Some(meta) } else { None };
93+
let (llsize, _) = size_of_val::size_and_align_of_dst(bx, tp_ty, meta);
94+
llsize
9795
}
9896
sym::min_align_of_val => {
9997
let tp_ty = fn_args.type_at(0);
100-
if let OperandValue::Pair(_, meta) = args[0].val {
101-
let (_, llalign) = glue::size_and_align_of_dst(bx, tp_ty, Some(meta));
102-
llalign
103-
} else {
104-
bx.const_usize(bx.layout_of(tp_ty).align.abi.bytes())
105-
}
98+
let meta =
99+
if let OperandValue::Pair(_, meta) = args[0].val { Some(meta) } else { None };
100+
let (_, llalign) = size_of_val::size_and_align_of_dst(bx, tp_ty, meta);
101+
llalign
106102
}
107103
sym::vtable_size | sym::vtable_align => {
108104
let vtable = args[0].immediate();

compiler/rustc_codegen_ssa/src/mir/operand.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ use super::place::PlaceRef;
22
use super::{FunctionCx, LocalRef};
33

44
use crate::base;
5-
use crate::glue;
5+
use crate::size_of_val;
66
use crate::traits::*;
77
use crate::MemFlags;
88

@@ -466,13 +466,13 @@ impl<'a, 'tcx, V: CodegenObject> OperandValue<V> {
466466
.ty;
467467

468468
let OperandValue::Ref(llptr, Some(llextra), _) = self else {
469-
bug!("store_unsized called with a sized value")
469+
bug!("store_unsized called with a sized value (or with an extern type)")
470470
};
471471

472472
// Allocate an appropriate region on the stack, and copy the value into it. Since alloca
473473
// doesn't support dynamic alignment, we allocate an extra align - 1 bytes, and align the
474474
// pointer manually.
475-
let (size, align) = glue::size_and_align_of_dst(bx, unsized_ty, Some(llextra));
475+
let (size, align) = size_of_val::size_and_align_of_dst(bx, unsized_ty, Some(llextra));
476476
let one = bx.const_usize(1);
477477
let align_minus_1 = bx.sub(align, one);
478478
let size_extra = bx.add(size, align_minus_1);

compiler/rustc_codegen_ssa/src/mir/place.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ use super::operand::OperandValue;
22
use super::{FunctionCx, LocalRef};
33

44
use crate::common::IntPredicate;
5-
use crate::glue;
5+
use crate::size_of_val;
66
use crate::traits::*;
77

88
use rustc_middle::mir;
@@ -179,7 +179,7 @@ impl<'a, 'tcx, V: CodegenObject> PlaceRef<'tcx, V> {
179179
let unaligned_offset = bx.cx().const_usize(offset.bytes());
180180

181181
// Get the alignment of the field
182-
let (_, mut unsized_align) = glue::size_and_align_of_dst(bx, field.ty, meta);
182+
let (_, mut unsized_align) = size_of_val::size_and_align_of_dst(bx, field.ty, meta);
183183

184184
// For packed types, we need to cap alignment.
185185
if let ty::Adt(def, _) = self.layout.ty.kind()

compiler/rustc_codegen_ssa/src/glue.rs compiler/rustc_codegen_ssa/src/size_of_val.rs

+31-5
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,11 @@
1-
//!
2-
//
3-
// Code relating to drop glue.
1+
//! Computing the size and alignment of a value.
42
3+
use crate::common;
54
use crate::common::IntPredicate;
65
use crate::meth;
76
use crate::traits::*;
7+
use rustc_hir::LangItem;
8+
use rustc_middle::ty::print::{with_no_trimmed_paths, with_no_visible_paths};
89
use rustc_middle::ty::{self, Ty};
910
use rustc_target::abi::WrappingRange;
1011

@@ -14,7 +15,7 @@ pub fn size_and_align_of_dst<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>>(
1415
info: Option<Bx::Value>,
1516
) -> (Bx::Value, Bx::Value) {
1617
let layout = bx.layout_of(t);
17-
debug!("size_and_align_of_dst(ty={}, info={:?}): layout: {:?}", t, info, layout);
18+
trace!("size_and_align_of_dst(ty={}, info={:?}): layout: {:?}", t, info, layout);
1819
if layout.is_sized() {
1920
let size = bx.const_usize(layout.size.bytes());
2021
let align = bx.const_usize(layout.align.abi.bytes());
@@ -51,7 +52,31 @@ pub fn size_and_align_of_dst<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>>(
5152
bx.const_usize(unit.align.abi.bytes()),
5253
)
5354
}
54-
_ => {
55+
ty::Foreign(_) => {
56+
// `extern` type. We cannot compute the size, so panic.
57+
let msg_str = with_no_visible_paths!({
58+
with_no_trimmed_paths!({
59+
format!("attempted to compute the size or alignment of extern type `{t}`")
60+
})
61+
});
62+
let msg = bx.const_str(&msg_str);
63+
64+
// Obtain the panic entry point.
65+
let (fn_abi, llfn) = common::build_langcall(bx, None, LangItem::PanicNounwind);
66+
67+
// Generate the call.
68+
// Cannot use `do_call` since we don't have a MIR terminator so we can't create a `TerminationCodegenHelper`.
69+
// (But we are in good company, this code is duplicated plenty of times.)
70+
let fn_ty = bx.fn_decl_backend_type(fn_abi);
71+
72+
bx.call(fn_ty, /* fn_attrs */ None, Some(fn_abi), llfn, &[msg.0, msg.1], None);
73+
74+
// This function does not return so we can now return whatever we want.
75+
let size = bx.const_usize(layout.size.bytes());
76+
let align = bx.const_usize(layout.align.abi.bytes());
77+
(size, align)
78+
}
79+
ty::Adt(..) | ty::Tuple(..) => {
5580
// First get the size of all statically known fields.
5681
// Don't use size_of because it also rounds up to alignment, which we
5782
// want to avoid, as the unsized field's alignment could be smaller.
@@ -122,5 +147,6 @@ pub fn size_and_align_of_dst<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>>(
122147

123148
(size, align)
124149
}
150+
_ => bug!("size_and_align_of_dst: {t} not supported"),
125151
}
126152
}

tests/ui/extern/extern-types-size_of_val.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
1-
// run-pass
1+
// run-fail
2+
// check-run-results
3+
// normalize-stderr-test "panicking\.rs:\d+:\d+:" -> "panicking.rs:"
24
#![feature(extern_types)]
35

46
use std::mem::{align_of_val, size_of_val};
@@ -10,6 +12,7 @@ extern "C" {
1012
fn main() {
1113
let x: &A = unsafe { &*(1usize as *const A) };
1214

15+
// These don't have a dynamic size, so this should panic.
1316
assert_eq!(size_of_val(x), 0);
1417
assert_eq!(align_of_val(x), 1);
1518
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
thread 'main' panicked at library/core/src/panicking.rs:
2+
attempted to compute the size or alignment of extern type `A`
3+
note: run with `RUST_BACKTRACE=1` environment variable to display a backtrace
4+
thread caused non-unwinding panic. aborting.

0 commit comments

Comments
 (0)