| // Copyright 2020 The Dawn Authors | 
 | // | 
 | // Licensed under the Apache License, Version 2.0 (the "License"); | 
 | // you may not use this file except in compliance with the License. | 
 | // You may obtain a copy of the License at | 
 | // | 
 | //     http://www.apache.org/licenses/LICENSE-2.0 | 
 | // | 
 | // Unless required by applicable law or agreed to in writing, software | 
 | // distributed under the License is distributed on an "AS IS" BASIS, | 
 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | // See the License for the specific language governing permissions and | 
 | // limitations under the License. | 
 |  | 
 | #ifndef DAWNNATIVE_PERSISTENTCACHE_H_ | 
 | #define DAWNNATIVE_PERSISTENTCACHE_H_ | 
 |  | 
 | #include "dawn_native/Error.h" | 
 |  | 
 | #include <mutex> | 
 | #include <vector> | 
 |  | 
 | namespace dawn::platform { | 
 |     class CachingInterface; | 
 | } | 
 |  | 
 | namespace dawn::native { | 
 |  | 
 |     using PersistentCacheKey = std::vector<uint8_t>; | 
 |  | 
 |     struct ScopedCachedBlob { | 
 |         std::unique_ptr<uint8_t[]> buffer; | 
 |         size_t bufferSize = 0; | 
 |     }; | 
 |  | 
 |     class DeviceBase; | 
 |  | 
 |     enum class PersistentKeyType { Shader }; | 
 |  | 
 |     // This class should always be thread-safe as it is used in Create*PipelineAsync() where it is | 
 |     // called asynchronously. | 
 |     // The thread-safety of any access to mCache (the function LoadData() and StoreData()) is | 
 |     // protected by mMutex. | 
 |     class PersistentCache { | 
 |       public: | 
 |         PersistentCache(DeviceBase* device); | 
 |  | 
 |         // Combines load/store operations into a single call. | 
 |         // If the load was successful, a non-empty blob is returned to the caller. | 
 |         // Else, the creation callback |createFn| gets invoked with a callback | 
 |         // |doCache| to store the newly created blob back in the cache. | 
 |         // | 
 |         // Example usage: | 
 |         // | 
 |         // ScopedCachedBlob cachedBlob = {}; | 
 |         // DAWN_TRY_ASSIGN(cachedBlob, GetOrCreate(key, [&](auto doCache)) { | 
 |         //      // Create a new blob to be stored | 
 |         //      doCache(newBlobPtr, newBlobSize); // store | 
 |         // })); | 
 |         // | 
 |         template <typename CreateFn> | 
 |         ResultOrError<ScopedCachedBlob> GetOrCreate(const PersistentCacheKey& key, | 
 |                                                     CreateFn&& createFn) { | 
 |             // Attempt to load an existing blob from the cache. | 
 |             ScopedCachedBlob blob = LoadData(key); | 
 |             if (blob.bufferSize > 0) { | 
 |                 return std::move(blob); | 
 |             } | 
 |  | 
 |             // Allow the caller to create a new blob to be stored for the given key. | 
 |             DAWN_TRY(createFn([this, key](const void* value, size_t size) { | 
 |                 this->StoreData(key, value, size); | 
 |             })); | 
 |  | 
 |             return std::move(blob); | 
 |         } | 
 |  | 
 |       private: | 
 |         // PersistentCache impl | 
 |         ScopedCachedBlob LoadData(const PersistentCacheKey& key); | 
 |         void StoreData(const PersistentCacheKey& key, const void* value, size_t size); | 
 |  | 
 |         dawn::platform::CachingInterface* GetPlatformCache(); | 
 |  | 
 |         DeviceBase* mDevice = nullptr; | 
 |  | 
 |         std::mutex mMutex; | 
 |         dawn::platform::CachingInterface* mCache = nullptr; | 
 |     }; | 
 | }  // namespace dawn::native | 
 |  | 
 | #endif  // DAWNNATIVE_PERSISTENTCACHE_H_ |