LLVM 22.0.0git
MachineStableHash.cpp
Go to the documentation of this file.
1//===- lib/CodeGen/MachineStableHash.cpp ----------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// Stable hashing for MachineInstr and MachineOperand. Useful or getting a
10// hash across runs, modules, etc.
11//
12//===----------------------------------------------------------------------===//
13
15#include "llvm/ADT/APFloat.h"
16#include "llvm/ADT/APInt.h"
17#include "llvm/ADT/STLExtras.h"
20#include "llvm/ADT/Statistic.h"
28#include "llvm/Config/llvm-config.h"
29#include "llvm/IR/Constants.h"
32#include "llvm/MC/MCSymbol.h"
35
36#define DEBUG_TYPE "machine-stable-hash"
37
38using namespace llvm;
39
40STATISTIC(StableHashBailingMachineBasicBlock,
41 "Number of encountered unsupported MachineOperands that were "
42 "MachineBasicBlocks while computing stable hashes");
43STATISTIC(StableHashBailingConstantPoolIndex,
44 "Number of encountered unsupported MachineOperands that were "
45 "ConstantPoolIndex while computing stable hashes");
46STATISTIC(StableHashBailingTargetIndexNoName,
47 "Number of encountered unsupported MachineOperands that were "
48 "TargetIndex with no name");
49STATISTIC(StableHashBailingGlobalAddress,
50 "Number of encountered unsupported MachineOperands that were "
51 "GlobalAddress while computing stable hashes");
52STATISTIC(StableHashBailingBlockAddress,
53 "Number of encountered unsupported MachineOperands that were "
54 "BlockAddress while computing stable hashes");
55STATISTIC(StableHashBailingMetadataUnsupported,
56 "Number of encountered unsupported MachineOperands that were "
57 "Metadata of an unsupported kind while computing stable hashes");
58
60 switch (MO.getType()) {
62 if (MO.getReg().isVirtual()) {
64 SmallVector<stable_hash> DefOpcodes;
65 for (auto &Def : MRI.def_instructions(MO.getReg()))
66 DefOpcodes.push_back(Def.getOpcode());
67 return stable_hash_combine(DefOpcodes);
68 }
69
70 // Register operands don't have target flags.
71 return stable_hash_combine(MO.getType(), MO.getReg().id(), MO.getSubReg(),
72 MO.isDef());
74 return stable_hash_combine(MO.getType(), MO.getTargetFlags(), MO.getImm());
77 auto Val = MO.isCImm() ? MO.getCImm()->getValue()
79 auto ValHash = stable_hash_combine(
80 ArrayRef<stable_hash>(Val.getRawData(), Val.getNumWords()));
81 return stable_hash_combine(MO.getType(), MO.getTargetFlags(), ValHash);
82 }
83
85 ++StableHashBailingMachineBasicBlock;
86 return 0;
88 ++StableHashBailingConstantPoolIndex;
89 return 0;
91 ++StableHashBailingBlockAddress;
92 return 0;
94 ++StableHashBailingMetadataUnsupported;
95 return 0;
97 const GlobalValue *GV = MO.getGlobal();
98 stable_hash GVHash = 0;
99 if (auto *GVar = dyn_cast<GlobalVariable>(GV))
100 GVHash = StructuralHash(*GVar);
101 if (!GVHash) {
102 if (!GV->hasName()) {
103 ++StableHashBailingGlobalAddress;
104 return 0;
105 }
106 GVHash = stable_hash_name(GV->getName());
107 }
108
109 return stable_hash_combine(MO.getType(), MO.getTargetFlags(), GVHash,
110 MO.getOffset());
111 }
112
114 if (const char *Name = MO.getTargetIndexName())
116 stable_hash_name(Name), MO.getOffset());
117 ++StableHashBailingTargetIndexNoName;
118 return 0;
119 }
120
124 MO.getIndex());
125
128 MO.getOffset(),
130
133 if (const MachineInstr *MI = MO.getParent()) {
134 if (const MachineBasicBlock *MBB = MI->getParent()) {
135 if (const MachineFunction *MF = MBB->getParent()) {
136 const TargetRegisterInfo *TRI = MF->getSubtarget().getRegisterInfo();
137 unsigned RegMaskSize =
138 MachineOperand::getRegMaskSize(TRI->getNumRegs());
139 const uint32_t *RegMask =
140 MO.isRegMask() ? MO.getRegMask() : MO.getRegLiveOut();
141 std::vector<llvm::stable_hash> RegMaskHashes(RegMask,
142 RegMask + RegMaskSize);
144 stable_hash_combine(RegMaskHashes));
145 }
146 }
147 }
148
149 assert(0 && "MachineOperand not associated with any MachineFunction");
150 return stable_hash_combine(MO.getType(), MO.getTargetFlags());
151 }
152
154 std::vector<llvm::stable_hash> ShuffleMaskHashes;
155
157 MO.getShuffleMask(), std::back_inserter(ShuffleMaskHashes),
158 [](int S) -> llvm::stable_hash { return llvm::stable_hash(S); });
159
161 stable_hash_combine(ShuffleMaskHashes));
162 }
164 auto SymbolName = MO.getMCSymbol()->getName();
166 stable_hash_name(SymbolName));
167 }
170 MO.getCFIIndex());
173 MO.getIntrinsicID());
176 MO.getPredicate());
179 MO.getInstrRefOpIndex());
180 }
181 llvm_unreachable("Invalid machine operand type");
182}
183
184/// A stable hash value for machine instructions.
185/// Returns 0 if no stable hash could be computed.
186/// The hashing and equality testing functions ignore definitions so this is
187/// useful for CSE, etc.
189 bool HashConstantPoolIndices,
190 bool HashMemOperands) {
191 // Build up a buffer of hash code components.
192 SmallVector<stable_hash, 16> HashComponents;
193 HashComponents.reserve(MI.getNumOperands() + MI.getNumMemOperands() + 2);
194 HashComponents.push_back(MI.getOpcode());
195 HashComponents.push_back(MI.getFlags());
196 for (const MachineOperand &MO : MI.operands()) {
197 if (!HashVRegs && MO.isReg() && MO.isDef() && MO.getReg().isVirtual())
198 continue; // Skip virtual register defs.
199
200 if (MO.isCPI()) {
201 HashComponents.push_back(stable_hash_combine(
202 MO.getType(), MO.getTargetFlags(), MO.getIndex()));
203 continue;
204 }
205
206 stable_hash StableHash = stableHashValue(MO);
207 if (!StableHash)
208 return 0;
209 HashComponents.push_back(StableHash);
210 }
211
212 for (const auto *Op : MI.memoperands()) {
213 if (!HashMemOperands)
214 break;
215 HashComponents.push_back(static_cast<unsigned>(Op->getSize().getValue()));
216 HashComponents.push_back(static_cast<unsigned>(Op->getFlags()));
217 HashComponents.push_back(static_cast<unsigned>(Op->getOffset()));
218 HashComponents.push_back(static_cast<unsigned>(Op->getSuccessOrdering()));
219 HashComponents.push_back(static_cast<unsigned>(Op->getAddrSpace()));
220 HashComponents.push_back(static_cast<unsigned>(Op->getSyncScopeID()));
221 HashComponents.push_back(static_cast<unsigned>(Op->getBaseAlign().value()));
222 HashComponents.push_back(static_cast<unsigned>(Op->getFailureOrdering()));
223 }
224
225 return stable_hash_combine(HashComponents);
226}
227
229 SmallVector<stable_hash> HashComponents;
230 // TODO: Hash more stuff like block alignment and branch probabilities.
231 for (const auto &MI : MBB)
232 HashComponents.push_back(stableHashValue(MI));
233 return stable_hash_combine(HashComponents);
234}
235
237 SmallVector<stable_hash> HashComponents;
238 // TODO: Hash lots more stuff like function alignment and stack objects.
239 for (const auto &MBB : MF)
240 HashComponents.push_back(stableHashValue(MBB));
241 return stable_hash_combine(HashComponents);
242}
unsigned const MachineRegisterInfo * MRI
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
This file declares a class to represent arbitrary precision floating point values and provide a varie...
This file implements a class to represent arbitrary precision integral constant values and operations...
MachineBasicBlock & MBB
This file contains the declarations for the subclasses of Constant, which represent the different fla...
IRTranslator LLVM IR MI
Register const TargetRegisterInfo * TRI
This file contains some templates that are useful if you are working with the STL at all.
This file defines the SmallVector class.
This file defines the 'Statistic' class, which is designed to be an easy way to expose various metric...
#define STATISTIC(VARNAME, DESC)
Definition Statistic.h:171
APInt bitcastToAPInt() const
Definition APFloat.h:1335
ArrayRef - Represent a constant reference to an array (0 or more elements consecutively in memory),...
Definition ArrayRef.h:41
const APFloat & getValueAPF() const
Definition Constants.h:320
const APInt & getValue() const
Return the constant as an APInt value reference.
Definition Constants.h:154
StringRef getName() const
getName - Get the symbol name.
Definition MCSymbol.h:188
MachineRegisterInfo & getRegInfo()
getRegInfo - Return information about the registers currently in use.
Representation of each machine instruction.
LLVM_ABI const MachineFunction * getMF() const
Return the function that contains the basic block that this instruction belongs to.
MachineOperand class - Representation of each machine instruction operand.
unsigned getSubReg() const
unsigned getInstrRefOpIndex() const
const GlobalValue * getGlobal() const
const uint32_t * getRegLiveOut() const
getRegLiveOut - Returns a bit mask of live-out registers.
const ConstantInt * getCImm() const
LLVM_ABI const char * getTargetIndexName() const
getTargetIndexName - If this MachineOperand is a TargetIndex that has a name, attempt to get the name...
bool isCImm() const
isCImm - Test if this is a MO_CImmediate operand.
int64_t getImm() const
unsigned getInstrRefInstrIndex() const
bool isRegMask() const
isRegMask - Tests if this is a MO_RegisterMask operand.
ArrayRef< int > getShuffleMask() const
unsigned getCFIIndex() const
MachineInstr * getParent()
getParent - Return the instruction that this operand belongs to.
static unsigned getRegMaskSize(unsigned NumRegs)
Returns number of elements needed for a regmask array.
unsigned getTargetFlags() const
MachineOperandType getType() const
getType - Returns the MachineOperandType for this operand.
const char * getSymbolName() const
Register getReg() const
getReg - Returns the register number.
Intrinsic::ID getIntrinsicID() const
const uint32_t * getRegMask() const
getRegMask - Returns a bit mask of registers preserved by this RegMask operand.
const ConstantFP * getFPImm() const
unsigned getPredicate() const
MCSymbol * getMCSymbol() const
@ MO_CFIIndex
MCCFIInstruction index.
@ MO_Immediate
Immediate operand.
@ MO_ConstantPoolIndex
Address of indexed Constant in Constant Pool.
@ MO_MCSymbol
MCSymbol reference (for debug/eh info)
@ MO_Predicate
Generic predicate for ISel.
@ MO_GlobalAddress
Address of a global value.
@ MO_RegisterMask
Mask of preserved registers.
@ MO_ShuffleMask
Other IR Constant for ISel (shuffle masks)
@ MO_CImmediate
Immediate >64bit operand.
@ MO_BlockAddress
Address of a basic block.
@ MO_DbgInstrRef
Integer indices referring to an instruction+operand.
@ MO_MachineBasicBlock
MachineBasicBlock reference.
@ MO_FrameIndex
Abstract Stack Frame Index.
@ MO_Register
Register operand.
@ MO_ExternalSymbol
Name of external global symbol.
@ MO_IntrinsicID
Intrinsic ID for ISel.
@ MO_JumpTableIndex
Address of indexed Jump Table for switch.
@ MO_TargetIndex
Target-dependent index+offset operand.
@ MO_Metadata
Metadata reference (for debug info)
@ MO_FPImmediate
Floating-point immediate operand.
@ MO_RegisterLiveOut
Mask of live-out registers.
int64_t getOffset() const
Return the offset from the symbol in this operand.
MachineRegisterInfo - Keep track of information for virtual and physical registers,...
constexpr bool isVirtual() const
Return true if the specified register number is in the virtual register namespace.
Definition Register.h:74
constexpr unsigned id() const
Definition Register.h:95
void reserve(size_type N)
void push_back(const T &Elt)
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
TargetRegisterInfo base class - We assume that the target defines a static array of TargetRegisterDes...
bool hasName() const
Definition Value.h:262
LLVM_ABI StringRef getName() const
Return a constant reference to the value's name.
Definition Value.cpp:322
#define llvm_unreachable(msg)
Marks that the current location is not supposed to be reachable.
This is an optimization pass for GlobalISel generic memory operations.
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
Definition Casting.h:643
uint64_t stable_hash
An opaque object representing a stable hash code.
OutputIt transform(R &&Range, OutputIt d_first, UnaryFunction F)
Wrapper function around std::transform to apply a function to a range and store the result elsewhere.
Definition STLExtras.h:1968
LLVM_ABI stable_hash stableHashValue(const MachineOperand &MO)
stable_hash stable_hash_name(StringRef Name)
DWARFExpression::Operation Op
stable_hash stable_hash_combine(ArrayRef< stable_hash > Buffer)
LLVM_ABI stable_hash StructuralHash(const Function &F, bool DetailedHash=false)
Returns a hash of the function F.