[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:
River Riddle 2020-04-11 23:00:11 -07:00
parent d985b0bf5c
commit 1fc6efaf6a
4 changed files with 35 additions and 36 deletions

View File

@ -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

View File

@ -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;

View File

@ -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) {

View File

@ -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);
}