Skip to content

Commit db8086e

Browse files
committed
Auto merge of #95548 - rcvalle:rust-cfi-2, r=nagisa
Add fine-grained LLVM CFI support to the Rust compiler This PR improves the LLVM Control Flow Integrity (CFI) support in the Rust compiler by providing forward-edge control flow protection for Rust-compiled code only by aggregating function pointers in groups identified by their return and parameter types. Forward-edge control flow protection for C or C++ and Rust -compiled code "mixed binaries" (i.e., for when C or C++ and Rust -compiled code share the same virtual address space) will be provided in later work as part of this project by identifying C char and integer type uses at the time types are encoded (see Type metadata in the design document in the tracking issue #89653). LLVM CFI can be enabled with -Zsanitizer=cfi and requires LTO (i.e., -Clto). Thank you again, `@eddyb,` `@nagisa,` `@pcc,` and `@tmiasko` for all the help!
2 parents fcad918 + f792f26 commit db8086e

25 files changed

+1732
-165
lines changed

Cargo.lock

+1
Original file line numberDiff line numberDiff line change
@@ -4519,6 +4519,7 @@ dependencies = [
45194519
name = "rustc_symbol_mangling"
45204520
version = "0.0.0"
45214521
dependencies = [
4522+
"bitflags",
45224523
"punycode",
45234524
"rustc-demangle",
45244525
"rustc_data_structures",

compiler/rustc_codegen_gcc/src/builder.rs

-10
Original file line numberDiff line numberDiff line change
@@ -784,16 +784,6 @@ impl<'a, 'gcc, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'gcc, 'tcx> {
784784
// TODO(antoyo)
785785
}
786786

787-
fn type_metadata(&mut self, _function: RValue<'gcc>, _typeid: String) {
788-
// Unsupported.
789-
}
790-
791-
fn typeid_metadata(&mut self, _typeid: String) -> RValue<'gcc> {
792-
// Unsupported.
793-
self.context.new_rvalue_from_int(self.int_type, 0)
794-
}
795-
796-
797787
fn store(&mut self, val: RValue<'gcc>, ptr: RValue<'gcc>, align: Align) -> RValue<'gcc> {
798788
self.store_with_flags(val, ptr, align, MemFlags::empty())
799789
}

compiler/rustc_codegen_gcc/src/type_.rs

+12-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
use std::convert::TryInto;
22

33
use gccjit::{RValue, Struct, Type};
4-
use rustc_codegen_ssa::traits::{BaseTypeMethods, DerivedTypeMethods};
4+
use rustc_codegen_ssa::traits::{BaseTypeMethods, DerivedTypeMethods, TypeMembershipMethods};
55
use rustc_codegen_ssa::common::TypeKind;
66
use rustc_middle::{bug, ty};
77
use rustc_middle::ty::layout::TyAndLayout;
@@ -290,3 +290,14 @@ pub fn struct_fields<'gcc, 'tcx>(cx: &CodegenCx<'gcc, 'tcx>, layout: TyAndLayout
290290

291291
(result, packed)
292292
}
293+
294+
impl<'gcc, 'tcx> TypeMembershipMethods<'tcx> for CodegenCx<'gcc, 'tcx> {
295+
fn set_type_metadata(&self, _function: RValue<'gcc>, _typeid: String) {
296+
// Unsupported.
297+
}
298+
299+
fn typeid_metadata(&self, _typeid: String) -> RValue<'gcc> {
300+
// Unsupported.
301+
self.context.new_rvalue_from_int(self.int_type, 0)
302+
}
303+
}

compiler/rustc_codegen_llvm/Cargo.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,6 @@ rustc_middle = { path = "../rustc_middle" }
1818
rustc-demangle = "0.1.21"
1919
rustc_attr = { path = "../rustc_attr" }
2020
rustc_codegen_ssa = { path = "../rustc_codegen_ssa" }
21-
rustc_symbol_mangling = { path = "../rustc_symbol_mangling" }
2221
rustc_data_structures = { path = "../rustc_data_structures" }
2322
rustc_errors = { path = "../rustc_errors" }
2423
rustc_fs_util = { path = "../rustc_fs_util" }
@@ -30,6 +29,7 @@ rustc_metadata = { path = "../rustc_metadata" }
3029
rustc_query_system = { path = "../rustc_query_system" }
3130
rustc_session = { path = "../rustc_session" }
3231
rustc_serialize = { path = "../rustc_serialize" }
32+
rustc_symbol_mangling = { path = "../rustc_symbol_mangling" }
3333
rustc_target = { path = "../rustc_target" }
3434
smallvec = { version = "1.8.1", features = ["union", "may_dangle"] }
3535
rustc_ast = { path = "../rustc_ast" }

