
This computes a structural hash while allowing for selective ignoring of certain operands based on a custom function that is provided. Instead of a single hash value, it now returns FunctionHashInfo which includes a hash value, an instruction mapping, and a map to track the operand location and its corresponding hash value that is ignored. Depends on https://github.com/llvm/llvm-project/pull/112621. This is a patch for https://discourse.llvm.org/t/rfc-global-function-merging/82608.
54 lines
2.1 KiB
C++
54 lines
2.1 KiB
C++
//===- StructuralHash.cpp - Function Hash Printing ------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines the StructuralHashPrinterPass which is used to show
|
|
// the structural hash of all functions in a module and the module itself.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/Analysis/StructuralHash.h"
|
|
#include "llvm/IR/Module.h"
|
|
#include "llvm/IR/StructuralHash.h"
|
|
#include "llvm/Support/Format.h"
|
|
|
|
using namespace llvm;
|
|
|
|
PreservedAnalyses StructuralHashPrinterPass::run(Module &M,
|
|
ModuleAnalysisManager &MAM) {
|
|
OS << "Module Hash: "
|
|
<< format("%016" PRIx64,
|
|
StructuralHash(M, Options != StructuralHashOptions::None))
|
|
<< "\n";
|
|
for (Function &F : M) {
|
|
if (F.isDeclaration())
|
|
continue;
|
|
if (Options == StructuralHashOptions::CallTargetIgnored) {
|
|
auto IgnoreOp = [&](const Instruction *I, unsigned OpndIdx) {
|
|
return I->getOpcode() == Instruction::Call &&
|
|
isa<Constant>(I->getOperand(OpndIdx));
|
|
};
|
|
auto FuncHashInfo = StructuralHashWithDifferences(F, IgnoreOp);
|
|
OS << "Function " << F.getName()
|
|
<< " Hash: " << format("%016" PRIx64, FuncHashInfo.FunctionHash)
|
|
<< "\n";
|
|
for (auto &[IndexPair, OpndHash] : *FuncHashInfo.IndexOperandHashMap) {
|
|
auto [InstIndex, OpndIndex] = IndexPair;
|
|
OS << "\tIgnored Operand Hash: " << format("%016" PRIx64, OpndHash)
|
|
<< " at (" << InstIndex << "," << OpndIndex << ")\n";
|
|
}
|
|
} else {
|
|
OS << "Function " << F.getName() << " Hash: "
|
|
<< format(
|
|
"%016" PRIx64,
|
|
StructuralHash(F, Options == StructuralHashOptions::Detailed))
|
|
<< "\n";
|
|
}
|
|
}
|
|
return PreservedAnalyses::all();
|
|
}
|