//===- StorageUniquer.cpp - Common Storage Class Uniquer ------------------===// // // Part of the MLIR 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 // //===----------------------------------------------------------------------===// #include "mlir/Support/StorageUniquer.h" #include "mlir/Support/LLVM.h" #include "llvm/Support/RWMutex.h" using namespace mlir; using namespace mlir::detail; namespace mlir { namespace detail { /// This is the implementation of the StorageUniquer class. struct StorageUniquerImpl { using BaseStorage = StorageUniquer::BaseStorage; using StorageAllocator = StorageUniquer::StorageAllocator; /// A lookup key for derived instances of storage objects. struct LookupKey { /// The known derived kind for the storage. unsigned kind; /// The known hash value of the key. unsigned hashValue; /// An equality function for comparing with an existing storage instance. function_ref isEqual; }; /// A utility wrapper object representing a hashed storage object. This class /// contains a storage object and an existing computed hash value. struct HashedStorage { unsigned hashValue; BaseStorage *storage; }; /// Get or create an instance of a complex derived type. BaseStorage * getOrCreate(unsigned kind, unsigned hashValue, function_ref isEqual, function_ref ctorFn) { LookupKey lookupKey{kind, hashValue, isEqual}; // Check for an existing instance in read-only mode. { llvm::sys::SmartScopedReader typeLock(mutex); auto it = storageTypes.find_as(lookupKey); if (it != storageTypes.end()) return it->storage; } // Acquire a writer-lock so that we can safely create the new type instance. llvm::sys::SmartScopedWriter typeLock(mutex); // Check for an existing instance again here, because another writer thread // may have already created one. auto existing = storageTypes.insert_as({}, lookupKey); if (!existing.second) return existing.first->storage; // Otherwise, construct and initialize the derived storage for this type // instance. BaseStorage *storage = initializeStorage(kind, ctorFn); *existing.first = HashedStorage{hashValue, storage}; return storage; } /// Get or create an instance of a simple derived type. BaseStorage * getOrCreate(unsigned kind, function_ref ctorFn) { // Check for an existing instance in read-only mode. { llvm::sys::SmartScopedReader typeLock(mutex); auto it = simpleTypes.find(kind); if (it != simpleTypes.end()) return it->second; } // Acquire a writer-lock so that we can safely create the new type instance. llvm::sys::SmartScopedWriter typeLock(mutex); // Check for an existing instance again here, because another writer thread // may have already created one. auto &result = simpleTypes[kind]; if (result) return result; // Otherwise, create and return a new storage instance. return result = initializeStorage(kind, ctorFn); } /// Erase an instance of a complex derived type. void erase(unsigned kind, unsigned hashValue, function_ref isEqual, function_ref cleanupFn) { LookupKey lookupKey{kind, hashValue, isEqual}; // Acquire a writer-lock so that we can safely erase the type instance. llvm::sys::SmartScopedWriter typeLock(mutex); auto existing = storageTypes.find_as(lookupKey); if (existing == storageTypes.end()) return; // Cleanup the storage and remove it from the map. cleanupFn(existing->storage); storageTypes.erase(existing); } //===--------------------------------------------------------------------===// // Instance Storage //===--------------------------------------------------------------------===// /// Utility to create and initialize a storage instance. BaseStorage * initializeStorage(unsigned kind, function_ref ctorFn) { BaseStorage *storage = ctorFn(allocator); storage->kind = kind; return storage; } /// Storage info for derived TypeStorage objects. struct StorageKeyInfo : DenseMapInfo { static HashedStorage getEmptyKey() { return HashedStorage{0, DenseMapInfo::getEmptyKey()}; } static HashedStorage getTombstoneKey() { return HashedStorage{0, DenseMapInfo::getTombstoneKey()}; } static unsigned getHashValue(const HashedStorage &key) { return key.hashValue; } static unsigned getHashValue(LookupKey key) { return key.hashValue; } static bool isEqual(const HashedStorage &lhs, const HashedStorage &rhs) { return lhs.storage == rhs.storage; } static bool isEqual(const LookupKey &lhs, const HashedStorage &rhs) { if (isEqual(rhs, getEmptyKey()) || isEqual(rhs, getTombstoneKey())) return false; // If the lookup kind matches the kind of the storage, then invoke the // equality function on the lookup key. return lhs.kind == rhs.storage->getKind() && lhs.isEqual(rhs.storage); } }; // Unique types with specific hashing or storage constraints. using StorageTypeSet = DenseSet; StorageTypeSet storageTypes; // Unique types with just the kind. DenseMap simpleTypes; // Allocator to use when constructing derived type instances. StorageUniquer::StorageAllocator allocator; // A mutex to keep type uniquing thread-safe. llvm::sys::SmartRWMutex mutex; }; } // end namespace detail } // namespace mlir StorageUniquer::StorageUniquer() : impl(new StorageUniquerImpl()) {} StorageUniquer::~StorageUniquer() {} /// Implementation for getting/creating an instance of a derived type with /// complex storage. auto StorageUniquer::getImpl( unsigned kind, unsigned hashValue, function_ref isEqual, std::function 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 ctorFn) -> BaseStorage * { return impl->getOrCreate(kind, ctorFn); } /// Implementation for erasing an instance of a derived type with complex /// storage. void StorageUniquer::eraseImpl(unsigned kind, unsigned hashValue, function_ref isEqual, std::function cleanupFn) { impl->erase(kind, hashValue, isEqual, cleanupFn); }