From: Alex Lorenz Date: Thu, 30 Jul 2015 16:54:38 +0000 (+0000) Subject: MIR Serialization: Serialize the machine basic block's successor weights. X-Git-Url: http://demsky.eecs.uci.edu/git/?a=commitdiff_plain;h=fbd94791e6881aad4ed1e3fd13c6abbdd2178df4;p=oota-llvm.git MIR Serialization: Serialize the machine basic block's successor weights. Reviewers: Duncan P. N. Exon Smith git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@243659 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/include/llvm/CodeGen/MIRYamlMapping.h b/include/llvm/CodeGen/MIRYamlMapping.h index e2a47295d0f..7ef4c6b6e39 100644 --- a/include/llvm/CodeGen/MIRYamlMapping.h +++ b/include/llvm/CodeGen/MIRYamlMapping.h @@ -73,6 +73,37 @@ template <> struct ScalarTraits { static bool mustQuote(StringRef Scalar) { return needsQuotes(Scalar); } }; +/// A wrapper around unsigned which contains a source range that's being set +/// during parsing. +struct UnsignedValue { + unsigned Value; + SMRange SourceRange; + + UnsignedValue() : Value(0) {} + UnsignedValue(unsigned Value) : Value(Value) {} + + bool operator==(const UnsignedValue &Other) const { + return Value == Other.Value; + } +}; + +template <> struct ScalarTraits { + static void output(const UnsignedValue &Value, void *Ctx, raw_ostream &OS) { + return ScalarTraits::output(Value.Value, Ctx, OS); + } + + static StringRef input(StringRef Scalar, void *Ctx, UnsignedValue &Value) { + if (const auto *Node = + reinterpret_cast(Ctx)->getCurrentNode()) + Value.SourceRange = Node->getSourceRange(); + return ScalarTraits::input(Scalar, Ctx, Value.Value); + } + + static bool mustQuote(StringRef Scalar) { + return ScalarTraits::mustQuote(Scalar); + } +}; + template <> struct ScalarEnumerationTraits { static void enumeration(yaml::IO &IO, MachineJumpTableInfo::JTEntryKind &EntryKind) { @@ -94,6 +125,7 @@ template <> struct ScalarEnumerationTraits { LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::yaml::StringValue) LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(llvm::yaml::FlowStringValue) +LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(llvm::yaml::UnsignedValue) namespace llvm { namespace yaml { @@ -139,8 +171,8 @@ struct MachineBasicBlock { unsigned Alignment = 0; bool IsLandingPad = false; bool AddressTaken = false; - // TODO: Serialize the successor weights. std::vector Successors; + std::vector SuccessorWeights; std::vector LiveIns; std::vector Instructions; }; @@ -156,6 +188,7 @@ template <> struct MappingTraits { YamlIO.mapOptional("isLandingPad", MBB.IsLandingPad); YamlIO.mapOptional("addressTaken", MBB.AddressTaken); YamlIO.mapOptional("successors", MBB.Successors); + YamlIO.mapOptional("weights", MBB.SuccessorWeights); YamlIO.mapOptional("liveins", MBB.LiveIns); YamlIO.mapOptional("instructions", MBB.Instructions); } diff --git a/include/llvm/CodeGen/MachineBasicBlock.h b/include/llvm/CodeGen/MachineBasicBlock.h index 5e5f45cae8f..aba64ff9a1a 100644 --- a/include/llvm/CodeGen/MachineBasicBlock.h +++ b/include/llvm/CodeGen/MachineBasicBlock.h @@ -25,6 +25,7 @@ class Pass; class BasicBlock; class MachineFunction; class MCSymbol; +class MIRPrinter; class SlotIndexes; class StringRef; class raw_ostream; @@ -424,6 +425,9 @@ public: /// which refer to fromMBB to refer to this. void transferSuccessorsAndUpdatePHIs(MachineBasicBlock *fromMBB); + /// Return true if any of the successors have weights attached to them. + bool hasSuccessorWeights() const { return !Weights.empty(); } + /// isPredecessor - Return true if the specified MBB is a predecessor of this /// block. bool isPredecessor(const MachineBasicBlock *MBB) const; @@ -685,6 +689,7 @@ private: const_weight_iterator getWeightIterator(const_succ_iterator I) const; friend class MachineBranchProbabilityInfo; + friend class MIRPrinter; /// getSuccWeight - Return weight of the edge from this block to MBB. This /// method should NOT be called directly, but by using getEdgeWeight method diff --git a/lib/CodeGen/MIRParser/MIRParser.cpp b/lib/CodeGen/MIRParser/MIRParser.cpp index d49085b6fe5..a179c593380 100644 --- a/lib/CodeGen/MIRParser/MIRParser.cpp +++ b/lib/CodeGen/MIRParser/MIRParser.cpp @@ -356,12 +356,24 @@ bool MIRParserImpl::initializeMachineBasicBlock( MBB.setIsLandingPad(YamlMBB.IsLandingPad); SMDiagnostic Error; // Parse the successors. + const auto &Weights = YamlMBB.SuccessorWeights; + bool HasWeights = !Weights.empty(); + if (HasWeights && Weights.size() != YamlMBB.Successors.size()) { + bool IsFew = Weights.size() < YamlMBB.Successors.size(); + return error(IsFew ? Weights.back().SourceRange.End + : Weights[YamlMBB.Successors.size()].SourceRange.Start, + Twine("too ") + (IsFew ? "few" : "many") + + " successor weights, expected " + + Twine(YamlMBB.Successors.size()) + ", have " + + Twine(Weights.size())); + } + size_t SuccessorIndex = 0; for (const auto &MBBSource : YamlMBB.Successors) { MachineBasicBlock *SuccMBB = nullptr; if (parseMBBReference(SuccMBB, MBBSource, MF, PFS)) return true; // TODO: Report an error when adding the same successor more than once. - MBB.addSuccessor(SuccMBB); + MBB.addSuccessor(SuccMBB, HasWeights ? Weights[SuccessorIndex++].Value : 0); } // Parse the liveins. for (const auto &LiveInSource : YamlMBB.LiveIns) { diff --git a/lib/CodeGen/MIRPrinter.cpp b/lib/CodeGen/MIRPrinter.cpp index 84cf09f904c..f34cef7bcb4 100644 --- a/lib/CodeGen/MIRPrinter.cpp +++ b/lib/CodeGen/MIRPrinter.cpp @@ -56,6 +56,10 @@ struct FrameIndexOperand { } }; +} // end anonymous namespace + +namespace llvm { + /// This class prints out the machine functions using the MIR serialization /// format. class MIRPrinter { @@ -88,6 +92,10 @@ private: void initRegisterMaskIds(const MachineFunction &MF); }; +} // end namespace llvm + +namespace { + /// This class prints out the machine instructions using the MIR serialization /// format. class MIPrinter { @@ -363,6 +371,11 @@ void MIRPrinter::convert(ModuleSlotTracker &MST, .printMBBReference(*SuccMBB); YamlMBB.Successors.push_back(StrOS.str()); } + if (MBB.hasSuccessorWeights()) { + for (auto I = MBB.succ_begin(), E = MBB.succ_end(); I != E; ++I) + YamlMBB.SuccessorWeights.push_back( + yaml::UnsignedValue(MBB.getSuccWeight(I))); + } // Print the live in registers. const auto *TRI = MBB.getParent()->getSubtarget().getRegisterInfo(); assert(TRI && "Expected target register info"); diff --git a/test/CodeGen/MIR/X86/successor-basic-blocks-few-weights.mir b/test/CodeGen/MIR/X86/successor-basic-blocks-few-weights.mir new file mode 100644 index 00000000000..93d2eead0bb --- /dev/null +++ b/test/CodeGen/MIR/X86/successor-basic-blocks-few-weights.mir @@ -0,0 +1,41 @@ +# RUN: not llc -march=x86-64 -start-after branch-folder -stop-after branch-folder -o /dev/null %s 2>&1 | FileCheck %s + +--- | + + define i32 @foo(i32 %a) { + entry: + %0 = icmp sle i32 %a, 10 + br i1 %0, label %less, label %exit + + less: + ret i32 0 + + exit: + ret i32 %a + } + +... +--- +name: foo +body: + - id: 0 + name: entry + successors: [ '%bb.1.less', '%bb.2.exit' ] +# CHECK: [[@LINE+1]]:23: too few successor weights, expected 2, have 1 + weights: [ 16 ] + liveins: [ '%edi' ] + instructions: + - 'CMP32ri8 %edi, 10, implicit-def %eflags' + - 'JG_1 %bb.2.exit, implicit killed %eflags' + - id: 1 + name: less + instructions: + - '%eax = MOV32r0 implicit-def dead %eflags' + - 'RETQ killed %eax' + - id: 2 + name: exit + liveins: [ '%edi' ] + instructions: + - '%eax = COPY killed %edi' + - 'RETQ killed %eax' +... diff --git a/test/CodeGen/MIR/X86/successor-basic-blocks-many-weights.mir b/test/CodeGen/MIR/X86/successor-basic-blocks-many-weights.mir new file mode 100644 index 00000000000..2c0e1932aad --- /dev/null +++ b/test/CodeGen/MIR/X86/successor-basic-blocks-many-weights.mir @@ -0,0 +1,41 @@ +# RUN: not llc -march=x86-64 -start-after branch-folder -stop-after branch-folder -o /dev/null %s 2>&1 | FileCheck %s + +--- | + + define i32 @foo(i32 %a) { + entry: + %0 = icmp sle i32 %a, 10 + br i1 %0, label %less, label %exit + + less: + ret i32 0 + + exit: + ret i32 %a + } + +... +--- +name: foo +body: + - id: 0 + name: entry + successors: [ '%bb.1.less', '%bb.2.exit' ] +# CHECK: [[@LINE+1]]:28: too many successor weights, expected 2, have 3 + weights: [ 16, 16, 16 ] + liveins: [ '%edi' ] + instructions: + - 'CMP32ri8 %edi, 10, implicit-def %eflags' + - 'JG_1 %bb.2.exit, implicit killed %eflags' + - id: 1 + name: less + instructions: + - '%eax = MOV32r0 implicit-def dead %eflags' + - 'RETQ killed %eax' + - id: 2 + name: exit + liveins: [ '%edi' ] + instructions: + - '%eax = COPY killed %edi' + - 'RETQ killed %eax' +... diff --git a/test/CodeGen/MIR/X86/successor-basic-blocks-weights.mir b/test/CodeGen/MIR/X86/successor-basic-blocks-weights.mir new file mode 100644 index 00000000000..c7cfeaae90d --- /dev/null +++ b/test/CodeGen/MIR/X86/successor-basic-blocks-weights.mir @@ -0,0 +1,46 @@ +# RUN: llc -march=x86-64 -start-after branch-folder -stop-after branch-folder -o /dev/null %s | FileCheck %s +# This test ensures that the MIR parser parses basic block successors and +# weights correctly. + +--- | + + define i32 @foo(i32 %a) { + entry: + %0 = icmp sle i32 %a, 10 + br i1 %0, label %less, label %exit + + less: + ret i32 0 + + exit: + ret i32 %a + } + +... +--- +name: foo +body: + # CHECK: name: entry + # CHECK: successors: [ '%bb.1.less', '%bb.2.exit' ] + # CHECK-NEXT: weights: [ 16, 32 ] + # CHECK: name: less + - id: 0 + name: entry + successors: [ '%bb.1.less', '%bb.2.exit' ] + weights: [ 16, 32 ] + liveins: [ '%edi' ] + instructions: + - 'CMP32ri8 %edi, 10, implicit-def %eflags' + - 'JG_1 %bb.2.exit, implicit killed %eflags' + - id: 1 + name: less + instructions: + - '%eax = MOV32r0 implicit-def dead %eflags' + - 'RETQ killed %eax' + - id: 2 + name: exit + liveins: [ '%edi' ] + instructions: + - '%eax = COPY killed %edi' + - 'RETQ killed %eax' +...