Skip to content

Commit f3bd723

Browse files
committed
Fix intcast, use it where appropriate
1 parent 7d1f36a commit f3bd723

File tree

7 files changed

+34
-22
lines changed

7 files changed

+34
-22
lines changed

src/librustc/ty/layout.rs

+17-1
Original file line numberDiff line numberDiff line change
@@ -1310,7 +1310,6 @@ impl<'a, 'gcx, 'tcx> Layout {
13101310
let discr_max = (variants.len() - 1) as i64;
13111311
assert!(discr_max >= 0);
13121312
let (min_ity, _) = Integer::repr_discr(tcx, ty, &hints[..], 0, discr_max);
1313-
13141313
let mut align = dl.aggregate_align;
13151314
let mut size = Size::from_bytes(0);
13161315

@@ -1351,6 +1350,23 @@ impl<'a, 'gcx, 'tcx> Layout {
13511350
return Err(LayoutError::SizeOverflow(ty));
13521351
}
13531352

1353+
let typeck_ity = Integer::from_attr(dl, def.discr_ty);
1354+
if typeck_ity < min_ity {
1355+
// It is a bug if Layout decided on a greater discriminant size than typeck for
1356+
// some reason at this point (based on values discriminant can take on). Mostly
1357+
// because this discriminant will be loaded, and then stored into variable of
1358+
// type calculated by typeck. Consider such case (a bug): typeck decided on
1359+
// byte-sized discriminant, but layout thinks we need a 16-bit to store all
1360+
// discriminant values. That would be a bug, because then, in trans, in order
1361+
// to store this 16-bit discriminant into 8-bit sized temporary some of the
1362+
// space necessary to represent would have to be discarded (or layout is wrong
1363+
// on thinking it needs 16 bits)
1364+
bug!("layout decided on a larger discriminant type ({:?}) than typeck ({:?})",
1365+
min_ity, typeck_ity);
1366+
// However, it is fine to make discr type however large (as an optimisation)
1367+
// after this point – we’ll just truncate the value we load in trans.
1368+
}
1369+
13541370
// Check to see if we should use a different type for the
13551371
// discriminant. We can safely use a type with the same size
13561372
// as the alignment of the first field of each variant.

src/librustc_llvm/ffi.rs

+5-5
Original file line numberDiff line numberDiff line change
@@ -1084,11 +1084,11 @@ extern "C" {
10841084
DestTy: TypeRef,
10851085
Name: *const c_char)
10861086
-> ValueRef;
1087-
pub fn LLVMBuildIntCast(B: BuilderRef,
1088-
Val: ValueRef,
1089-
DestTy: TypeRef,
1090-
Name: *const c_char)
1091-
-> ValueRef;
1087+
pub fn LLVMRustBuildIntCast(B: BuilderRef,
1088+
Val: ValueRef,
1089+
DestTy: TypeRef,
1090+
IsSized: bool)
1091+
-> ValueRef;
10921092
pub fn LLVMBuildFPCast(B: BuilderRef,
10931093
Val: ValueRef,
10941094
DestTy: TypeRef,

src/librustc_trans/adt.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -318,7 +318,7 @@ pub fn trans_get_discr<'a, 'tcx>(
318318
};
319319
match cast_to {
320320
None => val,
321-
Some(llty) => if is_discr_signed(&l) { bcx.sext(val, llty) } else { bcx.zext(val, llty) }
321+
Some(llty) => bcx.intcast(val, llty, is_discr_signed(&l))
322322
}
323323
}
324324

src/librustc_trans/base.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -536,7 +536,7 @@ pub fn call_memcpy<'a, 'tcx>(b: &Builder<'a, 'tcx>,
536536
let memcpy = ccx.get_intrinsic(&key);
537537
let src_ptr = b.pointercast(src, Type::i8p(ccx));
538538
let dst_ptr = b.pointercast(dst, Type::i8p(ccx));
539-
let size = b.intcast(n_bytes, ccx.int_type());
539+
let size = b.intcast(n_bytes, ccx.int_type(), false);
540540
let align = C_i32(ccx, align as i32);
541541
let volatile = C_bool(ccx, false);
542542
b.call(memcpy, &[dst_ptr, src_ptr, size, align, volatile], None);

src/librustc_trans/builder.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -780,10 +780,10 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
780780
}
781781
}
782782

783-
pub fn intcast(&self, val: ValueRef, dest_ty: Type) -> ValueRef {
783+
pub fn intcast(&self, val: ValueRef, dest_ty: Type, is_signed: bool) -> ValueRef {
784784
self.count_insn("intcast");
785785
unsafe {
786-
llvm::LLVMBuildIntCast(self.llbuilder, val, dest_ty.to_ref(), noname())
786+
llvm::LLVMRustBuildIntCast(self.llbuilder, val, dest_ty.to_ref(), is_signed)
787787
}
788788
}
789789

src/librustc_trans/mir/rvalue.rs

+2-12
Original file line numberDiff line numberDiff line change
@@ -286,17 +286,7 @@ impl<'a, 'tcx> MirContext<'a, 'tcx> {
286286

287287
let newval = match (r_t_in, r_t_out) {
288288
(CastTy::Int(_), CastTy::Int(_)) => {
289-
let srcsz = ll_t_in.int_width();
290-
let dstsz = ll_t_out.int_width();
291-
if srcsz == dstsz {
292-
bcx.bitcast(llval, ll_t_out)
293-
} else if srcsz > dstsz {
294-
bcx.trunc(llval, ll_t_out)
295-
} else if signed {
296-
bcx.sext(llval, ll_t_out)
297-
} else {
298-
bcx.zext(llval, ll_t_out)
299-
}
289+
bcx.intcast(llval, ll_t_out, signed)
300290
}
301291
(CastTy::Float, CastTy::Float) => {
302292
let srcsz = ll_t_in.float_width();
@@ -439,7 +429,7 @@ impl<'a, 'tcx> MirContext<'a, 'tcx> {
439429
let discr_ty = rvalue.ty(&*self.mir, bcx.tcx()).unwrap();
440430
let discr_type = type_of::immediate_type_of(bcx.ccx, discr_ty);
441431
let discr = adt::trans_get_discr(&bcx, enum_ty, discr_lvalue.llval,
442-
Some(discr_type), true);
432+
discr_lvalue.alignment, Some(discr_type), true);
443433
(bcx, OperandRef {
444434
val: OperandValue::Immediate(discr),
445435
ty: discr_ty

src/rustllvm/RustWrapper.cpp

+6
Original file line numberDiff line numberDiff line change
@@ -1308,6 +1308,12 @@ extern "C" LLVMRustVisibility LLVMRustGetVisibility(LLVMValueRef V) {
13081308
return toRust(LLVMGetVisibility(V));
13091309
}
13101310

1311+
// Oh hey, a binding that makes sense for once? (because LLVM’s own do not)
1312+
extern "C" LLVMValueRef LLVMRustBuildIntCast(LLVMBuilderRef B, LLVMValueRef Val,
1313+
LLVMTypeRef DestTy, bool isSigned) {
1314+
return wrap(unwrap(B)->CreateIntCast(unwrap(Val), unwrap(DestTy), isSigned, ""));
1315+
}
1316+
13111317
extern "C" void LLVMRustSetVisibility(LLVMValueRef V,
13121318
LLVMRustVisibility RustVisibility) {
13131319
LLVMSetVisibility(V, fromRust(RustVisibility));

0 commit comments

Comments
 (0)