Skip to content

Commit dbf1045

Browse files
committedJul 31, 2015
[MS ABI] Hook clang up to the new EH instructions
The new EH instructions make it possible for LLVM to generate .xdata tables that the MSVC personality routines will be happy about. Because this is experimental, hide it behind a -cc1 flag (-fnew-ms-eh). Differential Revision: http://reviews.llvm.org/D11405 llvm-svn: 243767
1 parent 654e130 commit dbf1045

10 files changed

+242
-44
lines changed
 

‎clang/include/clang/Driver/CC1Options.td

+2
Original file line numberDiff line numberDiff line change
@@ -486,6 +486,8 @@ def fblocks_runtime_optional : Flag<["-"], "fblocks-runtime-optional">,
486486
HelpText<"Weakly link in the blocks runtime">;
487487
def fsjlj_exceptions : Flag<["-"], "fsjlj-exceptions">,
488488
HelpText<"Use SjLj style exceptions">;
489+
def fnew_ms_eh: Flag<["-"], "fnew-ms-eh">,
490+
HelpText<"Use the new IR representation for MS exceptions">;
489491
def split_dwarf_file : Separate<["-"], "split-dwarf-file">,
490492
HelpText<"File name to use for split dwarf debug info output">;
491493
def fno_wchar : Flag<["-"], "fno-wchar">,

‎clang/include/clang/Frontend/CodeGenOptions.def

+1
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@ CODEGENOPT(CompressDebugSections, 1, 0) ///< -Wa,-compress-debug-sections
3333
CODEGENOPT(Autolink , 1, 1) ///< -fno-autolink
3434
CODEGENOPT(AsmVerbose , 1, 0) ///< -dA, -fverbose-asm.
3535
CODEGENOPT(ObjCAutoRefCountExceptions , 1, 0) ///< Whether ARC should be EH-safe.
36+
CODEGENOPT(NewMSEH , 1, 0) ///< Whether we should use the new IR representation for MS exceptions
3637
CODEGENOPT(CoverageExtraChecksum, 1, 0) ///< Whether we need a second checksum for functions in GCNO files.
3738
CODEGENOPT(CoverageNoFunctionNamesInData, 1, 0) ///< Do not include function names in GCDA files.
3839
CODEGENOPT(CoverageExitBlockBeforeBody, 1, 0) ///< Whether to emit the exit block before the body blocks in GCNO files.

‎clang/lib/CodeGen/CGCleanup.cpp

+19-1
Original file line numberDiff line numberDiff line change
@@ -247,6 +247,13 @@ void EHScopeStack::pushTerminate() {
247247
InnermostEHScope = stable_begin();
248248
}
249249

