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

[Asan] Add "funclet" OpBundle to generated runtime calls if required by EH personality #82533

Merged
merged 1 commit into from
Mar 8, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
18 changes: 18 additions & 0 deletions compiler-rt/test/asan/TestCases/Windows/issue64990.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
// Repro for the issue #64990: Asan with Windows EH generates __asan_xxx runtime calls without required funclet tokens
// RUN: %clang_cl_asan %Od %s -EHsc %Fe%t
// RUN: not %run %t 2>&1 | FileCheck %s

char buff1[6] = "hello";
char buff2[6] = "hello";

int main(int argc, char **argv) {
try {
throw 1;
} catch (...) {
// Make asan generate call to __asan_memcpy inside the EH pad.
__builtin_memcpy(buff1, buff2 + 3, 6);
}
return 0;
}

// CHECK: SUMMARY: AddressSanitizer: global-buffer-overflow {{.*}} in __asan_memcpy
62 changes: 56 additions & 6 deletions llvm/lib/Transforms/Instrumentation/AddressSanitizer.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@
#include "llvm/IR/DebugInfoMetadata.h"
#include "llvm/IR/DebugLoc.h"
#include "llvm/IR/DerivedTypes.h"
#include "llvm/IR/EHPersonalities.h"
#include "llvm/IR/Function.h"
#include "llvm/IR/GlobalAlias.h"
#include "llvm/IR/GlobalValue.h"
Expand Down Expand Up @@ -643,21 +644,70 @@ static uint64_t GetCtorAndDtorPriority(Triple &TargetTriple) {

namespace {
/// Helper RAII class to post-process inserted asan runtime calls during a
/// pass on a single Function. This is a no-op implementation, for a first NFC
/// commit. Coming up: detect and add "funclet" opBundle to function calls that
/// need them.
/// pass on a single Function. Upon end of scope, detects and applies the
/// required funclet OpBundle.
class RuntimeCallInserter {
Function *OwnerFn = nullptr;
bool TrackInsertedCalls = false;
SmallVector<CallInst *> InsertedCalls;

public:
RuntimeCallInserter(Function &Fn) : OwnerFn(&Fn) {}
RuntimeCallInserter(Function &Fn) : OwnerFn(&Fn) {
if (Fn.hasPersonalityFn()) {
auto Personality = classifyEHPersonality(Fn.getPersonalityFn());
if (isScopedEHPersonality(Personality))
TrackInsertedCalls = true;
}
}

~RuntimeCallInserter() {
if (InsertedCalls.empty())
return;
assert(TrackInsertedCalls && "Calls were wrongly tracked");

DenseMap<BasicBlock *, ColorVector> BlockColors = colorEHFunclets(*OwnerFn);
for (CallInst *CI : InsertedCalls) {
BasicBlock *BB = CI->getParent();
assert(BB && "Instruction doesn't belong to a BasicBlock");
assert(BB->getParent() == OwnerFn &&
"Instruction doesn't belong to the expected Function!");

ColorVector &Colors = BlockColors[BB];
// funclet opbundles are only valid in monochromatic BBs.
// Note that unreachable BBs are seen as colorless by colorEHFunclets()
// and will be DCE'ed later.
if (Colors.empty())
continue;
if (Colors.size() != 1) {
OwnerFn->getContext().emitError(
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Seems like a reasonable strategy. It's a little janky, but probably less user hostile than the current emergent behavior.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I had to add an extra check, to skip without error if Colors is empty, to not fail in case of unreachable BB (as it happens in the test).
I'll redo a pass on all the tests on Windows and Linux locally.

"Instruction's BasicBlock is not monochromatic");
continue;
}

BasicBlock *Color = Colors.front();
Instruction *EHPad = Color->getFirstNonPHI();

if (EHPad && EHPad->isEHPad()) {
// Replace CI with a clone with an added funclet OperandBundle
OperandBundleDef OB("funclet", EHPad);
auto *NewCall =
CallBase::addOperandBundle(CI, LLVMContext::OB_funclet, OB, CI);
NewCall->copyMetadata(*CI);
CI->replaceAllUsesWith(NewCall);
CI->eraseFromParent();
}
}
}

CallInst *createRuntimeCall(IRBuilder<> &IRB, FunctionCallee Callee,
ArrayRef<Value *> Args = {},
const Twine &Name = "") {
assert(IRB.GetInsertBlock()->getParent() == OwnerFn);
(void)OwnerFn;
return IRB.CreateCall(Callee, Args, Name, nullptr);

CallInst *Inst = IRB.CreateCall(Callee, Args, Name, nullptr);
if (TrackInsertedCalls)
InsertedCalls.push_back(Inst);
return Inst;
}
};

Expand Down
Loading
Loading