blob: 70fedeaca393920c47b9333a7e1bb0b7a93710e9 [file] [log] [blame]
Brian Salomon00a5eb82018-07-11 15:32:05 -04001/*
2 * Copyright 2018 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "include/utils/SkBase64.h"
9#include "src/core/SkMD5.h"
John Stilesf2c2d302021-04-09 17:56:58 -040010#include "src/core/SkReadBuffer.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "src/gpu/GrPersistentCacheUtils.h"
12#include "tools/gpu/MemoryCache.h"
Brian Osmanbe2062c2019-04-15 09:26:13 -040013
14#if defined(SK_VULKAN)
Mike Kleinc0bd9f92019-04-23 12:05:21 -050015#include "src/gpu/vk/GrVkGpu.h"
Brian Osmanbe2062c2019-04-15 09:26:13 -040016#endif
Brian Salomon00a5eb82018-07-11 15:32:05 -040017
18// Change this to 1 to log cache hits/misses/stores using SkDebugf.
19#define LOG_MEMORY_CACHE 0
20
21static SkString data_to_str(const SkData& data) {
22 size_t encodeLength = SkBase64::Encode(data.data(), data.size(), nullptr);
23 SkString str;
24 str.resize(encodeLength);
25 SkBase64::Encode(data.data(), data.size(), str.writable_str());
26 static constexpr size_t kMaxLength = 60;
27 static constexpr char kTail[] = "...";
28 static const size_t kTailLen = strlen(kTail);
29 bool overlength = encodeLength > kMaxLength;
30 if (overlength) {
31 str = SkString(str.c_str(), kMaxLength - kTailLen);
32 str.append(kTail);
33 }
34 return str;
35}
36
37namespace sk_gpu_test {
38
39sk_sp<SkData> MemoryCache::load(const SkData& key) {
40 auto result = fMap.find(key);
41 if (result == fMap.end()) {
42 if (LOG_MEMORY_CACHE) {
43 SkDebugf("Load Key: %s\n\tNot Found.\n\n", data_to_str(key).c_str());
44 }
45 ++fCacheMissCnt;
46 return nullptr;
47 }
48 if (LOG_MEMORY_CACHE) {
49 SkDebugf("Load Key: %s\n\tFound Data: %s\n\n", data_to_str(key).c_str(),
Brian Osman5aa11fb2019-04-08 16:40:36 -040050 data_to_str(*result->second.fData).c_str());
Brian Salomon00a5eb82018-07-11 15:32:05 -040051 }
Brian Osman5aa11fb2019-04-08 16:40:36 -040052 result->second.fHitCount++;
53 return result->second.fData;
Brian Salomon00a5eb82018-07-11 15:32:05 -040054}
55
Brian Osmanf0de96f2021-02-26 13:54:11 -050056void MemoryCache::store(const SkData& key, const SkData& data, const SkString& description) {
Brian Salomon00a5eb82018-07-11 15:32:05 -040057 if (LOG_MEMORY_CACHE) {
58 SkDebugf("Store Key: %s\n\tData: %s\n\n", data_to_str(key).c_str(),
59 data_to_str(data).c_str());
60 }
Brian Osman43f443f2020-06-05 11:11:36 -040061 ++fCacheStoreCnt;
Brian Osmanf0de96f2021-02-26 13:54:11 -050062 fMap[Key(key)] = Value(data, description);
Brian Osman5aa11fb2019-04-08 16:40:36 -040063}
64
65void MemoryCache::writeShadersToDisk(const char* path, GrBackendApi api) {
Brian Osmanbe2062c2019-04-15 09:26:13 -040066 if (GrBackendApi::kOpenGL != api && GrBackendApi::kVulkan != api) {
Brian Osman5aa11fb2019-04-08 16:40:36 -040067 return;
68 }
69
Brian Osman5aa11fb2019-04-08 16:40:36 -040070 for (auto it = fMap.begin(); it != fMap.end(); ++it) {
71 SkMD5 hash;
Brian Osmanbe2062c2019-04-15 09:26:13 -040072 size_t bytesToHash = it->first.fKey->size();
73#if defined(SK_VULKAN)
74 if (GrBackendApi::kVulkan == api) {
75 // Vulkan stores two kinds of data in the cache (shaders and pipelines). The last four
76 // bytes of the key identify which one we have. We only want to extract shaders.
77 // Additionally, we don't want to hash the tag bytes, so we get the same keys as GL,
78 // which is good for cross-checking code generation and performance.
79 GrVkGpu::PersistentCacheKeyType vkKeyType;
80 SkASSERT(bytesToHash >= sizeof(vkKeyType));
81 bytesToHash -= sizeof(vkKeyType);
82 memcpy(&vkKeyType, it->first.fKey->bytes() + bytesToHash, sizeof(vkKeyType));
83 if (vkKeyType != GrVkGpu::kShader_PersistentCacheKeyType) {
84 continue;
85 }
86 }
87#endif
88 hash.write(it->first.fKey->bytes(), bytesToHash);
Brian Osman5aa11fb2019-04-08 16:40:36 -040089 SkMD5::Digest digest = hash.finish();
90 SkString md5;
91 for (int i = 0; i < 16; ++i) {
92 md5.appendf("%02x", digest.data[i]);
93 }
94
Brian Osmanbe2062c2019-04-15 09:26:13 -040095 SkSL::Program::Inputs inputsIgnored[kGrShaderTypeCount];
96 SkSL::String shaders[kGrShaderTypeCount];
97 const SkData* data = it->second.fData.get();
Brian Osman9e1ef992021-03-03 14:24:37 -050098 const SkString& description = it->second.fDescription;
Brian Osman9e4e4c72020-06-10 07:19:34 -040099 SkReadBuffer reader(data->data(), data->size());
Brian Osman1facd5e2020-03-16 16:21:24 -0400100 GrPersistentCacheUtils::GetType(&reader); // Shader type tag
Brian Osmana66081d2019-09-03 14:59:26 -0400101 GrPersistentCacheUtils::UnpackCachedShaders(&reader, shaders,
Brian Osmana085a412019-04-25 09:44:43 -0400102 inputsIgnored, kGrShaderTypeCount);
Brian Osman5aa11fb2019-04-08 16:40:36 -0400103
Brian Osman9e1ef992021-03-03 14:24:37 -0500104 // Even with the SPIR-V switches, it seems like we must use .spv, or malisc tries to
105 // run glslang on the input.
106 {
107 const char* ext = GrBackendApi::kOpenGL == api ? "frag" : "frag.spv";
108 SkString filename = SkStringPrintf("%s/%s.%s", path, md5.c_str(), ext);
109 SkFILEWStream file(filename.c_str());
110 file.write(shaders[kFragment_GrShaderType].c_str(),
111 shaders[kFragment_GrShaderType].size());
112 }
113 {
114 const char* ext = GrBackendApi::kOpenGL == api ? "vert" : "vert.spv";
115 SkString filename = SkStringPrintf("%s/%s.%s", path, md5.c_str(), ext);
116 SkFILEWStream file(filename.c_str());
117 file.write(shaders[kVertex_GrShaderType].c_str(),
118 shaders[kVertex_GrShaderType].size());
119 }
120
121 if (!description.isEmpty()) {
122 const char* ext = "key";
123 SkString filename = SkStringPrintf("%s/%s.%s", path, md5.c_str(), ext);
124 SkFILEWStream file(filename.c_str());
125 file.write(description.c_str(), description.size());
126 }
Brian Osman5aa11fb2019-04-08 16:40:36 -0400127 }
Brian Salomon00a5eb82018-07-11 15:32:05 -0400128}
129
130} // namespace sk_gpu_test