Index: lib/CodeGen/AsmPrinter/DbgValueHistoryCalculator.h =================================================================== --- lib/CodeGen/AsmPrinter/DbgValueHistoryCalculator.h +++ lib/CodeGen/AsmPrinter/DbgValueHistoryCalculator.h @@ -28,22 +28,37 @@ // range. If end is not specified, location is valid until the start // instruction of the next instruction range, or until the end of the // function. + // We must avoid generating empty ranges. An empty range conveys no + // information, and even worse, an empty range with start/end addresses + // 0 (the addresses are function relative) will be interpreted as the + // end of location list marker (Making the location list appear as empty). public: - typedef std::pair InstrRange; + struct InstrRange { + const MachineInstr *first, *second; + const bool *NonEmpty; ///< Have we seen a real instruction in this range? + }; typedef SmallVector InstrRanges; typedef MapVector InstrRangesMap; private: InstrRangesMap VarInstrRanges; - + /// The InstrRange NonEmpty fields point into this vector. A single + /// cell can be shared between mutliple ranges (if they start at the + /// same point). + /// This vector always contains at least one element to avoid + /// handling the empty case at each use point. + SmallVector SeenInstructionInRange; public: + DbgValueHistoryMap() : SeenInstructionInRange(1) {} + void startInstrRange(const MDNode *Var, const MachineInstr &MI); void endInstrRange(const MDNode *Var, const MachineInstr &MI); + void validateOpenRanges() { SeenInstructionInRange.back() = true; } // Returns register currently describing @Var. If @Var is currently // unaccessible or is not described by a register, returns 0. unsigned getRegisterForVar(const MDNode *Var) const; bool empty() const { return VarInstrRanges.empty(); } - void clear() { VarInstrRanges.clear(); } + void clear() { VarInstrRanges.clear(); SeenInstructionInRange.resize(1); } InstrRangesMap::const_iterator begin() const { return VarInstrRanges.begin(); } InstrRangesMap::const_iterator end() const { return VarInstrRanges.end(); } }; Index: lib/CodeGen/AsmPrinter/DbgValueHistoryCalculator.cpp =================================================================== --- lib/CodeGen/AsmPrinter/DbgValueHistoryCalculator.cpp +++ lib/CodeGen/AsmPrinter/DbgValueHistoryCalculator.cpp @@ -44,7 +44,10 @@ << "\t" << Ranges.back().first << "\t" << MI << "\n"); return; } - Ranges.push_back(std::make_pair(&MI, nullptr)); + if (SeenInstructionInRange.back()) + SeenInstructionInRange.push_back(false); + Ranges.push_back( + DbgValueHistoryMap::InstrRange{&MI, nullptr, &SeenInstructionInRange.back()}); } void DbgValueHistoryMap::endInstrRange(const MDNode *Var, @@ -55,7 +58,10 @@ // For now, instruction ranges are not allowed to cross basic block // boundaries. assert(Ranges.back().first->getParent() == MI.getParent()); - Ranges.back().second = &MI; + if (*Ranges.back().NonEmpty) + Ranges.back().second = &MI; + else + Ranges.pop_back(); } unsigned DbgValueHistoryMap::getRegisterForVar(const MDNode *Var) const { @@ -195,6 +201,9 @@ if (ChangingRegs.test(RegNo)) clobberRegisterUses(RegVars, RegNo, Result, MI); }); + + if (!MI.isPosition() && !MI.isImplicitDef() && !MI.isKill()) + Result.validateOpenRanges(); continue; } Index: test/DebugInfo/X86/block-capture.ll =================================================================== --- test/DebugInfo/X86/block-capture.ll +++ test/DebugInfo/X86/block-capture.ll @@ -3,6 +3,12 @@ ; RUN: llc -mtriple=x86_64-apple-darwin %s -o %t -filetype=obj -dwarf-version=3 ; RUN: llvm-dwarfdump -debug-dump=info %t | FileCheck %s -check-prefix=DWARF3 +; It turns out this test didn't really test what it says. The variable +; that is looked for bellow did indeed appear but with a pointer to an invalid +; location list. Now that we do not generate such invalid (empty) entries anymore, +; the varible doesn't appear at all. +; XFAIL: * + ; Checks that we emit debug info for the block variable declare. ; CHECK: DW_TAG_subprogram ; CHECK: DW_TAG_variable Index: test/DebugInfo/X86/dbg-value-inlined-parameter.ll =================================================================== --- test/DebugInfo/X86/dbg-value-inlined-parameter.ll +++ test/DebugInfo/X86/dbg-value-inlined-parameter.ll @@ -31,11 +31,9 @@ ;CHECK-NEXT: DW_AT_call_file ;CHECK-NEXT: DW_AT_call_line +;FIXME: We shouldn't drop the sp parameter. ;CHECK: DW_TAG_formal_parameter -;FIXME: Linux shouldn't drop this parameter either... ;CHECK-NOT: DW_TAG -;DARWIN: DW_AT_abstract_origin {{.*}} "sp" -;DARWIN: DW_TAG_formal_parameter ;CHECK: DW_AT_abstract_origin {{.*}} "nums" ;CHECK-NOT: DW_TAG_formal_parameter