250+
void EHScopeStack::pushCatchEnd(llvm::BasicBlock *CatchEndBlockBB) {
251+
char *Buffer = allocate(EHCatchEndScope::getSize());
252+
auto *CES = new (Buffer) EHCatchEndScope(InnermostEHScope);
253+
CES->setCachedEHDispatchBlock(CatchEndBlockBB);
254+
InnermostEHScope = stable_begin();
255+
}
256+
250257
/// Remove any 'null' fixups on the stack. However, we can't pop more
251258
/// fixups than the fixup depth on the innermost normal cleanup, or
252259
/// else fixups that we try to add to that cleanup will end up in the
@@ -896,6 +903,14 @@ void CodeGenFunction::PopCleanupBlock(bool FallthroughIsBranchThrough) {
896903
CGBuilderTy::InsertPoint SavedIP = Builder.saveAndClearIP();
897904

898905
EmitBlock(EHEntry);
906+
llvm::BasicBlock *NextAction = getEHDispatchBlock(EHParent);
907+
if (CGM.getCodeGenOpts().NewMSEH &&
908+
EHPersonality::get(*this).isMSVCPersonality()) {
909+
if (NextAction)
910+
Builder.CreateCleanupPad(VoidTy, NextAction);
911+
else
912+
Builder.CreateCleanupPad(VoidTy, {});
913+
}
899914

900915
// We only actually emit the cleanup code if the cleanup is either
901916
// active or was used before it was deactivated.
@@ -905,7 +920,10 @@ void CodeGenFunction::PopCleanupBlock(bool FallthroughIsBranchThrough) {
905920
EmitCleanup(*this, Fn, cleanupFlags, EHActiveFlag);
906921
}
907922

908-
Builder.CreateBr(getEHDispatchBlock(EHParent));
923+
if (CGM.getCodeGenOpts().NewMSEH && EHPersonality::get(*this).isMSVCPersonality())
924+
Builder.CreateCleanupRet(NextAction);
925+
else
926+
Builder.CreateBr(NextAction);
909927

910928
Builder.restoreIP(SavedIP);
911929

‎clang/lib/CodeGen/CGCleanup.h

+34-4
Original file line numberDiff line numberDiff line change
@@ -40,9 +40,9 @@ class EHScope {
4040

4141
class CommonBitFields {
4242
friend class EHScope;
43-
unsigned Kind : 2;
43+
unsigned Kind : 3;
4444
};
45-
enum { NumCommonBits = 2 };
45+
enum { NumCommonBits = 3 };
4646

4747
protected:
4848
class CatchBitFields {
@@ -81,7 +81,7 @@ class EHScope {
8181
/// The number of fixups required by enclosing scopes (not including
8282
/// this one). If this is the top cleanup scope, all the fixups
8383
/// from this index onwards belong to this scope.
84-
unsigned FixupDepth : 32 - 18 - NumCommonBits; // currently 13
84+
unsigned FixupDepth : 32 - 18 - NumCommonBits; // currently 12
8585
};
8686

8787
class FilterBitFields {
@@ -99,7 +99,7 @@ class EHScope {
9999
};
100100

101101
public:
102-
enum Kind { Cleanup, Catch, Terminate, Filter };
102+
enum Kind { Cleanup, Catch, Terminate, Filter, CatchEnd };
103103

104104
EHScope(Kind kind, EHScopeStack::stable_iterator enclosingEHScope)
105105
: CachedLandingPad(nullptr), CachedEHDispatchBlock(nullptr),
@@ -466,6 +466,17 @@ class EHTerminateScope : public EHScope {
466466
}
467467
};
468468

469+
class EHCatchEndScope : public EHScope {
470+
public:
471+
EHCatchEndScope(EHScopeStack::stable_iterator enclosingEHScope)
472+
: EHScope(CatchEnd, enclosingEHScope) {}
473+
static size_t getSize() { return sizeof(EHCatchEndScope); }
474+
475+
static bool classof(const EHScope *scope) {
476+
return scope->getKind() == CatchEnd;
477+
}
478+
};
479+
469480
/// A non-stable pointer into the scope stack.
470481
class EHScopeStack::iterator {
471482
char *Ptr;
@@ -503,6 +514,10 @@ class EHScopeStack::iterator {
503514
case EHScope::Terminate:
504515
Size = EHTerminateScope::getSize();
505516
break;
517+
518+
case EHScope::CatchEnd:
519+
Size = EHCatchEndScope::getSize();
520+
break;
506521
}
507522
Ptr += llvm::RoundUpToAlignment(Size, ScopeStackAlignment);
508523
return *this;
@@ -551,6 +566,14 @@ inline void EHScopeStack::popTerminate() {
551566
deallocate(EHTerminateScope::getSize());
552567
}
553568

569+
inline void EHScopeStack::popCatchEnd() {
570+
assert(!empty() && "popping exception stack when not empty");
571+
572+
EHCatchEndScope &scope = cast<EHCatchEndScope>(*begin());
573+
InnermostEHScope = scope.getEnclosingEHScope();
574+
deallocate(EHCatchEndScope::getSize());
575+
}
576+
554577
inline EHScopeStack::iterator EHScopeStack::find(stable_iterator sp) const {
555578
assert(sp.isValid() && "finding invalid savepoint");
556579
assert(sp.Size <= stable_begin().Size && "finding savepoint after pop");
@@ -588,6 +611,13 @@ struct EHPersonality {
588611
static const EHPersonality MSVC_except_handler;
589612
static const EHPersonality MSVC_C_specific_handler;
590613
static const EHPersonality MSVC_CxxFrameHandler3;
614+
615+
bool isMSVCPersonality() const {
616+
return this == &MSVC_except_handler || this == &MSVC_C_specific_handler ||
617+
this == &MSVC_CxxFrameHandler3;
618+
}
619+
620+
bool isMSVCXXPersonality() const { return this == &MSVC_CxxFrameHandler3; }
591621
};
592622
}
593623
}

0 commit comments

Comments
 (0)
Please sign in to comment.