compiler/rustc_codegen_llvm/src/builder.rs

-26
Original file line numberDiff line numberDiff line change
@@ -626,32 +626,6 @@ impl<'a, 'll, 'tcx> BuilderMethods<'a, 'tcx> for Builder<'a, 'll, 'tcx> {
626626
}
627627
}
628628

629-
fn type_metadata(&mut self, function: &'ll Value, typeid: String) {
630-
let typeid_metadata = self.typeid_metadata(typeid);
631-
let v = [self.const_usize(0), typeid_metadata];
632-
unsafe {
633-
llvm::LLVMGlobalSetMetadata(
634-
function,
635-
llvm::MD_type as c_uint,
636-
llvm::LLVMValueAsMetadata(llvm::LLVMMDNodeInContext(
637-
self.cx.llcx,
638-
v.as_ptr(),
639-
v.len() as c_uint,
640-
)),
641-
)
642-
}
643-
}
644-
645-
fn typeid_metadata(&mut self, typeid: String) -> Self::Value {
646-
unsafe {
647-
llvm::LLVMMDStringInContext(
648-
self.cx.llcx,
649-
typeid.as_ptr() as *const c_char,
650-
typeid.as_bytes().len() as c_uint,
651-
)
652-
}
653-
}
654-
655629
fn store(&mut self, val: &'ll Value, ptr: &'ll Value, align: Align) -> &'ll Value {
656630
self.store_with_flags(val, ptr, align, MemFlags::empty())
657631
}

compiler/rustc_codegen_llvm/src/declare.rs

+8
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,9 @@ use crate::llvm;
1818
use crate::llvm::AttributePlace::Function;
1919
use crate::type_::Type;
2020
use crate::value::Value;
21+
use rustc_codegen_ssa::traits::TypeMembershipMethods;
2122
use rustc_middle::ty::Ty;
23+
use rustc_symbol_mangling::typeid::typeid_for_fnabi;
2224
use smallvec::SmallVec;
2325
use tracing::debug;
2426

@@ -97,6 +99,12 @@ impl<'ll, 'tcx> CodegenCx<'ll, 'tcx> {
9799
fn_abi.llvm_type(self),
98100
);
99101
fn_abi.apply_attrs_llfn(self, llfn);
102+
103+
if self.tcx.sess.is_sanitizer_cfi_enabled() {
104+
let typeid = typeid_for_fnabi(self.tcx, fn_abi);
105+
self.set_type_metadata(llfn, typeid);
106+
}
107+
100108
llfn
101109
}
102110

compiler/rustc_codegen_llvm/src/type_.rs

+29-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ use rustc_target::abi::{AddressSpace, Align, Integer, Size};
1919
use std::fmt;
2020
use std::ptr;
2121

22-
use libc::c_uint;
22+
use libc::{c_char, c_uint};
2323

2424
impl PartialEq for Type {
2525
fn eq(&self, other: &Self) -> bool {
@@ -289,3 +289,31 @@ impl<'ll, 'tcx> LayoutTypeMethods<'tcx> for CodegenCx<'ll, 'tcx> {
289289
ty.llvm_type(self)
290290
}
291291
}
292+
293+
impl<'ll, 'tcx> TypeMembershipMethods<'tcx> for CodegenCx<'ll, 'tcx> {
294+
fn set_type_metadata(&self, function: &'ll Value, typeid: String) {
295+
let typeid_metadata = self.typeid_metadata(typeid);
296+
let v = [self.const_usize(0), typeid_metadata];
297+
unsafe {
298+
llvm::LLVMGlobalSetMetadata(
299+
function,
300+
llvm::MD_type as c_uint,
301+
llvm::LLVMValueAsMetadata(llvm::LLVMMDNodeInContext(
302+
self.llcx,
303+
v.as_ptr(),
304+
v.len() as c_uint,
305+
)),
306+
)
307+
}
308+
}
309+
310+
fn typeid_metadata(&self, typeid: String) -> &'ll Value {
311+
unsafe {
312+
llvm::LLVMMDStringInContext(
313+
self.llcx,
314+
typeid.as_ptr() as *const c_char,
315+
typeid.len() as c_uint,
316+
)
317+
}
318+
}
319+
}

