68#define PASS_KEY "x86-lvi-load"
69#define DEBUG_TYPE PASS_KEY
71STATISTIC(NumFences,
"Number of LFENCEs inserted for LVI mitigation");
72STATISTIC(NumFunctionsConsidered,
"Number of functions analyzed");
73STATISTIC(NumFunctionsMitigated,
"Number of functions for which mitigations "
75STATISTIC(NumGadgets,
"Number of LVI gadgets detected during analysis");
83 cl::desc(
"Don't treat conditional branches as disclosure gadgets. This "
84 "may improve performance, at the cost of security."),
90 "For each function, emit a dot graph depicting potential LVI gadgets"),
95 cl::desc(
"For each function, emit a dot graph depicting potential LVI "
96 "gadgets, and do not insert any fences"),
101 cl::desc(
"For each function, emit a dot graph to stdout depicting "
102 "potential LVI gadgets, used for testing purposes only"),
106typedef int (*
OptimizeCutT)(
unsigned int *Nodes,
unsigned int NodesSize,
107 unsigned int *Edges,
int *EdgeValues,
108 int *CutEdges ,
unsigned int EdgesSize);
114 static constexpr int GadgetEdgeSentinel = -1;
115 static constexpr MachineInstr *
const ArgNodeSentinel =
nullptr;
118 using Node = GraphT::Node;
119 using Edge = GraphT::Edge;
121 MachineGadgetGraph(std::unique_ptr<
Node[]> Nodes,
122 std::unique_ptr<Edge[]> Edges, size_type NodesSize,
123 size_type EdgesSize,
int NumFences = 0,
int NumGadgets = 0)
124 : GraphT(std::move(Nodes), std::move(Edges), NodesSize, EdgesSize),
125 NumFences(NumFences), NumGadgets(NumGadgets) {}
126 static inline bool isCFGEdge(
const Edge &
E) {
127 return E.getValue() != GadgetEdgeSentinel;
129 static inline bool isGadgetEdge(
const Edge &
E) {
130 return E.getValue() == GadgetEdgeSentinel;
137 "X86 Load Value Injection (LVI) Load Hardening";
143 StringRef getPassName()
const override {
return X86LVILHPassName; }
150class X86LoadValueInjectionLoadHardeningImpl {
152 X86LoadValueInjectionLoadHardeningImpl() =
default;
160 using Edge = MachineGadgetGraph::Edge;
161 using Node = MachineGadgetGraph::Node;
162 using EdgeSet = MachineGadgetGraph::EdgeSet;
163 using NodeSet = MachineGadgetGraph::NodeSet;
169 std::unique_ptr<MachineGadgetGraph>
174 std::unique_ptr<MachineGadgetGraph> Graph)
const;
176 std::unique_ptr<MachineGadgetGraph> Graph)
const;
177 int elimMitigatedEdgesAndNodes(MachineGadgetGraph &
G,
180 std::unique_ptr<MachineGadgetGraph>
181 trimMitigatedEdges(std::unique_ptr<MachineGadgetGraph> Graph)
const;
183 EdgeSet &CutEdges )
const;
187 return MI && (
MI->getOpcode() == X86::LFENCE ||
188 (STI->useLVIControlFlowIntegrity() &&
MI->isCall()));
212 if (
Node->getValue() == MachineGadgetGraph::ArgNodeSentinel)
217 OS << *
Node->getValue();
223 if (
MI == MachineGadgetGraph::ArgNodeSentinel)
224 return "color = blue";
225 if (
MI->getOpcode() == X86::LFENCE)
226 return "color = green";
232 int EdgeVal = (*E.getCurrent()).getValue();
233 return EdgeVal >= 0 ?
"label = " + std::to_string(EdgeVal)
234 :
"color = red, style = \"dashed\"";
240char X86LoadValueInjectionLoadHardeningLegacy::ID = 0;
242void X86LoadValueInjectionLoadHardeningLegacy::getAnalysisUsage(
247 AU.
addRequired<MachineDominanceFrontierWrapperPass>();
252 MachineGadgetGraph *
G) {
254 "Speculative gadgets for \"" + MF.
getName() +
"\" function");
257bool X86LoadValueInjectionLoadHardeningImpl::run(
258 MachineFunction &MF,
const MachineLoopInfo &MLI,
259 const MachineDominatorTree &MDT,
const MachineDominanceFrontier &MDF) {
268 ++NumFunctionsConsidered;
272 std::unique_ptr<MachineGadgetGraph> Graph = getGadgetGraph(MF, MLI, MDT, MDF);
274 if (Graph ==
nullptr)
284 std::error_code FileError;
285 std::string FileName =
"lvi.";
288 raw_fd_ostream FileOut(FileName, FileError);
290 errs() << FileError.message();
301 std::string ErrorMsg;
304 if (!ErrorMsg.empty())
311 FencesInserted = hardenLoadsWithPlugin(MF, std::move(Graph));
313 FencesInserted = hardenLoadsWithHeuristic(MF, std::move(Graph));
316 if (FencesInserted > 0)
317 ++NumFunctionsMitigated;
318 NumFences += FencesInserted;
319 return (FencesInserted > 0);
322std::unique_ptr<MachineGadgetGraph>
323X86LoadValueInjectionLoadHardeningImpl::getGadgetGraph(
324 MachineFunction &MF,
const MachineLoopInfo &MLI,
325 const MachineDominatorTree &MDT,
326 const MachineDominanceFrontier &MDF)
const {
330 DataFlowGraph DFG{MF, *
TII, *
TRI, MDT, MDF};
335 GraphBuilder Builder;
336 using GraphIter = GraphBuilder::BuilderNodeRef;
337 DenseMap<MachineInstr *, GraphIter> NodeMap;
338 int FenceCount = 0, GadgetCount = 0;
339 auto MaybeAddNode = [&NodeMap, &Builder](MachineInstr *
MI) {
342 auto I = Builder.addVertex(
MI);
344 return std::pair<GraphIter, bool>{
I,
true};
346 return std::pair<GraphIter, bool>{
Ref->getSecond(),
false};
353 DenseMap<NodeId, std::vector<NodeId>> Transmitters;
357 auto AnalyzeDef = [&](NodeAddr<DefNode *> SourceDef) {
358 SmallSet<NodeId, 8> UsesVisited, DefsVisited;
359 std::function<void(NodeAddr<DefNode *>)> AnalyzeDefUseChain =
360 [&](NodeAddr<DefNode *>
Def) {
367 for (
auto UseID :
L.getAllReachedUses(DefReg, Def)) {
368 auto Use = DFG.addr<UseNode *>(UseID);
371 for (
const auto&
I :
L.getRealUses(
Phi.
Id)) {
372 if (DFG.getPRI().alias(RegisterRef(
I.first), DefReg)) {
373 for (
const auto &UA :
I.second)
374 Uses.emplace(UA.first);
385 for (
auto UseID :
Uses) {
386 if (!UsesVisited.
insert(UseID).second)
389 auto Use = DFG.addr<UseNode *>(UseID);
403 if (instrUsesRegToAccessMemory(
UseMI, UseMO.
getReg()) ||
415 for (
const auto &ChildDef :
416 Owner.Addr->members_if(DataFlowGraph::IsDef, DFG)) {
417 if (!DefsVisited.
insert(ChildDef.Id).second)
421 if (
Def.
Id == ChildDef.Id)
424 AnalyzeDefUseChain(ChildDef);
427 for (
auto TransmitterId : Transmitters[ChildDef.Id])
428 Transmitters[
Def.
Id].push_back(TransmitterId);
434 auto &DefTransmitters = Transmitters[
Def.
Id];
439 DefTransmitters.end());
443 AnalyzeDefUseChain(SourceDef);
444 auto &SourceDefTransmitters = Transmitters[SourceDef.Id];
445 if (SourceDefTransmitters.empty())
448 MachineInstr *
Source = SourceDef.Addr->getFlags() & NodeAttrs::PhiRef
449 ? MachineGadgetGraph::ArgNodeSentinel
450 : SourceDef.Addr->getOp().getParent();
451 auto GadgetSource = MaybeAddNode(Source);
453 for (
auto TransmitterId : SourceDefTransmitters) {
454 MachineInstr *
Sink = DFG.addr<StmtNode *>(TransmitterId).Addr->getCode();
455 auto GadgetSink = MaybeAddNode(Sink);
457 Builder.addEdge(MachineGadgetGraph::GadgetEdgeSentinel,
458 GadgetSource.first, GadgetSink.first);
463 LLVM_DEBUG(
dbgs() <<
"Analyzing def-use chains to find gadgets\n");
465 NodeAddr<BlockNode *> EntryBlock = DFG.getFunc().Addr->getEntryBlock(DFG);
466 for (NodeAddr<PhiNode *> ArgPhi :
467 EntryBlock.Addr->members_if(DataFlowGraph::IsPhi, DFG)) {
468 NodeList Defs = ArgPhi.Addr->members_if(DataFlowGraph::IsDef, DFG);
472 for (NodeAddr<BlockNode *> BA : DFG.getFunc().Addr->members(DFG)) {
473 for (NodeAddr<StmtNode *> SA :
474 BA.Addr->members_if(DataFlowGraph::IsCode<NodeAttrs::Stmt>, DFG)) {
475 MachineInstr *
MI = SA.Addr->getCode();
479 }
else if (
MI->mayLoad()) {
480 NodeList Defs = SA.Addr->members_if(DataFlowGraph::IsDef, DFG);
487 if (GadgetCount == 0)
489 NumGadgets += GadgetCount;
492 SmallPtrSet<MachineBasicBlock *, 8> BlocksVisited;
493 std::function<void(MachineBasicBlock *, GraphIter,
unsigned)> TraverseCFG =
494 [&](MachineBasicBlock *
MBB, GraphIter GI,
unsigned ParentDepth) {
499 auto BeginBB = MaybeAddNode(&*NI);
500 Builder.addEdge(ParentDepth, GI, BeginBB.first);
506 while (++NI !=
MBB->
end()) {
508 if (
Ref != NodeMap.
end()) {
509 Builder.addEdge(LoopDepth, GI,
Ref->getSecond());
510 GI =
Ref->getSecond();
517 auto EndBB = MaybeAddNode(&*
T);
519 Builder.addEdge(LoopDepth, GI, EndBB.first);
524 TraverseCFG(Succ, GI, LoopDepth);
528 GraphIter ArgNode = MaybeAddNode(MachineGadgetGraph::ArgNodeSentinel).first;
529 TraverseCFG(&MF.
front(), ArgNode, 0);
530 std::unique_ptr<MachineGadgetGraph>
G{Builder.get(FenceCount, GadgetCount)};
536int X86LoadValueInjectionLoadHardeningImpl::elimMitigatedEdgesAndNodes(
537 MachineGadgetGraph &
G, EdgeSet &ElimEdges ,
538 NodeSet &ElimNodes )
const {
539 if (
G.NumFences > 0) {
542 for (
const Edge &
E :
G.edges()) {
543 const Node *Dest =
E.getDest();
544 if (isFence(Dest->getValue())) {
547 for (
const Edge &DE : Dest->edges())
548 ElimEdges.insert(DE);
554 int RemainingGadgets = 0;
555 NodeSet ReachableNodes{
G};
556 for (
const Node &
RootN :
G.nodes()) {
561 ReachableNodes.
clear();
562 std::function<void(
const Node *,
bool)> FindReachableNodes =
563 [&](
const Node *
N,
bool FirstNode) {
566 for (
const Edge &
E :
N->edges()) {
567 const Node *Dest =
E.getDest();
568 if (MachineGadgetGraph::isCFGEdge(
E) && !ElimEdges.contains(
E) &&
569 !ReachableNodes.contains(*Dest))
570 FindReachableNodes(Dest,
false);
573 FindReachableNodes(&
RootN,
true);
577 if (MachineGadgetGraph::isGadgetEdge(
E)) {
578 if (ReachableNodes.contains(*
E.getDest())) {
587 return RemainingGadgets;
590std::unique_ptr<MachineGadgetGraph>
591X86LoadValueInjectionLoadHardeningImpl::trimMitigatedEdges(
592 std::unique_ptr<MachineGadgetGraph> Graph)
const {
593 NodeSet ElimNodes{*Graph};
594 EdgeSet ElimEdges{*Graph};
595 int RemainingGadgets =
596 elimMitigatedEdgesAndNodes(*Graph, ElimEdges, ElimNodes);
597 if (ElimEdges.empty() && ElimNodes.
empty()) {
598 Graph->NumFences = 0;
599 Graph->NumGadgets = RemainingGadgets;
601 Graph = GraphBuilder::trim(*Graph, ElimNodes, ElimEdges, 0 ,
607int X86LoadValueInjectionLoadHardeningImpl::hardenLoadsWithPlugin(
608 MachineFunction &MF, std::unique_ptr<MachineGadgetGraph> Graph)
const {
609 int FencesInserted = 0;
613 Graph = trimMitigatedEdges(std::move(Graph));
615 if (Graph->NumGadgets == 0)
619 EdgeSet CutEdges{*Graph};
620 auto Nodes = std::make_unique<unsigned int[]>(Graph->nodes_size() +
622 auto Edges = std::make_unique<unsigned int[]>(Graph->edges_size());
623 auto EdgeCuts = std::make_unique<int[]>(Graph->edges_size());
624 auto EdgeValues = std::make_unique<int[]>(Graph->edges_size());
625 for (
const Node &
N : Graph->nodes()) {
626 Nodes[Graph->getNodeIndex(
N)] = Graph->getEdgeIndex(*
N.edges_begin());
628 Nodes[Graph->nodes_size()] = Graph->edges_size();
629 for (
const Edge &
E : Graph->edges()) {
630 Edges[Graph->getEdgeIndex(
E)] = Graph->getNodeIndex(*
E.getDest());
631 EdgeValues[Graph->getEdgeIndex(
E)] =
E.getValue();
633 OptimizeCut(Nodes.get(), Graph->nodes_size(), Edges.get(), EdgeValues.get(),
634 EdgeCuts.get(), Graph->edges_size());
635 for (
int I = 0;
I < Graph->edges_size(); ++
I)
642 FencesInserted += insertFences(MF, *Graph, CutEdges);
644 LLVM_DEBUG(
dbgs() <<
"Inserted " << FencesInserted <<
" fences\n");
646 Graph = GraphBuilder::trim(*Graph, NodeSet{*Graph}, CutEdges);
649 return FencesInserted;
652int X86LoadValueInjectionLoadHardeningImpl::hardenLoadsWithHeuristic(
653 MachineFunction &MF, std::unique_ptr<MachineGadgetGraph> Graph)
const {
656 if (Graph->NumFences > 0) {
658 Graph = trimMitigatedEdges(std::move(Graph));
662 if (Graph->NumGadgets == 0)
666 EdgeSet CutEdges{*Graph};
669 DenseMap<const Node *, SmallVector<const Edge *, 2>> IngressEdgeMap;
670 for (
const Edge &
E : Graph->edges())
671 if (MachineGadgetGraph::isCFGEdge(
E))
672 IngressEdgeMap[
E.getDest()].push_back(&
E);
682 for (
const Node &
N : Graph->nodes()) {
683 for (
const Edge &
E :
N.edges()) {
684 if (!MachineGadgetGraph::isGadgetEdge(
E))
689 for (
const Edge &EgressEdge :
N.edges())
690 if (MachineGadgetGraph::isCFGEdge(EgressEdge))
693 int EgressCutCost = 0, IngressCutCost = 0;
694 for (
const Edge *EgressEdge : EgressEdges)
695 if (!CutEdges.contains(*EgressEdge))
696 EgressCutCost += EgressEdge->getValue();
697 for (
const Edge *IngressEdge : IngressEdges)
698 if (!CutEdges.contains(*IngressEdge))
699 IngressCutCost += IngressEdge->getValue();
702 IngressCutCost < EgressCutCost ? IngressEdges : EgressEdges;
703 for (
const Edge *
E : EdgesToCut)
711 int FencesInserted = insertFences(MF, *Graph, CutEdges);
713 LLVM_DEBUG(
dbgs() <<
"Inserted " << FencesInserted <<
" fences\n");
715 return FencesInserted;
718int X86LoadValueInjectionLoadHardeningImpl::insertFences(
719 MachineFunction &MF, MachineGadgetGraph &
G,
720 EdgeSet &CutEdges )
const {
721 int FencesInserted = 0;
722 for (
const Node &
N :
G.nodes()) {
723 for (
const Edge &
E :
N.edges()) {
724 if (CutEdges.contains(
E)) {
725 MachineInstr *
MI =
N.getValue(), *Prev;
726 MachineBasicBlock *
MBB;
728 if (
MI == MachineGadgetGraph::ArgNodeSentinel) {
733 }
else if (
MI->isBranch()) {
734 MBB =
MI->getParent();
736 Prev =
MI->getPrevNode();
739 for (
const Edge &
E :
N.edges()) {
740 if (MachineGadgetGraph::isCFGEdge(
E))
744 MBB =
MI->getParent();
745 InsertionPt =
MI->getNextNode() ?
MI->getNextNode() :
MBB->
end();
746 Prev = InsertionPt ==
MBB->
end()
748 : InsertionPt->getPrevNode();
751 if ((InsertionPt ==
MBB->
end() || !isFence(&*InsertionPt)) &&
752 (!Prev || !isFence(Prev))) {
759 return FencesInserted;
762bool X86LoadValueInjectionLoadHardeningImpl::instrUsesRegToAccessMemory(
764 if (!
MI.mayLoadOrStore() ||
MI.getOpcode() == X86::MFENCE ||
765 MI.getOpcode() == X86::SFENCE ||
MI.getOpcode() == X86::LFENCE)
769 if (MemRefBeginIdx < 0) {
770 LLVM_DEBUG(
dbgs() <<
"Warning: unable to obtain memory operand for loading "
776 const MachineOperand &BaseMO =
778 const MachineOperand &IndexMO =
786bool X86LoadValueInjectionLoadHardeningImpl::instrUsesRegToBranch(
788 if (!
MI.isConditionalBranch())
790 for (
const MachineOperand &Use :
MI.uses())
796bool X86LoadValueInjectionLoadHardeningLegacy::runOnMachineFunction(
797 MachineFunction &MF) {
801 if (!
F.hasOptNone() && skipFunction(
F))
805 if (!MF.
getSubtarget<X86Subtarget>().useLVILoadHardening()) {
809 const auto &MLI = getAnalysis<MachineLoopInfoWrapperPass>().getLI();
810 const auto &MDT = getAnalysis<MachineDominatorTreeWrapperPass>().getDomTree();
811 const auto &MDF = getAnalysis<MachineDominanceFrontierWrapperPass>().getMDF();
813 X86LoadValueInjectionLoadHardeningImpl Impl;
814 return Impl.run(MF, MLI, MDT, MDF);
828 X86LoadValueInjectionLoadHardeningImpl Impl;
829 const bool Modified = Impl.run(MF, MLI, MDT, MDF);
836 "X86 LVI load hardening",
false,
false)
844 return new X86LoadValueInjectionLoadHardeningLegacy();
MachineInstrBuilder & UseMI
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static ManagedStatic< DebugCounterOwner > Owner
This file defines the DenseMap class.
const HexagonInstrInfo * TII
Description: ImmutableGraph is a fast DAG implementation that cannot be modified, except by creating ...
Register const TargetRegisterInfo * TRI
Promote Memory to Register
#define INITIALIZE_PASS_DEPENDENCY(depName)
#define INITIALIZE_PASS_END(passName, arg, name, cfg, analysis)
#define INITIALIZE_PASS_BEGIN(passName, arg, name, cfg, analysis)
Remove Loads Into Fake Uses
std::pair< BasicBlock *, BasicBlock * > Edge
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)
PassT::Result & getResult(IRUnitT &IR, ExtraArgTs... ExtraArgs)
Get the result of an analysis pass for a given IR unit.
Represent the analysis usage information of a pass.
AnalysisUsage & addRequired()
LLVM_ABI void setPreservesCFG()
This function should be called by the pass, iff they do not:
Represents analyses that only rely on functions' control flow.
iterator find(const_arg_type_t< KeyT > Val)
std::pair< iterator, bool > try_emplace(KeyT &&Key, Ts &&...Args)
bool contains(const_arg_type_t< KeyT > Val) const
Return true if the specified key is in the map, false otherwise.
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.
LLVM_ABI iterator getFirstTerminator()
Returns an iterator to the first terminator instruction of this basic block.
iterator_range< succ_iterator > successors()
MachineInstrBundleIterator< MachineInstr > iterator
Analysis pass which computes a MachineDominatorTree.
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.
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.
Analysis pass that exposes the MachineLoopInfo for a machine function.
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.
A set of analyses that are preserved following a run of a transformation pass.
static PreservedAnalyses all()
Construct a special preserved set that preserves all passes.
PreservedAnalyses & preserveSet()
Mark an analysis set as preserved.
Wrapper class representing virtual and physical registers.
constexpr bool isValid() const
std::pair< iterator, bool > insert(PtrType Ptr)
Inserts Ptr if and only if there is no element in the container equal to Ptr.
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)
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...
PreservedAnalyses run(MachineFunction &MF, MachineFunctionAnalysisManager &MFAM)
const X86InstrInfo * getInstrInfo() const override
const X86RegisterInfo * getRegisterInfo() const override
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.
std::string & str()
Returns the string's reference.
This class provides a portable interface to dynamic libraries which also might be known as shared lib...
static LLVM_ABI 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...
LLVM_ABI 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< DefNode * > Def
NodeAddr< PhiNode * > Phi
NodeAddr< UseNode * > Use
std::set< NodeId > NodeSet
SmallVector< Node, 4 > NodeList
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.
LLVM_ABI 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="")
AnalysisManager< MachineFunction > MachineFunctionAnalysisManager
auto unique(Range &&R, Predicate P)
LLVM_ABI PreservedAnalyses getMachineFunctionPassPreservedAnalyses()
Returns the minimum set of Analyses that all machine function passes must preserve.
FunctionPass * createX86LoadValueInjectionLoadHardeningLegacyPass()
void sort(IteratorTy Start, IteratorTy End)
LLVM_ABI 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.
LLVM_ABI void report_fatal_error(Error Err, bool gen_crash_diag=true)
class LLVM_GSL_OWNER SmallVector
Forward declaration of SmallVector so that calculateSmallVectorDefaultInlinedElements can reference s...
LLVM_ABI 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.
static std::string getNodeAttributes(NodeRef Node, GraphType *)
Traits::ChildIteratorType ChildIteratorType
std::string getNodeLabel(NodeRef Node, GraphType *)
static std::string getEdgeAttributes(NodeRef, ChildIteratorType E, GraphType *)
MachineGadgetGraph GraphType
Traits::ChildEdgeIteratorType ChildEdgeIteratorType
llvm::GraphTraits< GraphType * > Traits
DOTGraphTraits(bool IsSimple=false)
DefaultDOTGraphTraits(bool simple=false)
typename GraphType::UnknownGraphTypeError NodeRef
uint16_t getAttrs() const
uint16_t getFlags() const
RegisterRef getRegRef(const DataFlowGraph &G) const
Node getOwner(const DataFlowGraph &G)