mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2024-11-24 14:20:17 +00:00
[BOLT] Fix empty function emission in non-relocation mode
In non-relocation mode, every function is emitted in its own section. If a function is empty, RuntimeDyld will still allocate 1-byte section for the function and initialize it with zero. As a result, we will overwrite the first byte of the original function contents with zero. Such scenario can happen when the input function had only NOP instructions which BOLT removes by default. Even though such functions likely cause undefined behavior, it's better to preserve their contents. Reviewed By: yota9 Differential Revision: https://reviews.llvm.org/D133978
This commit is contained in:
parent
1076b31da8
commit
9742c25b98
@ -1074,6 +1074,14 @@ public:
|
||||
return N;
|
||||
}
|
||||
|
||||
/// Return true if function has instructions to emit.
|
||||
bool hasNonPseudoInstructions() const {
|
||||
for (const BinaryBasicBlock &BB : blocks())
|
||||
if (BB.getNumNonPseudos() > 0)
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
/// Return MC symbol associated with the function.
|
||||
/// All references to the function should use this symbol.
|
||||
MCSymbol *getSymbol(const FragmentNum Fragment = FragmentNum::main()) {
|
||||
|
@ -287,6 +287,11 @@ bool BinaryEmitter::emitFunction(BinaryFunction &Function,
|
||||
if (Function.getState() == BinaryFunction::State::Empty)
|
||||
return false;
|
||||
|
||||
// Avoid emitting function without instructions when overwriting the original
|
||||
// function in-place. Otherwise, emit the empty function to define the symbol.
|
||||
if (!BC.HasRelocations && !Function.hasNonPseudoInstructions())
|
||||
return false;
|
||||
|
||||
MCSection *Section =
|
||||
BC.getCodeSection(Function.getCodeSectionName(FF.getFragmentNum()));
|
||||
Streamer.switchSection(Section);
|
||||
|
30
bolt/test/X86/nop-function.s
Normal file
30
bolt/test/X86/nop-function.s
Normal file
@ -0,0 +1,30 @@
|
||||
## Check that BOLT preserves nop instruction if it's the only instruction
|
||||
## in a function.
|
||||
|
||||
# REQUIRES: system-linux
|
||||
|
||||
# RUN: llvm-mc -filetype=obj -triple x86_64-unknown-linux %s -o %t.o
|
||||
# RUN: ld.lld %t.o -o %t.exe -q
|
||||
# RUN: llvm-bolt %t.exe -o %t.bolt.exe --relocs=0
|
||||
# RUN: llvm-objdump -d %t.bolt.exe | FileCheck %s
|
||||
|
||||
.text
|
||||
.globl nop_function
|
||||
.type nop_function,@function
|
||||
nop_function:
|
||||
.cfi_startproc
|
||||
nop
|
||||
# CHECK: <nop_function>:
|
||||
# CHECK-NEXT: nop
|
||||
|
||||
.size nop_function, .-nop_function
|
||||
.cfi_endproc
|
||||
|
||||
|
||||
.globl _start
|
||||
.type _start,@function
|
||||
_start:
|
||||
.cfi_startproc
|
||||
call nop_function
|
||||
.size _start, .-_start
|
||||
.cfi_endproc
|
Loading…
Reference in New Issue
Block a user