compiler/rustc_codegen_ssa/src/mir/block.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -20,7 +20,7 @@ use rustc_middle::ty::print::{with_no_trimmed_paths, with_no_visible_paths};
2020
use rustc_middle::ty::{self, Instance, Ty, TypeVisitable};
2121
use rustc_span::source_map::Span;
2222
use rustc_span::{sym, Symbol};
23-
use rustc_symbol_mangling::typeid_for_fnabi;
23+
use rustc_symbol_mangling::typeid::typeid_for_fnabi;
2424
use rustc_target::abi::call::{ArgAbi, FnAbi, PassMode};
2525
use rustc_target::abi::{self, HasDataLayout, WrappingRange};
2626
use rustc_target::spec::abi::Abi;
@@ -918,7 +918,7 @@ impl<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>> FunctionCx<'a, 'tcx, Bx> {
918918
// FIXME(rcvalle): Add support for generalized identifiers.
919919
// FIXME(rcvalle): Create distinct unnamed MDNodes for internal identifiers.
920920
let typeid = typeid_for_fnabi(bx.tcx(), fn_abi);
921-
let typeid_metadata = bx.typeid_metadata(typeid);
921+
let typeid_metadata = self.cx.typeid_metadata(typeid);
922922

923923
// Test whether the function pointer is associated with the type identifier.
924924
let cond = bx.type_test(fn_ptr, typeid_metadata);

compiler/rustc_codegen_ssa/src/mir/mod.rs

-8
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@ use rustc_middle::mir;
33
use rustc_middle::mir::interpret::ErrorHandled;
44
use rustc_middle::ty::layout::{FnAbiOf, HasTyCtxt, TyAndLayout};
55
use rustc_middle::ty::{self, Instance, Ty, TypeFoldable, TypeVisitable};
6-
use rustc_symbol_mangling::typeid_for_fnabi;
76
use rustc_target::abi::call::{FnAbi, PassMode};
87

98
use std::iter;
@@ -247,13 +246,6 @@ pub fn codegen_mir<'a, 'tcx, Bx: BuilderMethods<'a, 'tcx>>(
247246
for (bb, _) in traversal::reverse_postorder(&mir) {
248247
fx.codegen_block(bb);
249248
}
250-
251-
// For backends that support CFI using type membership (i.e., testing whether a given pointer
252-
// is associated with a type identifier).
253-
if cx.tcx().sess.is_sanitizer_cfi_enabled() {
254-
let typeid = typeid_for_fnabi(cx.tcx(), fn_abi);
255-
bx.type_metadata(llfn, typeid);
256-
}
257249
}
258250

259251
/// Produces, for each argument, a `Value` pointing at the

compiler/rustc_codegen_ssa/src/traits/builder.rs

-2
Original file line numberDiff line numberDiff line change
@@ -160,8 +160,6 @@ pub trait BuilderMethods<'a, 'tcx>:
160160

161161
fn range_metadata(&mut self, load: Self::Value, range: WrappingRange);
162162
fn nonnull_metadata(&mut self, load: Self::Value);
163-
fn type_metadata(&mut self, function: Self::Function, typeid: String);
164-
fn typeid_metadata(&mut self, typeid: String) -> Self::Value;
165163

166164
fn store(&mut self, val: Self::Value, ptr: Self::Value, align: Align) -> Self::Value;
167165
fn store_with_flags(

compiler/rustc_codegen_ssa/src/traits/mod.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,8 @@ pub use self::intrinsic::IntrinsicCallMethods;
4040
pub use self::misc::MiscMethods;
4141
pub use self::statics::{StaticBuilderMethods, StaticMethods};
4242
pub use self::type_::{
43-
ArgAbiMethods, BaseTypeMethods, DerivedTypeMethods, LayoutTypeMethods, TypeMethods,
43+
ArgAbiMethods, BaseTypeMethods, DerivedTypeMethods, LayoutTypeMethods, TypeMembershipMethods,
44+
TypeMethods,
4445
};
4546
pub use self::write::{ModuleBufferMethods, ThinBufferMethods, WriteBackendMethods};
4647

compiler/rustc_codegen_ssa/src/traits/type_.rs

+15-2
Original file line numberDiff line numberDiff line change
@@ -117,6 +117,13 @@ pub trait LayoutTypeMethods<'tcx>: Backend<'tcx> {
117117
) -> Self::Type;
118118
}
119119

120+
// For backends that support CFI using type membership (i.e., testing whether a given pointer is
121+
// associated with a type identifier).
122+
pub trait TypeMembershipMethods<'tcx>: Backend<'tcx> {
123+
fn set_type_metadata(&self, function: Self::Function, typeid: String);
124+
fn typeid_metadata(&self, typeid: String) -> Self::Value;
125+
}
126+
120127
pub trait ArgAbiMethods<'tcx>: HasCodegen<'tcx> {
121128
fn store_fn_arg(
122129
&mut self,
@@ -133,6 +140,12 @@ pub trait ArgAbiMethods<'tcx>: HasCodegen<'tcx> {
133140
fn arg_memory_ty(&self, arg_abi: &ArgAbi<'tcx, Ty<'tcx>>) -> Self::Type;
134141
}
135142

136-
pub trait TypeMethods<'tcx>: DerivedTypeMethods<'tcx> + LayoutTypeMethods<'tcx> {}
143+
pub trait TypeMethods<'tcx>:
144+
DerivedTypeMethods<'tcx> + LayoutTypeMethods<'tcx> + TypeMembershipMethods<'tcx>
145+
{
146+
}
137147

138-
impl<'tcx, T> TypeMethods<'tcx> for T where Self: DerivedTypeMethods<'tcx> + LayoutTypeMethods<'tcx> {}
148+
impl<'tcx, T> TypeMethods<'tcx> for T where
149+
Self: DerivedTypeMethods<'tcx> + LayoutTypeMethods<'tcx> + TypeMembershipMethods<'tcx>
150+
{
151+
}

compiler/rustc_symbol_mangling/Cargo.toml

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ edition = "2021"
77
doctest = false
88

99
[dependencies]
10+
bitflags = "1.2.1"
1011
tracing = "0.1"
1112
punycode = "0.4.0"
1213
rustc-demangle = "0.1.21"

compiler/rustc_symbol_mangling/src/lib.rs

+2-7
Original file line numberDiff line numberDiff line change
@@ -102,16 +102,16 @@ use rustc_middle::middle::codegen_fn_attrs::CodegenFnAttrs;
102102
use rustc_middle::mir::mono::{InstantiationMode, MonoItem};
103103
use rustc_middle::ty::query::Providers;
104104
use rustc_middle::ty::subst::SubstsRef;
105-
use rustc_middle::ty::{self, Instance, Ty, TyCtxt};
105+
use rustc_middle::ty::{self, Instance, TyCtxt};
106106
use rustc_session::config::SymbolManglingVersion;
107-
use rustc_target::abi::call::FnAbi;
108107

109108
use tracing::debug;
110109

111110
mod legacy;
112111
mod v0;
113112

114113
pub mod test;
114+
pub mod typeid;
115115

116116
/// This function computes the symbol name for the given `instance` and the
117117
/// given instantiating crate. That is, if you know that instance X is
@@ -150,11 +150,6 @@ fn symbol_name_provider<'tcx>(tcx: TyCtxt<'tcx>, instance: Instance<'tcx>) -> ty
150150
ty::SymbolName::new(tcx, &symbol_name)
151151
}
152152

153-
/// This function computes the typeid for the given function ABI.
154-
pub fn typeid_for_fnabi<'tcx>(tcx: TyCtxt<'tcx>, fn_abi: &FnAbi<'tcx, Ty<'tcx>>) -> String {
155-
v0::mangle_typeid_for_fnabi(tcx, fn_abi)
156-
}
157-
158153
pub fn typeid_for_trait_ref<'tcx>(
159154
tcx: TyCtxt<'tcx>,
160155
trait_ref: ty::PolyExistentialTraitRef<'tcx>,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// For more information about type metadata and type metadata identifiers for cross-language LLVM
2+
// CFI support, see Type metadata in the design document in the tracking issue #89653.
3+
4+
use rustc_middle::ty::{FnSig, Ty, TyCtxt};
5+
use rustc_target::abi::call::FnAbi;
6+
7+
mod typeid_itanium_cxx_abi;
8+
use typeid_itanium_cxx_abi::TypeIdOptions;
9+
10+
/// Returns a type metadata identifier for the specified FnAbi.
11+
pub fn typeid_for_fnabi<'tcx>(tcx: TyCtxt<'tcx>, fn_abi: &FnAbi<'tcx, Ty<'tcx>>) -> String {
12+
typeid_itanium_cxx_abi::typeid_for_fnabi(tcx, fn_abi, TypeIdOptions::NO_OPTIONS)
13+
}
14+
15+
/// Returns a type metadata identifier for the specified FnSig.
16+
pub fn typeid_for_fnsig<'tcx>(tcx: TyCtxt<'tcx>, fn_sig: &FnSig<'tcx>) -> String {
17+
typeid_itanium_cxx_abi::typeid_for_fnsig(tcx, fn_sig, TypeIdOptions::NO_OPTIONS)
18+
}

0 commit comments

Comments
 (0)