Skip to content

Commit 7af3406

Browse files
committed
Set metadata for vtable-related loads
Give LLVM much more information about vtable pointers. Without the extra information, LLVM has to be rather pessimistic about vtables, preventing a number of obvious optimisations. * Makes the vtable pointer argument noalias and readonly. * Marks loads of the vtable pointer as nonnull. * Marks load from the vtable with `!invariant.load` metadata. Fixes #39992
1 parent a17e5e2 commit 7af3406

File tree

5 files changed

+34
-6
lines changed

5 files changed

+34
-6
lines changed

src/librustc_trans/abi.rs

+4
Original file line numberDiff line numberDiff line change
@@ -506,7 +506,11 @@ impl FnType {
506506
if let Some(inner) = rust_ptr_attrs(ty, &mut data) {
507507
data.attrs.set(ArgAttribute::NonNull);
508508
if ccx.tcx().struct_tail(inner).is_trait() {
509+
// vtables can be safely marked non-null, readonly
510+
// and noalias.
509511
info.attrs.set(ArgAttribute::NonNull);
512+
info.attrs.set(ArgAttribute::ReadOnly);
513+
info.attrs.set(ArgAttribute::NoAlias);
510514
}
511515
}
512516
args.push(data);

src/librustc_trans/base.rs

+9-2
Original file line numberDiff line numberDiff line change
@@ -472,8 +472,15 @@ pub fn load_fat_ptr<'a, 'tcx>(
472472
b.load(ptr, alignment.to_align())
473473
};
474474

475-
// FIXME: emit metadata on `meta`.
476-
let meta = b.load(get_meta(b, src), alignment.to_align());
475+
let meta = get_meta(b, src);
476+
let meta_ty = val_ty(meta);
477+
// If the 'meta' field is a pointer, it's a vtable, so use load_nonnull
478+
// instead
479+
let meta = if meta_ty.element_type().kind() == llvm::TypeKind::Pointer {
480+
b.load_nonnull(meta, None)
481+
} else {
482+
b.load(meta, None)
483+
};
477484

478485
(ptr, meta)
479486
}

src/librustc_trans/builder.rs

+7
Original file line numberDiff line numberDiff line change
@@ -1149,6 +1149,13 @@ impl<'a, 'tcx> Builder<'a, 'tcx> {
11491149
}
11501150
}
11511151

1152+
pub fn set_invariant_load(&self, load: ValueRef) {
1153+
unsafe {
1154+
llvm::LLVMSetMetadata(load, llvm::MD_invariant_load as c_uint,
1155+
llvm::LLVMMDNodeInContext(self.ccx.llcx(), ptr::null(), 0));
1156+
}
1157+
}
1158+
11521159
/// Returns the ptr value that should be used for storing `val`.
11531160
fn check_store<'b>(&self,
11541161
val: ValueRef,

src/librustc_trans/glue.rs

+9-1
Original file line numberDiff line numberDiff line change
@@ -386,7 +386,15 @@ pub fn size_and_align_of_dst<'a, 'tcx>(bcx: &Builder<'a, 'tcx>, t: Ty<'tcx>, inf
386386
let info = bcx.pointercast(info, Type::int(bcx.ccx).ptr_to());
387387
let size_ptr = bcx.gepi(info, &[1]);
388388
let align_ptr = bcx.gepi(info, &[2]);
389-
(bcx.load(size_ptr, None), bcx.load(align_ptr, None))
389+
390+
let size = bcx.load(size_ptr, None);
391+
let align = bcx.load(align_ptr, None);
392+
393+
// Vtable loads are invariant
394+
bcx.set_invariant_load(size);
395+
bcx.set_invariant_load(align);
396+
397+
(size, align)
390398
}
391399
ty::TySlice(_) | ty::TyStr => {
392400
let unit_ty = t.sequence_element_type(bcx.tcx());

src/librustc_trans/meth.rs

+5-3
Original file line numberDiff line numberDiff line change
@@ -30,13 +30,15 @@ const VTABLE_OFFSET: usize = 3;
3030
/// Extracts a method from a trait object's vtable, at the specified index.
3131
pub fn get_virtual_method<'a, 'tcx>(bcx: &Builder<'a, 'tcx>,
3232
llvtable: ValueRef,
33-
vtable_index: usize)
34-
-> ValueRef {
33+
vtable_index: usize) -> ValueRef {
3534
// Load the data pointer from the object.
3635
debug!("get_virtual_method(vtable_index={}, llvtable={:?})",
3736
vtable_index, Value(llvtable));
3837

39-
bcx.load(bcx.gepi(llvtable, &[vtable_index + VTABLE_OFFSET]), None)
38+
let ptr = bcx.load_nonnull(bcx.gepi(llvtable, &[vtable_index + VTABLE_OFFSET]), None);
39+
// Vtable loads are invariant
40+
bcx.set_invariant_load(ptr);
41+
ptr
4042
}
4143

4244
/// Generate a shim function that allows an object type like `SomeTrait` to

0 commit comments

Comments
 (0)