2018-07-11 19:32:05 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2018 Google Inc.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/utils/SkBase64.h"
|
|
|
|
#include "src/core/SkMD5.h"
|
2021-04-09 21:56:58 +00:00
|
|
|
#include "src/core/SkReadBuffer.h"
|
2022-04-07 15:20:24 +00:00
|
|
|
#include "src/gpu/ganesh/GrPersistentCacheUtils.h"
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "tools/gpu/MemoryCache.h"
|
2019-04-15 13:26:13 +00:00
|
|
|
|
|
|
|
#if defined(SK_VULKAN)
|
2022-04-07 15:20:24 +00:00
|
|
|
#include "src/gpu/ganesh/vk/GrVkGpu.h"
|
2019-04-15 13:26:13 +00:00
|
|
|
#endif
|
2018-07-11 19:32:05 +00:00
|
|
|
|
|
|
|
// Change this to 1 to log cache hits/misses/stores using SkDebugf.
|
|
|
|
#define LOG_MEMORY_CACHE 0
|
|
|
|
|
|
|
|
static SkString data_to_str(const SkData& data) {
|
|
|
|
size_t encodeLength = SkBase64::Encode(data.data(), data.size(), nullptr);
|
|
|
|
SkString str;
|
|
|
|
str.resize(encodeLength);
|
|
|
|
SkBase64::Encode(data.data(), data.size(), str.writable_str());
|
|
|
|
static constexpr size_t kMaxLength = 60;
|
|
|
|
static constexpr char kTail[] = "...";
|
|
|
|
static const size_t kTailLen = strlen(kTail);
|
|
|
|
bool overlength = encodeLength > kMaxLength;
|
|
|
|
if (overlength) {
|
|
|
|
str = SkString(str.c_str(), kMaxLength - kTailLen);
|
|
|
|
str.append(kTail);
|
|
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace sk_gpu_test {
|
|
|
|
|
|
|
|
sk_sp<SkData> MemoryCache::load(const SkData& key) {
|
|
|
|
auto result = fMap.find(key);
|
|
|
|
if (result == fMap.end()) {
|
|
|
|
if (LOG_MEMORY_CACHE) {
|
|
|
|
SkDebugf("Load Key: %s\n\tNot Found.\n\n", data_to_str(key).c_str());
|
|
|
|
}
|
|
|
|
++fCacheMissCnt;
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
if (LOG_MEMORY_CACHE) {
|
|
|
|
SkDebugf("Load Key: %s\n\tFound Data: %s\n\n", data_to_str(key).c_str(),
|
2019-04-08 20:40:36 +00:00
|
|
|
data_to_str(*result->second.fData).c_str());
|
2018-07-11 19:32:05 +00:00
|
|
|
}
|
2019-04-08 20:40:36 +00:00
|
|
|
result->second.fHitCount++;
|
|
|
|
return result->second.fData;
|
2018-07-11 19:32:05 +00:00
|
|
|
}
|
|
|
|
|
2021-02-26 18:54:11 +00:00
|
|
|
void MemoryCache::store(const SkData& key, const SkData& data, const SkString& description) {
|
2018-07-11 19:32:05 +00:00
|
|
|
if (LOG_MEMORY_CACHE) {
|
|
|
|
SkDebugf("Store Key: %s\n\tData: %s\n\n", data_to_str(key).c_str(),
|
|
|
|
data_to_str(data).c_str());
|
|
|
|
}
|
2020-06-05 15:11:36 +00:00
|
|
|
++fCacheStoreCnt;
|
2021-02-26 18:54:11 +00:00
|
|
|
fMap[Key(key)] = Value(data, description);
|
2019-04-08 20:40:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void MemoryCache::writeShadersToDisk(const char* path, GrBackendApi api) {
|
2019-04-15 13:26:13 +00:00
|
|
|
if (GrBackendApi::kOpenGL != api && GrBackendApi::kVulkan != api) {
|
2019-04-08 20:40:36 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto it = fMap.begin(); it != fMap.end(); ++it) {
|
|
|
|
SkMD5 hash;
|
2019-04-15 13:26:13 +00:00
|
|
|
size_t bytesToHash = it->first.fKey->size();
|
|
|
|
#if defined(SK_VULKAN)
|
|
|
|
if (GrBackendApi::kVulkan == api) {
|
|
|
|
// Vulkan stores two kinds of data in the cache (shaders and pipelines). The last four
|
|
|
|
// bytes of the key identify which one we have. We only want to extract shaders.
|
|
|
|
// Additionally, we don't want to hash the tag bytes, so we get the same keys as GL,
|
|
|
|
// which is good for cross-checking code generation and performance.
|
|
|
|
GrVkGpu::PersistentCacheKeyType vkKeyType;
|
|
|
|
SkASSERT(bytesToHash >= sizeof(vkKeyType));
|
|
|
|
bytesToHash -= sizeof(vkKeyType);
|
|
|
|
memcpy(&vkKeyType, it->first.fKey->bytes() + bytesToHash, sizeof(vkKeyType));
|
|
|
|
if (vkKeyType != GrVkGpu::kShader_PersistentCacheKeyType) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
hash.write(it->first.fKey->bytes(), bytesToHash);
|
2019-04-08 20:40:36 +00:00
|
|
|
SkMD5::Digest digest = hash.finish();
|
|
|
|
SkString md5;
|
|
|
|
for (int i = 0; i < 16; ++i) {
|
|
|
|
md5.appendf("%02x", digest.data[i]);
|
|
|
|
}
|
|
|
|
|
2019-04-15 13:26:13 +00:00
|
|
|
SkSL::Program::Inputs inputsIgnored[kGrShaderTypeCount];
|
2022-02-02 21:51:18 +00:00
|
|
|
std::string shaders[kGrShaderTypeCount];
|
2019-04-15 13:26:13 +00:00
|
|
|
const SkData* data = it->second.fData.get();
|
2021-03-03 19:24:37 +00:00
|
|
|
const SkString& description = it->second.fDescription;
|
2020-06-10 11:19:34 +00:00
|
|
|
SkReadBuffer reader(data->data(), data->size());
|
2020-03-16 20:21:24 +00:00
|
|
|
GrPersistentCacheUtils::GetType(&reader); // Shader type tag
|
2019-09-03 18:59:26 +00:00
|
|
|
GrPersistentCacheUtils::UnpackCachedShaders(&reader, shaders,
|
2019-04-25 13:44:43 +00:00
|
|
|
inputsIgnored, kGrShaderTypeCount);
|
2019-04-08 20:40:36 +00:00
|
|
|
|
2021-03-03 19:24:37 +00:00
|
|
|
// Even with the SPIR-V switches, it seems like we must use .spv, or malisc tries to
|
|
|
|
// run glslang on the input.
|
|
|
|
{
|
|
|
|
const char* ext = GrBackendApi::kOpenGL == api ? "frag" : "frag.spv";
|
|
|
|
SkString filename = SkStringPrintf("%s/%s.%s", path, md5.c_str(), ext);
|
|
|
|
SkFILEWStream file(filename.c_str());
|
|
|
|
file.write(shaders[kFragment_GrShaderType].c_str(),
|
|
|
|
shaders[kFragment_GrShaderType].size());
|
|
|
|
}
|
|
|
|
{
|
|
|
|
const char* ext = GrBackendApi::kOpenGL == api ? "vert" : "vert.spv";
|
|
|
|
SkString filename = SkStringPrintf("%s/%s.%s", path, md5.c_str(), ext);
|
|
|
|
SkFILEWStream file(filename.c_str());
|
|
|
|
file.write(shaders[kVertex_GrShaderType].c_str(),
|
|
|
|
shaders[kVertex_GrShaderType].size());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!description.isEmpty()) {
|
|
|
|
const char* ext = "key";
|
|
|
|
SkString filename = SkStringPrintf("%s/%s.%s", path, md5.c_str(), ext);
|
|
|
|
SkFILEWStream file(filename.c_str());
|
|
|
|
file.write(description.c_str(), description.size());
|
|
|
|
}
|
2019-04-08 20:40:36 +00:00
|
|
|
}
|
2018-07-11 19:32:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace sk_gpu_test
|