blob: 37bf52fd2b43e9f0bbdbe33affbe3c61ee621dea [file] [log] [blame]
Brian Osman6b797fe2019-04-08 13:56:36 -04001/*
2 * Copyright 2019 Google LLC.
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
8#ifndef GrPersistentCacheEntry_DEFINED
9#define GrPersistentCacheEntry_DEFINED
10
Mike Kleinc0bd9f92019-04-23 12:05:21 -050011#include "include/core/SkData.h"
12#include "include/private/GrTypesPriv.h"
Brian Osman9e4e4c72020-06-10 07:19:34 -040013#include "src/core/SkReadBuffer.h"
14#include "src/core/SkWriteBuffer.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050015#include "src/sksl/SkSLString.h"
16#include "src/sksl/ir/SkSLProgram.h"
Brian Osman6b797fe2019-04-08 13:56:36 -040017
Brian Osmana5a010b2019-04-08 15:01:32 -040018// The GrPersistentCache stores opaque blobs, as far as clients are concerned. It's helpful to
19// inspect certain kinds of cached data within our tools, so for those cases (GLSL, SPIR-V), we
20// put the serialization logic here, to be shared by the backend code and the tool code.
Brian Osman6b797fe2019-04-08 13:56:36 -040021namespace GrPersistentCacheUtils {
22
Brian Osman4524e842019-09-24 16:03:41 -040023struct ShaderMetadata {
24 SkSL::Program::Settings* fSettings = nullptr;
25 SkTArray<SkSL::String> fAttributeNames;
26 bool fHasCustomColorOutput = false;
27 bool fHasSecondaryColorOutput = false;
28};
29
Brian Osman1facd5e2020-03-16 16:21:24 -040030// Increment this whenever the serialization format of cached shaders changes
Brian Osman9e4e4c72020-06-10 07:19:34 -040031static constexpr int kCurrentVersion = 2;
Brian Osman1facd5e2020-03-16 16:21:24 -040032
Brian Osmana085a412019-04-25 09:44:43 -040033static inline sk_sp<SkData> PackCachedShaders(SkFourByteTag shaderType,
34 const SkSL::String shaders[],
35 const SkSL::Program::Inputs inputs[],
Brian Osmaned58e002019-09-06 14:42:43 -040036 int numInputs,
Brian Osman4524e842019-09-24 16:03:41 -040037 const ShaderMetadata* meta = nullptr) {
Brian Osmana085a412019-04-25 09:44:43 -040038 // For consistency (so tools can blindly pack and unpack cached shaders), we always write
39 // kGrShaderTypeCount inputs. If the backend gives us fewer, we just replicate the last one.
40 SkASSERT(numInputs >= 1 && numInputs <= kGrShaderTypeCount);
Brian Osman6b797fe2019-04-08 13:56:36 -040041
Brian Osman9e4e4c72020-06-10 07:19:34 -040042 SkBinaryWriteBuffer writer;
43 writer.writeInt(kCurrentVersion);
44 writer.writeUInt(shaderType);
Brian Osmana5a010b2019-04-08 15:01:32 -040045 for (int i = 0; i < kGrShaderTypeCount; ++i) {
Brian Osman9e4e4c72020-06-10 07:19:34 -040046 writer.writeByteArray(shaders[i].c_str(), shaders[i].size());
47 writer.writePad32(&inputs[std::min(i, numInputs - 1)], sizeof(SkSL::Program::Inputs));
Brian Osmana5a010b2019-04-08 15:01:32 -040048 }
Brian Osman4524e842019-09-24 16:03:41 -040049 writer.writeBool(SkToBool(meta));
50 if (meta) {
51 writer.writeBool(SkToBool(meta->fSettings));
52 if (meta->fSettings) {
53 writer.writeBool(meta->fSettings->fFlipY);
54 writer.writeBool(meta->fSettings->fFragColorIsInOut);
55 writer.writeBool(meta->fSettings->fForceHighPrecision);
56 }
57
58 writer.writeInt(meta->fAttributeNames.count());
59 for (const auto& attr : meta->fAttributeNames) {
Brian Osman9e4e4c72020-06-10 07:19:34 -040060 writer.writeByteArray(attr.c_str(), attr.size());
Brian Osman4524e842019-09-24 16:03:41 -040061 }
62
63 writer.writeBool(meta->fHasCustomColorOutput);
64 writer.writeBool(meta->fHasSecondaryColorOutput);
Brian Osmaned58e002019-09-06 14:42:43 -040065 }
Brian Osmane6ef03d2019-04-11 14:38:27 -040066 return writer.snapshotAsData();
Brian Osmana5a010b2019-04-08 15:01:32 -040067}
68
Brian Osman9e4e4c72020-06-10 07:19:34 -040069static SkFourByteTag GetType(SkReadBuffer* reader) {
Brian Osman1facd5e2020-03-16 16:21:24 -040070 constexpr SkFourByteTag kInvalidTag = ~0;
Brian Osman9e4e4c72020-06-10 07:19:34 -040071 int version = reader->readInt();
72 SkFourByteTag typeTag = reader->readUInt();
73 return reader->validate(version == kCurrentVersion) ? typeTag : kInvalidTag;
Brian Osman1facd5e2020-03-16 16:21:24 -040074}
75
Brian Osman9e4e4c72020-06-10 07:19:34 -040076static inline bool UnpackCachedShaders(SkReadBuffer* reader,
Brian Osmana66081d2019-09-03 14:59:26 -040077 SkSL::String shaders[],
78 SkSL::Program::Inputs inputs[],
Brian Osmaned58e002019-09-06 14:42:43 -040079 int numInputs,
Brian Osman4524e842019-09-24 16:03:41 -040080 ShaderMetadata* meta = nullptr) {
Brian Osmana5a010b2019-04-08 15:01:32 -040081 for (int i = 0; i < kGrShaderTypeCount; ++i) {
Brian Osman9e4e4c72020-06-10 07:19:34 -040082 size_t shaderLen = 0;
83 const char* shaderBuf = static_cast<const char*>(reader->skipByteArray(&shaderLen));
84 if (shaderBuf) {
85 shaders[i].assign(shaderBuf, shaderLen);
86 }
Brian Osmana085a412019-04-25 09:44:43 -040087
88 // GL, for example, only wants one set of Inputs
89 if (i < numInputs) {
Brian Osman9e4e4c72020-06-10 07:19:34 -040090 reader->readPad32(&inputs[i], sizeof(inputs[i]));
Brian Osmana085a412019-04-25 09:44:43 -040091 } else {
Brian Osmana66081d2019-09-03 14:59:26 -040092 reader->skip(sizeof(SkSL::Program::Inputs));
Brian Osmana085a412019-04-25 09:44:43 -040093 }
Brian Osmana5a010b2019-04-08 15:01:32 -040094 }
Brian Osman4524e842019-09-24 16:03:41 -040095 if (reader->readBool() && meta) {
96 SkASSERT(meta->fSettings != nullptr);
97
98 if (reader->readBool()) {
99 meta->fSettings->fFlipY = reader->readBool();
100 meta->fSettings->fFragColorIsInOut = reader->readBool();
101 meta->fSettings->fForceHighPrecision = reader->readBool();
102 }
103
104 meta->fAttributeNames.resize(reader->readInt());
Brian Osman9e4e4c72020-06-10 07:19:34 -0400105 for (auto& attr : meta->fAttributeNames) {
106 size_t attrLen = 0;
107 const char* attrName = static_cast<const char*>(reader->skipByteArray(&attrLen));
108 if (attrName) {
109 attr.assign(attrName, attrLen);
110 }
Brian Osman4524e842019-09-24 16:03:41 -0400111 }
112
113 meta->fHasCustomColorOutput = reader->readBool();
114 meta->fHasSecondaryColorOutput = reader->readBool();
Brian Osmaned58e002019-09-06 14:42:43 -0400115 }
Brian Osman9e4e4c72020-06-10 07:19:34 -0400116
117 if (!reader->isValid()) {
118 for (int i = 0; i < kGrShaderTypeCount; ++i) {
119 shaders[i].clear();
120 }
121 }
122 return reader->isValid();
Brian Osmana5a010b2019-04-08 15:01:32 -0400123}
124
John Stilesa6841be2020-08-06 14:11:56 -0400125} // namespace GrPersistentCacheUtils
Brian Osman6b797fe2019-04-08 13:56:36 -0400126
127#endif