1 | /* |
2 | * Copyright 2014 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 | |
8 | #ifndef GrProgramDesc_DEFINED |
9 | #define GrProgramDesc_DEFINED |
10 | |
11 | #include "include/private/GrTypesPriv.h" |
12 | #include "include/private/SkTArray.h" |
13 | #include "include/private/SkTo.h" |
14 | |
15 | class GrCaps; |
16 | class GrProgramInfo; |
17 | class GrRenderTarget; |
18 | class GrShaderCaps; |
19 | |
20 | /** This class is used to generate a generic program cache key. The Dawn, Metal and Vulkan |
21 | * backends derive backend-specific versions which add additional information. |
22 | */ |
23 | class GrProgramDesc { |
24 | public: |
25 | GrProgramDesc(const GrProgramDesc& other) : fKey(other.fKey) {} // for SkLRUCache |
26 | |
27 | bool isValid() const { return !fKey.empty(); } |
28 | |
29 | // Returns this as a uint32_t array to be used as a key in the program cache. |
30 | const uint32_t* asKey() const { |
31 | return reinterpret_cast<const uint32_t*>(fKey.begin()); |
32 | } |
33 | |
34 | // Gets the number of bytes in asKey(). It will be a 4-byte aligned value. |
35 | uint32_t keyLength() const { |
36 | SkASSERT(0 == (fKey.count() % 4)); |
37 | return fKey.count(); |
38 | } |
39 | |
40 | GrProgramDesc& operator= (const GrProgramDesc& other) { |
41 | uint32_t keyLength = other.keyLength(); |
42 | fKey.reset(SkToInt(keyLength)); |
43 | memcpy(fKey.begin(), other.fKey.begin(), keyLength); |
44 | return *this; |
45 | } |
46 | |
47 | bool operator== (const GrProgramDesc& that) const { |
48 | if (this->keyLength() != that.keyLength()) { |
49 | return false; |
50 | } |
51 | |
52 | SkASSERT(SkIsAlign4(this->keyLength())); |
53 | int l = this->keyLength() >> 2; |
54 | const uint32_t* aKey = this->asKey(); |
55 | const uint32_t* bKey = that.asKey(); |
56 | for (int i = 0; i < l; ++i) { |
57 | if (aKey[i] != bKey[i]) { |
58 | return false; |
59 | } |
60 | } |
61 | return true; |
62 | } |
63 | |
64 | bool operator!= (const GrProgramDesc& other) const { |
65 | return !(*this == other); |
66 | } |
67 | |
68 | uint32_t initialKeyLength() const { return this->header().fInitialKeyLength; } |
69 | |
70 | protected: |
71 | friend class GrDawnCaps; |
72 | friend class GrD3DCaps; |
73 | friend class GrGLCaps; |
74 | friend class GrMockCaps; |
75 | friend class GrMtlCaps; |
76 | friend class GrVkCaps; |
77 | |
78 | friend class GrGLGpu; // for ProgramCache to access BuildFromData |
79 | |
80 | // Creates an uninitialized key that must be populated by Build |
81 | GrProgramDesc() {} |
82 | |
83 | /** |
84 | * Builds a program descriptor. |
85 | * |
86 | * @param desc The built descriptor |
87 | * @param renderTarget The target of the draw |
88 | * @param programInfo Program information need to build the key |
89 | * @param caps the caps |
90 | **/ |
91 | static bool Build(GrProgramDesc*, GrRenderTarget*, const GrProgramInfo&, const GrCaps&); |
92 | |
93 | // This is strictly an OpenGL call since the other backends have additional data in their keys. |
94 | static bool BuildFromData(GrProgramDesc* desc, const void* keyData, size_t keyLength) { |
95 | if (!SkTFitsIn<int>(keyLength)) { |
96 | return false; |
97 | } |
98 | desc->fKey.reset(SkToInt(keyLength)); |
99 | memcpy(desc->fKey.begin(), keyData, keyLength); |
100 | return true; |
101 | } |
102 | |
103 | // TODO: this should be removed and converted to just data added to the key |
104 | struct { |
105 | // Set to uniquely identify any swizzling of the shader's output color(s). |
106 | uint16_t ; |
107 | uint8_t ; // Can be packed into 4 bits if required. |
108 | uint8_t ; |
109 | // Set to uniquely identify the rt's origin, or 0 if the shader does not require this info. |
110 | uint32_t : 2; |
111 | uint32_t : 1; |
112 | uint32_t : 1; |
113 | uint32_t : 1; |
114 | // This is the key size (in bytes) after core key construction. It doesn't include any |
115 | // portions added by the platform-specific backends. |
116 | uint32_t : 27; |
117 | }; |
118 | static_assert(sizeof(KeyHeader) == 8); |
119 | |
120 | const KeyHeader& () const { return *this->atOffset<KeyHeader, kHeaderOffset>(); } |
121 | |
122 | template<typename T, size_t OFFSET> T* atOffset() { |
123 | return reinterpret_cast<T*>(reinterpret_cast<intptr_t>(fKey.begin()) + OFFSET); |
124 | } |
125 | |
126 | template<typename T, size_t OFFSET> const T* atOffset() const { |
127 | return reinterpret_cast<const T*>(reinterpret_cast<intptr_t>(fKey.begin()) + OFFSET); |
128 | } |
129 | |
130 | // The key, stored in fKey, is composed of two parts: |
131 | // 1. Header struct defined above. |
132 | // 2. A Backend specific payload which includes the per-processor keys. |
133 | enum KeyOffsets { |
134 | = 0, |
135 | = SkAlign4(sizeof(KeyHeader)), |
136 | // This is the offset into the backend-specific part of the key, which includes |
137 | // per-processor keys. |
138 | kProcessorKeysOffset = kHeaderOffset + kHeaderSize, |
139 | }; |
140 | |
141 | enum { |
142 | kMaxPreallocProcessors = 8, |
143 | kIntsPerProcessor = 4, // This is an overestimate of the average effect key size. |
144 | kPreAllocSize = kHeaderOffset + kHeaderSize + |
145 | kMaxPreallocProcessors * sizeof(uint32_t) * kIntsPerProcessor, |
146 | }; |
147 | |
148 | SkSTArray<kPreAllocSize, uint8_t, true>& key() { return fKey; } |
149 | |
150 | private: |
151 | SkSTArray<kPreAllocSize, uint8_t, true> fKey; |
152 | }; |
153 | |
154 | #endif |
155 | |