Skip to content

Commit e102148

Browse files
committed
Auto merge of #121577 - erikdesjardins:struct, r=<try>
Stop using LLVM struct types for alloca, byval, sret, and many GEPs This is an extension of #98615, extending the removal from field offsets to most places that it's feasible right now. (It might make sense to split this PR up, but I want to test perf with everything.) For `alloca`, `byval`, and `sret`, the type has no semantic meaning, only the size matters\*†. Using `[N x i8]` is a more direct way to specify that we want `N` bytes, and avoids relying on LLVM's layout algorithm. Particularly for `alloca`, it is likely that a future LLVM will change to a representation where you only specify the size. For GEPs, upstream LLVM is in the beginning stages of [migrating to `ptradd`](https://discourse.llvm.org/t/rfc-replacing-getelementptr-with-ptradd/68699). LLVM 19 will [canonicalize](llvm/llvm-project#68882) all constant-offset GEPs to i8, which is the same thing we do here. \*: Since we always explicitly specify the alignment. For `byval`, this wasn't the case until #112157. †: For `byval`, the hidden copy may be impacted by padding in the LLVM struct type, i.e. padding bytes may not be copied. (I'm not sure if this is done today, but I think it would be legal.) But we manually pad our LLVM struct types specifically to avoid there ever being LLVM-visible padding, so that shouldn't be an issue here. r? `@ghost`
2 parents 34aab62 + 3ae7ba6 commit e102148

Some content is hidden

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

42 files changed

+268
-333
lines changed

compiler/rustc_codegen_gcc/src/builder.rs

+21-34
Original file line numberDiff line numberDiff line change
@@ -734,7 +734,18 @@ impl<'a, 'gcc, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'gcc, 'tcx> {
734734
self.gcc_checked_binop(oop, typ, lhs, rhs)
735735
}
736736

