1 /*
2 * Copyright 2016 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 #include "vk/GrVkProgramBuilder.h"
9
10 #include "vk/GrVkGpu.h"
11 #include "vk/GrVkRenderPass.h"
12 #include "vk/GrVkProgram.h"
13
CreateProgram(GrVkGpu * gpu,const DrawArgs & args,GrPrimitiveType primitiveType,const GrVkRenderPass & renderPass)14 GrVkProgram* GrVkProgramBuilder::CreateProgram(GrVkGpu* gpu,
15 const DrawArgs& args,
16 GrPrimitiveType primitiveType,
17 const GrVkRenderPass& renderPass) {
18 // create a builder. This will be handed off to effects so they can use it to add
19 // uniforms, varyings, textures, etc
20 GrVkProgramBuilder builder(gpu, args);
21
22 GrGLSLExpr4 inputColor;
23 GrGLSLExpr4 inputCoverage;
24
25 if (!builder.emitAndInstallProcs(&inputColor,
26 &inputCoverage,
27 gpu->vkCaps().maxSampledTextures())) {
28 builder.cleanupFragmentProcessors();
29 return nullptr;
30 }
31
32 return builder.finalize(args, primitiveType, renderPass);
33 }
34
GrVkProgramBuilder(GrVkGpu * gpu,const DrawArgs & args)35 GrVkProgramBuilder::GrVkProgramBuilder(GrVkGpu* gpu, const DrawArgs& args)
36 : INHERITED(args)
37 , fGpu(gpu)
38 , fVaryingHandler(this)
39 , fUniformHandler(this) {
40 }
41
caps() const42 const GrCaps* GrVkProgramBuilder::caps() const {
43 return fGpu->caps();
44 }
glslCaps() const45 const GrGLSLCaps* GrVkProgramBuilder::glslCaps() const {
46 return fGpu->vkCaps().glslCaps();
47 }
48
finalizeFragmentOutputColor(GrGLSLShaderVar & outputColor)49 void GrVkProgramBuilder::finalizeFragmentOutputColor(GrGLSLShaderVar& outputColor) {
50 outputColor.setLayoutQualifier("location = 0");
51 }
52
emitSamplers(const GrProcessor & processor,GrGLSLTextureSampler::TextureSamplerArray * outSamplers)53 void GrVkProgramBuilder::emitSamplers(const GrProcessor& processor,
54 GrGLSLTextureSampler::TextureSamplerArray* outSamplers) {
55 int numTextures = processor.numTextures();
56 UniformHandle* localSamplerUniforms = fSamplerUniforms.push_back_n(numTextures);
57 SkString name;
58 for (int t = 0; t < numTextures; ++t) {
59 name.printf("%d", t);
60 localSamplerUniforms[t] =
61 fUniformHandler.addUniform(kFragment_GrShaderFlag,
62 kSampler2D_GrSLType, kDefault_GrSLPrecision,
63 name.c_str());
64 outSamplers->emplace_back(localSamplerUniforms[t], processor.textureAccess(t));
65 }
66 }
67
visibility_to_vk_stage_flags(uint32_t visibility)68 VkShaderStageFlags visibility_to_vk_stage_flags(uint32_t visibility) {
69 VkShaderStageFlags flags = 0;
70
71 if (visibility & kVertex_GrShaderFlag) {
72 flags |= VK_SHADER_STAGE_VERTEX_BIT;
73 }
74 if (visibility & kGeometry_GrShaderFlag) {
75 flags |= VK_SHADER_STAGE_GEOMETRY_BIT;
76 }
77 if (visibility & kFragment_GrShaderFlag) {
78 flags |= VK_SHADER_STAGE_FRAGMENT_BIT;
79 }
80 return flags;
81 }
82
vk_shader_stage_to_shaderc_kind(VkShaderStageFlagBits stage)83 shaderc_shader_kind vk_shader_stage_to_shaderc_kind(VkShaderStageFlagBits stage) {
84 if (VK_SHADER_STAGE_VERTEX_BIT == stage) {
85 return shaderc_glsl_vertex_shader;
86 }
87 SkASSERT(VK_SHADER_STAGE_FRAGMENT_BIT == stage);
88 return shaderc_glsl_fragment_shader;
89 }
90
CreateVkShaderModule(const GrVkGpu * gpu,VkShaderStageFlagBits stage,const GrGLSLShaderBuilder & builder,VkShaderModule * shaderModule,VkPipelineShaderStageCreateInfo * stageInfo)91 bool GrVkProgramBuilder::CreateVkShaderModule(const GrVkGpu* gpu,
92 VkShaderStageFlagBits stage,
93 const GrGLSLShaderBuilder& builder,
94 VkShaderModule* shaderModule,
95 VkPipelineShaderStageCreateInfo* stageInfo) {
96 SkString shaderString;
97 for (int i = 0; i < builder.fCompilerStrings.count(); ++i) {
98 if (builder.fCompilerStrings[i]) {
99 shaderString.append(builder.fCompilerStrings[i]);
100 shaderString.append("\n");
101 }
102 }
103
104 shaderc_compiler_t compiler = gpu->shadercCompiler();
105
106 shaderc_compile_options_t options = shaderc_compile_options_initialize();
107 shaderc_compile_options_set_forced_version_profile(options, 140, shaderc_profile_none);
108
109 shaderc_shader_kind shadercStage = vk_shader_stage_to_shaderc_kind(stage);
110 shaderc_compilation_result_t result = shaderc_compile_into_spv(compiler,
111 shaderString.c_str(),
112 strlen(shaderString.c_str()),
113 shadercStage,
114 "shader",
115 "main",
116 options);
117 shaderc_compile_options_release(options);
118 #ifdef SK_DEBUG
119 if (shaderc_result_get_num_errors(result)) {
120 SkDebugf("%s\n", shaderString.c_str());
121 SkDebugf("%s\n", shaderc_result_get_error_message(result));
122 return false;
123 }
124 #endif
125
126 VkShaderModuleCreateInfo moduleCreateInfo;
127 memset(&moduleCreateInfo, 0, sizeof(VkShaderModuleCreateInfo));
128 moduleCreateInfo.sType = VK_STRUCTURE_TYPE_SHADER_MODULE_CREATE_INFO;
129 moduleCreateInfo.pNext = nullptr;
130 moduleCreateInfo.flags = 0;
131 moduleCreateInfo.codeSize = shaderc_result_get_length(result);
132 moduleCreateInfo.pCode = (const uint32_t*)shaderc_result_get_bytes(result);
133
134 VkResult err = GR_VK_CALL(gpu->vkInterface(), CreateShaderModule(gpu->device(),
135 &moduleCreateInfo,
136 nullptr,
137 shaderModule));
138 shaderc_result_release(result);
139 if (err) {
140 return false;
141 }
142
143 memset(stageInfo, 0, sizeof(VkPipelineShaderStageCreateInfo));
144 stageInfo->sType = VK_STRUCTURE_TYPE_PIPELINE_SHADER_STAGE_CREATE_INFO;
145 stageInfo->pNext = nullptr;
146 stageInfo->flags = 0;
147 stageInfo->stage = stage;
148 stageInfo->module = *shaderModule;
149 stageInfo->pName = "main";
150 stageInfo->pSpecializationInfo = nullptr;
151
152 return true;
153 }
154
finalize(const DrawArgs & args,GrPrimitiveType primitiveType,const GrVkRenderPass & renderPass)155 GrVkProgram* GrVkProgramBuilder::finalize(const DrawArgs& args,
156 GrPrimitiveType primitiveType,
157 const GrVkRenderPass& renderPass) {
158 VkDescriptorSetLayout dsLayout[2];
159 VkPipelineLayout pipelineLayout;
160 VkShaderModule vertShaderModule;
161 VkShaderModule fragShaderModule;
162
163 uint32_t numSamplers = fSamplerUniforms.count();
164
165 SkAutoTDeleteArray<VkDescriptorSetLayoutBinding> dsSamplerBindings(
166 new VkDescriptorSetLayoutBinding[numSamplers]);
167 for (uint32_t i = 0; i < numSamplers; ++i) {
168 UniformHandle uniHandle = fSamplerUniforms[i];
169 GrVkUniformHandler::UniformInfo uniformInfo = fUniformHandler.getUniformInfo(uniHandle);
170 SkASSERT(kSampler2D_GrSLType == uniformInfo.fVariable.getType());
171 SkASSERT(0 == uniformInfo.fSetNumber);
172 SkASSERT(uniformInfo.fBinding == i);
173 dsSamplerBindings[i].binding = uniformInfo.fBinding;
174 dsSamplerBindings[i].descriptorType = VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER;
175 dsSamplerBindings[i].descriptorCount = 1;
176 dsSamplerBindings[i].stageFlags = visibility_to_vk_stage_flags(uniformInfo.fVisibility);
177 dsSamplerBindings[i].pImmutableSamplers = nullptr;
178 }
179
180 VkDescriptorSetLayoutCreateInfo dsSamplerLayoutCreateInfo;
181 memset(&dsSamplerLayoutCreateInfo, 0, sizeof(VkDescriptorSetLayoutCreateInfo));
182 dsSamplerLayoutCreateInfo.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO;
183 dsSamplerLayoutCreateInfo.pNext = nullptr;
184 dsSamplerLayoutCreateInfo.flags = 0;
185 dsSamplerLayoutCreateInfo.bindingCount = fSamplerUniforms.count();
186 // Setting to nullptr fixes an error in the param checker validation layer. Even though
187 // bindingCount is 0 (which is valid), it still tries to validate pBindings unless it is null.
188 dsSamplerLayoutCreateInfo.pBindings = fSamplerUniforms.count() ? dsSamplerBindings.get() :
189 nullptr;
190
191 GR_VK_CALL_ERRCHECK(fGpu->vkInterface(),
192 CreateDescriptorSetLayout(fGpu->device(),
193 &dsSamplerLayoutCreateInfo,
194 nullptr,
195 &dsLayout[GrVkUniformHandler::kSamplerDescSet]));
196
197 // Create Uniform Buffer Descriptor
198 // We always attach uniform buffers to descriptor set 1. The vertex uniform buffer will have
199 // binding 0 and the fragment binding 1.
200 VkDescriptorSetLayoutBinding dsUniBindings[2];
201 memset(&dsUniBindings, 0, 2 * sizeof(VkDescriptorSetLayoutBinding));
202 dsUniBindings[0].binding = GrVkUniformHandler::kVertexBinding;
203 dsUniBindings[0].descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER;
204 dsUniBindings[0].descriptorCount = fUniformHandler.hasVertexUniforms() ? 1 : 0;
205 dsUniBindings[0].stageFlags = VK_SHADER_STAGE_VERTEX_BIT;
206 dsUniBindings[0].pImmutableSamplers = nullptr;
207 dsUniBindings[1].binding = GrVkUniformHandler::kFragBinding;
208 dsUniBindings[1].descriptorType = VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER;
209 dsUniBindings[1].descriptorCount = fUniformHandler.hasFragmentUniforms() ? 1 : 0;
210 dsUniBindings[1].stageFlags = VK_SHADER_STAGE_FRAGMENT_BIT;
211 dsUniBindings[1].pImmutableSamplers = nullptr;
212
213 VkDescriptorSetLayoutCreateInfo dsUniformLayoutCreateInfo;
214 memset(&dsUniformLayoutCreateInfo, 0, sizeof(VkDescriptorSetLayoutCreateInfo));
215 dsUniformLayoutCreateInfo.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_LAYOUT_CREATE_INFO;
216 dsUniformLayoutCreateInfo.pNext = nullptr;
217 dsUniformLayoutCreateInfo.flags = 0;
218 dsUniformLayoutCreateInfo.bindingCount = 2;
219 dsUniformLayoutCreateInfo.pBindings = dsUniBindings;
220
221 GR_VK_CALL_ERRCHECK(fGpu->vkInterface(), CreateDescriptorSetLayout(
222 fGpu->device(),
223 &dsUniformLayoutCreateInfo,
224 nullptr,
225 &dsLayout[GrVkUniformHandler::kUniformBufferDescSet]));
226
227 // Create the VkPipelineLayout
228 VkPipelineLayoutCreateInfo layoutCreateInfo;
229 memset(&layoutCreateInfo, 0, sizeof(VkPipelineLayoutCreateFlags));
230 layoutCreateInfo.sType = VK_STRUCTURE_TYPE_PIPELINE_LAYOUT_CREATE_INFO;
231 layoutCreateInfo.pNext = 0;
232 layoutCreateInfo.flags = 0;
233 layoutCreateInfo.setLayoutCount = 2;
234 layoutCreateInfo.pSetLayouts = dsLayout;
235 layoutCreateInfo.pushConstantRangeCount = 0;
236 layoutCreateInfo.pPushConstantRanges = nullptr;
237
238 GR_VK_CALL_ERRCHECK(fGpu->vkInterface(), CreatePipelineLayout(fGpu->device(),
239 &layoutCreateInfo,
240 nullptr,
241 &pipelineLayout));
242
243 // We need to enable the following extensions so that the compiler can correctly make spir-v
244 // from our glsl shaders.
245 fVS.extensions().appendf("#extension GL_ARB_separate_shader_objects : enable\n");
246 fFS.extensions().appendf("#extension GL_ARB_separate_shader_objects : enable\n");
247 fVS.extensions().appendf("#extension GL_ARB_shading_language_420pack : enable\n");
248 fFS.extensions().appendf("#extension GL_ARB_shading_language_420pack : enable\n");
249
250 this->finalizeShaders();
251
252 VkPipelineShaderStageCreateInfo shaderStageInfo[2];
253 SkAssertResult(CreateVkShaderModule(fGpu,
254 VK_SHADER_STAGE_VERTEX_BIT,
255 fVS,
256 &vertShaderModule,
257 &shaderStageInfo[0]));
258
259 SkAssertResult(CreateVkShaderModule(fGpu,
260 VK_SHADER_STAGE_FRAGMENT_BIT,
261 fFS,
262 &fragShaderModule,
263 &shaderStageInfo[1]));
264
265 GrVkResourceProvider& resourceProvider = fGpu->resourceProvider();
266 GrVkPipeline* pipeline = resourceProvider.createPipeline(*args.fPipeline,
267 *args.fPrimitiveProcessor,
268 shaderStageInfo,
269 2,
270 primitiveType,
271 renderPass,
272 pipelineLayout);
273 GR_VK_CALL(fGpu->vkInterface(), DestroyShaderModule(fGpu->device(), vertShaderModule,
274 nullptr));
275 GR_VK_CALL(fGpu->vkInterface(), DestroyShaderModule(fGpu->device(), fragShaderModule,
276 nullptr));
277
278 if (!pipeline) {
279 GR_VK_CALL(fGpu->vkInterface(), DestroyPipelineLayout(fGpu->device(), pipelineLayout,
280 nullptr));
281 GR_VK_CALL(fGpu->vkInterface(), DestroyDescriptorSetLayout(fGpu->device(), dsLayout[0],
282 nullptr));
283 GR_VK_CALL(fGpu->vkInterface(), DestroyDescriptorSetLayout(fGpu->device(), dsLayout[1],
284 nullptr));
285 return nullptr;
286 }
287
288
289 GrVkDescriptorPool::DescriptorTypeCounts typeCounts;
290 typeCounts.setTypeCount(VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, 2);
291 SkASSERT(numSamplers < 256);
292 typeCounts.setTypeCount(VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER, (uint8_t)numSamplers);
293 GrVkDescriptorPool* descriptorPool =
294 fGpu->resourceProvider().findOrCreateCompatibleDescriptorPool(typeCounts);
295
296 VkDescriptorSetAllocateInfo dsAllocateInfo;
297 memset(&dsAllocateInfo, 0, sizeof(VkDescriptorSetAllocateInfo));
298 dsAllocateInfo.sType = VK_STRUCTURE_TYPE_DESCRIPTOR_SET_ALLOCATE_INFO;
299 dsAllocateInfo.pNext = nullptr;
300 dsAllocateInfo.descriptorPool = descriptorPool->descPool();
301 dsAllocateInfo.descriptorSetCount = 2;
302 dsAllocateInfo.pSetLayouts = dsLayout;
303
304 VkDescriptorSet descriptorSets[2];
305 GR_VK_CALL_ERRCHECK(fGpu->vkInterface(), AllocateDescriptorSets(fGpu->device(),
306 &dsAllocateInfo,
307 descriptorSets));
308
309 return new GrVkProgram(fGpu,
310 pipeline,
311 pipelineLayout,
312 dsLayout,
313 descriptorPool,
314 descriptorSets,
315 fUniformHandles,
316 fUniformHandler.fUniforms,
317 fUniformHandler.fCurrentVertexUBOOffset,
318 fUniformHandler.fCurrentFragmentUBOOffset,
319 numSamplers,
320 fGeometryProcessor,
321 fXferProcessor,
322 fFragmentProcessors);
323 }
324