• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 //
2 // Copyright 2016 The ANGLE Project Authors. All rights reserved.
3 // Use of this source code is governed by a BSD-style license that can be
4 // found in the LICENSE file.
5 //
6 // ProgramVk.cpp:
7 //    Implements the class methods for ProgramVk.
8 //
9 
10 #include "libANGLE/renderer/vulkan/ProgramVk.h"
11 
12 #include "common/debug.h"
13 #include "common/utilities.h"
14 #include "libANGLE/Context.h"
15 #include "libANGLE/ProgramLinkedResources.h"
16 #include "libANGLE/renderer/glslang_wrapper_utils.h"
17 #include "libANGLE/renderer/renderer_utils.h"
18 #include "libANGLE/renderer/vulkan/BufferVk.h"
19 #include "libANGLE/renderer/vulkan/GlslangWrapperVk.h"
20 #include "libANGLE/renderer/vulkan/TextureVk.h"
21 
22 namespace rx
23 {
24 
25 namespace
26 {
27 // Identical to Std140 encoder in all aspects, except it ignores opaque uniform types.
28 class VulkanDefaultBlockEncoder : public sh::Std140BlockEncoder
29 {
30   public:
advanceOffset(GLenum type,const std::vector<unsigned int> & arraySizes,bool isRowMajorMatrix,int arrayStride,int matrixStride)31     void advanceOffset(GLenum type,
32                        const std::vector<unsigned int> &arraySizes,
33                        bool isRowMajorMatrix,
34                        int arrayStride,
35                        int matrixStride) override
36     {
37         if (gl::IsOpaqueType(type))
38         {
39             return;
40         }
41 
42         sh::Std140BlockEncoder::advanceOffset(type, arraySizes, isRowMajorMatrix, arrayStride,
43                                               matrixStride);
44     }
45 };
46 
InitDefaultUniformBlock(const std::vector<sh::ShaderVariable> & uniforms,sh::BlockLayoutMap * blockLayoutMapOut,size_t * blockSizeOut)47 void InitDefaultUniformBlock(const std::vector<sh::ShaderVariable> &uniforms,
48                              sh::BlockLayoutMap *blockLayoutMapOut,
49                              size_t *blockSizeOut)
50 {
51     if (uniforms.empty())
52     {
53         *blockSizeOut = 0;
54         return;
55     }
56 
57     VulkanDefaultBlockEncoder blockEncoder;
58     sh::GetActiveUniformBlockInfo(uniforms, "", &blockEncoder, blockLayoutMapOut);
59 
60     size_t blockSize = blockEncoder.getCurrentOffset();
61 
62     // TODO(jmadill): I think we still need a valid block for the pipeline even if zero sized.
63     if (blockSize == 0)
64     {
65         *blockSizeOut = 0;
66         return;
67     }
68 
69     *blockSizeOut = blockSize;
70     return;
71 }
72 
73 template <typename T>
UpdateDefaultUniformBlock(GLsizei count,uint32_t arrayIndex,int componentCount,const T * v,const sh::BlockMemberInfo & layoutInfo,angle::MemoryBuffer * uniformData)74 void UpdateDefaultUniformBlock(GLsizei count,
75                                uint32_t arrayIndex,
76                                int componentCount,
77                                const T *v,
78                                const sh::BlockMemberInfo &layoutInfo,
79                                angle::MemoryBuffer *uniformData)
80 {
81     const int elementSize = sizeof(T) * componentCount;
82 
83     uint8_t *dst = uniformData->data() + layoutInfo.offset;
84     if (layoutInfo.arrayStride == 0 || layoutInfo.arrayStride == elementSize)
85     {
86         uint32_t arrayOffset = arrayIndex * layoutInfo.arrayStride;
87         uint8_t *writePtr    = dst + arrayOffset;
88         ASSERT(writePtr + (elementSize * count) <= uniformData->data() + uniformData->size());
89         memcpy(writePtr, v, elementSize * count);
90     }
91     else
92     {
93         // Have to respect the arrayStride between each element of the array.
94         int maxIndex = arrayIndex + count;
95         for (int writeIndex = arrayIndex, readIndex = 0; writeIndex < maxIndex;
96              writeIndex++, readIndex++)
97         {
98             const int arrayOffset = writeIndex * layoutInfo.arrayStride;
99             uint8_t *writePtr     = dst + arrayOffset;
100             const T *readPtr      = v + (readIndex * componentCount);
101             ASSERT(writePtr + elementSize <= uniformData->data() + uniformData->size());
102             memcpy(writePtr, readPtr, elementSize);
103         }
104     }
105 }
106 
107 template <typename T>
ReadFromDefaultUniformBlock(int componentCount,uint32_t arrayIndex,T * dst,const sh::BlockMemberInfo & layoutInfo,const angle::MemoryBuffer * uniformData)108 void ReadFromDefaultUniformBlock(int componentCount,
109                                  uint32_t arrayIndex,
110                                  T *dst,
111                                  const sh::BlockMemberInfo &layoutInfo,
112                                  const angle::MemoryBuffer *uniformData)
113 {
114     ASSERT(layoutInfo.offset != -1);
115 
116     const int elementSize = sizeof(T) * componentCount;
117     const uint8_t *source = uniformData->data() + layoutInfo.offset;
118 
119     if (layoutInfo.arrayStride == 0 || layoutInfo.arrayStride == elementSize)
120     {
121         const uint8_t *readPtr = source + arrayIndex * layoutInfo.arrayStride;
122         memcpy(dst, readPtr, elementSize);
123     }
124     else
125     {
126         // Have to respect the arrayStride between each element of the array.
127         const int arrayOffset  = arrayIndex * layoutInfo.arrayStride;
128         const uint8_t *readPtr = source + arrayOffset;
129         memcpy(dst, readPtr, elementSize);
130     }
131 }
132 
133 class Std140BlockLayoutEncoderFactory : public gl::CustomBlockLayoutEncoderFactory
134 {
135   public:
makeEncoder()136     sh::BlockLayoutEncoder *makeEncoder() override { return new sh::Std140BlockEncoder(); }
137 };
138 }  // anonymous namespace
139 
140 // ProgramVk implementation.
ProgramVk(const gl::ProgramState & state)141 ProgramVk::ProgramVk(const gl::ProgramState &state) : ProgramImpl(state)
142 {
143     GlslangWrapperVk::ResetGlslangProgramInterfaceInfo(&mGlslangProgramInterfaceInfo);
144     mExecutable.setProgram(this);
145 }
146 
147 ProgramVk::~ProgramVk() = default;
148 
destroy(const gl::Context * context)149 void ProgramVk::destroy(const gl::Context *context)
150 {
151     ContextVk *contextVk = vk::GetImpl(context);
152     reset(contextVk);
153 }
154 
reset(ContextVk * contextVk)155 void ProgramVk::reset(ContextVk *contextVk)
156 {
157     mOriginalShaderInfo.release(contextVk);
158 
159     GlslangWrapperVk::ResetGlslangProgramInterfaceInfo(&mGlslangProgramInterfaceInfo);
160 
161     mExecutable.reset(contextVk);
162 }
163 
load(const gl::Context * context,gl::BinaryInputStream * stream,gl::InfoLog & infoLog)164 std::unique_ptr<rx::LinkEvent> ProgramVk::load(const gl::Context *context,
165                                                gl::BinaryInputStream *stream,
166                                                gl::InfoLog &infoLog)
167 {
168     ContextVk *contextVk = vk::GetImpl(context);
169     gl::ShaderMap<size_t> requiredBufferSize;
170     requiredBufferSize.fill(0);
171 
172     reset(contextVk);
173 
174     mOriginalShaderInfo.load(stream);
175     mExecutable.load(stream);
176 
177     // Deserializes the uniformLayout data of mDefaultUniformBlocks
178     for (gl::ShaderType shaderType : gl::AllShaderTypes())
179     {
180         const size_t uniformCount = stream->readInt<size_t>();
181         for (unsigned int uniformIndex = 0; uniformIndex < uniformCount; ++uniformIndex)
182         {
183             sh::BlockMemberInfo blockInfo;
184             gl::LoadBlockMemberInfo(stream, &blockInfo);
185             mDefaultUniformBlocks[shaderType].uniformLayout.push_back(blockInfo);
186         }
187     }
188 
189     // Deserializes required uniform block memory sizes
190     for (gl::ShaderType shaderType : gl::AllShaderTypes())
191     {
192         requiredBufferSize[shaderType] = stream->readInt<size_t>();
193     }
194 
195     // Initialize and resize the mDefaultUniformBlocks' memory
196     angle::Result status = resizeUniformBlockMemory(contextVk, requiredBufferSize);
197     if (status != angle::Result::Continue)
198     {
199         return std::make_unique<LinkEventDone>(status);
200     }
201 
202     status = mExecutable.createPipelineLayout(contextVk, mState.getExecutable(), nullptr);
203     return std::make_unique<LinkEventDone>(status);
204 }
205 
save(const gl::Context * context,gl::BinaryOutputStream * stream)206 void ProgramVk::save(const gl::Context *context, gl::BinaryOutputStream *stream)
207 {
208     mOriginalShaderInfo.save(stream);
209     mExecutable.save(stream);
210 
211     // Serializes the uniformLayout data of mDefaultUniformBlocks
212     for (gl::ShaderType shaderType : gl::AllShaderTypes())
213     {
214         const size_t uniformCount = mDefaultUniformBlocks[shaderType].uniformLayout.size();
215         stream->writeInt(uniformCount);
216         for (unsigned int uniformIndex = 0; uniformIndex < uniformCount; ++uniformIndex)
217         {
218             sh::BlockMemberInfo &blockInfo =
219                 mDefaultUniformBlocks[shaderType].uniformLayout[uniformIndex];
220             gl::WriteBlockMemberInfo(stream, blockInfo);
221         }
222     }
223 
224     // Serializes required uniform block memory sizes
225     for (gl::ShaderType shaderType : gl::AllShaderTypes())
226     {
227         stream->writeInt(mDefaultUniformBlocks[shaderType].uniformData.size());
228     }
229 }
230 
setBinaryRetrievableHint(bool retrievable)231 void ProgramVk::setBinaryRetrievableHint(bool retrievable)
232 {
233     // Nothing to do here yet.
234 }
235 
setSeparable(bool separable)236 void ProgramVk::setSeparable(bool separable)
237 {
238     // Nothing to do here yet.
239 }
240 
241 // TODO: http://anglebug.com/3570: Move/Copy all of the necessary information into
242 // the ProgramExecutable, so this function can be removed.
fillProgramStateMap(gl::ShaderMap<const gl::ProgramState * > * programStatesOut)243 void ProgramVk::fillProgramStateMap(gl::ShaderMap<const gl::ProgramState *> *programStatesOut)
244 {
245     for (gl::ShaderType shaderType : gl::AllShaderTypes())
246     {
247         (*programStatesOut)[shaderType] = nullptr;
248         if (mState.getExecutable().hasLinkedShaderStage(shaderType))
249         {
250             (*programStatesOut)[shaderType] = &mState;
251         }
252     }
253 }
254 
link(const gl::Context * context,const gl::ProgramLinkedResources & resources,gl::InfoLog & infoLog,const gl::ProgramMergedVaryings & mergedVaryings)255 std::unique_ptr<LinkEvent> ProgramVk::link(const gl::Context *context,
256                                            const gl::ProgramLinkedResources &resources,
257                                            gl::InfoLog &infoLog,
258                                            const gl::ProgramMergedVaryings &mergedVaryings)
259 {
260     ANGLE_TRACE_EVENT0("gpu.angle", "ProgramVk::link");
261 
262     ContextVk *contextVk = vk::GetImpl(context);
263     // Link resources before calling GetShaderSource to make sure they are ready for the set/binding
264     // assignment done in that function.
265     linkResources(resources);
266 
267     reset(contextVk);
268     mExecutable.clearVariableInfoMap();
269 
270     // Gather variable info and compiled SPIR-V binaries.
271     gl::ShaderMap<const angle::spirv::Blob *> spirvBlobs;
272     GlslangWrapperVk::GetShaderCode(contextVk->getFeatures(), mState, resources,
273                                     &mGlslangProgramInterfaceInfo, &spirvBlobs,
274                                     &mExecutable.mVariableInfoMap);
275 
276     // Compile the shaders.
277     angle::Result status = mOriginalShaderInfo.initShaders(
278         mState.getExecutable().getLinkedShaderStages(), spirvBlobs, mExecutable.mVariableInfoMap);
279     if (status != angle::Result::Continue)
280     {
281         return std::make_unique<LinkEventDone>(status);
282     }
283 
284     status = initDefaultUniformBlocks(context);
285     if (status != angle::Result::Continue)
286     {
287         return std::make_unique<LinkEventDone>(status);
288     }
289 
290     if (contextVk->getFeatures().enablePrecisionQualifiers.enabled)
291     {
292         mExecutable.resolvePrecisionMismatch(mergedVaryings);
293     }
294 
295     // TODO(jie.a.chen@intel.com): Parallelize linking.
296     // http://crbug.com/849576
297     status = mExecutable.createPipelineLayout(contextVk, mState.getExecutable(), nullptr);
298     return std::make_unique<LinkEventDone>(status);
299 }
300 
linkResources(const gl::ProgramLinkedResources & resources)301 void ProgramVk::linkResources(const gl::ProgramLinkedResources &resources)
302 {
303     Std140BlockLayoutEncoderFactory std140EncoderFactory;
304     gl::ProgramLinkedResourcesLinker linker(&std140EncoderFactory);
305 
306     linker.linkResources(mState, resources);
307 }
308 
initDefaultUniformBlocks(const gl::Context * glContext)309 angle::Result ProgramVk::initDefaultUniformBlocks(const gl::Context *glContext)
310 {
311     ContextVk *contextVk = vk::GetImpl(glContext);
312 
313     // Process vertex and fragment uniforms into std140 packing.
314     gl::ShaderMap<sh::BlockLayoutMap> layoutMap;
315     gl::ShaderMap<size_t> requiredBufferSize;
316     requiredBufferSize.fill(0);
317 
318     generateUniformLayoutMapping(layoutMap, requiredBufferSize);
319     initDefaultUniformLayoutMapping(layoutMap);
320 
321     // All uniform initializations are complete, now resize the buffers accordingly and return
322     return resizeUniformBlockMemory(contextVk, requiredBufferSize);
323 }
324 
generateUniformLayoutMapping(gl::ShaderMap<sh::BlockLayoutMap> & layoutMap,gl::ShaderMap<size_t> & requiredBufferSize)325 void ProgramVk::generateUniformLayoutMapping(gl::ShaderMap<sh::BlockLayoutMap> &layoutMap,
326                                              gl::ShaderMap<size_t> &requiredBufferSize)
327 {
328     const gl::ProgramExecutable &glExecutable = mState.getExecutable();
329 
330     for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
331     {
332         gl::Shader *shader = mState.getAttachedShader(shaderType);
333 
334         if (shader)
335         {
336             const std::vector<sh::ShaderVariable> &uniforms = shader->getUniforms();
337             InitDefaultUniformBlock(uniforms, &layoutMap[shaderType],
338                                     &requiredBufferSize[shaderType]);
339         }
340     }
341 }
342 
initDefaultUniformLayoutMapping(gl::ShaderMap<sh::BlockLayoutMap> & layoutMap)343 void ProgramVk::initDefaultUniformLayoutMapping(gl::ShaderMap<sh::BlockLayoutMap> &layoutMap)
344 {
345     // Init the default block layout info.
346     const auto &uniforms                      = mState.getUniforms();
347     const gl::ProgramExecutable &glExecutable = mState.getExecutable();
348 
349     for (const gl::VariableLocation &location : mState.getUniformLocations())
350     {
351         gl::ShaderMap<sh::BlockMemberInfo> layoutInfo;
352 
353         if (location.used() && !location.ignored)
354         {
355             const auto &uniform = uniforms[location.index];
356             if (uniform.isInDefaultBlock() && !uniform.isSampler() && !uniform.isImage() &&
357                 !uniform.isFragmentInOut)
358             {
359                 std::string uniformName = uniform.name;
360                 if (uniform.isArray())
361                 {
362                     // Gets the uniform name without the [0] at the end.
363                     uniformName = gl::StripLastArrayIndex(uniformName);
364                     ASSERT(uniformName.size() != uniform.name.size());
365                 }
366 
367                 bool found = false;
368 
369                 for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
370                 {
371                     auto it = layoutMap[shaderType].find(uniformName);
372                     if (it != layoutMap[shaderType].end())
373                     {
374                         found                  = true;
375                         layoutInfo[shaderType] = it->second;
376                     }
377                 }
378 
379                 ASSERT(found);
380             }
381         }
382 
383         for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
384         {
385             mDefaultUniformBlocks[shaderType].uniformLayout.push_back(layoutInfo[shaderType]);
386         }
387     }
388 }
389 
resizeUniformBlockMemory(ContextVk * contextVk,gl::ShaderMap<size_t> & requiredBufferSize)390 angle::Result ProgramVk::resizeUniformBlockMemory(ContextVk *contextVk,
391                                                   gl::ShaderMap<size_t> &requiredBufferSize)
392 {
393     const gl::ProgramExecutable &glExecutable = mState.getExecutable();
394 
395     for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
396     {
397         if (requiredBufferSize[shaderType] > 0)
398         {
399             if (!mDefaultUniformBlocks[shaderType].uniformData.resize(
400                     requiredBufferSize[shaderType]))
401             {
402                 ANGLE_VK_CHECK(contextVk, false, VK_ERROR_OUT_OF_HOST_MEMORY);
403             }
404 
405             // Initialize uniform buffer memory to zero by default.
406             mDefaultUniformBlocks[shaderType].uniformData.fill(0);
407             mDefaultUniformBlocksDirty.set(shaderType);
408         }
409     }
410 
411     return angle::Result::Continue;
412 }
413 
validate(const gl::Caps & caps,gl::InfoLog * infoLog)414 GLboolean ProgramVk::validate(const gl::Caps &caps, gl::InfoLog *infoLog)
415 {
416     // No-op. The spec is very vague about the behavior of validation.
417     return GL_TRUE;
418 }
419 
420 template <typename T>
setUniformImpl(GLint location,GLsizei count,const T * v,GLenum entryPointType)421 void ProgramVk::setUniformImpl(GLint location, GLsizei count, const T *v, GLenum entryPointType)
422 {
423     const gl::VariableLocation &locationInfo  = mState.getUniformLocations()[location];
424     const gl::LinkedUniform &linkedUniform    = mState.getUniforms()[locationInfo.index];
425     const gl::ProgramExecutable &glExecutable = mState.getExecutable();
426 
427     ASSERT(!linkedUniform.isSampler());
428 
429     if (linkedUniform.typeInfo->type == entryPointType)
430     {
431         for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
432         {
433             DefaultUniformBlock &uniformBlock     = mDefaultUniformBlocks[shaderType];
434             const sh::BlockMemberInfo &layoutInfo = uniformBlock.uniformLayout[location];
435 
436             // Assume an offset of -1 means the block is unused.
437             if (layoutInfo.offset == -1)
438             {
439                 continue;
440             }
441 
442             const GLint componentCount = linkedUniform.typeInfo->componentCount;
443             UpdateDefaultUniformBlock(count, locationInfo.arrayIndex, componentCount, v, layoutInfo,
444                                       &uniformBlock.uniformData);
445             mDefaultUniformBlocksDirty.set(shaderType);
446         }
447     }
448     else
449     {
450         for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
451         {
452             DefaultUniformBlock &uniformBlock     = mDefaultUniformBlocks[shaderType];
453             const sh::BlockMemberInfo &layoutInfo = uniformBlock.uniformLayout[location];
454 
455             // Assume an offset of -1 means the block is unused.
456             if (layoutInfo.offset == -1)
457             {
458                 continue;
459             }
460 
461             const GLint componentCount = linkedUniform.typeInfo->componentCount;
462 
463             ASSERT(linkedUniform.typeInfo->type == gl::VariableBoolVectorType(entryPointType));
464 
465             GLint initialArrayOffset =
466                 locationInfo.arrayIndex * layoutInfo.arrayStride + layoutInfo.offset;
467             for (GLint i = 0; i < count; i++)
468             {
469                 GLint elementOffset = i * layoutInfo.arrayStride + initialArrayOffset;
470                 GLint *dst =
471                     reinterpret_cast<GLint *>(uniformBlock.uniformData.data() + elementOffset);
472                 const T *source = v + i * componentCount;
473 
474                 for (int c = 0; c < componentCount; c++)
475                 {
476                     dst[c] = (source[c] == static_cast<T>(0)) ? GL_FALSE : GL_TRUE;
477                 }
478             }
479 
480             mDefaultUniformBlocksDirty.set(shaderType);
481         }
482     }
483 }
484 
485 template <typename T>
getUniformImpl(GLint location,T * v,GLenum entryPointType) const486 void ProgramVk::getUniformImpl(GLint location, T *v, GLenum entryPointType) const
487 {
488     const gl::VariableLocation &locationInfo = mState.getUniformLocations()[location];
489     const gl::LinkedUniform &linkedUniform   = mState.getUniforms()[locationInfo.index];
490 
491     ASSERT(!linkedUniform.isSampler() && !linkedUniform.isImage());
492 
493     const gl::ShaderType shaderType = linkedUniform.getFirstShaderTypeWhereActive();
494     ASSERT(shaderType != gl::ShaderType::InvalidEnum);
495 
496     const DefaultUniformBlock &uniformBlock = mDefaultUniformBlocks[shaderType];
497     const sh::BlockMemberInfo &layoutInfo   = uniformBlock.uniformLayout[location];
498 
499     ASSERT(linkedUniform.typeInfo->componentType == entryPointType ||
500            linkedUniform.typeInfo->componentType == gl::VariableBoolVectorType(entryPointType));
501 
502     if (gl::IsMatrixType(linkedUniform.type))
503     {
504         const uint8_t *ptrToElement = uniformBlock.uniformData.data() + layoutInfo.offset +
505                                       (locationInfo.arrayIndex * layoutInfo.arrayStride);
506         GetMatrixUniform(linkedUniform.type, v, reinterpret_cast<const T *>(ptrToElement), false);
507     }
508     else
509     {
510         ReadFromDefaultUniformBlock(linkedUniform.typeInfo->componentCount, locationInfo.arrayIndex,
511                                     v, layoutInfo, &uniformBlock.uniformData);
512     }
513 }
514 
setUniform1fv(GLint location,GLsizei count,const GLfloat * v)515 void ProgramVk::setUniform1fv(GLint location, GLsizei count, const GLfloat *v)
516 {
517     setUniformImpl(location, count, v, GL_FLOAT);
518 }
519 
setUniform2fv(GLint location,GLsizei count,const GLfloat * v)520 void ProgramVk::setUniform2fv(GLint location, GLsizei count, const GLfloat *v)
521 {
522     setUniformImpl(location, count, v, GL_FLOAT_VEC2);
523 }
524 
setUniform3fv(GLint location,GLsizei count,const GLfloat * v)525 void ProgramVk::setUniform3fv(GLint location, GLsizei count, const GLfloat *v)
526 {
527     setUniformImpl(location, count, v, GL_FLOAT_VEC3);
528 }
529 
setUniform4fv(GLint location,GLsizei count,const GLfloat * v)530 void ProgramVk::setUniform4fv(GLint location, GLsizei count, const GLfloat *v)
531 {
532     setUniformImpl(location, count, v, GL_FLOAT_VEC4);
533 }
534 
setUniform1iv(GLint location,GLsizei count,const GLint * v)535 void ProgramVk::setUniform1iv(GLint location, GLsizei count, const GLint *v)
536 {
537     const gl::VariableLocation &locationInfo = mState.getUniformLocations()[location];
538     const gl::LinkedUniform &linkedUniform   = mState.getUniforms()[locationInfo.index];
539     if (linkedUniform.isSampler())
540     {
541         // We could potentially cache some indexing here. For now this is a no-op since the mapping
542         // is handled entirely in ContextVk.
543         return;
544     }
545 
546     setUniformImpl(location, count, v, GL_INT);
547 }
548 
setUniform2iv(GLint location,GLsizei count,const GLint * v)549 void ProgramVk::setUniform2iv(GLint location, GLsizei count, const GLint *v)
550 {
551     setUniformImpl(location, count, v, GL_INT_VEC2);
552 }
553 
setUniform3iv(GLint location,GLsizei count,const GLint * v)554 void ProgramVk::setUniform3iv(GLint location, GLsizei count, const GLint *v)
555 {
556     setUniformImpl(location, count, v, GL_INT_VEC3);
557 }
558 
setUniform4iv(GLint location,GLsizei count,const GLint * v)559 void ProgramVk::setUniform4iv(GLint location, GLsizei count, const GLint *v)
560 {
561     setUniformImpl(location, count, v, GL_INT_VEC4);
562 }
563 
setUniform1uiv(GLint location,GLsizei count,const GLuint * v)564 void ProgramVk::setUniform1uiv(GLint location, GLsizei count, const GLuint *v)
565 {
566     setUniformImpl(location, count, v, GL_UNSIGNED_INT);
567 }
568 
setUniform2uiv(GLint location,GLsizei count,const GLuint * v)569 void ProgramVk::setUniform2uiv(GLint location, GLsizei count, const GLuint *v)
570 {
571     setUniformImpl(location, count, v, GL_UNSIGNED_INT_VEC2);
572 }
573 
setUniform3uiv(GLint location,GLsizei count,const GLuint * v)574 void ProgramVk::setUniform3uiv(GLint location, GLsizei count, const GLuint *v)
575 {
576     setUniformImpl(location, count, v, GL_UNSIGNED_INT_VEC3);
577 }
578 
setUniform4uiv(GLint location,GLsizei count,const GLuint * v)579 void ProgramVk::setUniform4uiv(GLint location, GLsizei count, const GLuint *v)
580 {
581     setUniformImpl(location, count, v, GL_UNSIGNED_INT_VEC4);
582 }
583 
584 template <int cols, int rows>
setUniformMatrixfv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)585 void ProgramVk::setUniformMatrixfv(GLint location,
586                                    GLsizei count,
587                                    GLboolean transpose,
588                                    const GLfloat *value)
589 {
590     const gl::VariableLocation &locationInfo  = mState.getUniformLocations()[location];
591     const gl::LinkedUniform &linkedUniform    = mState.getUniforms()[locationInfo.index];
592     const gl::ProgramExecutable &glExecutable = mState.getExecutable();
593 
594     for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
595     {
596         DefaultUniformBlock &uniformBlock     = mDefaultUniformBlocks[shaderType];
597         const sh::BlockMemberInfo &layoutInfo = uniformBlock.uniformLayout[location];
598 
599         // Assume an offset of -1 means the block is unused.
600         if (layoutInfo.offset == -1)
601         {
602             continue;
603         }
604 
605         SetFloatUniformMatrixGLSL<cols, rows>::Run(
606             locationInfo.arrayIndex, linkedUniform.getArraySizeProduct(), count, transpose, value,
607             uniformBlock.uniformData.data() + layoutInfo.offset);
608 
609         mDefaultUniformBlocksDirty.set(shaderType);
610     }
611 }
612 
setUniformMatrix2fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)613 void ProgramVk::setUniformMatrix2fv(GLint location,
614                                     GLsizei count,
615                                     GLboolean transpose,
616                                     const GLfloat *value)
617 {
618     setUniformMatrixfv<2, 2>(location, count, transpose, value);
619 }
620 
setUniformMatrix3fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)621 void ProgramVk::setUniformMatrix3fv(GLint location,
622                                     GLsizei count,
623                                     GLboolean transpose,
624                                     const GLfloat *value)
625 {
626     setUniformMatrixfv<3, 3>(location, count, transpose, value);
627 }
628 
setUniformMatrix4fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)629 void ProgramVk::setUniformMatrix4fv(GLint location,
630                                     GLsizei count,
631                                     GLboolean transpose,
632                                     const GLfloat *value)
633 {
634     setUniformMatrixfv<4, 4>(location, count, transpose, value);
635 }
636 
setUniformMatrix2x3fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)637 void ProgramVk::setUniformMatrix2x3fv(GLint location,
638                                       GLsizei count,
639                                       GLboolean transpose,
640                                       const GLfloat *value)
641 {
642     setUniformMatrixfv<2, 3>(location, count, transpose, value);
643 }
644 
setUniformMatrix3x2fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)645 void ProgramVk::setUniformMatrix3x2fv(GLint location,
646                                       GLsizei count,
647                                       GLboolean transpose,
648                                       const GLfloat *value)
649 {
650     setUniformMatrixfv<3, 2>(location, count, transpose, value);
651 }
652 
setUniformMatrix2x4fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)653 void ProgramVk::setUniformMatrix2x4fv(GLint location,
654                                       GLsizei count,
655                                       GLboolean transpose,
656                                       const GLfloat *value)
657 {
658     setUniformMatrixfv<2, 4>(location, count, transpose, value);
659 }
660 
setUniformMatrix4x2fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)661 void ProgramVk::setUniformMatrix4x2fv(GLint location,
662                                       GLsizei count,
663                                       GLboolean transpose,
664                                       const GLfloat *value)
665 {
666     setUniformMatrixfv<4, 2>(location, count, transpose, value);
667 }
668 
setUniformMatrix3x4fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)669 void ProgramVk::setUniformMatrix3x4fv(GLint location,
670                                       GLsizei count,
671                                       GLboolean transpose,
672                                       const GLfloat *value)
673 {
674     setUniformMatrixfv<3, 4>(location, count, transpose, value);
675 }
676 
setUniformMatrix4x3fv(GLint location,GLsizei count,GLboolean transpose,const GLfloat * value)677 void ProgramVk::setUniformMatrix4x3fv(GLint location,
678                                       GLsizei count,
679                                       GLboolean transpose,
680                                       const GLfloat *value)
681 {
682     setUniformMatrixfv<4, 3>(location, count, transpose, value);
683 }
684 
getUniformfv(const gl::Context * context,GLint location,GLfloat * params) const685 void ProgramVk::getUniformfv(const gl::Context *context, GLint location, GLfloat *params) const
686 {
687     getUniformImpl(location, params, GL_FLOAT);
688 }
689 
getUniformiv(const gl::Context * context,GLint location,GLint * params) const690 void ProgramVk::getUniformiv(const gl::Context *context, GLint location, GLint *params) const
691 {
692     getUniformImpl(location, params, GL_INT);
693 }
694 
getUniformuiv(const gl::Context * context,GLint location,GLuint * params) const695 void ProgramVk::getUniformuiv(const gl::Context *context, GLint location, GLuint *params) const
696 {
697     getUniformImpl(location, params, GL_UNSIGNED_INT);
698 }
699 
calcUniformUpdateRequiredSpace(ContextVk * contextVk,const gl::ProgramExecutable & glExecutable,gl::ShaderMap<VkDeviceSize> & uniformOffsets) const700 size_t ProgramVk::calcUniformUpdateRequiredSpace(ContextVk *contextVk,
701                                                  const gl::ProgramExecutable &glExecutable,
702                                                  gl::ShaderMap<VkDeviceSize> &uniformOffsets) const
703 {
704     size_t requiredSpace = 0;
705     for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
706     {
707         if (mDefaultUniformBlocksDirty[shaderType])
708         {
709             uniformOffsets[shaderType] = requiredSpace;
710             requiredSpace += getDefaultUniformAlignedSize(contextVk, shaderType);
711         }
712     }
713     return requiredSpace;
714 }
715 
updateUniforms(ContextVk * contextVk)716 angle::Result ProgramVk::updateUniforms(ContextVk *contextVk)
717 {
718     ASSERT(hasDirtyUniforms());
719 
720     bool anyNewBufferAllocated                = false;
721     uint8_t *bufferData                       = nullptr;
722     VkDeviceSize bufferOffset                 = 0;
723     uint32_t offsetIndex                      = 0;
724     const gl::ProgramExecutable &glExecutable = mState.getExecutable();
725     gl::ShaderMap<VkDeviceSize> offsets;  // offset to the beginning of bufferData
726     size_t requiredSpace;
727 
728     // We usually only update uniform data for shader stages that are actually dirty. But when the
729     // buffer for uniform data have switched, because all shader stages are using the same buffer,
730     // we then must update uniform data for all shader stages to keep all shader stages' uniform
731     // data in the same buffer.
732     requiredSpace = calcUniformUpdateRequiredSpace(contextVk, glExecutable, offsets);
733     ASSERT(requiredSpace > 0);
734 
735     // Allocate space from dynamicBuffer. Always try to allocate from the current buffer first.
736     // If that failed, we deal with fall out and try again.
737     vk::DynamicBuffer *defaultUniformStorage = contextVk->getDefaultUniformStorage();
738     if (!defaultUniformStorage->allocateFromCurrentBuffer(requiredSpace, &bufferData,
739                                                           &bufferOffset))
740     {
741         for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
742         {
743             if (!mDefaultUniformBlocks[shaderType].uniformData.empty())
744             {
745                 mDefaultUniformBlocksDirty.set(shaderType);
746             }
747         }
748 
749         requiredSpace = calcUniformUpdateRequiredSpace(contextVk, glExecutable, offsets);
750         ANGLE_TRY(defaultUniformStorage->allocate(contextVk, requiredSpace, &bufferData, nullptr,
751                                                   &bufferOffset, &anyNewBufferAllocated));
752     }
753 
754     // Update buffer memory by immediate mapping. This immediate update only works once.
755     for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
756     {
757         if (mDefaultUniformBlocksDirty[shaderType])
758         {
759             const angle::MemoryBuffer &uniformData = mDefaultUniformBlocks[shaderType].uniformData;
760             memcpy(&bufferData[offsets[shaderType]], uniformData.data(), uniformData.size());
761             mExecutable.mDynamicUniformDescriptorOffsets[offsetIndex] =
762                 static_cast<uint32_t>(bufferOffset + offsets[shaderType]);
763             mDefaultUniformBlocksDirty.reset(shaderType);
764         }
765         ++offsetIndex;
766     }
767     ANGLE_TRY(defaultUniformStorage->flush(contextVk));
768 
769     vk::BufferHelper *defaultUniformBuffer = defaultUniformStorage->getCurrentBuffer();
770     if (mExecutable.getCurrentDefaultUniformBufferSerial() !=
771         defaultUniformBuffer->getBufferSerial())
772     {
773         // We need to reinitialize the descriptor sets if we newly allocated buffers since we can't
774         // modify the descriptor sets once initialized.
775         vk::UniformsAndXfbDescriptorDesc defaultUniformsDesc;
776         vk::UniformsAndXfbDescriptorDesc *uniformsAndXfbBufferDesc;
777 
778         if (glExecutable.hasTransformFeedbackOutput())
779         {
780             const gl::State &glState = contextVk->getState();
781             TransformFeedbackVk *transformFeedbackVk =
782                 vk::GetImpl(glState.getCurrentTransformFeedback());
783             uniformsAndXfbBufferDesc = &transformFeedbackVk->getTransformFeedbackDesc();
784             uniformsAndXfbBufferDesc->updateDefaultUniformBuffer(
785                 defaultUniformBuffer->getBufferSerial());
786         }
787         else
788         {
789             defaultUniformsDesc.updateDefaultUniformBuffer(defaultUniformBuffer->getBufferSerial());
790             uniformsAndXfbBufferDesc = &defaultUniformsDesc;
791         }
792 
793         bool newDescriptorSetAllocated;
794         ANGLE_TRY(mExecutable.allocUniformAndXfbDescriptorSet(contextVk, *uniformsAndXfbBufferDesc,
795                                                               &newDescriptorSetAllocated));
796         if (newDescriptorSetAllocated)
797         {
798             // Update the descriptor set with the bufferInfo
799             for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
800             {
801                 mExecutable.updateDefaultUniformsDescriptorSet(
802                     shaderType, mDefaultUniformBlocks[shaderType], defaultUniformBuffer, contextVk);
803             }
804             mExecutable.updateTransformFeedbackDescriptorSetImpl(mState, contextVk);
805         }
806     }
807 
808     return angle::Result::Continue;
809 }
810 
setAllDefaultUniformsDirty()811 void ProgramVk::setAllDefaultUniformsDirty()
812 {
813     const gl::ProgramExecutable &glExecutable = mState.getExecutable();
814     for (const gl::ShaderType shaderType : glExecutable.getLinkedShaderStages())
815     {
816         setShaderUniformDirtyBit(shaderType);
817     }
818 }
819 
onProgramBind()820 void ProgramVk::onProgramBind()
821 {
822     // Because all programs share default uniform buffers, when we switch programs, we have to
823     // re-update all uniform data. We could do more tracking to avoid update if the context's
824     // current uniform buffer is still the same buffer we last time used and buffer has not been
825     // recycled. But statistics gathered on gfxbench shows that app always update uniform data on
826     // program bind anyway, so not really worth it to add more tracking logic here.
827     setAllDefaultUniformsDirty();
828 }
829 }  // namespace rx
830