Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

rustc emits unnecessary abort / trap instructions after calls to divergent functions #52998

Closed
japaric opened this issue Aug 2, 2018 · 4 comments
Labels
T-compiler Relevant to the compiler team, which will review and decide on the PR/issue.

Comments

@japaric
Copy link
Member

japaric commented Aug 2, 2018

STR

$ cat foo.rs
#![crate_type = "lib"]

#[no_mangle]
pub fn foo() -> ! {
    extern "Rust" {
        fn bar() -> !;
    }

    unsafe { bar() }
}
$ rustc -C panic=abort --emit=llvm-ir,link -C opt-level=3 foo.rs

$ cat foo.ll
; ModuleID = 'foo0-8787f43e282added376259c1adb08b80.rs'
source_filename = "foo0-8787f43e282added376259c1adb08b80.rs"
target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
target triple = "x86_64-unknown-linux-gnu"

; Function Attrs: noreturn nounwind
define void @foo() unnamed_addr #0 {
start:
  tail call void @bar()
  unreachable
}

; Function Attrs: noreturn nounwind
declare void @bar() unnamed_addr #0

attributes #0 = { noreturn nounwind "probe-stack"="__rust_probestack" }

Is the unreachable after calling bar really necessary? bar is noreturn nounwind.

$ objdump -Cd libfoo.rlib
In archive libfoo.rlib:

foo.foo0.rcgu.o:     file format elf64-x86-64

Disassembly of section .text.foo:

0000000000000000 <foo>:
   0:   50                      push   %rax
   1:   e8 00 00 00 00          callq  6 <foo+0x6>
   6:   0f 0b                   ud2

The ud2 instruction is dead code because bar never returns.

If I manually use llc on the optimized .ll file I get this:

$ llc -filetype=obj foo.ll

$ objdump -Cd foo.o
foo.o:     file format elf64-x86-64

Disassembly of section .text:

0000000000000000 <foo>:
   0:   50                      push   %rax
   1:   e8 00 00 00 00          callq  6 <foo+0x6>

(I see the same if I run llc on an unoptimized .ll file)

Which is what I expected to get from rustc. Is there some LLVM pass that we are disabling that
prevents LLVM from removing the ud2 instruction? I also observe this behavior when compiling for ARM.

Meta

$ rustc -V
rustc 1.29.0-nightly (97085f9fb 2018-08-01)

$ # the llc is from a Rust build I had around
$ llc -version | head -n5
LLVM (http://llvm.org/):
  LLVM version 7.0.0svn
  Optimized build.
  Default target: x86_64-unknown-linux-gnu
  Host CPU: skylake

cc @nagisa

@the8472
Copy link
Member

the8472 commented Aug 2, 2018

see #45920

@nagisa
Copy link
Member

nagisa commented Aug 3, 2018

This behaviour is very much intended.

We should probably have a codegen flag/target switch to control this behaviour, but otherwise this seems like a non-issue to me.

@nagisa nagisa added the T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. label Aug 3, 2018
@japaric
Copy link
Member Author

japaric commented Aug 7, 2018

Thanks for your replies, @the8472 @nagisa.

I see that there is a target option to disable the unreachable -> trap lowering but disabling it opens my programs to UB in safe code whenever the user defines functions like main to be infinite loops with no side effects.

It seems that the only way to remove these trap instructions is to fix the original issue where LLVM misoptimizes infinite loops with no side effects (#28728).

I'm going to close this issue as it's a consequence of #28728.

@japaric japaric closed this as completed Aug 7, 2018
@hanna-kruppe
Copy link
Contributor

The codegen option has little to do with #28728. It doesn't fix the soundness bug (though it does mitigate some simpler instances of it, downgrading them e.g. from wild reads to SIGILL) and the rationale for generating the trap instructions applies to any UB that can cause control to fall off the end of a function, including UB caused by wrong user code. It's a hardening measure, not something ensuring soundness.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
T-compiler Relevant to the compiler team, which will review and decide on the PR/issue.
Projects
None yet
Development

No branches or pull requests

4 participants