69#define PASS_KEY "x86-lvi-load"
70#define DEBUG_TYPE PASS_KEY
72STATISTIC(NumFences,
"Number of LFENCEs inserted for LVI mitigation");
73STATISTIC(NumFunctionsConsidered,
"Number of functions analyzed");
74STATISTIC(NumFunctionsMitigated,
"Number of functions for which mitigations "
76STATISTIC(NumGadgets,
"Number of LVI gadgets detected during analysis");
84 cl::desc(
"Don't treat conditional branches as disclosure gadgets. This "
85 "may improve performance, at the cost of security."),
91 "For each function, emit a dot graph depicting potential LVI gadgets"),
96 cl::desc(
"For each function, emit a dot graph depicting potential LVI "
97 "gadgets, and do not insert any fences"),
102 cl::desc(
"For each function, emit a dot graph to stdout depicting "
103 "potential LVI gadgets, used for testing purposes only"),
107typedef int (*
OptimizeCutT)(
unsigned int *Nodes,
unsigned int NodesSize,
108 unsigned int *Edges,
int *EdgeValues,
109 int *CutEdges ,
unsigned int EdgesSize);
115 static constexpr int GadgetEdgeSentinel = -1;
116 static constexpr MachineInstr *
const ArgNodeSentinel =
nullptr;
119 using Node =
typename GraphT::Node;
120 using Edge =
typename GraphT::Edge;
122 MachineGadgetGraph(std::unique_ptr<
Node[]> Nodes,
123 std::unique_ptr<Edge[]> Edges, size_type NodesSize,
124 size_type EdgesSize,
int NumFences = 0,
int NumGadgets = 0)
125 : GraphT(std::move(Nodes), std::move(Edges), NodesSize, EdgesSize),
126 NumFences(NumFences), NumGadgets(NumGadgets) {}
127 static inline bool isCFGEdge(
const Edge &E) {
128 return E.getValue() != GadgetEdgeSentinel;
130 static inline bool isGadgetEdge(
const Edge &E) {
131 return E.getValue() == GadgetEdgeSentinel;
142 return "X86 Load Value Injection (LVI) Load Hardening";
151 using Edge = MachineGadgetGraph::Edge;
152 using Node = MachineGadgetGraph::Node;
153 using EdgeSet = MachineGadgetGraph::EdgeSet;
154 using NodeSet = MachineGadgetGraph::NodeSet;
160 std::unique_ptr<MachineGadgetGraph>
165 std::unique_ptr<MachineGadgetGraph> Graph)
const;
167 std::unique_ptr<MachineGadgetGraph> Graph)
const;
168 int elimMitigatedEdgesAndNodes(MachineGadgetGraph &
G,
171 std::unique_ptr<MachineGadgetGraph>
172 trimMitigatedEdges(std::unique_ptr<MachineGadgetGraph> Graph)
const;
174 EdgeSet &CutEdges )
const;
175 bool instrUsesRegToAccessMemory(
const MachineInstr &
I,
unsigned Reg)
const;
176 bool instrUsesRegToBranch(
const MachineInstr &
I,
unsigned Reg)
const;
178 return MI && (
MI->getOpcode() == X86::LFENCE ||
179 (STI->useLVIControlFlowIntegrity() &&
MI->isCall()));
203 if (Node->getValue() == MachineGadgetGraph::ArgNodeSentinel)
208 OS << *Node->getValue();
214 if (
MI == MachineGadgetGraph::ArgNodeSentinel)
215 return "color = blue";
216 if (
MI->getOpcode() == X86::LFENCE)
217 return "color = green";
223 int EdgeVal = (*
E.getCurrent()).getValue();
224 return EdgeVal >= 0 ?
"label = " + std::to_string(EdgeVal)
225 :
"color = red, style = \"dashed\"";
231constexpr MachineInstr *MachineGadgetGraph::ArgNodeSentinel;
232constexpr int MachineGadgetGraph::GadgetEdgeSentinel;
234char X86LoadValueInjectionLoadHardeningPass::ID = 0;
236void X86LoadValueInjectionLoadHardeningPass::getAnalysisUsage(
246 MachineGadgetGraph *
G) {
248 "Speculative gadgets for \"" + MF.
getName() +
"\" function");
251bool X86LoadValueInjectionLoadHardeningPass::runOnMachineFunction(
256 if (!STI->useLVILoadHardening())
265 if (!
F.hasOptNone() && skipFunction(
F))
268 ++NumFunctionsConsidered;
269 TII = STI->getInstrInfo();
270 TRI = STI->getRegisterInfo();
272 const auto &MLI = getAnalysis<MachineLoopInfoWrapperPass>().getLI();
273 const auto &MDT = getAnalysis<MachineDominatorTreeWrapperPass>().getDomTree();
274 const auto &MDF = getAnalysis<MachineDominanceFrontier>();
275 std::unique_ptr<MachineGadgetGraph> Graph = getGadgetGraph(MF, MLI, MDT, MDF);
277 if (Graph ==
nullptr)
288 std::string FileName =
"lvi.";
304 std::string ErrorMsg;
307 if (!ErrorMsg.empty())
314 FencesInserted = hardenLoadsWithPlugin(MF, std::move(Graph));
316 FencesInserted = hardenLoadsWithHeuristic(MF, std::move(Graph));
319 if (FencesInserted > 0)
320 ++NumFunctionsMitigated;
321 NumFences += FencesInserted;
322 return (FencesInserted > 0);
325std::unique_ptr<MachineGadgetGraph>
326X86LoadValueInjectionLoadHardeningPass::getGadgetGraph(
333 DataFlowGraph DFG{MF, *
TII, *
TRI, MDT, MDF};
338 GraphBuilder Builder;
339 using GraphIter =
typename GraphBuilder::BuilderNodeRef;
341 int FenceCount = 0, GadgetCount = 0;
344 if (
Ref == NodeMap.
end()) {
345 auto I = Builder.addVertex(
MI);
347 return std::pair<GraphIter, bool>{
I,
true};
349 return std::pair<GraphIter, bool>{
Ref->getSecond(),
false};
360 auto AnalyzeDef = [&](NodeAddr<DefNode *> SourceDef) {
362 std::function<void(NodeAddr<DefNode *>)> AnalyzeDefUseChain =
363 [&](NodeAddr<DefNode *>
Def) {
369 RegisterRef DefReg =
Def.Addr->getRegRef(DFG);
370 for (
auto UseID :
L.getAllReachedUses(DefReg, Def)) {
371 auto Use = DFG.addr<UseNode *>(UseID);
372 if (
Use.Addr->getFlags() & NodeAttrs::PhiRef) {
373 NodeAddr<PhiNode *>
Phi =
Use.Addr->getOwner(DFG);
374 for (
const auto&
I :
L.getRealUses(
Phi.Id)) {
375 if (DFG.getPRI().alias(RegisterRef(
I.first), DefReg)) {
376 for (
const auto &UA :
I.second)
377 Uses.emplace(UA.first);
388 for (
auto UseID :
Uses) {
389 if (!UsesVisited.
insert(UseID).second)
392 auto Use = DFG.addr<UseNode *>(UseID);
393 assert(!(
Use.Addr->getFlags() & NodeAttrs::PhiRef));
406 if (instrUsesRegToAccessMemory(
UseMI, UseMO.
getReg()) ||
409 Transmitters[
Def.Id].push_back(
Use.Addr->getOwner(DFG).Id);
417 NodeAddr<InstrNode *> Owner{
Use.Addr->getOwner(DFG)};
419 for (
const auto &ChildDef :
420 Owner.Addr->members_if(DataFlowGraph::IsDef, DFG)) {
421 if (!DefsVisited.
insert(ChildDef.Id).second)
423 if (
Def.Addr->getAttrs() & NodeAttrs::Dead)
425 if (
Def.Id == ChildDef.Id)
428 AnalyzeDefUseChain(ChildDef);
431 for (
auto TransmitterId : Transmitters[ChildDef.Id])
432 Transmitters[
Def.Id].push_back(TransmitterId);
438 auto &DefTransmitters = Transmitters[
Def.Id];
443 DefTransmitters.end());
447 AnalyzeDefUseChain(SourceDef);
448 auto &SourceDefTransmitters = Transmitters[SourceDef.Id];
449 if (SourceDefTransmitters.empty())
453 ? MachineGadgetGraph::ArgNodeSentinel
455 auto GadgetSource = MaybeAddNode(Source);
457 for (
auto TransmitterId : SourceDefTransmitters) {
459 auto GadgetSink = MaybeAddNode(Sink);
461 Builder.addEdge(MachineGadgetGraph::GadgetEdgeSentinel,
462 GadgetSource.first, GadgetSink.first);
467 LLVM_DEBUG(
dbgs() <<
"Analyzing def-use chains to find gadgets\n");
469 NodeAddr<BlockNode *> EntryBlock = DFG.getFunc().Addr->getEntryBlock(DFG);
470 for (NodeAddr<PhiNode *> ArgPhi :
471 EntryBlock.Addr->members_if(DataFlowGraph::IsPhi, DFG)) {
472 NodeList Defs = ArgPhi.Addr->members_if(DataFlowGraph::IsDef, DFG);
476 for (NodeAddr<BlockNode *> BA : DFG.getFunc().Addr->members(DFG)) {
477 for (NodeAddr<StmtNode *> SA :
478 BA.Addr->members_if(DataFlowGraph::IsCode<NodeAttrs::Stmt>, DFG)) {
483 }
else if (
MI->mayLoad()) {
484 NodeList Defs = SA.Addr->members_if(DataFlowGraph::IsDef, DFG);
491 if (GadgetCount == 0)
493 NumGadgets += GadgetCount;
503 auto BeginBB = MaybeAddNode(&*NI);
504 Builder.addEdge(ParentDepth, GI, BeginBB.first);
510 while (++NI !=
MBB->
end()) {
512 if (
Ref != NodeMap.
end()) {
513 Builder.addEdge(LoopDepth, GI,
Ref->getSecond());
514 GI =
Ref->getSecond();
521 auto EndBB = MaybeAddNode(&*
T);
523 Builder.addEdge(LoopDepth, GI, EndBB.first);
528 TraverseCFG(Succ, GI, LoopDepth);
532 GraphIter ArgNode = MaybeAddNode(MachineGadgetGraph::ArgNodeSentinel).first;
533 TraverseCFG(&MF.
front(), ArgNode, 0);
534 std::unique_ptr<MachineGadgetGraph>
G{Builder.get(FenceCount, GadgetCount)};
540int X86LoadValueInjectionLoadHardeningPass::elimMitigatedEdgesAndNodes(
541 MachineGadgetGraph &
G, EdgeSet &ElimEdges ,
543 if (
G.NumFences > 0) {
546 for (
const Edge &E :
G.edges()) {
547 const Node *Dest = E.getDest();
548 if (isFence(Dest->getValue())) {
551 for (
const Edge &DE : Dest->edges())
552 ElimEdges.insert(DE);
558 int RemainingGadgets = 0;
560 for (
const Node &RootN :
G.nodes()) {
561 if (
llvm::none_of(RootN.edges(), MachineGadgetGraph::isGadgetEdge))
565 ReachableNodes.clear();
566 std::function<void(
const Node *,
bool)> FindReachableNodes =
567 [&](
const Node *
N,
bool FirstNode) {
569 ReachableNodes.insert(*
N);
570 for (
const Edge &E :
N->edges()) {
571 const Node *Dest = E.getDest();
572 if (MachineGadgetGraph::isCFGEdge(E) && !ElimEdges.contains(E) &&
573 !ReachableNodes.contains(*Dest))
574 FindReachableNodes(Dest,
false);
577 FindReachableNodes(&RootN,
true);
580 for (
const Edge &E : RootN.edges()) {
581 if (MachineGadgetGraph::isGadgetEdge(E)) {
582 if (ReachableNodes.contains(*E.getDest())) {
591 return RemainingGadgets;
594std::unique_ptr<MachineGadgetGraph>
595X86LoadValueInjectionLoadHardeningPass::trimMitigatedEdges(
596 std::unique_ptr<MachineGadgetGraph> Graph)
const {
598 EdgeSet ElimEdges{*Graph};
599 int RemainingGadgets =
600 elimMitigatedEdgesAndNodes(*Graph, ElimEdges, ElimNodes);
601 if (ElimEdges.empty() && ElimNodes.
empty()) {
602 Graph->NumFences = 0;
603 Graph->NumGadgets = RemainingGadgets;
605 Graph = GraphBuilder::trim(*Graph, ElimNodes, ElimEdges, 0 ,
611int X86LoadValueInjectionLoadHardeningPass::hardenLoadsWithPlugin(
612 MachineFunction &MF, std::unique_ptr<MachineGadgetGraph> Graph)
const {
613 int FencesInserted = 0;
617 Graph = trimMitigatedEdges(std::move(Graph));
619 if (Graph->NumGadgets == 0)
623 EdgeSet CutEdges{*Graph};
624 auto Nodes = std::make_unique<unsigned int[]>(Graph->nodes_size() +
626 auto Edges = std::make_unique<unsigned int[]>(Graph->edges_size());
627 auto EdgeCuts = std::make_unique<int[]>(Graph->edges_size());
628 auto EdgeValues = std::make_unique<int[]>(Graph->edges_size());
629 for (
const Node &
N : Graph->nodes()) {
630 Nodes[Graph->getNodeIndex(
N)] = Graph->getEdgeIndex(*
N.edges_begin());
632 Nodes[Graph->nodes_size()] = Graph->edges_size();
633 for (
const Edge &E : Graph->edges()) {
634 Edges[Graph->getEdgeIndex(E)] = Graph->getNodeIndex(*E.getDest());
635 EdgeValues[Graph->getEdgeIndex(E)] = E.getValue();
637 OptimizeCut(Nodes.get(), Graph->nodes_size(), Edges.get(), EdgeValues.get(),
638 EdgeCuts.get(), Graph->edges_size());
639 for (
int I = 0;
I < Graph->edges_size(); ++
I)
646 FencesInserted += insertFences(MF, *Graph, CutEdges);
648 LLVM_DEBUG(
dbgs() <<
"Inserted " << FencesInserted <<
" fences\n");
650 Graph = GraphBuilder::trim(*Graph,
NodeSet{*Graph}, CutEdges);
653 return FencesInserted;
656int X86LoadValueInjectionLoadHardeningPass::hardenLoadsWithHeuristic(
657 MachineFunction &MF, std::unique_ptr<MachineGadgetGraph> Graph)
const {
660 if (Graph->NumFences > 0) {
662 Graph = trimMitigatedEdges(std::move(Graph));
666 if (Graph->NumGadgets == 0)
670 EdgeSet CutEdges{*Graph};
674 for (
const Edge &E : Graph->edges())
675 if (MachineGadgetGraph::isCFGEdge(E))
676 IngressEdgeMap[E.getDest()].push_back(&E);
686 for (
const Node &
N : Graph->nodes()) {
687 for (
const Edge &E :
N.edges()) {
688 if (!MachineGadgetGraph::isGadgetEdge(E))
693 for (
const Edge &EgressEdge :
N.edges())
694 if (MachineGadgetGraph::isCFGEdge(EgressEdge))
697 int EgressCutCost = 0, IngressCutCost = 0;
698 for (
const Edge *EgressEdge : EgressEdges)
699 if (!CutEdges.contains(*EgressEdge))
700 EgressCutCost += EgressEdge->getValue();
701 for (
const Edge *IngressEdge : IngressEdges)
702 if (!CutEdges.contains(*IngressEdge))
703 IngressCutCost += IngressEdge->getValue();
706 IngressCutCost < EgressCutCost ? IngressEdges : EgressEdges;
707 for (
const Edge *E : EdgesToCut)
715 int FencesInserted = insertFences(MF, *Graph, CutEdges);
717 LLVM_DEBUG(
dbgs() <<
"Inserted " << FencesInserted <<
" fences\n");
719 return FencesInserted;
722int X86LoadValueInjectionLoadHardeningPass::insertFences(
724 EdgeSet &CutEdges )
const {
725 int FencesInserted = 0;
726 for (
const Node &
N :
G.nodes()) {
727 for (
const Edge &E :
N.edges()) {
728 if (CutEdges.contains(E)) {
732 if (
MI == MachineGadgetGraph::ArgNodeSentinel) {
737 }
else if (
MI->isBranch()) {
738 MBB =
MI->getParent();
740 Prev =
MI->getPrevNode();
743 for (
const Edge &E :
N.edges()) {
744 if (MachineGadgetGraph::isCFGEdge(E))
748 MBB =
MI->getParent();
749 InsertionPt =
MI->getNextNode() ?
MI->getNextNode() :
MBB->
end();
750 Prev = InsertionPt ==
MBB->
end()
752 : InsertionPt->getPrevNode();
755 if ((InsertionPt ==
MBB->
end() || !isFence(&*InsertionPt)) &&
756 (!Prev || !isFence(Prev))) {
763 return FencesInserted;
766bool X86LoadValueInjectionLoadHardeningPass::instrUsesRegToAccessMemory(
768 if (!
MI.mayLoadOrStore() ||
MI.getOpcode() == X86::MFENCE ||
769 MI.getOpcode() == X86::SFENCE ||
MI.getOpcode() == X86::LFENCE)
773 if (MemRefBeginIdx < 0) {
774 LLVM_DEBUG(
dbgs() <<
"Warning: unable to obtain memory operand for loading "
784 return (BaseMO.
isReg() && BaseMO.
getReg() != X86::NoRegister &&
785 TRI->regsOverlap(BaseMO.
getReg(), Reg)) ||
786 (IndexMO.
isReg() && IndexMO.
getReg() != X86::NoRegister &&
787 TRI->regsOverlap(IndexMO.
getReg(), Reg));
790bool X86LoadValueInjectionLoadHardeningPass::instrUsesRegToBranch(
792 if (!
MI.isConditionalBranch())
795 if (
Use.isReg() &&
Use.getReg() == Reg)
801 "X86 LVI load hardening",
false,
false)
809 return new X86LoadValueInjectionLoadHardeningPass();
MachineInstrBuilder & UseMI
AMDGPU Mark last scratch load
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
This file defines the DenseMap class.
Rewrite Partial Register Uses
const HexagonInstrInfo * TII
Description: ImmutableGraph is a fast DAG implementation that cannot be modified, except by creating ...
unsigned const TargetRegisterInfo * TRI
#define INITIALIZE_PASS_DEPENDENCY(depName)
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
assert(ImpDefSCC.getReg()==AMDGPU::SCC &&ImpDefSCC.isDef())
This file defines the SmallSet class.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
int(* OptimizeCutT)(unsigned int *Nodes, unsigned int NodesSize, unsigned int *Edges, int *EdgeValues, int *CutEdges, unsigned int EdgesSize)
static cl::opt< bool > EmitDot(PASS_KEY "-dot", cl::desc("For each function, emit a dot graph depicting potential LVI gadgets"), cl::init(false), cl::Hidden)
static cl::opt< std::string > OptimizePluginPath(PASS_KEY "-opt-plugin", cl::desc("Specify a plugin to optimize LFENCE insertion"), cl::Hidden)
static void writeGadgetGraph(raw_ostream &OS, MachineFunction &MF, MachineGadgetGraph *G)
static cl::opt< bool > EmitDotVerify(PASS_KEY "-dot-verify", cl::desc("For each function, emit a dot graph to stdout depicting " "potential LVI gadgets, used for testing purposes only"), cl::init(false), cl::Hidden)
static llvm::sys::DynamicLibrary OptimizeDL
static OptimizeCutT OptimizeCut
static cl::opt< bool > EmitDotOnly(PASS_KEY "-dot-only", cl::desc("For each function, emit a dot graph depicting potential LVI " "gadgets, and do not insert any fences"), cl::init(false), cl::Hidden)
static cl::opt< bool > NoConditionalBranches(PASS_KEY "-no-cbranch", cl::desc("Don't treat conditional branches as disclosure gadgets. This " "may improve performance, at the cost of security."), cl::init(false), cl::Hidden)
Represent the analysis usage information of a pass.
AnalysisUsage & addRequired()
void setPreservesCFG()
This function should be called by the pass, iff they do not:
iterator find(const_arg_type_t< KeyT > Val)
bool contains(const_arg_type_t< KeyT > Val) const
Return true if the specified key is in the map, false otherwise.
virtual std::string message() const
Return the error message as a string.
This class wraps a filename and another Error.
FunctionPass class - This class is used to implement most global optimizations.
unsigned getLoopDepth(const BlockT *BB) const
Return the loop nesting level of the specified block.
iterator getFirstTerminator()
Returns an iterator to the first terminator instruction of this basic block.
iterator_range< succ_iterator > successors()
Analysis pass which computes a MachineDominatorTree.
DominatorTree Class - Concrete subclass of DominatorTreeBase that is used to compute a normal dominat...
MachineFunctionPass - This class adapts the FunctionPass interface to allow convenient creation of pa...
void getAnalysisUsage(AnalysisUsage &AU) const override
getAnalysisUsage - Subclasses that override getAnalysisUsage must call this.
virtual bool runOnMachineFunction(MachineFunction &MF)=0
runOnMachineFunction - This method must be overloaded to perform the desired machine code transformat...
const TargetSubtargetInfo & getSubtarget() const
getSubtarget - Return the subtarget for which this machine code is being compiled.
StringRef getName() const
getName - Return the name of the corresponding LLVM function.
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
Function & getFunction()
Return the LLVM function that this machine code represents.
const MachineBasicBlock & front() const
Representation of each machine instruction.
const MachineBasicBlock * getParent() const
MachineOperand class - Representation of each machine instruction operand.
bool isReg() const
isReg - Tests if this is a MO_Register operand.
MachineInstr * getParent()
getParent - Return the instruction that this operand belongs to.
Register getReg() const
getReg - Returns the register number.
A NodeSet contains a set of SUnit DAG nodes with additional information that assigns a priority to th...
virtual void print(raw_ostream &OS, const Module *M) const
print - Print out the internal state of the pass.
virtual StringRef getPassName() const
getPassName - Return a nice clean name for a pass.
SmallSet - This maintains a set of unique values, optimizing for the case when the set is small (less...
std::pair< const_iterator, bool > insert(const T &V)
insert - Insert an element into the set if it isn't already there.
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringRef - Represent a constant reference to a string, i.e.
TargetInstrInfo - Interface to description of machine instruction set.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
A Use represents the edge between a Value definition and its users.
A raw_ostream that writes to a file descriptor.
This class implements an extremely fast bulk output stream that can only output to a stream.
A raw_ostream that writes to an std::string.
This class provides a portable interface to dynamic libraries which also might be known as shared lib...
static DynamicLibrary getPermanentLibrary(const char *filename, std::string *errMsg=nullptr)
This function permanently loads the dynamic library at the given path using the library load operatio...
void * getAddressOfSymbol(const char *symbolName)
Searches through the library for the symbol symbolName.
bool isValid() const
Returns true if the object refers to a valid library.
unsigned ID
LLVM IR allows to use arbitrary numbers as calling convention identifiers.
int getFirstAddrOperandIdx(const MachineInstr &MI)
Return the index of the instruction's first address operand, if it has a memory reference,...
initializer< Ty > init(const Ty &Val)
NodeAddr< PhiNode * > Phi
NodeAddr< DefNode * > Def
std::set< NodeId > NodeSet
This is an optimization pass for GlobalISel generic memory operations.
UnaryFunction for_each(R &&Range, UnaryFunction F)
Provide wrappers to std::for_each which take ranges instead of having to pass begin/end explicitly.
MachineInstrBuilder BuildMI(MachineFunction &MF, const MIMetadata &MIMD, const MCInstrDesc &MCID)
Builder interface. Specify how to create the initial instruction itself.
FunctionPass * createX86LoadValueInjectionLoadHardeningPass()
raw_fd_ostream & outs()
This returns a reference to a raw_fd_ostream for standard output.
raw_ostream & WriteGraph(raw_ostream &O, const GraphType &G, bool ShortNames=false, const Twine &Title="")
auto unique(Range &&R, Predicate P)
void sort(IteratorTy Start, IteratorTy End)
raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
bool none_of(R &&Range, UnaryPredicate P)
Provide wrappers to std::none_of which take ranges instead of having to pass begin/end explicitly.
void report_fatal_error(Error Err, bool gen_crash_diag=true)
Report a serious error, calling any installed error handler.
raw_fd_ostream & errs()
This returns a reference to a raw_ostream for standard error.
@ Ref
The access may reference the value stored in memory.
typename Traits::NodeRef NodeRef
static std::string getNodeAttributes(NodeRef Node, GraphType *)
typename Traits::ChildIteratorType ChildIteratorType
std::string getNodeLabel(NodeRef Node, GraphType *)
static std::string getEdgeAttributes(NodeRef, ChildIteratorType E, GraphType *)
MachineGadgetGraph GraphType
typename Traits::EdgeRef EdgeRef
typename Traits::ChildEdgeIteratorType ChildEdgeIteratorType
DOTGraphTraits(bool IsSimple=false)
DOTGraphTraits - Template class that can be specialized to customize how graphs are converted to 'dot...
DefaultDOTGraphTraits - This class provides the default implementations of all of the DOTGraphTraits ...
typename GraphType::UnknownGraphTypeError NodeRef