mirror of
https://github.com/rust-lang/rust.git
synced 2024-11-24 07:44:10 +00:00
004aa15b47
This commit adds cross-language LLVM Control Flow Integrity (CFI) support to the Rust compiler by adding the `-Zsanitizer-cfi-normalize-integers` option to be used with Clang `-fsanitize-cfi-icall-normalize-integers` for normalizing integer types (see https://reviews.llvm.org/D139395). It provides 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). For more information about LLVM CFI and cross-language LLVM CFI support for the Rust compiler, see design document in the tracking issue #89653. Cross-language LLVM CFI can be enabled with -Zsanitizer=cfi and -Zsanitizer-cfi-normalize-integers, and requires proper (i.e., non-rustc) LTO (i.e., -Clinker-plugin-lto).
45 lines
1.7 KiB
Rust
45 lines
1.7 KiB
Rust
// Verifies that generalized KCFI type metadata for functions are emitted.
|
|
//
|
|
// revisions: aarch64 x86_64
|
|
// [aarch64] compile-flags: --target aarch64-unknown-none
|
|
// [aarch64] needs-llvm-components: aarch64
|
|
// [x86_64] compile-flags: --target x86_64-unknown-none
|
|
// [x86_64] needs-llvm-components:
|
|
// compile-flags: -Cno-prepopulate-passes -Zsanitizer=kcfi -Zsanitizer-cfi-generalize-pointers
|
|
|
|
#![crate_type="lib"]
|
|
#![feature(no_core, lang_items)]
|
|
#![no_core]
|
|
|
|
#[lang="sized"]
|
|
trait Sized { }
|
|
#[lang="copy"]
|
|
trait Copy { }
|
|
|
|
impl Copy for i32 {}
|
|
|
|
pub fn foo(f: fn(i32) -> i32, arg: i32) -> i32 {
|
|
// CHECK-LABEL: define{{.*}}foo
|
|
// CHECK-SAME: {{.*}}!{{<unknown kind #36>|kcfi_type}} ![[TYPE1:[0-9]+]]
|
|
// CHECK: {{%.+}} = call {{(noundef )*}}i32 %f(i32 {{(noundef )*}}%arg){{.*}}[ "kcfi"(i32 233085384) ]
|
|
f(arg)
|
|
}
|
|
|
|
pub fn bar(f: fn(i32, i32) -> i32, arg1: i32, arg2: i32) -> i32 {
|
|
// CHECK-LABEL: define{{.*}}bar
|
|
// CHECK-SAME: {{.*}}!{{<unknown kind #36>|kcfi_type}} ![[TYPE2:[0-9]+]]
|
|
// CHECK: {{%.+}} = call {{(noundef )*}}i32 %f(i32 {{(noundef )*}}%arg1, i32 {{(noundef )*}}%arg2){{.*}}[ "kcfi"(i32 435418021) ]
|
|
f(arg1, arg2)
|
|
}
|
|
|
|
pub fn baz(f: fn(i32, i32, i32) -> i32, arg1: i32, arg2: i32, arg3: i32) -> i32 {
|
|
// CHECK-LABEL: define{{.*}}baz
|
|
// CHECK-SAME: {{.*}}!{{<unknown kind #36>|kcfi_type}} ![[TYPE3:[0-9]+]]
|
|
// CHECK: {{%.+}} = call {{(noundef )*}}i32 %f(i32 {{(noundef )*}}%arg1, i32 {{(noundef )*}}%arg2, i32 {{(noundef )*}}%arg3){{.*}}[ "kcfi"(i32 -1003721339) ]
|
|
f(arg1, arg2, arg3)
|
|
}
|
|
|
|
// CHECK: ![[TYPE1]] = !{i32 -1741689296}
|
|
// CHECK: ![[TYPE2]] = !{i32 489439372}
|
|
// CHECK: ![[TYPE3]] = !{i32 2026563871}
|