//-------------------------------------------------------------------------------------- // File: SharedResourcePool.h // // Copyright (c) Microsoft Corporation. // Licensed under the MIT License. // // http://go.microsoft.com/fwlink/?LinkId=248929 // http://go.microsoft.com/fwlink/?LinkID=615561 //-------------------------------------------------------------------------------------- #pragma once #include #include #include "PlatformHelpers.h" namespace DirectX { // Pool manager ensures that only a single TData instance is created for each unique TKey. // This is used to avoid duplicate resource creation, so that for instance a caller can // create any number of SpriteBatch instances, but these can internally share shaders and // vertex buffer if more than one SpriteBatch uses the same underlying D3D device. template class SharedResourcePool { public: SharedResourcePool() noexcept(false) : mResourceMap(std::make_shared()) { } SharedResourcePool(SharedResourcePool const&) = delete; SharedResourcePool& operator= (SharedResourcePool const&) = delete; // Allocates or looks up the shared TData instance for the specified key. std::shared_ptr DemandCreate(TKey key, TConstructorArgs... args) { std::lock_guard lock(mResourceMap->mutex); // Return an existing instance? auto pos = mResourceMap->find(key); if (pos != mResourceMap->end()) { auto existingValue = pos->second.lock(); if (existingValue) return existingValue; else mResourceMap->erase(pos); } // Allocate a new instance. auto newValue = std::make_shared(key, mResourceMap, args...); auto entry = std::make_pair(key, newValue); mResourceMap->insert(entry); return std::move(newValue); } private: // Keep track of all allocated TData instances. struct ResourceMap : public std::map> { std::mutex mutex; }; std::shared_ptr mResourceMap; // Wrap TData with our own subclass, so we can hook the destructor // to remove instances from our pool before they are freed. struct WrappedData : public TData { WrappedData(TKey key, std::shared_ptr const& resourceMap, TConstructorArgs... args) : TData(key, args...), mKey(key), mResourceMap(resourceMap) { } WrappedData(WrappedData&&) = default; WrappedData& operator= (WrappedData&&) = default; WrappedData(WrappedData const&) = delete; WrappedData& operator= (WrappedData const&) = delete; ~WrappedData() { const std::lock_guard lock(mResourceMap->mutex); auto const pos = mResourceMap->find(mKey); // Check for weak reference expiry before erasing, in case DemandCreate runs on // a different thread at the same time as a previous instance is being destroyed. // We mustn't erase replacement objects that have just been added! if (pos != mResourceMap->end() && pos->second.expired()) { mResourceMap->erase(pos); } } TKey mKey; std::shared_ptr mResourceMap; }; }; }