You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
91 lines
2.4 KiB
91 lines
2.4 KiB
//===-- DifferenceEngine.h - Module comparator ------------------*- C++ -*-===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This header defines the interface to the LLVM difference engine,
|
|
// which structurally compares functions within a module.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_TOOLS_LLVM_DIFF_DIFFERENCEENGINE_H
|
|
#define LLVM_TOOLS_LLVM_DIFF_DIFFERENCEENGINE_H
|
|
|
|
#include "DiffConsumer.h"
|
|
#include "DiffLog.h"
|
|
#include "llvm/ADT/StringRef.h"
|
|
#include <utility>
|
|
|
|
namespace llvm {
|
|
class Function;
|
|
class GlobalValue;
|
|
class Instruction;
|
|
class LLVMContext;
|
|
class Module;
|
|
class Twine;
|
|
class Value;
|
|
|
|
/// A class for performing structural comparisons of LLVM assembly.
|
|
class DifferenceEngine {
|
|
public:
|
|
/// A RAII object for recording the current context.
|
|
struct Context {
|
|
Context(DifferenceEngine &Engine, Value *L, Value *R) : Engine(Engine) {
|
|
Engine.consumer.enterContext(L, R);
|
|
}
|
|
|
|
~Context() {
|
|
Engine.consumer.exitContext();
|
|
}
|
|
|
|
private:
|
|
DifferenceEngine &Engine;
|
|
};
|
|
|
|
/// An oracle for answering whether two values are equivalent as
|
|
/// operands.
|
|
class Oracle {
|
|
virtual void anchor();
|
|
public:
|
|
virtual bool operator()(Value *L, Value *R) = 0;
|
|
|
|
protected:
|
|
virtual ~Oracle() {}
|
|
};
|
|
|
|
DifferenceEngine(Consumer &consumer)
|
|
: consumer(consumer), globalValueOracle(nullptr) {}
|
|
|
|
void diff(Module *L, Module *R);
|
|
void diff(Function *L, Function *R);
|
|
void log(StringRef text) {
|
|
consumer.log(text);
|
|
}
|
|
LogBuilder logf(StringRef text) {
|
|
return LogBuilder(consumer, text);
|
|
}
|
|
Consumer& getConsumer() const { return consumer; }
|
|
|
|
/// Installs an oracle to decide whether two global values are
|
|
/// equivalent as operands. Without an oracle, global values are
|
|
/// considered equivalent as operands precisely when they have the
|
|
/// same name.
|
|
void setGlobalValueOracle(Oracle *oracle) {
|
|
globalValueOracle = oracle;
|
|
}
|
|
|
|
/// Determines whether two global values are equivalent.
|
|
bool equivalentAsOperands(GlobalValue *L, GlobalValue *R);
|
|
|
|
private:
|
|
Consumer &consumer;
|
|
Oracle *globalValueOracle;
|
|
};
|
|
}
|
|
|
|
#endif
|