From c586bc3d760c4cb158aba5c59ca15809a6957142 Mon Sep 17 00:00:00 2001 From: Gary Guo Date: Wed, 11 May 2022 23:08:02 +0100 Subject: [PATCH 1/3] Prevent unwinding when `-C panic=abort` is used regardless declared ABI --- compiler/rustc_middle/src/ty/layout.rs | 8 ++++++++ src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs | 6 +++--- 2 files changed, 11 insertions(+), 3 deletions(-) diff --git a/compiler/rustc_middle/src/ty/layout.rs b/compiler/rustc_middle/src/ty/layout.rs index c8055100d30..630a89cc7cb 100644 --- a/compiler/rustc_middle/src/ty/layout.rs +++ b/compiler/rustc_middle/src/ty/layout.rs @@ -2888,6 +2888,14 @@ pub fn fn_can_unwind<'tcx>(tcx: TyCtxt<'tcx>, fn_def_id: Option, abi: Spe return false; } + // With `-C panic=abort`, all non-FFI functions are required to not unwind. + // + // Note that this is true regardless ABI specified on the function -- a `extern "C-unwind"` + // function defined in Rust is also required to abort. + if tcx.sess.panic_strategy() == PanicStrategy::Abort && !tcx.is_foreign_item(did) { + return false; + } + // With -Z panic-in-drop=abort, drop_in_place never unwinds. // // This is not part of `codegen_fn_attrs` as it can differ between crates diff --git a/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs b/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs index 398937a04c9..5334a1c004d 100644 --- a/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs +++ b/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs @@ -1,7 +1,7 @@ // compile-flags: -C panic=abort -// Test that `nounwind` atributes are not applied to `C-unwind` extern functions -// even when the code is compiled with `panic=abort`. +// Test that `nounwind` atributes are also applied to extern `C-unwind` Rust functions +// when the code is compiled with `panic=abort`. #![crate_type = "lib"] #![feature(c_unwind)] @@ -19,4 +19,4 @@ pub unsafe extern "C-unwind" fn rust_item_that_can_unwind() { // Now, make sure that the LLVM attributes for this functions are correct. First, make // sure that the first item is correctly marked with the `nounwind` attribute: // -// CHECK-NOT: attributes #0 = { {{.*}}nounwind{{.*}} } +// CHECK: attributes #0 = { {{.*}}nounwind{{.*}} } From a315bb444678a31de0effcb3071c6a9c57d25219 Mon Sep 17 00:00:00 2001 From: Gary Guo Date: Thu, 12 May 2022 14:20:13 +0100 Subject: [PATCH 2/3] Expand c-unwind-abi-panic-abort test --- .../codegen/unwind-abis/c-unwind-abi-panic-abort.rs | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs b/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs index 5334a1c004d..af6879880ae 100644 --- a/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs +++ b/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs @@ -6,17 +6,22 @@ #![crate_type = "lib"] #![feature(c_unwind)] -extern "C-unwind" { - fn may_unwind(); -} - // CHECK: @rust_item_that_can_unwind() unnamed_addr #0 #[no_mangle] pub unsafe extern "C-unwind" fn rust_item_that_can_unwind() { + // CHECK: call void @_ZN4core9panicking15panic_no_unwind may_unwind(); } +extern "C-unwind" { + // CHECK: @may_unwind() unnamed_addr #1 + fn may_unwind(); +} + // Now, make sure that the LLVM attributes for this functions are correct. First, make // sure that the first item is correctly marked with the `nounwind` attribute: // // CHECK: attributes #0 = { {{.*}}nounwind{{.*}} } +// +// Now, check that foreign item is correctly marked without the `nounwind` attribute. +// CHECK-NOT: attributes #1 = { {{.*}}nounwind{{.*}} } From f86e409f0906deefd0871ba823c086dd58b50870 Mon Sep 17 00:00:00 2001 From: Gary Guo Date: Fri, 13 May 2022 19:18:02 +0100 Subject: [PATCH 3/3] Don't hardcode attribute id in codegen test --- src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs b/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs index af6879880ae..e817d5715a1 100644 --- a/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs +++ b/src/test/codegen/unwind-abis/c-unwind-abi-panic-abort.rs @@ -6,7 +6,7 @@ #![crate_type = "lib"] #![feature(c_unwind)] -// CHECK: @rust_item_that_can_unwind() unnamed_addr #0 +// CHECK: @rust_item_that_can_unwind() unnamed_addr [[ATTR0:#[0-9]+]] #[no_mangle] pub unsafe extern "C-unwind" fn rust_item_that_can_unwind() { // CHECK: call void @_ZN4core9panicking15panic_no_unwind @@ -14,14 +14,14 @@ pub unsafe extern "C-unwind" fn rust_item_that_can_unwind() { } extern "C-unwind" { - // CHECK: @may_unwind() unnamed_addr #1 + // CHECK: @may_unwind() unnamed_addr [[ATTR1:#[0-9]+]] fn may_unwind(); } // Now, make sure that the LLVM attributes for this functions are correct. First, make // sure that the first item is correctly marked with the `nounwind` attribute: // -// CHECK: attributes #0 = { {{.*}}nounwind{{.*}} } +// CHECK: attributes [[ATTR0]] = { {{.*}}nounwind{{.*}} } // // Now, check that foreign item is correctly marked without the `nounwind` attribute. -// CHECK-NOT: attributes #1 = { {{.*}}nounwind{{.*}} } +// CHECK-NOT: attributes [[ATTR1]] = { {{.*}}nounwind{{.*}} }