MC: Add support for .cfi_startproc simple

This commit allows LLVM MC to process .cfi_startproc directives when
they are followed by an additional `simple' identifier. This signals to
elide the emission of target specific CFI instructions that would
normally occur initially.

This fixes PR16587.

Differential Revision: http://llvm-reviews.chandlerc.com/D2624


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@200227 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
David Majnemer 2014-01-27 17:20:25 +00:00
parent bb0bb73da1
commit 0fd23cd6c2
8 changed files with 49 additions and 23 deletions

View File

@ -438,7 +438,7 @@ struct MCDwarfFrameInfo {
MCDwarfFrameInfo() MCDwarfFrameInfo()
: Begin(0), End(0), Personality(0), Lsda(0), Function(0), Instructions(), : Begin(0), End(0), Personality(0), Lsda(0), Function(0), Instructions(),
PersonalityEncoding(), LsdaEncoding(0), CompactUnwindEncoding(0), PersonalityEncoding(), LsdaEncoding(0), CompactUnwindEncoding(0),
IsSignalFrame(false) {} IsSignalFrame(false), IsSimple(false) {}
MCSymbol *Begin; MCSymbol *Begin;
MCSymbol *End; MCSymbol *End;
const MCSymbol *Personality; const MCSymbol *Personality;
@ -449,6 +449,7 @@ struct MCDwarfFrameInfo {
unsigned LsdaEncoding; unsigned LsdaEncoding;
uint32_t CompactUnwindEncoding; uint32_t CompactUnwindEncoding;
bool IsSignalFrame; bool IsSignalFrame;
bool IsSimple;
}; };
class MCDwarfFrameEmitter { class MCDwarfFrameEmitter {

View File

@ -626,7 +626,7 @@ public:
virtual void EmitCompactUnwindEncoding(uint32_t CompactUnwindEncoding); virtual void EmitCompactUnwindEncoding(uint32_t CompactUnwindEncoding);
virtual void EmitCFISections(bool EH, bool Debug); virtual void EmitCFISections(bool EH, bool Debug);
void EmitCFIStartProc(); void EmitCFIStartProc(bool IsSimple);
void EmitCFIEndProc(); void EmitCFIEndProc();
virtual void EmitCFIDefCfa(int64_t Register, int64_t Offset); virtual void EmitCFIDefCfa(int64_t Register, int64_t Offset);
virtual void EmitCFIDefCfaOffset(int64_t Offset); virtual void EmitCFIDefCfaOffset(int64_t Offset);

View File

@ -113,7 +113,7 @@ void DwarfCFIException::beginFunction(const MachineFunction *MF) {
if (!shouldEmitPersonality && !shouldEmitMoves) if (!shouldEmitPersonality && !shouldEmitMoves)
return; return;
Asm->OutStreamer.EmitCFIStartProc(); Asm->OutStreamer.EmitCFIStartProc(/*IsSimple=*/false);
// Indicate personality routine, if any. // Indicate personality routine, if any.
if (!shouldEmitPersonality) if (!shouldEmitPersonality)

View File

@ -925,6 +925,8 @@ void MCAsmStreamer::EmitCFIStartProcImpl(MCDwarfFrameInfo &Frame) {
} }
OS << "\t.cfi_startproc"; OS << "\t.cfi_startproc";
if (Frame.IsSimple)
OS << " simple";
EmitEOL(); EmitEOL();
} }

View File

@ -889,7 +889,8 @@ namespace {
unsigned personalityEncoding, unsigned personalityEncoding,
const MCSymbol *lsda, const MCSymbol *lsda,
bool IsSignalFrame, bool IsSignalFrame,
unsigned lsdaEncoding); unsigned lsdaEncoding,
bool IsSimple);
MCSymbol *EmitFDE(MCStreamer &streamer, MCSymbol *EmitFDE(MCStreamer &streamer,
const MCSymbol &cieStart, const MCSymbol &cieStart,
const MCDwarfFrameInfo &frame); const MCDwarfFrameInfo &frame);
@ -1199,7 +1200,8 @@ const MCSymbol &FrameEmitterImpl::EmitCIE(MCStreamer &streamer,
unsigned personalityEncoding, unsigned personalityEncoding,
const MCSymbol *lsda, const MCSymbol *lsda,
bool IsSignalFrame, bool IsSignalFrame,
unsigned lsdaEncoding) { unsigned lsdaEncoding,
bool IsSimple) {
MCContext &context = streamer.getContext(); MCContext &context = streamer.getContext();
const MCRegisterInfo *MRI = context.getRegisterInfo(); const MCRegisterInfo *MRI = context.getRegisterInfo();
const MCObjectFileInfo *MOFI = context.getObjectFileInfo(); const MCObjectFileInfo *MOFI = context.getObjectFileInfo();
@ -1298,9 +1300,11 @@ const MCSymbol &FrameEmitterImpl::EmitCIE(MCStreamer &streamer,
// Initial Instructions // Initial Instructions
const MCAsmInfo *MAI = context.getAsmInfo(); const MCAsmInfo *MAI = context.getAsmInfo();
const std::vector<MCCFIInstruction> &Instructions = if (!IsSimple) {
MAI->getInitialFrameState(); const std::vector<MCCFIInstruction> &Instructions =
EmitCFIInstructions(streamer, Instructions, NULL); MAI->getInitialFrameState();
EmitCFIInstructions(streamer, Instructions, NULL);
}
// Padding // Padding
streamer.EmitValueToAlignment(IsEH ? 4 : MAI->getPointerSize()); streamer.EmitValueToAlignment(IsEH ? 4 : MAI->getPointerSize());
@ -1386,18 +1390,20 @@ MCSymbol *FrameEmitterImpl::EmitFDE(MCStreamer &streamer,
namespace { namespace {
struct CIEKey { struct CIEKey {
static const CIEKey getEmptyKey() { return CIEKey(0, 0, -1, false); } static const CIEKey getEmptyKey() { return CIEKey(0, 0, -1, false, false); }
static const CIEKey getTombstoneKey() { return CIEKey(0, -1, 0, false); } static const CIEKey getTombstoneKey() { return CIEKey(0, -1, 0, false, false); }
CIEKey(const MCSymbol* Personality_, unsigned PersonalityEncoding_, CIEKey(const MCSymbol* Personality_, unsigned PersonalityEncoding_,
unsigned LsdaEncoding_, bool IsSignalFrame_) : unsigned LsdaEncoding_, bool IsSignalFrame_, bool IsSimple_) :
Personality(Personality_), PersonalityEncoding(PersonalityEncoding_), Personality(Personality_), PersonalityEncoding(PersonalityEncoding_),
LsdaEncoding(LsdaEncoding_), IsSignalFrame(IsSignalFrame_) { LsdaEncoding(LsdaEncoding_), IsSignalFrame(IsSignalFrame_),
IsSimple(IsSimple_) {
} }
const MCSymbol* Personality; const MCSymbol* Personality;
unsigned PersonalityEncoding; unsigned PersonalityEncoding;
unsigned LsdaEncoding; unsigned LsdaEncoding;
bool IsSignalFrame; bool IsSignalFrame;
bool IsSimple;
}; };
} }
@ -1414,14 +1420,16 @@ namespace llvm {
return static_cast<unsigned>(hash_combine(Key.Personality, return static_cast<unsigned>(hash_combine(Key.Personality,
Key.PersonalityEncoding, Key.PersonalityEncoding,
Key.LsdaEncoding, Key.LsdaEncoding,
Key.IsSignalFrame)); Key.IsSignalFrame,
Key.IsSimple));
} }
static bool isEqual(const CIEKey &LHS, static bool isEqual(const CIEKey &LHS,
const CIEKey &RHS) { const CIEKey &RHS) {
return LHS.Personality == RHS.Personality && return LHS.Personality == RHS.Personality &&
LHS.PersonalityEncoding == RHS.PersonalityEncoding && LHS.PersonalityEncoding == RHS.PersonalityEncoding &&
LHS.LsdaEncoding == RHS.LsdaEncoding && LHS.LsdaEncoding == RHS.LsdaEncoding &&
LHS.IsSignalFrame == RHS.IsSignalFrame; LHS.IsSignalFrame == RHS.IsSignalFrame &&
LHS.IsSimple == RHS.IsSimple;
} }
}; };
} }
@ -1465,13 +1473,14 @@ void MCDwarfFrameEmitter::Emit(MCStreamer &Streamer, MCAsmBackend *MAB,
for (unsigned i = 0, n = FrameArray.size(); i < n; ++i) { for (unsigned i = 0, n = FrameArray.size(); i < n; ++i) {
const MCDwarfFrameInfo &Frame = FrameArray[i]; const MCDwarfFrameInfo &Frame = FrameArray[i];
CIEKey Key(Frame.Personality, Frame.PersonalityEncoding, CIEKey Key(Frame.Personality, Frame.PersonalityEncoding,
Frame.LsdaEncoding, Frame.IsSignalFrame); Frame.LsdaEncoding, Frame.IsSignalFrame, Frame.IsSimple);
const MCSymbol *&CIEStart = IsEH ? CIEStarts[Key] : DummyDebugKey; const MCSymbol *&CIEStart = IsEH ? CIEStarts[Key] : DummyDebugKey;
if (!CIEStart) if (!CIEStart)
CIEStart = &Emitter.EmitCIE(Streamer, Frame.Personality, CIEStart = &Emitter.EmitCIE(Streamer, Frame.Personality,
Frame.PersonalityEncoding, Frame.Lsda, Frame.PersonalityEncoding, Frame.Lsda,
Frame.IsSignalFrame, Frame.IsSignalFrame,
Frame.LsdaEncoding); Frame.LsdaEncoding,
Frame.IsSimple);
FDEEnd = Emitter.EmitFDE(Streamer, *CIEStart, Frame); FDEEnd = Emitter.EmitFDE(Streamer, *CIEStart, Frame);

View File

@ -2796,9 +2796,14 @@ bool AsmParser::parseDirectiveCFISections() {
} }
/// parseDirectiveCFIStartProc /// parseDirectiveCFIStartProc
/// ::= .cfi_startproc /// ::= .cfi_startproc [simple]
bool AsmParser::parseDirectiveCFIStartProc() { bool AsmParser::parseDirectiveCFIStartProc() {
getStreamer().EmitCFIStartProc(); StringRef Simple;
if (getLexer().isNot(AsmToken::EndOfStatement))
if (parseIdentifier(Simple) || Simple != "simple")
return TokError("unexpected token in .cfi_startproc directive");
getStreamer().EmitCFIStartProc(!Simple.empty());
return false; return false;
} }

View File

@ -248,12 +248,13 @@ void MCStreamer::EmitCFISections(bool EH, bool Debug) {
EmitDebugFrame = Debug; EmitDebugFrame = Debug;
} }
void MCStreamer::EmitCFIStartProc() { void MCStreamer::EmitCFIStartProc(bool IsSimple) {
MCDwarfFrameInfo *CurFrame = getCurrentFrameInfo(); MCDwarfFrameInfo *CurFrame = getCurrentFrameInfo();
if (CurFrame && !CurFrame->End) if (CurFrame && !CurFrame->End)
report_fatal_error("Starting a frame before finishing the previous one!"); report_fatal_error("Starting a frame before finishing the previous one!");
MCDwarfFrameInfo Frame; MCDwarfFrameInfo Frame;
Frame.IsSimple = IsSimple;
EmitCFIStartProcImpl(Frame); EmitCFIStartProcImpl(Frame);
FrameInfos.push_back(Frame); FrameInfos.push_back(Frame);

View File

@ -212,6 +212,11 @@ f36:
nop nop
.cfi_endproc .cfi_endproc
f37:
.cfi_startproc simple
nop
.cfi_endproc
// CHECK: Section { // CHECK: Section {
// CHECK: Index: 4 // CHECK: Index: 4
// CHECK-NEXT: Name: .eh_frame // CHECK-NEXT: Name: .eh_frame
@ -221,7 +226,7 @@ f36:
// CHECK-NEXT: ] // CHECK-NEXT: ]
// CHECK-NEXT: Address: 0x0 // CHECK-NEXT: Address: 0x0
// CHECK-NEXT: Offset: 0x68 // CHECK-NEXT: Offset: 0x68
// CHECK-NEXT: Size: 1736 // CHECK-NEXT: Size: 1776
// CHECK-NEXT: Link: 0 // CHECK-NEXT: Link: 0
// CHECK-NEXT: Info: 0 // CHECK-NEXT: Info: 0
// CHECK-NEXT: AddressAlignment: 8 // CHECK-NEXT: AddressAlignment: 8
@ -337,7 +342,9 @@ f36:
// CHECK-NEXT: 0690: 00000000 1C000000 00000000 017A5052 // CHECK-NEXT: 0690: 00000000 1C000000 00000000 017A5052
// CHECK-NEXT: 06A0: 00017810 0A980000 00000000 00001B0C // CHECK-NEXT: 06A0: 00017810 0A980000 00000000 00001B0C
// CHECK-NEXT: 06B0: 07089001 10000000 24000000 00000000 // CHECK-NEXT: 06B0: 07089001 10000000 24000000 00000000
// CHECK-NEXT: 06C0: 01000000 00000000 // CHECK-NEXT: 06C0: 01000000 00000000 10000000 00000000
// CHECK-NEXT: 06D0: 017A5200 01781001 1B000000 10000000
// CHECK-NEXT: 06E0: 18000000 00000000 01000000 00000000
// CHECK-NEXT: ) // CHECK-NEXT: )
// CHECK-NEXT: } // CHECK-NEXT: }
@ -348,8 +355,8 @@ f36:
// CHECK-NEXT: Flags [ // CHECK-NEXT: Flags [
// CHECK-NEXT: ] // CHECK-NEXT: ]
// CHECK-NEXT: Address: 0x0 // CHECK-NEXT: Address: 0x0
// CHECK-NEXT: Offset: 0xE30 // CHECK-NEXT: Offset: 0xE70
// CHECK-NEXT: Size: 1728 // CHECK-NEXT: Size: 1752
// CHECK-NEXT: Link: 7 // CHECK-NEXT: Link: 7
// CHECK-NEXT: Info: 4 // CHECK-NEXT: Info: 4
// CHECK-NEXT: AddressAlignment: 8 // CHECK-NEXT: AddressAlignment: 8
@ -427,5 +434,6 @@ f36:
// CHECK-NEXT: 0x688 R_X86_64_PC32 .text 0x21 // CHECK-NEXT: 0x688 R_X86_64_PC32 .text 0x21
// CHECK-NEXT: 0x6A6 R_X86_64_PC64 foo 0x0 // CHECK-NEXT: 0x6A6 R_X86_64_PC64 foo 0x0
// CHECK-NEXT: 0x6BC R_X86_64_PC32 .text 0x22 // CHECK-NEXT: 0x6BC R_X86_64_PC32 .text 0x22
// CHECK-NEXT: 0x6E4 R_X86_64_PC32 .text 0x23
// CHECK-NEXT: ] // CHECK-NEXT: ]
// CHECK: } // CHECK: }