737-
fn alloca(&mut self, ty: Type<'gcc>, align: Align) -> RValue<'gcc> {
737+
fn alloca(&mut self, size: Size, align: Align) -> RValue<'gcc> {
738+
let ty = self.cx.type_array(self.cx.type_i8(), size.bytes()).get_aligned(align.bytes());
739+
// TODO(antoyo): It might be better to return a LValue, but fixing the rustc API is non-trivial.
740+
self.stack_var_count.set(self.stack_var_count.get() + 1);
741+
self.current_func().new_local(None, ty, &format!("stack_var_{}", self.stack_var_count.get())).get_address(None)
742+
}
743+
744+
fn dynamic_alloca(&mut self, _len: RValue<'gcc>, _align: Align) -> RValue<'gcc> {
745+
unimplemented!();
746+
}
747+
748+
fn typed_alloca(&mut self, ty: Type<'gcc>, align: Align) -> RValue<'gcc> {
738749
// FIXME(antoyo): this check that we don't call get_aligned() a second time on a type.
739750
// Ideally, we shouldn't need to do this check.
740751
let aligned_type =
@@ -749,10 +760,6 @@ impl<'a, 'gcc, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'gcc, 'tcx> {
749760
self.current_func().new_local(None, aligned_type, &format!("stack_var_{}", self.stack_var_count.get())).get_address(None)
750761
}
751762

752-
fn byte_array_alloca(&mut self, _len: RValue<'gcc>, _align: Align) -> RValue<'gcc> {
753-
unimplemented!();
754-
}
755-
756763
fn load(&mut self, pointee_ty: Type<'gcc>, ptr: RValue<'gcc>, align: Align) -> RValue<'gcc> {
757764
let block = self.llbb();
758765
let function = block.get_function();
@@ -834,10 +841,17 @@ impl<'a, 'gcc, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'gcc, 'tcx> {
834841
}
835842
else if let abi::Abi::ScalarPair(ref a, ref b) = place.layout.abi {
836843
let b_offset = a.size(self).align_to(b.align(self).abi);
837-
let pair_type = place.layout.gcc_type(self);
838844

839845
let mut load = |i, scalar: &abi::Scalar, align| {
840-
let llptr = self.struct_gep(pair_type, place.llval, i as u64);
846+
let llptr = if i == 0 {
847+
place.llval
848+
} else {
849+
self.inbounds_gep(
850+
self.type_i8(),
851+
place.llval,
852+
&[self.const_usize(b_offset.bytes())],
853+
)
854+
};
841855
let llty = place.layout.scalar_pair_element_gcc_type(self, i);
842856
let load = self.load(llty, llptr, align);
843857
scalar_load_metadata(self, load, scalar);
@@ -971,33 +985,6 @@ impl<'a, 'gcc, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'gcc, 'tcx> {
971985
result.get_address(None)
972986
}
973987

974-
fn struct_gep(&mut self, value_type: Type<'gcc>, ptr: RValue<'gcc>, idx: u64) -> RValue<'gcc> {
975-
// FIXME(antoyo): it would be better if the API only called this on struct, not on arrays.
976-
assert_eq!(idx as usize as u64, idx);
977-
let value = ptr.dereference(None).to_rvalue();
978-
979-
if value_type.dyncast_array().is_some() {
980-
let index = self.context.new_rvalue_from_long(self.u64_type, i64::try_from(idx).expect("i64::try_from"));
981-
let element = self.context.new_array_access(None, value, index);
982-
element.get_address(None)
983-
}
984-
else if let Some(vector_type) = value_type.dyncast_vector() {
985-
let array_type = vector_type.get_element_type().make_pointer();
986-
let array = self.bitcast(ptr, array_type);
987-
let index = self.context.new_rvalue_from_long(self.u64_type, i64::try_from(idx).expect("i64::try_from"));
988-
let element = self.context.new_array_access(None, array, index);
989-
element.get_address(None)
990-
}
991-
else if let Some(struct_type) = value_type.is_struct() {
992-
// NOTE: due to opaque pointers now being used, we need to bitcast here.
993-
let ptr = self.bitcast_if_needed(ptr, value_type.make_pointer());
994-
ptr.dereference_field(None, struct_type.get_field(idx as i32)).get_address(None)
995-
}
996-
else {
997-
panic!("Unexpected type {:?}", value_type);
998-
}
999-
}
1000-
1001988
/* Casts */
1002989
fn trunc(&mut self, value: RValue<'gcc>, dest_ty: Type<'gcc>) -> RValue<'gcc> {
1003990
// TODO(antoyo): check that it indeed truncate the value.

compiler/rustc_codegen_gcc/src/intrinsic/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -490,7 +490,7 @@ impl<'gcc, 'tcx> ArgAbiExt<'gcc, 'tcx> for ArgAbi<'tcx, Ty<'tcx>> {
490490
// We instead thus allocate some scratch space...
491491
let scratch_size = cast.size(bx);
492492
let scratch_align = cast.align(bx);
493-
let llscratch = bx.alloca(cast.gcc_type(bx), scratch_align);
493+
let llscratch = bx.alloca(scratch_size, scratch_align);
494494
bx.lifetime_start(llscratch, scratch_size);
495495

496496
// ... where we first store the value...

compiler/rustc_codegen_gcc/src/intrinsic/simd.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ use rustc_middle::span_bug;
1616
use rustc_middle::ty::layout::HasTyCtxt;
1717
use rustc_middle::ty::{self, Ty};
1818
use rustc_span::{sym, Span, Symbol};
19-
use rustc_target::abi::Align;
19+
use rustc_target::abi::{Align, Size};
2020

2121
use crate::builder::Builder;
2222
#[cfg(feature = "master")]
@@ -363,7 +363,7 @@ pub fn generic_simd_intrinsic<'a, 'gcc, 'tcx>(
363363
let ze = bx.zext(result, bx.type_ix(expected_bytes * 8));
364364

365365
// Convert the integer to a byte array
366-
let ptr = bx.alloca(bx.type_ix(expected_bytes * 8), Align::ONE);
366+
let ptr = bx.alloca(Size::from_bytes(expected_bytes), Align::ONE);
367367
bx.store(ze, ptr, Align::ONE);
368368
let array_ty = bx.type_array(bx.type_i8(), expected_bytes);
369369
let ptr = bx.pointercast(ptr, bx.cx.type_ptr_to(array_ty));

compiler/rustc_codegen_gcc/src/type_of.rs

-23
Original file line numberDiff line numberDiff line change
@@ -151,7 +151,6 @@ pub trait LayoutGccExt<'tcx> {
151151
fn immediate_gcc_type<'gcc>(&self, cx: &CodegenCx<'gcc, 'tcx>) -> Type<'gcc>;
152152
fn scalar_gcc_type_at<'gcc>(&self, cx: &CodegenCx<'gcc, 'tcx>, scalar: &abi::Scalar, offset: Size) -> Type<'gcc>;
153153
fn scalar_pair_element_gcc_type<'gcc>(&self, cx: &CodegenCx<'gcc, 'tcx>, index: usize) -> Type<'gcc>;
154-
fn gcc_field_index(&self, index: usize) -> u64;
155154
fn pointee_info_at<'gcc>(&self, cx: &CodegenCx<'gcc, 'tcx>, offset: Size) -> Option<PointeeInfo>;
156155
}
157156

@@ -304,24 +303,6 @@ impl<'tcx> LayoutGccExt<'tcx> for TyAndLayout<'tcx> {
304303
self.scalar_gcc_type_at(cx, scalar, offset)
305304
}
306305

307-
fn gcc_field_index(&self, index: usize) -> u64 {
308-
match self.abi {
309-
Abi::Scalar(_) | Abi::ScalarPair(..) => {
310-
bug!("TyAndLayout::gcc_field_index({:?}): not applicable", self)
311-
}
312-
_ => {}
313-
}
314-
match self.fields {
315-
FieldsShape::Primitive | FieldsShape::Union(_) => {
316-
bug!("TyAndLayout::gcc_field_index({:?}): not applicable", self)
317-
}
318-
319-
FieldsShape::Array { .. } => index as u64,
320-
321-
FieldsShape::Arbitrary { .. } => 1 + (self.fields.memory_index(index) as u64) * 2,
322-
}
323-
}
324-
325306
fn pointee_info_at<'a>(&self, cx: &CodegenCx<'a, 'tcx>, offset: Size) -> Option<PointeeInfo> {
326307
if let Some(&pointee) = cx.pointee_infos.borrow().get(&(self.ty, offset)) {
327308
return pointee;
@@ -351,10 +332,6 @@ impl<'gcc, 'tcx> LayoutTypeMethods<'tcx> for CodegenCx<'gcc, 'tcx> {
351332
layout.is_gcc_scalar_pair()
352333
}
353334

354-
fn backend_field_index(&self, layout: TyAndLayout<'tcx>, index: usize) -> u64 {
355-
layout.gcc_field_index(index)
356-
}
357-
358335
fn scalar_pair_element_backend_type(&self, layout: TyAndLayout<'tcx>, index: usize, _immediate: bool) -> Type<'gcc> {
359336
layout.scalar_pair_element_gcc_type(self, index)
360337
}

compiler/rustc_codegen_llvm/src/abi.rs

+17-5
Original file line numberDiff line numberDiff line change
@@ -234,7 +234,7 @@ impl<'ll, 'tcx> ArgAbiExt<'ll, 'tcx> for ArgAbi<'tcx, Ty<'tcx>> {
234234
// We instead thus allocate some scratch space...
235235
let scratch_size = cast.size(bx);
236236
let scratch_align = cast.align(bx);
237-
let llscratch = bx.alloca(cast.llvm_type(bx), scratch_align);
237+
let llscratch = bx.alloca(scratch_size, scratch_align);
238238
bx.lifetime_start(llscratch, scratch_size);
239239

240240
// ... where we first store the value...
@@ -424,7 +424,10 @@ impl<'ll, 'tcx> FnAbiLlvmExt<'ll, 'tcx> for FnAbi<'tcx, Ty<'tcx>> {
424424
PassMode::Indirect { attrs, meta_attrs: _, on_stack } => {
425425
assert!(!on_stack);
426426
let i = apply(attrs);
427-
let sret = llvm::CreateStructRetAttr(cx.llcx, self.ret.layout.llvm_type(cx));
427+
let sret = llvm::CreateStructRetAttr(
428+
cx.llcx,
429+
cx.type_array(cx.type_i8(), self.ret.layout.size.bytes()),
430+
);
428431
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Argument(i), &[sret]);
429432
}
430433
PassMode::Cast { cast, pad_i32: _ } => {
@@ -437,7 +440,10 @@ impl<'ll, 'tcx> FnAbiLlvmExt<'ll, 'tcx> for FnAbi<'tcx, Ty<'tcx>> {
437440
PassMode::Ignore => {}
438441
PassMode::Indirect { attrs, meta_attrs: None, on_stack: true } => {
439442
let i = apply(attrs);
440-
let byval = llvm::CreateByValAttr(cx.llcx, arg.layout.llvm_type(cx));
443+
let byval = llvm::CreateByValAttr(
444+
cx.llcx,
445+
cx.type_array(cx.type_i8(), arg.layout.size.bytes()),
446+
);
441447
attributes::apply_to_llfn(llfn, llvm::AttributePlace::Argument(i), &[byval]);
442448
}
443449
PassMode::Direct(attrs)
@@ -486,7 +492,10 @@ impl<'ll, 'tcx> FnAbiLlvmExt<'ll, 'tcx> for FnAbi<'tcx, Ty<'tcx>> {
486492
PassMode::Indirect { attrs, meta_attrs: _, on_stack } => {
487493
assert!(!on_stack);
488494
let i = apply(bx.cx, attrs);
489-
let sret = llvm::CreateStructRetAttr(bx.cx.llcx, self.ret.layout.llvm_type(bx));
495+
let sret = llvm::CreateStructRetAttr(
496+
bx.cx.llcx,
497+
bx.cx.type_array(bx.cx.type_i8(), self.ret.layout.size.bytes()),
498+
);
490499
attributes::apply_to_callsite(callsite, llvm::AttributePlace::Argument(i), &[sret]);
491500
}
492501
PassMode::Cast { cast, pad_i32: _ } => {
@@ -513,7 +522,10 @@ impl<'ll, 'tcx> FnAbiLlvmExt<'ll, 'tcx> for FnAbi<'tcx, Ty<'tcx>> {
513522
PassMode::Ignore => {}
514523
PassMode::Indirect { attrs, meta_attrs: None, on_stack: true } => {
515524
let i = apply(bx.cx, attrs);
516-
let byval = llvm::CreateByValAttr(bx.cx.llcx, arg.layout.llvm_type(bx));
525+
let byval = llvm::CreateByValAttr(
526+
bx.cx.llcx,
527+
bx.cx.type_array(bx.cx.type_i8(), arg.layout.size.bytes()),
528+
);
517529
attributes::apply_to_callsite(
518530
callsite,
519531
llvm::AttributePlace::Argument(i),

compiler/rustc_codegen_llvm/src/builder.rs

+16-13
Original file line numberDiff line numberDiff line change
@@ -466,7 +466,9 @@ impl<'a, 'll, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'll, 'tcx> {
466466
val
467467
}
468468

469-
fn alloca(&mut self, ty: &'ll Type, align: Align) -> &'ll Value {
469+
fn alloca(&mut self, size: Size, align: Align) -> &'ll Value {
470+
let ty = self.cx().type_array(self.cx().type_i8(), size.bytes());
471+
470472
let mut bx = Builder::with_cx(self.cx);
471473
bx.position_at_start(unsafe { llvm::LLVMGetFirstBasicBlock(self.llfn()) });
472474
unsafe {
@@ -476,10 +478,20 @@ impl<'a, 'll, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'll, 'tcx> {
476478
}
477479
}
478480

479-
fn byte_array_alloca(&mut self, len: &'ll Value, align: Align) -> &'ll Value {
481+
fn dynamic_alloca(&mut self, size: &'ll Value, align: Align) -> &'ll Value {
480482
unsafe {
481483
let alloca =
482-
llvm::LLVMBuildArrayAlloca(self.llbuilder, self.cx().type_i8(), len, UNNAMED);
484+
llvm::LLVMBuildArrayAlloca(self.llbuilder, self.cx().type_i8(), size, UNNAMED);
485+
llvm::LLVMSetAlignment(alloca, align.bytes() as c_uint);
486+
alloca
487+
}
488+
}
489+
490+
fn typed_alloca(&mut self, ty: &'ll Type, align: Align) -> &'ll Value {
491+
let mut bx = Builder::with_cx(self.cx);
492+
bx.position_at_start(unsafe { llvm::LLVMGetFirstBasicBlock(self.llfn()) });
493+
unsafe {
494+
let alloca = llvm::LLVMBuildAlloca(bx.llbuilder, ty, UNNAMED);
483495
llvm::LLVMSetAlignment(alloca, align.bytes() as c_uint);
484496
alloca
485497
}
@@ -603,11 +615,7 @@ impl<'a, 'll, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'll, 'tcx> {
603615
let llptr = if i == 0 {
604616
place.llval
605617
} else {
606-
self.inbounds_gep(
607-
self.type_i8(),
608-
place.llval,
609-
&[self.const_usize(b_offset.bytes())],
610-
)
618+
self.inbounds_ptradd(place.llval, self.const_usize(b_offset.bytes()))
611619
};
612620
let llty = place.layout.scalar_pair_element_llvm_type(self, i, false);
613621
let load = self.load(llty, llptr, align);
@@ -778,11 +786,6 @@ impl<'a, 'll, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'll, 'tcx> {
778786
}
779787
}
780788

781-
fn struct_gep(&mut self, ty: &'ll Type, ptr: &'ll Value, idx: u64) -> &'ll Value {
782-
assert_eq!(idx as c_uint as u64, idx);
783-
unsafe { llvm::LLVMBuildStructGEP2(self.llbuilder, ty, ptr, idx as c_uint, UNNAMED) }
784-
}
785-
786789
/* Casts */
787790
fn trunc(&mut self, val: &'ll Value, dest_ty: &'ll Type) -> &'ll Value {
788791
unsafe { llvm::LLVMBuildTrunc(self.llbuilder, val, dest_ty, UNNAMED) }

compiler/rustc_codegen_llvm/src/intrinsic.rs

+5-4
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ use rustc_middle::ty::layout::{FnAbiOf, HasTyCtxt, LayoutOf};
1818
use rustc_middle::ty::{self, GenericArgsRef, Ty};
1919
use rustc_middle::{bug, span_bug};
2020
use rustc_span::{sym, symbol::kw, Span, Symbol};
21-
use rustc_target::abi::{self, Align, HasDataLayout, Primitive};
21+
use rustc_target::abi::{self, Align, HasDataLayout, Primitive, Size};
2222
use rustc_target::spec::{HasTargetSpec, PanicStrategy};
2323

2424
use std::cmp::Ordering;
@@ -565,7 +565,8 @@ fn codegen_msvc_try<'ll>(
565565
//
566566
// More information can be found in libstd's seh.rs implementation.
567567
let ptr_align = bx.tcx().data_layout.pointer_align.abi;
568-
let slot = bx.alloca(bx.type_ptr(), ptr_align);
568+
let ptr_size = bx.tcx().data_layout.pointer_size;
569+
let slot = bx.alloca(ptr_size, ptr_align);
569570
let try_func_ty = bx.type_func(&[bx.type_ptr()], bx.type_void());
570571
bx.invoke(try_func_ty, None, None, try_func, &[data], normal, catchswitch, None);
571572

@@ -838,7 +839,7 @@ fn codegen_emcc_try<'ll>(
838839
let ptr_align = bx.tcx().data_layout.pointer_align.abi;
839840
let i8_align = bx.tcx().data_layout.i8_align.abi;
840841
let catch_data_type = bx.type_struct(&[bx.type_ptr(), bx.type_bool()], false);
841-
let catch_data = bx.alloca(catch_data_type, ptr_align);
842+
let catch_data = bx.typed_alloca(catch_data_type, ptr_align);
842843
let catch_data_0 =
843844
bx.inbounds_gep(catch_data_type, catch_data, &[bx.const_usize(0), bx.const_usize(0)]);
844845
bx.store(ptr, catch_data_0, ptr_align);
@@ -1289,7 +1290,7 @@ fn generic_simd_intrinsic<'ll, 'tcx>(
12891290
let ze = bx.zext(i_, bx.type_ix(expected_bytes * 8));
12901291

12911292
// Convert the integer to a byte array
1292-
let ptr = bx.alloca(bx.type_ix(expected_bytes * 8), Align::ONE);
1293+
let ptr = bx.alloca(Size::from_bytes(expected_bytes), Align::ONE);
12931294
bx.store(ze, ptr, Align::ONE);
12941295
let array_ty = bx.type_array(bx.type_i8(), expected_bytes);
12951296
return Ok(bx.load(array_ty, ptr, Align::ONE));

compiler/rustc_codegen_llvm/src/llvm/ffi.rs

-7
Original file line numberDiff line numberDiff line change
@@ -1301,13 +1301,6 @@ extern "C" {
13011301
NumIndices: c_uint,
13021302
Name: *const c_char,
13031303
) -> &'a Value;
1304-
pub fn LLVMBuildStructGEP2<'a>(
1305-
B: &Builder<'a>,
1306-
Ty: &'a Type,
1307-
Pointer: &'a Value,
1308-
Idx: c_uint,
1309-
Name: *const c_char,
1310-
) -> &'a Value;
13111304

13121305
// Casts
13131306
pub fn LLVMBuildTrunc<'a>(

compiler/rustc_codegen_llvm/src/type_.rs

-3
Original file line numberDiff line numberDiff line change
@@ -250,9 +250,6 @@ impl<'ll, 'tcx> LayoutTypeMethods<'tcx> for CodegenCx<'ll, 'tcx> {
250250
fn is_backend_scalar_pair(&self, layout: TyAndLayout<'tcx>) -> bool {
251251
layout.is_llvm_scalar_pair()
252252
}
253-
fn backend_field_index(&self, layout: TyAndLayout<'tcx>, index: usize) -> u64 {
254-
layout.llvm_field_index(self, index)
255-
}
256253
fn scalar_pair_element_backend_type(
257254
&self,
258255
layout: TyAndLayout<'tcx>,

compiler/rustc_codegen_llvm/src/type_of.rs

-37
Original file line numberDiff line numberDiff line change
@@ -174,7 +174,6 @@ pub trait LayoutLlvmExt<'tcx> {
174174
index: usize,
175175
immediate: bool,
176176
) -> &'a Type;
177-
fn llvm_field_index<'a>(&self, cx: &CodegenCx<'a, 'tcx>, index: usize) -> u64;
178177
fn scalar_copy_llvm_type<'a>(&self, cx: &CodegenCx<'a, 'tcx>) -> Option<&'a Type>;
179178
}
180179

@@ -324,42 +323,6 @@ impl<'tcx> LayoutLlvmExt<'tcx> for TyAndLayout<'tcx> {
324323
self.scalar_llvm_type_at(cx, scalar)
325324
}
326325

327-
fn llvm_field_index<'a>(&self, cx: &CodegenCx<'a, 'tcx>, index: usize) -> u64 {
328-
match self.abi {
329-
Abi::Scalar(_) | Abi::ScalarPair(..) => {
330-
bug!("TyAndLayout::llvm_field_index({:?}): not applicable", self)
331-
}
332-
_ => {}
333-
}
334-
match self.fields {
335-
FieldsShape::Primitive | FieldsShape::Union(_) => {
336-
bug!("TyAndLayout::llvm_field_index({:?}): not applicable", self)
337-
}
338-
339-
FieldsShape::Array { .. } => index as u64,
340-
341-
FieldsShape::Arbitrary { .. } => {
342-
let variant_index = match self.variants {
343-
Variants::Single { index } => Some(index),
344-
_ => None,
345-
};
346-
347-
// Look up llvm field if indexes do not match memory order due to padding. If
348-
// `field_remapping` is `None` no padding was used and the llvm field index
349-
// matches the memory index.
350-
match cx.type_lowering.borrow().get(&(self.ty, variant_index)) {
351-
Some(TypeLowering { field_remapping: Some(ref remap), .. }) => {
352-
remap[index] as u64
353-
}
354-
Some(_) => self.fields.memory_index(index) as u64,
355-
None => {
356-
bug!("TyAndLayout::llvm_field_index({:?}): type info not found", self)
357-
}
358-
}
359-
}
360-
}
361-
}
362-
363326
fn scalar_copy_llvm_type<'a>(&self, cx: &CodegenCx<'a, 'tcx>) -> Option<&'a Type> {
364327
debug_assert!(self.is_sized());
365328

0 commit comments

Comments
 (0)