mirror of
https://github.com/RPCSX/llvm.git
synced 2025-04-03 16:51:42 +00:00
Memoize InlineAsms into the LLVMContext and delete them on shutdown.
Fixes PR803. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@99143 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
f65b0e9b45
commit
bf48a9b6db
@ -24,8 +24,17 @@ namespace llvm {
|
|||||||
class PointerType;
|
class PointerType;
|
||||||
class FunctionType;
|
class FunctionType;
|
||||||
class Module;
|
class Module;
|
||||||
|
struct InlineAsmKeyType;
|
||||||
|
template<class ValType, class TypeClass, class ConstantClass, bool HasLargeKey>
|
||||||
|
class ConstantUniqueMap;
|
||||||
|
template<class ConstantClass, class TypeClass, class ValType>
|
||||||
|
struct ConstantCreator;
|
||||||
|
|
||||||
class InlineAsm : public Value {
|
class InlineAsm : public Value {
|
||||||
|
friend struct ConstantCreator<InlineAsm, PointerType, InlineAsmKeyType>;
|
||||||
|
friend class ConstantUniqueMap<InlineAsmKeyType, PointerType, InlineAsm,
|
||||||
|
false>;
|
||||||
|
|
||||||
InlineAsm(const InlineAsm &); // do not implement
|
InlineAsm(const InlineAsm &); // do not implement
|
||||||
void operator=(const InlineAsm&); // do not implement
|
void operator=(const InlineAsm&); // do not implement
|
||||||
|
|
||||||
@ -33,10 +42,14 @@ class InlineAsm : public Value {
|
|||||||
bool HasSideEffects;
|
bool HasSideEffects;
|
||||||
bool IsAlignStack;
|
bool IsAlignStack;
|
||||||
|
|
||||||
InlineAsm(const FunctionType *Ty, StringRef AsmString,
|
InlineAsm(const PointerType *Ty, const std::string &AsmString,
|
||||||
StringRef Constraints, bool hasSideEffects,
|
const std::string &Constraints, bool hasSideEffects,
|
||||||
bool isAlignStack = false);
|
bool isAlignStack);
|
||||||
virtual ~InlineAsm();
|
virtual ~InlineAsm();
|
||||||
|
|
||||||
|
/// When the ConstantUniqueMap merges two types and makes two InlineAsms
|
||||||
|
/// identical, it destroys one of them with this method.
|
||||||
|
void destroyConstant();
|
||||||
public:
|
public:
|
||||||
|
|
||||||
/// InlineAsm::get - Return the specified uniqued inline asm string.
|
/// InlineAsm::get - Return the specified uniqued inline asm string.
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
#ifndef LLVM_CONSTANTSCONTEXT_H
|
#ifndef LLVM_CONSTANTSCONTEXT_H
|
||||||
#define LLVM_CONSTANTSCONTEXT_H
|
#define LLVM_CONSTANTSCONTEXT_H
|
||||||
|
|
||||||
|
#include "llvm/InlineAsm.h"
|
||||||
#include "llvm/Instructions.h"
|
#include "llvm/Instructions.h"
|
||||||
#include "llvm/Operator.h"
|
#include "llvm/Operator.h"
|
||||||
#include "llvm/Support/Debug.h"
|
#include "llvm/Support/Debug.h"
|
||||||
@ -327,6 +328,39 @@ struct ExprMapKeyType {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct InlineAsmKeyType {
|
||||||
|
InlineAsmKeyType(StringRef AsmString,
|
||||||
|
StringRef Constraints, bool hasSideEffects,
|
||||||
|
bool isAlignStack)
|
||||||
|
: asm_string(AsmString), constraints(Constraints),
|
||||||
|
has_side_effects(hasSideEffects), is_align_stack(isAlignStack) {}
|
||||||
|
std::string asm_string;
|
||||||
|
std::string constraints;
|
||||||
|
bool has_side_effects;
|
||||||
|
bool is_align_stack;
|
||||||
|
bool operator==(const InlineAsmKeyType& that) const {
|
||||||
|
return this->asm_string == that.asm_string &&
|
||||||
|
this->constraints == that.constraints &&
|
||||||
|
this->has_side_effects == that.has_side_effects &&
|
||||||
|
this->is_align_stack == that.is_align_stack;
|
||||||
|
}
|
||||||
|
bool operator<(const InlineAsmKeyType& that) const {
|
||||||
|
if (this->asm_string != that.asm_string)
|
||||||
|
return this->asm_string < that.asm_string;
|
||||||
|
if (this->constraints != that.constraints)
|
||||||
|
return this->constraints < that.constraints;
|
||||||
|
if (this->has_side_effects != that.has_side_effects)
|
||||||
|
return this->has_side_effects < that.has_side_effects;
|
||||||
|
if (this->is_align_stack != that.is_align_stack)
|
||||||
|
return this->is_align_stack < that.is_align_stack;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool operator!=(const InlineAsmKeyType& that) const {
|
||||||
|
return !(*this == that);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// The number of operands for each ConstantCreator::create method is
|
// The number of operands for each ConstantCreator::create method is
|
||||||
// determined by the ConstantTraits template.
|
// determined by the ConstantTraits template.
|
||||||
// ConstantCreator - A class that is used to create constants by
|
// ConstantCreator - A class that is used to create constants by
|
||||||
@ -517,6 +551,23 @@ struct ConstantKeyData<UndefValue> {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
template<>
|
||||||
|
struct ConstantCreator<InlineAsm, PointerType, InlineAsmKeyType> {
|
||||||
|
static InlineAsm *create(const PointerType *Ty, const InlineAsmKeyType &Key) {
|
||||||
|
return new InlineAsm(Ty, Key.asm_string, Key.constraints,
|
||||||
|
Key.has_side_effects, Key.is_align_stack);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template<>
|
||||||
|
struct ConstantKeyData<InlineAsm> {
|
||||||
|
typedef InlineAsmKeyType ValType;
|
||||||
|
static ValType getValType(InlineAsm *Asm) {
|
||||||
|
return InlineAsmKeyType(Asm->getAsmString(), Asm->getConstraintString(),
|
||||||
|
Asm->hasSideEffects(), Asm->isAlignStack());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
template<class ValType, class TypeClass, class ConstantClass,
|
template<class ValType, class TypeClass, class ConstantClass,
|
||||||
bool HasLargeKey = false /*true for arrays and structs*/ >
|
bool HasLargeKey = false /*true for arrays and structs*/ >
|
||||||
class ConstantUniqueMap : public AbstractTypeUser {
|
class ConstantUniqueMap : public AbstractTypeUser {
|
||||||
|
@ -12,6 +12,8 @@
|
|||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
#include "llvm/InlineAsm.h"
|
#include "llvm/InlineAsm.h"
|
||||||
|
#include "ConstantsContext.h"
|
||||||
|
#include "LLVMContextImpl.h"
|
||||||
#include "llvm/DerivedTypes.h"
|
#include "llvm/DerivedTypes.h"
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <cctype>
|
#include <cctype>
|
||||||
@ -23,28 +25,29 @@ InlineAsm::~InlineAsm() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// NOTE: when memoizing the function type, we have to be careful to handle the
|
|
||||||
// case when the type gets refined.
|
|
||||||
|
|
||||||
InlineAsm *InlineAsm::get(const FunctionType *Ty, StringRef AsmString,
|
InlineAsm *InlineAsm::get(const FunctionType *Ty, StringRef AsmString,
|
||||||
StringRef Constraints, bool hasSideEffects,
|
StringRef Constraints, bool hasSideEffects,
|
||||||
bool isAlignStack) {
|
bool isAlignStack) {
|
||||||
// FIXME: memoize!
|
InlineAsmKeyType Key(AsmString, Constraints, hasSideEffects, isAlignStack);
|
||||||
return new InlineAsm(Ty, AsmString, Constraints, hasSideEffects,
|
LLVMContextImpl *pImpl = Ty->getContext().pImpl;
|
||||||
isAlignStack);
|
return pImpl->InlineAsms.getOrCreate(PointerType::getUnqual(Ty), Key);
|
||||||
}
|
}
|
||||||
|
|
||||||
InlineAsm::InlineAsm(const FunctionType *Ty, StringRef asmString,
|
InlineAsm::InlineAsm(const PointerType *Ty, const std::string &asmString,
|
||||||
StringRef constraints, bool hasSideEffects,
|
const std::string &constraints, bool hasSideEffects,
|
||||||
bool isAlignStack)
|
bool isAlignStack)
|
||||||
: Value(PointerType::getUnqual(Ty),
|
: Value(Ty, Value::InlineAsmVal),
|
||||||
Value::InlineAsmVal),
|
|
||||||
AsmString(asmString),
|
AsmString(asmString),
|
||||||
Constraints(constraints), HasSideEffects(hasSideEffects),
|
Constraints(constraints), HasSideEffects(hasSideEffects),
|
||||||
IsAlignStack(isAlignStack) {
|
IsAlignStack(isAlignStack) {
|
||||||
|
|
||||||
// Do various checks on the constraint string and type.
|
// Do various checks on the constraint string and type.
|
||||||
assert(Verify(Ty, constraints) && "Function type not legal for constraints!");
|
assert(Verify(getFunctionType(), constraints) &&
|
||||||
|
"Function type not legal for constraints!");
|
||||||
|
}
|
||||||
|
|
||||||
|
void InlineAsm::destroyConstant() {
|
||||||
|
delete this;
|
||||||
}
|
}
|
||||||
|
|
||||||
const FunctionType *InlineAsm::getFunctionType() const {
|
const FunctionType *InlineAsm::getFunctionType() const {
|
||||||
|
@ -135,6 +135,8 @@ public:
|
|||||||
|
|
||||||
DenseMap<std::pair<Function*, BasicBlock*> , BlockAddress*> BlockAddresses;
|
DenseMap<std::pair<Function*, BasicBlock*> , BlockAddress*> BlockAddresses;
|
||||||
ConstantUniqueMap<ExprMapKeyType, Type, ConstantExpr> ExprConstants;
|
ConstantUniqueMap<ExprMapKeyType, Type, ConstantExpr> ExprConstants;
|
||||||
|
|
||||||
|
ConstantUniqueMap<InlineAsmKeyType, PointerType, InlineAsm> InlineAsms;
|
||||||
|
|
||||||
ConstantInt *TheTrueVal;
|
ConstantInt *TheTrueVal;
|
||||||
ConstantInt *TheFalseVal;
|
ConstantInt *TheFalseVal;
|
||||||
@ -224,6 +226,7 @@ public:
|
|||||||
AggZeroConstants.freeConstants();
|
AggZeroConstants.freeConstants();
|
||||||
NullPtrConstants.freeConstants();
|
NullPtrConstants.freeConstants();
|
||||||
UndefValueConstants.freeConstants();
|
UndefValueConstants.freeConstants();
|
||||||
|
InlineAsms.freeConstants();
|
||||||
for (IntMapTy::iterator I = IntConstants.begin(), E = IntConstants.end();
|
for (IntMapTy::iterator I = IntConstants.begin(), E = IntConstants.end();
|
||||||
I != E; ++I) {
|
I != E; ++I) {
|
||||||
if (I->second->use_empty())
|
if (I->second->use_empty())
|
||||||
|
Loading…
x
Reference in New Issue
Block a user