mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-04-02 21:22:44 +00:00
[mlir][StorageUniquer] Replace all usages of std::function with function_ref.
Summary: std::function has a notoriously large amount of malloc traffic, whereas function_ref is a cheaper and more efficient alternative. Differential Revision: https://reviews.llvm.org/D77959
This commit is contained in:
parent
d985b0bf5c
commit
1fc6efaf6a
@ -92,13 +92,16 @@ public:
|
||||
template <typename T, typename... Args>
|
||||
static T get(MLIRContext *ctx, unsigned kind, Args &&... args) {
|
||||
return ctx->getAttributeUniquer().get<typename T::ImplType>(
|
||||
getInitFn(ctx, T::getTypeID()), kind, std::forward<Args>(args)...);
|
||||
[ctx](AttributeStorage *storage) {
|
||||
initializeAttributeStorage(storage, ctx, T::getTypeID());
|
||||
},
|
||||
kind, std::forward<Args>(args)...);
|
||||
}
|
||||
|
||||
private:
|
||||
/// Returns a functor used to initialize new attribute storage instances.
|
||||
static std::function<void(AttributeStorage *)> getInitFn(MLIRContext *ctx,
|
||||
TypeID attrID);
|
||||
/// Initialize the given attribute storage instance.
|
||||
static void initializeAttributeStorage(AttributeStorage *storage,
|
||||
MLIRContext *ctx, TypeID attrID);
|
||||
};
|
||||
} // namespace detail
|
||||
|
||||
|
@ -123,7 +123,7 @@ public:
|
||||
/// function is used for derived types that have complex storage or uniquing
|
||||
/// constraints.
|
||||
template <typename Storage, typename Arg, typename... Args>
|
||||
Storage *get(std::function<void(Storage *)> initFn, unsigned kind, Arg &&arg,
|
||||
Storage *get(function_ref<void(Storage *)> initFn, unsigned kind, Arg &&arg,
|
||||
Args &&... args) {
|
||||
// Construct a value of the derived key type.
|
||||
auto derivedKey =
|
||||
@ -133,19 +133,17 @@ public:
|
||||
unsigned hashValue = getHash<Storage>(kind, derivedKey);
|
||||
|
||||
// Generate an equality function for the derived storage.
|
||||
std::function<bool(const BaseStorage *)> isEqual =
|
||||
[&derivedKey](const BaseStorage *existing) {
|
||||
return static_cast<const Storage &>(*existing) == derivedKey;
|
||||
};
|
||||
auto isEqual = [&derivedKey](const BaseStorage *existing) {
|
||||
return static_cast<const Storage &>(*existing) == derivedKey;
|
||||
};
|
||||
|
||||
// Generate a constructor function for the derived storage.
|
||||
std::function<BaseStorage *(StorageAllocator &)> ctorFn =
|
||||
[&](StorageAllocator &allocator) {
|
||||
auto *storage = Storage::construct(allocator, derivedKey);
|
||||
if (initFn)
|
||||
initFn(storage);
|
||||
return storage;
|
||||
};
|
||||
auto ctorFn = [&](StorageAllocator &allocator) {
|
||||
auto *storage = Storage::construct(allocator, derivedKey);
|
||||
if (initFn)
|
||||
initFn(storage);
|
||||
return storage;
|
||||
};
|
||||
|
||||
// Get an instance for the derived storage.
|
||||
return static_cast<Storage *>(getImpl(kind, hashValue, isEqual, ctorFn));
|
||||
@ -156,7 +154,7 @@ public:
|
||||
/// function is used for derived types that use no additional storage or
|
||||
/// uniquing outside of the kind.
|
||||
template <typename Storage>
|
||||
Storage *get(std::function<void(Storage *)> initFn, unsigned kind) {
|
||||
Storage *get(function_ref<void(Storage *)> initFn, unsigned kind) {
|
||||
auto ctorFn = [&](StorageAllocator &allocator) {
|
||||
auto *storage = new (allocator.allocate<Storage>()) Storage();
|
||||
if (initFn)
|
||||
@ -178,10 +176,9 @@ public:
|
||||
unsigned hashValue = getHash<Storage>(kind, derivedKey);
|
||||
|
||||
// Generate an equality function for the derived storage.
|
||||
std::function<bool(const BaseStorage *)> isEqual =
|
||||
[&derivedKey](const BaseStorage *existing) {
|
||||
return static_cast<const Storage &>(*existing) == derivedKey;
|
||||
};
|
||||
auto isEqual = [&derivedKey](const BaseStorage *existing) {
|
||||
return static_cast<const Storage &>(*existing) == derivedKey;
|
||||
};
|
||||
|
||||
// Attempt to erase the storage instance.
|
||||
eraseImpl(kind, hashValue, isEqual, [](BaseStorage *storage) {
|
||||
@ -194,18 +191,18 @@ private:
|
||||
/// complex storage.
|
||||
BaseStorage *getImpl(unsigned kind, unsigned hashValue,
|
||||
function_ref<bool(const BaseStorage *)> isEqual,
|
||||
std::function<BaseStorage *(StorageAllocator &)> ctorFn);
|
||||
function_ref<BaseStorage *(StorageAllocator &)> ctorFn);
|
||||
|
||||
/// Implementation for getting/creating an instance of a derived type with
|
||||
/// default storage.
|
||||
BaseStorage *getImpl(unsigned kind,
|
||||
std::function<BaseStorage *(StorageAllocator &)> ctorFn);
|
||||
function_ref<BaseStorage *(StorageAllocator &)> ctorFn);
|
||||
|
||||
/// Implementation for erasing an instance of a derived type with complex
|
||||
/// storage.
|
||||
void eraseImpl(unsigned kind, unsigned hashValue,
|
||||
function_ref<bool(const BaseStorage *)> isEqual,
|
||||
std::function<void(BaseStorage *)> cleanupFn);
|
||||
function_ref<void(BaseStorage *)> cleanupFn);
|
||||
|
||||
/// The internal implementation class.
|
||||
std::unique_ptr<detail::StorageUniquerImpl> impl;
|
||||
|
@ -624,16 +624,15 @@ StorageUniquer &MLIRContext::getAttributeUniquer() {
|
||||
return getImpl().attributeUniquer;
|
||||
}
|
||||
|
||||
/// Returns a functor used to initialize new attribute storage instances.
|
||||
std::function<void(AttributeStorage *)>
|
||||
AttributeUniquer::getInitFn(MLIRContext *ctx, TypeID attrID) {
|
||||
return [ctx, attrID](AttributeStorage *storage) {
|
||||
storage->initializeDialect(lookupDialectForSymbol(ctx, attrID));
|
||||
/// Initialize the given attribute storage instance.
|
||||
void AttributeUniquer::initializeAttributeStorage(AttributeStorage *storage,
|
||||
MLIRContext *ctx,
|
||||
TypeID attrID) {
|
||||
storage->initializeDialect(lookupDialectForSymbol(ctx, attrID));
|
||||
|
||||
// If the attribute did not provide a type, then default to NoneType.
|
||||
if (!storage->getType())
|
||||
storage->setType(NoneType::get(ctx));
|
||||
};
|
||||
// If the attribute did not provide a type, then default to NoneType.
|
||||
if (!storage->getType())
|
||||
storage->setType(NoneType::get(ctx));
|
||||
}
|
||||
|
||||
BoolAttr BoolAttr::get(bool value, MLIRContext *context) {
|
||||
|
@ -176,14 +176,14 @@ StorageUniquer::~StorageUniquer() {}
|
||||
auto StorageUniquer::getImpl(
|
||||
unsigned kind, unsigned hashValue,
|
||||
function_ref<bool(const BaseStorage *)> isEqual,
|
||||
std::function<BaseStorage *(StorageAllocator &)> ctorFn) -> BaseStorage * {
|
||||
function_ref<BaseStorage *(StorageAllocator &)> ctorFn) -> BaseStorage * {
|
||||
return impl->getOrCreate(kind, hashValue, isEqual, ctorFn);
|
||||
}
|
||||
|
||||
/// Implementation for getting/creating an instance of a derived type with
|
||||
/// default storage.
|
||||
auto StorageUniquer::getImpl(
|
||||
unsigned kind, std::function<BaseStorage *(StorageAllocator &)> ctorFn)
|
||||
unsigned kind, function_ref<BaseStorage *(StorageAllocator &)> ctorFn)
|
||||
-> BaseStorage * {
|
||||
return impl->getOrCreate(kind, ctorFn);
|
||||
}
|
||||
@ -192,6 +192,6 @@ auto StorageUniquer::getImpl(
|
||||
/// storage.
|
||||
void StorageUniquer::eraseImpl(unsigned kind, unsigned hashValue,
|
||||
function_ref<bool(const BaseStorage *)> isEqual,
|
||||
std::function<void(BaseStorage *)> cleanupFn) {
|
||||
function_ref<void(BaseStorage *)> cleanupFn) {
|
||||
impl->erase(kind, hashValue, isEqual, cleanupFn);
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user