TDME2 1.9.121
VKGL3CoreShaderProgram.cpp
Go to the documentation of this file.
2
3#if defined(_MSC_VER)
4 // this suppresses a warning redefinition of APIENTRY macro
5 #define NOMINMAX
6 #include <windows.h>
7#endif
8#define GLFW_INCLUDE_VULKAN
9#include <GLFW/glfw3.h>
10
11#include <ext/vulkan/glslang/Public/ShaderLang.h>
12#include <ext/vulkan/spirv/GlslangToSpv.h>
13#include <ext/vulkan/OGLCompilersDLL/InitializeDll.h>
14
15#include <map>
16#include <stack>
17#include <string>
18#include <unordered_map>
19#include <unordered_set>
20#include <vector>
21
22#include <tdme/tdme.h>
24#include <tdme/math/Math.h>
34
35using std::to_string;
36
37using std::map;
38using std::stack;
39using std::string;
40using std::to_string;
41using std::unordered_map;
42using std::unordered_set;
43using std::vector;
44
46
57
58void VKGL3CoreShaderProgram::shaderInitResources(TBuiltInResource &resources) {
59 resources.maxLights = 32;
60 resources.maxClipPlanes = 6;
61 resources.maxTextureUnits = 32;
62 resources.maxTextureCoords = 32;
63 resources.maxVertexAttribs = 64;
64 resources.maxVertexUniformComponents = 4096;
65 resources.maxVaryingFloats = 64;
66 resources.maxVertexTextureImageUnits = 32;
67 resources.maxCombinedTextureImageUnits = 80;
68 resources.maxTextureImageUnits = 32;
69 resources.maxFragmentUniformComponents = 4096;
70 resources.maxDrawBuffers = 32;
71 resources.maxVertexUniformVectors = 128;
72 resources.maxVaryingVectors = 8;
73 resources.maxFragmentUniformVectors = 16;
74 resources.maxVertexOutputVectors = 16;
75 resources.maxFragmentInputVectors = 15;
76 resources.minProgramTexelOffset = -8;
77 resources.maxProgramTexelOffset = 7;
78 resources.maxClipDistances = 8;
79 resources.maxComputeWorkGroupCountX = 65535;
80 resources.maxComputeWorkGroupCountY = 65535;
81 resources.maxComputeWorkGroupCountZ = 65535;
82 resources.maxComputeWorkGroupSizeX = 1024;
83 resources.maxComputeWorkGroupSizeY = 1024;
84 resources.maxComputeWorkGroupSizeZ = 64;
85 resources.maxComputeUniformComponents = 1024;
86 resources.maxComputeTextureImageUnits = 16;
87 resources.maxComputeImageUniforms = 8;
88 resources.maxComputeAtomicCounters = 8;
89 resources.maxComputeAtomicCounterBuffers = 1;
90 resources.maxVaryingComponents = 60;
91 resources.maxVertexOutputComponents = 64;
92 resources.maxGeometryInputComponents = 64;
93 resources.maxGeometryOutputComponents = 128;
94 resources.maxFragmentInputComponents = 128;
95 resources.maxImageUnits = 8;
96 resources.maxCombinedImageUnitsAndFragmentOutputs = 8;
97 resources.maxCombinedShaderOutputResources = 8;
98 resources.maxImageSamples = 0;
99 resources.maxVertexImageUniforms = 0;
100 resources.maxTessControlImageUniforms = 0;
101 resources.maxTessEvaluationImageUniforms = 0;
102 resources.maxGeometryImageUniforms = 0;
103 resources.maxFragmentImageUniforms = 8;
104 resources.maxCombinedImageUniforms = 8;
105 resources.maxGeometryTextureImageUnits = 16;
106 resources.maxGeometryOutputVertices = 256;
107 resources.maxGeometryTotalOutputComponents = 1024;
108 resources.maxGeometryUniformComponents = 1024;
109 resources.maxGeometryVaryingComponents = 64;
110 resources.maxTessControlInputComponents = 128;
111 resources.maxTessControlOutputComponents = 128;
112 resources.maxTessControlTextureImageUnits = 16;
113 resources.maxTessControlUniformComponents = 1024;
114 resources.maxTessControlTotalOutputComponents = 4096;
115 resources.maxTessEvaluationInputComponents = 128;
116 resources.maxTessEvaluationOutputComponents = 128;
117 resources.maxTessEvaluationTextureImageUnits = 16;
118 resources.maxTessEvaluationUniformComponents = 1024;
119 resources.maxTessPatchComponents = 120;
120 resources.maxPatchVertices = 32;
121 resources.maxTessGenLevel = 64;
122 resources.maxViewports = 16;
123 resources.maxVertexAtomicCounters = 0;
124 resources.maxTessControlAtomicCounters = 0;
125 resources.maxTessEvaluationAtomicCounters = 0;
126 resources.maxGeometryAtomicCounters = 0;
127 resources.maxFragmentAtomicCounters = 8;
128 resources.maxCombinedAtomicCounters = 8;
129 resources.maxAtomicCounterBindings = 1;
130 resources.maxVertexAtomicCounterBuffers = 0;
131 resources.maxTessControlAtomicCounterBuffers = 0;
132 resources.maxTessEvaluationAtomicCounterBuffers = 0;
133 resources.maxGeometryAtomicCounterBuffers = 0;
134 resources.maxFragmentAtomicCounterBuffers = 1;
135 resources.maxCombinedAtomicCounterBuffers = 1;
136 resources.maxAtomicCounterBufferSize = 16384;
137 resources.maxTransformFeedbackBuffers = 4;
138 resources.maxTransformFeedbackInterleavedComponents = 64;
139 resources.maxCullDistances = 8;
140 resources.maxCombinedClipAndCullDistances = 8;
141 resources.maxSamples = 4;
142 resources.maxMeshOutputVerticesNV = 256;
143 resources.maxMeshOutputPrimitivesNV = 512;
144 resources.maxMeshWorkGroupSizeX_NV = 32;
145 resources.maxMeshWorkGroupSizeY_NV = 1;
146 resources.maxMeshWorkGroupSizeZ_NV = 1;
147 resources.maxTaskWorkGroupSizeX_NV = 32;
148 resources.maxTaskWorkGroupSizeY_NV = 1;
149 resources.maxTaskWorkGroupSizeZ_NV = 1;
150 resources.maxMeshViewCountNV = 4;
151 resources.limits.nonInductiveForLoops = 1;
152 resources.limits.whileLoops = 1;
153 resources.limits.doWhileLoops = 1;
154 resources.limits.generalUniformIndexing = 1;
155 resources.limits.generalAttributeMatrixVectorIndexing = 1;
156 resources.limits.generalVaryingIndexing = 1;
157 resources.limits.generalSamplerIndexing = 1;
158 resources.limits.generalVariableIndexing = 1;
159 resources.limits.generalConstantMatrixVectorIndexing = 1;
160}
161
162EShLanguage VKGL3CoreShaderProgram::shaderFindLanguage(const VkShaderStageFlagBits shaderType) {
163 switch (shaderType) {
164 case VK_SHADER_STAGE_VERTEX_BIT:
165 return EShLangVertex;
166 case VK_SHADER_STAGE_TESSELLATION_CONTROL_BIT:
167 return EShLangTessControl;
168 case VK_SHADER_STAGE_TESSELLATION_EVALUATION_BIT:
169 return EShLangTessEvaluation;
170 case VK_SHADER_STAGE_GEOMETRY_BIT:
171 return EShLangGeometry;
172 case VK_SHADER_STAGE_FRAGMENT_BIT:
173 return EShLangFragment;
174 case VK_SHADER_STAGE_COMPUTE_BIT:
175 return EShLangCompute;
176 default:
177 return EShLangVertex;
178 }
179}
180
181int VKGL3CoreShaderProgram::determineAlignment(const unordered_map<string, vector<string>>& structs, const vector<string>& uniforms) {
183 auto alignmentMax = 0;
184 for (auto uniform: uniforms) {
185 t.tokenize(uniform, "\t ;");
186 string uniformType;
187 string uniformName;
188 auto isArray = false;
189 auto arraySize = 1;
190 if (t.hasMoreTokens() == true) uniformType = t.nextToken();
191 while (t.hasMoreTokens() == true) uniformName = t.nextToken();
192 if (uniformName.find('[') != -1 && uniformName.find(']') != -1) {
193 uniformName = StringTools::substring(uniformName, 0, uniformName.find('['));
194 }
195 if (uniformType == "int") {
196 uint32_t size = sizeof(int32_t);
197 alignmentMax = Math::max(alignmentMax, size);
198 } else
199 if (uniformType == "float") {
200 uint32_t size = sizeof(float);
201 alignmentMax = Math::max(alignmentMax, size);
202 } else
203 if (uniformType == "vec2") {
204 uint32_t size = sizeof(float) * 2;
205 alignmentMax = Math::max(alignmentMax, size);
206 } else
207 if (uniformType == "vec3") {
208 uint32_t size = sizeof(float) * 4;
209 alignmentMax = Math::max(alignmentMax, size);
210 } else
211 if (uniformType == "vec4") {
212 uint32_t size = sizeof(float) * 4;
213 alignmentMax = Math::max(alignmentMax, size);
214 } else
215 if (uniformType == "mat3") {
216 uint32_t size = sizeof(float) * 4;
217 alignmentMax = Math::max(alignmentMax, size);
218 } else
219 if (uniformType == "mat4") {
220 uint32_t size = sizeof(float) * 4;
221 alignmentMax = Math::max(alignmentMax, size);
222 } else
223 if (uniformType == "sampler2D") {
224 // no op
225 } else
226 if (uniformType == "samplerCube") {
227 // no op
228 } else {
229 if (structs.find(uniformType) != structs.end()) {
230 uint32_t size = determineAlignment(structs, structs.find(uniformType)->second);
231 alignmentMax = Math::max(alignmentMax, size);
232 } else {
233 return false;
234 }
235 }
236 }
237 return alignmentMax;
238}
239
240bool VKGL3CoreShaderProgram::addToShaderUniformBufferObject(VKRenderer::VKRenderer::shader_type& shader, const unordered_map<string, string>& definitionValues, const unordered_map<string, vector<string>>& structs, const vector<string>& uniforms, const string& prefix, unordered_set<string>& uniformStructsArrays, string& uniformsBlock) {
242 for (auto uniform: uniforms) {
243 t.tokenize(uniform, "\t ;");
244 string uniformType;
245 string uniformName;
246 auto isArray = false;
247 auto arraySize = 1;
248 if (t.hasMoreTokens() == true) uniformType = t.nextToken();
249 while (t.hasMoreTokens() == true) uniformName = t.nextToken();
250 if (uniformName.find('[') != -1 && uniformName.find(']') != string::npos) {
251 isArray = true;
252 auto arraySizeString = StringTools::substring(uniformName, uniformName.find('[') + 1, uniformName.find(']'));
253 for (auto definitionValueIt: definitionValues) arraySizeString = StringTools::replace(arraySizeString, definitionValueIt.first, definitionValueIt.second);
254 if (Integer::is(arraySizeString) == false) {
255 Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Unknown array size: " + uniform);
256 }
257 arraySize = Integer::parse(arraySizeString);
258 uniformName = StringTools::substring(uniformName, 0, uniformName.find('['));
259 if (uniformType != "sampler2D" && uniformType != "samplerCube") uniformStructsArrays.insert(uniformName);
260 }
261 if (uniformType == "int") {
262 for (auto i = 0; i < arraySize; i++) {
263 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
264 uint32_t size = sizeof(int32_t);
265 uint32_t alignment = Math::max(isArray == true?16:0, sizeof(int32_t));
266 auto position = align(alignment, shader.uboSize);
267 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
268 {
269 .name = prefix + uniformName + suffix,
270 .newName = prefix + uniformName + suffix,
272 .position = position,
273 .size = size,
274 .textureUnit = -1
275 };
276 shader.uboSize = position + size;
277 }
278 } else
279 if (uniformType == "float") {
280 for (auto i = 0; i < arraySize; i++) {
281 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
282 uint32_t size = sizeof(float);
283 uint32_t alignment = Math::max(isArray == true?16:0, sizeof(float));
284 auto position = align(alignment, shader.uboSize);
285 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
286 {
287 .name = prefix + uniformName + suffix,
288 .newName = prefix + uniformName + suffix,
290 .position = position,
291 .size = size,
292 .textureUnit = -1
293 };
294 shader.uboSize = position + size;
295 }
296 } else
297 if (uniformType == "vec2") {
298 for (auto i = 0; i < arraySize; i++) {
299 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
300 uint32_t size = sizeof(float) * 2;
301 uint32_t alignment = Math::max(isArray == true?16:0, sizeof(float) * 2);
302 auto position = align(alignment, shader.uboSize);
303 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
304 {
305 .name = prefix + uniformName + suffix,
306 .newName = prefix + uniformName + suffix,
308 .position = position,
309 .size = size,
310 .textureUnit = -1
311 };
312 shader.uboSize = position + size;
313 }
314 } else
315 if (uniformType == "vec3") {
316 for (auto i = 0; i < arraySize; i++) {
317 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
318 uint32_t size = sizeof(float) * 3;
319 uint32_t alignment = Math::max(isArray == true?16:0, sizeof(float) * 4);
320 auto position = align(alignment, shader.uboSize);
321 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
322 {
323 .name = prefix + uniformName + suffix,
324 .newName = prefix + uniformName + suffix,
326 .position = position,
327 .size = size,
328 .textureUnit = -1
329 };
330 shader.uboSize = position + size;
331 }
332 } else
333 if (uniformType == "vec4") {
334 for (auto i = 0; i < arraySize; i++) {
335 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
336 uint32_t size = sizeof(float) * 4;
337 uint32_t alignment = Math::max(isArray == true?16:0, sizeof(float) * 4);
338 auto position = align(alignment, shader.uboSize);
339 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
340 {
341 .name = prefix + uniformName + suffix,
342 .newName = prefix + uniformName + suffix,
344 .position = position,
345 .size = size,
346 .textureUnit = -1
347 };
348 shader.uboSize = position + size;
349 }
350 } else
351 if (uniformType == "mat3") {
352 for (auto i = 0; i < arraySize; i++) {
353 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
354 uint32_t size = sizeof(float) * 12;
355 uint32_t alignment = Math::max(isArray == true?16:0, sizeof(float) * 4);
356 auto position = align(alignment, shader.uboSize);
357 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
358 {
359 .name = prefix + uniformName + suffix,
360 .newName = prefix + uniformName + suffix,
362 .position = position,
363 .size = size,
364 .textureUnit = -1
365 };
366 shader.uboSize = position + size;
367 }
368 } else
369 if (uniformType == "mat4") {
370 for (auto i = 0; i < arraySize; i++) {
371 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
372 uint32_t size = sizeof(float) * 16;
373 uint32_t alignment = Math::max(isArray == true?16:0, sizeof(float) * 4);
374 auto position = align(alignment, shader.uboSize);
375 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
376 {
377 .name = prefix + uniformName + suffix,
378 .newName = prefix + uniformName + suffix,
380 .position = position,
381 .size = size,
382 .textureUnit = -1
383 };
384 shader.uboSize = position + size;
385 }
386 } else
387 if (uniformType == "sampler2D") {
388 for (auto i = 0; i < arraySize; i++) {
389 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
390 auto newSuffix = isArray == true?"_" + to_string(i):"";
391 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
392 {
393 .name = prefix + uniformName + suffix,
394 .newName = prefix + uniformName + newSuffix,
396 .position = -1,
397 .size = 0,
398 .textureUnit = -1
399 };
400 }
401 continue;
402 } else
403 if (uniformType == "samplerCube") {
404 for (auto i = 0; i < arraySize; i++) {
405 auto suffix = isArray == true?"[" + to_string(i) + "]":"";
406 auto newSuffix = isArray == true?"_" + to_string(i):"";
407 shader.uniforms[prefix + uniformName + suffix] = new VKRenderer::shader_type::uniform_type
408 {
409 .name = prefix + uniformName + suffix,
410 .newName = prefix + uniformName + newSuffix,
412 .position = -1,
413 .size = 0,
414 .textureUnit = -1
415 };
416 }
417 continue;
418 } else {
419 if (structs.find(uniformType) != structs.end()) {
420 for (auto i = 0; i < arraySize; i++) {
421 auto structPrefix = prefix + uniformName + (isArray == true?"[" + to_string(i) + "]":"") + ".";
422 string uniformsBlockIgnore;
423 auto alignment = Math::max(16, determineAlignment(structs, structs.find(uniformType)->second));
424 shader.uboSize = align(alignment, shader.uboSize);
425 auto success = addToShaderUniformBufferObject(shader, definitionValues, structs, structs.find(uniformType)->second, structPrefix, uniformStructsArrays, uniformsBlockIgnore);
426 shader.uboSize = align(alignment, shader.uboSize);
427 if (success == false) return false;
428 if (isArray == false) uniformStructsArrays.insert(uniformName);
429 }
430 } else {
431 Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Unknown uniform type: " + uniformType + "@" + prefix + uniform);
432 return false;
433 }
434 }
435 uniformsBlock+= uniform + "\n";
436 }
437 return true;
438}
439
440void VKGL3CoreShaderProgram::loadShader(VKRenderer::shader_type& shader, int32_t type, const string& pathName, const string& fileName, const string& definitions, const string& functions)
441{
442 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): INIT: " + pathName + "/" + fileName + ": " + definitions);
443
444 shader.valid = true;
445 shader.type = (VkShaderStageFlagBits)type;
446 shader.file = pathName + "/" + fileName;
447
448 // cache + hash
449 shader.cacheId = "shader-" + to_string(shader.id) + "-" + StringTools::replace(shader.file, "/", "_");
450 shader.hash = SHA256::encode(shader.cacheId + "." + to_string(type) + definitions + functions);
451
452 // do we have a cached shader already?
453 if (FileSystem::getInstance()->fileExists("shader/vk/" + shader.cacheId + ".properties") == true) {
454 Properties vkShaderCache;
455 vkShaderCache.load("shader/vk", shader.cacheId + ".properties");
456 if (shader.hash != vkShaderCache.get("shader.hash", "")) {
457 Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Invalid hash id");
458 shader.valid = false;
459 }
460 return;
461 }
462
463 // shader source
464 auto shaderSource = StringTools::replace(
465 StringTools::replace(
466 FileSystem::getInstance()->getContentAsString(pathName, fileName),
467 "{$DEFINITIONS}",
468 "#define HAVE_VULKAN\n\n" + definitions + "\n\n"
469 ),
470 "{$FUNCTIONS}",
471 functions + "\n\n"
472 );
473
474 // do some shader adjustments
475 {
476 // pre parse shader code
477 vector<string> newShaderSourceLines;
478 vector<string> uniforms;
479 shaderSource = StringTools::replace(shaderSource, "\r", "");
480 shaderSource = StringTools::replace(shaderSource, "\t", " ");
481 shaderSource = StringTools::replace(shaderSource, "#version 430 core", "#version 430 core\n#extension GL_EXT_scalar_block_layout: require\n\n");
482 shaderSource = StringTools::replace(shaderSource, "#version 330 core", "#version 430 core\n#extension GL_EXT_scalar_block_layout: require\n\n");
484 t.tokenize(shaderSource, "\n");
486 vector<string> definitions;
487 unordered_map<string, string> definitionValues;
488 unordered_map<string, vector<string>> structs;
489 string currentStruct;
490 stack<string> testedDefinitions;
491 vector<bool> matchedDefinitions;
492 vector<bool> hadMatchedDefinitions;
493 auto uboUniformCount = 0;
494 auto multiLineComment = false;
495 while (t.hasMoreTokens() == true) {
496 auto matchedAllDefinitions = true;
497 for (auto matchedDefinition: matchedDefinitions) matchedAllDefinitions&= matchedDefinition;
498 auto line = StringTools::trim(t.nextToken());
499 if (StringTools::startsWith(line, "//") == true) continue;
500 auto position = string::npos;
501 if (StringTools::startsWith(line, "/*") == true) {
502 multiLineComment = true;
503 } else
504 if (multiLineComment == true) {
505 if (StringTools::endsWith(line, "*/") == true) multiLineComment = false;
506 } else
507 if (StringTools::startsWith(line, "#if defined(") == true ||
508 StringTools::startsWith(line, "#if !defined(") == true ||
509 StringTools::startsWith(line, "#ifdef ") == true ||
510 StringTools::startsWith(line, "#ifndef ") == true) {
511 auto inverted = StringTools::startsWith(line, "#if !defined(") == true || StringTools::startsWith(line, "#ifndef ") == true;
512 string definition;
513 if (StringTools::startsWith(line, "#if ") == true) {
514 definition = StringTools::trim(StringTools::substring(line, string(inverted == false?"#if defined(":"#if !defined(").size(), (position = line.find(")")) != string::npos?position:line.size()));
515 } else {
516 definition = StringTools::trim(StringTools::substring(line, string(inverted == false?"#ifdef ":"#ifndef ").size()));
517 }
518 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have preprocessor test begin: " + definition);
519 testedDefinitions.push(definition);
520 bool matched = false;
521 for (auto availableDefinition: definitions) {
522 if (definition == availableDefinition) {
523 matched = true;
524 break;
525 }
526 }
527 if (inverted == true) matched = !matched;
528 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have preprocessor test begin: " + definition + ": " + to_string(matched));
529 matchedDefinitions.push_back(matched);
530 hadMatchedDefinitions.push_back(matched);
531 newShaderSourceLines.push_back("// " + line);
532 } else
533 if (StringTools::startsWith(line, "#elif defined(") == true ||
534 StringTools::startsWith(line, "#elif !defined(") == true) {
535 auto inverted = StringTools::startsWith(line, "#elif !defined(") == true;
536 // remove old test from stack
537 if (testedDefinitions.size() == 0) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have preprocessor else end: invalid depth"); else {
538 testedDefinitions.pop();
539 matchedDefinitions.pop_back();
540 }
541 newShaderSourceLines.push_back("// " + line);
542 // do new test
543 auto definition = StringTools::trim(StringTools::substring(line, string(inverted == false?"#elif defined(":"#elif !defined(").size(), (position = line.find(")")) != string::npos?position:line.size()));
544 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have preprocessor test else if: " + definition);
545 testedDefinitions.push(definition);
546 bool matched = false;
547 for (auto availableDefinition: definitions) {
548 if (definition == availableDefinition) {
549 matched = true;
550 break;
551 }
552 }
553 if (inverted == true) matched = !matched;
554 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have preprocessor else test begin: " + definition + ": " + to_string(matched));
555 matchedDefinitions.push_back(matched);
556 if (matched == true) hadMatchedDefinitions[hadMatchedDefinitions.size() - 1] = matched;
557 } else
558 if (StringTools::startsWith(line, "#define ") == true) {
559 auto definition = StringTools::trim(StringTools::substring(line, string("#define ").size()));
560 if (definition.find(' ') != string::npos) {
561 t2.tokenize(definition, " ");
562 definition = t2.nextToken();
563 string value;
564 while (t2.hasMoreTokens() == true) value+= t2.nextToken();
565 definitionValues[definition] = value;
566 newShaderSourceLines.push_back((matchedAllDefinitions == true?"":"// ") + line);
567 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have define with value: " + definition + " --> " + value);
568 } else {
569 definitions.push_back(definition);
570 newShaderSourceLines.push_back((matchedAllDefinitions == true?"":"// ") + line);
571 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have define: " + definition);
572 }
573 } else
574 if (StringTools::startsWith(line, "#else") == true) {
575 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have preprocessor else: " + line);
576 matchedDefinitions[matchedDefinitions.size() - 1] = !matchedDefinitions[matchedDefinitions.size() - 1] && hadMatchedDefinitions[matchedDefinitions.size() - 1] == false;
577 newShaderSourceLines.push_back("// " + line);
578 matchedAllDefinitions = true;
579 for (auto matchedDefinition: matchedDefinitions) matchedAllDefinitions&= matchedDefinition;
580 } else
581 if (StringTools::startsWith(line, "#endif") == true) {
582 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have preprocessor test end: " + line);
583 if (testedDefinitions.size() == 0) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have preprocessor test end: invalid depth"); else {
584 testedDefinitions.pop();
585 matchedDefinitions.pop_back();
586 hadMatchedDefinitions.pop_back();
587 }
588 newShaderSourceLines.push_back("// " + line);
589 } else
590 if (matchedAllDefinitions == true) {
591 if (StringTools::startsWith(line, "struct ") == true) {
592 currentStruct = StringTools::trim(StringTools::substring(line, string("struct ").size(), line.find('{')));
593 } else
594 if (currentStruct.size() > 0) {
595 if (line == "};") {
596 currentStruct.clear();
597 } else {
598 structs[currentStruct].push_back(line);
599 }
600 } else
601 if (currentStruct.size() > 0) {
602
603 }
604 if ((StringTools::startsWith(line, "uniform ")) == true) {
605 string uniform;
606 if (line.find("sampler2D") != string::npos) {
607 uniform = StringTools::substring(line, string("uniform").size() + 1);
608 t2.tokenize(uniform, " ;");
609 string uniformType;
610 string uniformName;
611 if (t2.hasMoreTokens() == true) uniformType = t2.nextToken();
612 while (t2.hasMoreTokens() == true) uniformName = t2.nextToken();
613 auto isArray = false;
614 auto arraySize = 1;
615 if (uniformName.find('[') != string::npos && uniformName.find(']') != string::npos) {
616 isArray = true;
617 auto arraySizeString = StringTools::substring(uniformName, uniformName.find('[') + 1, uniformName.find(']'));
618 for (auto definitionValueIt: definitionValues) arraySizeString = StringTools::replace(arraySizeString, definitionValueIt.first, definitionValueIt.second);
619 arraySize = Integer::parse(arraySizeString);
620 uniformName = StringTools::substring(uniformName, 0, uniformName.find('['));
621 }
622 for (auto i = 0; i < arraySize; i++) {
623 auto suffix = isArray == true?"_" + to_string(i):"";
624 newShaderSourceLines.push_back("layout(set = 1, binding = {$SAMPLER2D_BINDING_" + uniformName + suffix + "_IDX}) uniform sampler2D " + uniformName + suffix + ";");
625 }
626 shader.samplers++;
627 } else
628 if (line.find("samplerCube") != string::npos) {
629 uniform = StringTools::substring(line, string("uniform").size() + 1);
630 t2.tokenize(uniform, " ;");
631 string uniformType;
632 string uniformName;
633 if (t2.hasMoreTokens() == true) uniformType = t2.nextToken();
634 while (t2.hasMoreTokens() == true) uniformName = t2.nextToken();
635 auto isArray = false;
636 auto arraySize = 1;
637 if (uniformName.find('[') != string::npos && uniformName.find(']') != string::npos) {
638 isArray = true;
639 auto arraySizeString = StringTools::substring(uniformName, uniformName.find('[') + 1, uniformName.find(']'));
640 for (auto definitionValueIt: definitionValues) arraySizeString = StringTools::replace(arraySizeString, definitionValueIt.first, definitionValueIt.second);
641 arraySize = Integer::parse(arraySizeString);
642 uniformName = StringTools::substring(uniformName, 0, uniformName.find('['));
643 }
644 for (auto i = 0; i < arraySize; i++) {
645 auto suffix = isArray == true?"_" + to_string(i):"";
646 newShaderSourceLines.push_back("layout(set = 1, binding = {$SAMPLERCUBE_BINDING_" + uniformName + suffix + "_IDX}) uniform samplerCube " + uniformName + suffix + ";");
647 }
648 shader.samplers++;
649 } else {
650 uniform = StringTools::substring(line, string("uniform").size() + 1);
651 uboUniformCount++;
652 }
653 uniforms.push_back(uniform);
654 newShaderSourceLines.push_back("// " + line);
655 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have uniform: " + uniform);
656 } else
657 if (StringTools::startsWith(line, "out ") == true || StringTools::startsWith(line, "flat out ") == true) {
658 if (shader.type == SHADER_VERTEX_SHADER) {
659 t2.tokenize(line, " ;");
660 bool flat = false;
661 auto inOutType = t2.hasMoreTokens() == true?t2.nextToken():string();
662 if (inOutType == "flat") {
663 flat = true;
664 auto inOutType = t2.hasMoreTokens() == true?t2.nextToken():string();
665 }
666 auto outType = t2.hasMoreTokens() == true?t2.nextToken():string();
667 auto outName = t2.hasMoreTokens() == true?t2.nextToken():string();
668 auto outLocation = 0;
669 for (auto& attributeLayout: shader.attributeLayouts) {
670 if (attributeLayout.type == "mat3") {
671 outLocation+= 3;
672 } else
673 if (attributeLayout.type == "mat4") {
674 outLocation+= 4;
675 } else {
676 outLocation++;
677 }
678 }
679 shader.attributeLayouts.push_back(
680 {
681 .name = outName,
682 .type = outType,
683 .location = static_cast<uint8_t>(outLocation)
684 }
685 );
686 if (VERBOSE == true) {
687 Console::println(
688 "inOutType: " + inOutType + " / " +
689 "outType: " + outType + " / " +
690 "outName: " + outName + " / " +
691 "location: " + to_string(outLocation)
692 );
693 }
694 newShaderSourceLines.push_back("layout (location = " + to_string(outLocation) + ") " + line);
695 } else
696 if (shader.type == SHADER_FRAGMENT_SHADER) {
697 newShaderSourceLines.push_back("layout (location = 0) " + line);
698 }
699 } else
700 if (StringTools::startsWith(line, "in ") == true || StringTools::startsWith(line, "flat in ") == true) {
701 if (shader.type == SHADER_FRAGMENT_SHADER) {
702 t2.tokenize(line, " ;");
703 bool flat = false;
704 auto inOutType = t2.hasMoreTokens() == true?t2.nextToken():string();
705 if (inOutType == "flat") {
706 flat = true;
707 auto inOutType = t2.hasMoreTokens() == true?t2.nextToken():string();
708 }
709 auto inType = t2.hasMoreTokens() == true?t2.nextToken():string();
710 auto inName = t2.hasMoreTokens() == true?t2.nextToken():string();
711 if (VERBOSE == true) {
712 Console::println("layout (location = {$IN_ATTRIBUTE_LOCATION_" + inName + "_IDX}) " + line);
713 }
714 newShaderSourceLines.push_back("layout (location = {$IN_ATTRIBUTE_LOCATION_" + inName + "_IDX}) " + line);
715 }
716 } else
717 if (StringTools::startsWith(line, "layout") == true && line.find("binding=") != string::npos) {
718 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Have layout with binding: " + line);
719 t2.tokenize(line, "(,)= \t");
720 while (t2.hasMoreTokens() == true) {
721 auto token = t2.nextToken();
722 if (token == "binding" && t2.hasMoreTokens() == true) {
723 auto nextToken = t2.nextToken();
724 shader.maxBindings = Math::max(Integer::parse(nextToken), shader.maxBindings);
725 break;
726 }
727 }
728 newShaderSourceLines.push_back(line);
729 } else {
730 newShaderSourceLines.push_back(line);
731 }
732 } else {
733 newShaderSourceLines.push_back("// " + line);
734 }
735 }
736
737 // generate new uniform block
738 unordered_set<string> uniformStructsArrays;
739 string uniformsBlock = "";
740
741 // replace uniforms to use ubo
742 if (uniforms.size() > 0) {
743 if (uboUniformCount > 0) {
744 uniformsBlock+= "layout(set = 0, std140, column_major, binding={$UBO_BINDING_IDX}) uniform UniformBufferObject\n";
745 uniformsBlock+= "{\n";
746 }
747 string uniformsBlockIgnore;
748 addToShaderUniformBufferObject(shader, definitionValues, structs, uniforms, "", uniformStructsArrays, uboUniformCount > 0?uniformsBlock:uniformsBlockIgnore);
749 if (uboUniformCount > 0) uniformsBlock+= "} ubo_generated;\n";
750 if (VERBOSE == true) Console::println("Shader UBO size: " + to_string(shader.uboSize));
751 }
752
753 // construct new shader from vector and flip y, also inject uniforms
754 shaderSource.clear();
755 auto injectedUniformsAt = -1;
756 auto injectedYFlip = false;
757 // inject uniform before first method
758 for (auto i = 0; i < newShaderSourceLines.size(); i++) {
759 auto line = newShaderSourceLines[i];
760 if (StringTools::startsWith(line, "//") == true) continue;
761 if (line.find('(') != string::npos &&
762 line.find(')') != string::npos &&
763 StringTools::startsWith(line, "struct ") == false &&
764 StringTools::startsWith(line, "layout ") == false &&
765 StringTools::startsWith(line, "layout(") == false &&
766 StringTools::startsWith(line, "#") == false) {
767 injectedUniformsAt = i - 1;
768 break;
769 }
770 }
771 for (int i = newShaderSourceLines.size() - 1; i >= 0; i--) {
772 auto line = newShaderSourceLines[i] + "\n";
773 if (i == injectedUniformsAt) {
774 shaderSource = uniformsBlock + line + shaderSource;
775 } else
776 if (StringTools::startsWith(line, "//") == true) {
777 shaderSource = line + shaderSource;
778 // rename uniforms to ubo uniforms
779 } else {
780 for (auto& uniformIt: shader.uniforms) {
781 if (uniformIt.second->type == VKRenderer::shader_type::uniform_type::TYPE_SAMPLER2D) {
782 if (uniformIt.second->name != uniformIt.second->newName) {
783 line = StringTools::replace(
784 line,
785 uniformIt.second->name,
786 uniformIt.second->newName
787 );
788 }
789 } else
790 if (uniformIt.second->type == VKRenderer::shader_type::uniform_type::TYPE_SAMPLERCUBE) {
791 if (uniformIt.second->name != uniformIt.second->newName) {
792 line = StringTools::replace(
793 line,
794 uniformIt.second->name,
795 uniformIt.second->newName
796 );
797 }
798 } else {
799 auto uniformName = uniformIt.second->name;
800 line = StringTools::regexReplace(
801 line,
802 "(\\b)" + uniformName + "(\\b)",
803 "$1ubo_generated." + uniformName + "$2"
804 );
805 // TODO: this is a workaround until we have a better shader parser/adapt code
806 line = StringTools::replace(line, "ubo_generated.ubo_generated.", "ubo_generated.");
807 }
808 }
809 // rename arrays and structs to ubo uniforms
810 for (auto& uniformName: uniformStructsArrays) {
811 line = StringTools::regexReplace(
812 line,
813 "(\\b)" + uniformName + "(\\b)",
814 "$1ubo_generated." + uniformName + "$2"
815 );
816 // TODO: this is a workaround until we have a better shader parser/adapt code
817 line = StringTools::replace(line, "ubo_generated.ubo_generated.", "ubo_generated.");
818 }
819 // inject gl_Position flip before last } from main
820 if (type == SHADER_VERTEX_SHADER && injectedYFlip == false && StringTools::startsWith(line, "}") == true) {
821 shaderSource =
822 "gl_Position.y*= -1.0;\ngl_Position.z = (gl_Position.z + gl_Position.w) / 2.0;\n" +
823 line +
824 shaderSource;
825 injectedYFlip = true;
826 } else {
827 shaderSource = line + shaderSource;
828 }
829 }
830 }
831 if (type == SHADER_VERTEX_SHADER && injectedYFlip == false) {
832 Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Could not inject OpenGL GL like Y and Z correction math");
833 }
834
835 // debug uniforms
836 for (auto& uniformIt: shader.uniforms) {
837 if (VERBOSE == true) Console::println("VKGL3CoreShaderProgram::" + string(__FUNCTION__) + "(): Uniform: " + uniformIt.second->name + ": " + to_string(uniformIt.second->position) + " / " + to_string(uniformIt.second->size));
838 }
839 }
840
841 shader.definitions = definitions;
842 shader.source = shaderSource;
843}
844
846 map<string, int32_t> uniformsByName;
847 auto useCache = false;
848
849 // check if shaders are valid
850 for (auto shader: program.shaders) {
851 if (shader->valid == false) {
852 Console::println("VKGL3CoreShaderProgram::linkProgram(): Cached shader is invalid. Please recreate VK shader cache or delete VK shader cache files");
853 return false;
854 }
855 }
856
857 // check if we can use a cache
858 if (FileSystem::getInstance()->fileExists("shader/vk/program-" + to_string(program.id) + ".properties") == true) {
859 // use cache
860 useCache = true;
861
862 //
863 Properties vkProgramCache;
864 vkProgramCache.load("shader/vk", "program-" + to_string(program.id) + ".properties");
865 if (Integer::parse(vkProgramCache.get("program.id", "-1")) != program.id) {
866 Console::println("VKGL3CoreShaderProgram::linkProgram(): program id mismatch");
867 return false;
868 }
869 program.layoutBindings = Integer::parse(vkProgramCache.get("program.layout_bindings", "-1"));
870
871 // read shaders from cache
872 auto shaderIdx = 0;
873 for (auto shader: program.shaders) {
874 //
875 shader->cacheId = vkProgramCache.get("program.shader_" + to_string(shaderIdx) + "_cacheid", "");
876 shader->source = FileSystem::getInstance()->getContentAsString("shader/vk", shader->cacheId + ".glsl");
877 shader->definitions = FileSystem::getInstance()->getContentAsString("shader/vk", shader->cacheId + ".definitions");
878 vector<uint8_t> spirv8;
879 FileSystem::getInstance()->getContent("shader/vk", shader->cacheId + ".spirv", spirv8);
880 shader->spirv.resize(spirv8.size() / 4);
881 for (auto i = 0; i < spirv8.size() / 4; i++) {
882 shader->spirv[i] =
883 (static_cast<uint32_t>(spirv8[i * 4 + 0])) +
884 (static_cast<uint32_t>(spirv8[i * 4 + 1]) << 8) +
885 (static_cast<uint32_t>(spirv8[i * 4 + 2]) << 16) +
886 (static_cast<uint32_t>(spirv8[i * 4 + 3]) << 24);
887 }
888 {
889 // use shader caches to load shaders
890 Properties vkShaderCache;
891 vkShaderCache.load("shader/vk", shader->cacheId + ".properties");
892 if (Integer::parse(vkShaderCache.get("shader.id", "-1")) != shader->id) {
893 Console::println("VKGL3CoreShaderProgram::linkProgram(): shader id mismatch");
894 return false;
895 }
896 shader->type = static_cast<VkShaderStageFlagBits>(Integer::parse(vkShaderCache.get("shader.type", "-1")));
897 shader->file = vkShaderCache.get("shader.file", "");
898 shader->maxBindings = Integer::parse(vkShaderCache.get("shader.max_bindings", "-1"));
899
900 // vert->frag layout attributes
901 {
902 auto i = 0;
903 while (vkShaderCache.get("shader.attributelayout_name_" + to_string(i), "").empty() == false) {
904 auto outName = vkShaderCache.get("shader.attributelayout_name_" + to_string(i), "");
905 auto outType = vkShaderCache.get("shader.attributelayout_type_" + to_string(i), "");
906 uint8_t outLocation = Integer::parse(vkShaderCache.get("shader.attributelayout_location_" + to_string(i), "-1"));
907 shader->attributeLayouts.push_back(
908 {
909 .name = outName,
910 .type = outType,
911 .location = static_cast<uint8_t>(outLocation)
912 }
913 );
914 i++;
915 }
916 shader->attributeLayouts.shrink_to_fit();
917 }
918
919 // ubo + uniforms
920 {
921 shader->uboSize = Integer::parse(vkShaderCache.get("shader.ubo_size", "-1"));
922 shader->uboBindingIdx = Integer::parse(vkShaderCache.get("shader.ubo_bindingidx", "-1"));
923 auto i = 0;
924 while (vkShaderCache.get("shader.uniform_name_" + to_string(i), "").empty() == false) {
925 auto name = vkShaderCache.get("shader.uniform_name_" + to_string(i), "");
926 auto newName = vkShaderCache.get("shader.uniform_newname_" + to_string(i), "");
927 auto type = static_cast<VKRenderer::shader_type::uniform_type::uniform_type_enum>(Integer::parse(vkShaderCache.get("shader.uniform_type_" + to_string(i), "-1")));
928 int32_t position = Integer::parse(vkShaderCache.get("shader.uniform_position_" + to_string(i), "-1"));
929 uint32_t size = Integer::parse(vkShaderCache.get("shader.uniform_size_" + to_string(i), "0"));
930 shader->uniforms[name] = new VKRenderer::shader_type::uniform_type
931 {
932 .name = name,
933 .newName = newName,
934 .type = type,
935 .position = position,
936 .size = size,
937 .textureUnit = -1
938 };
939 i++;
940 }
941 }
942 }
943 shaderIdx++;
944 }
945
946 // uniforms by name
947 auto uniformIdx = 1;
948 for (auto shader: program.shaders) {
949 for (auto& uniformIt: shader->uniforms) {
950 auto& uniform = *uniformIt.second;
951 uniformsByName[uniform.name] = uniformIdx++;
952 }
953 // binding idx
954 }
955 } else {
956 // nope, no cache
957 auto bindingIdx = 0;
958 for (auto shader: program.shaders) {
959 //
960 bindingIdx = Math::max(shader->maxBindings + 1, bindingIdx);
961 }
962
963 //
964 for (auto shader: program.shaders) {
965 // do we need a uniform buffer object for this shader stage?
966 if (shader->uboSize > 0) {
967 // yep, inject UBO index
968 shader->uboBindingIdx = bindingIdx;
969 shader->source = StringTools::replace(shader->source, "{$UBO_BINDING_IDX}", to_string(bindingIdx));
970 bindingIdx++;
971 }
972 }
973
974 // bind samplers, set up ingoing attribute layout indices, compile shaders
975 auto uniformIdx = 1;
976 VKRenderer::shader_type* shaderLast = nullptr;
977 for (auto shader: program.shaders) {
978 // set up sampler2D and samplerCube binding indices
979 for (auto& uniformIt: shader->uniforms) {
980 auto& uniform = *uniformIt.second;
981 //
983 shader->source = StringTools::replace(shader->source, "{$SAMPLER2D_BINDING_" + uniform.newName + "_IDX}", to_string(bindingIdx));
984 uniform.position = bindingIdx++;
985 } else
987 shader->source = StringTools::replace(shader->source, "{$SAMPLERCUBE_BINDING_" + uniform.newName + "_IDX}", to_string(bindingIdx));
988 uniform.position = bindingIdx++;
989 }
990 uniformsByName[uniform.name] = uniformIdx++;
991 }
992
993 // set up ingoing attributes layout indices
994 if (shaderLast != nullptr) {
995 for (auto& attributeLayout: shaderLast->attributeLayouts) {
996 shader->source = StringTools::replace(shader->source, "{$IN_ATTRIBUTE_LOCATION_" + attributeLayout.name + "_IDX}", to_string(attributeLayout.location));
997 }
998 }
999
1000 // compile shader
1001 EShLanguage stage = shaderFindLanguage(shader->type);
1002 glslang::TShader glslShader(stage);
1003 glslang::TProgram glslProgram;
1004 const char *shaderStrings[1];
1005 TBuiltInResource resources;
1006 shaderInitResources(resources);
1007
1008 // Enable SPIR-V and Vulkan rules when parsing GLSL
1009 EShMessages messages = (EShMessages)(EShMsgSpvRules | EShMsgVulkanRules);
1010
1011 shaderStrings[0] = shader->source.c_str();
1012 glslShader.setStrings(shaderStrings, 1);
1013
1014 if (!glslShader.parse(&resources, 100, false, messages)) {
1015 // be verbose
1016 Console::println(
1017 string(
1018 string("VKGL3CoreShaderProgram::") +
1019 string(__FUNCTION__) +
1020 string("[") +
1021 to_string(shader->id) +
1022 string("]") +
1023 string(": parsing failed: ") +
1024 shader->file + ": " +
1025 glslShader.getInfoLog() + ": " +
1026 glslShader.getInfoDebugLog()
1027 )
1028 );
1029 Console::println(shader->source);
1030 return false;
1031 }
1032
1033 glslProgram.addShader(&glslShader);
1034 if (glslProgram.link(messages) == false) {
1035 // be verbose
1036 Console::println(
1037 string(
1038 string("VKGL3CoreShaderProgram::") +
1039 string(__FUNCTION__) +
1040 string("[") +
1041 to_string(shader->id) +
1042 string("]") +
1043 string(": linking failed: ") +
1044 shader->file + ": " +
1045 glslShader.getInfoLog() + ": " +
1046 glslShader.getInfoDebugLog()
1047 )
1048 );
1049 Console::println(shader->source);
1050 return false;
1051 }
1052
1053 glslang::GlslangToSpv(*glslProgram.getIntermediate(stage), shader->spirv);
1054
1055 //
1056 shaderLast = shader;
1057 }
1058
1059 // total bindings of program
1060 program.layoutBindings = bindingIdx;
1061 }
1062
1063 //
1064 auto uniformMaxId = 0;
1065 for (auto& uniformIt: uniformsByName) {
1066 auto uniformName = uniformIt.first;
1067 auto uniformId = uniformIt.second;
1068 program.uniforms[uniformId] = uniformName;
1069 if (uniformId > uniformMaxId) uniformMaxId = uniformId;
1070 }
1071
1072 // prepare indexed uniform lists per shader
1073 for (auto shader: program.shaders) {
1074 shader->uniformList.resize(uniformMaxId + 1);
1075 }
1076 for (auto& it: program.uniforms) {
1077 auto uniformId = it.first;
1078 auto uniformName = it.second;
1079
1080 //
1081 for (auto shader: program.shaders) {
1082 auto shaderUniformIt = shader->uniforms.find(uniformName);
1083 if (shaderUniformIt == shader->uniforms.end()) {
1084 continue;
1085 }
1086 auto uniform = shaderUniformIt->second;
1087 shader->uniformList[uniformId] = uniform;
1089 uniform->type == VKRenderer::shader_type::uniform_type::TYPE_SAMPLERCUBE) shader->samplerUniformList.push_back(uniform);
1090 }
1091 }
1092
1093 // print shaders with more than SAMPLER_HASH_MAX samplers as our hashing depends of 4 samplers max
1094 if (program.type == 1/*PROGRAM_OBJECTS*/) {
1095 for (auto shader: program.shaders) {
1096 if (shader->samplerUniformList.size() > SAMPLER_HASH_MAX) {
1097 Console::println(
1098 string("VKGL3CoreShaderProgram::") +
1099 string(__FUNCTION__) +
1100 string("[") +
1101 to_string(shader->id) +
1102 string("]") +
1103 string(": warning: more than ") + to_string(SAMPLER_HASH_MAX) + string(" samplers @ ") +
1104 shader->file
1105 );
1106 for (auto samplerUniform: shader->samplerUniformList) {
1107 Console::println("\t" + samplerUniform->name);
1108 }
1109 }
1110 }
1111 }
1112
1113 //
1114 for (auto shader: program.shaders) {
1115 shader->samplerUniformList.shrink_to_fit();
1116 }
1117
1118
1119 // store if not using cache
1120 if (useCache == false) {
1121 // store program properties
1122 {
1123 Properties vkProgramCache;
1124 vkProgramCache.put("program.id", to_string(program.id));
1125 vkProgramCache.put("program.layout_bindings", to_string(program.layoutBindings));
1126 auto i = 0;
1127 for (auto& shader: program.shaders) {
1128 vkProgramCache.put("program.shader_" + to_string(i) + "_cacheid", shader->cacheId);
1129 i++;
1130 }
1131 vkProgramCache.store("shader/vk", "program-" + to_string(program.id) + ".properties");
1132 }
1133
1134 // store shader properties
1135 for (auto shader: program.shaders) {
1136 Properties vkShaderCache;
1137 vkShaderCache.put("shader.type", to_string(shader->type));
1138 vkShaderCache.put("shader.file", shader->file);
1139 vkShaderCache.put("shader.id", to_string(shader->id));
1140 vkShaderCache.put("shader.hash", shader->hash);
1141 vkShaderCache.put("shader.max_bindings", to_string(shader->maxBindings));
1142 vkShaderCache.put("shader.ubo_size", to_string(shader->uboSize));
1143 vkShaderCache.put("shader.ubo_bindingidx", to_string(shader->uboBindingIdx));
1144 // attribute layouts
1145 {
1146 auto i = 0;
1147 for (auto& attribute: shader->attributeLayouts) {
1148 vkShaderCache.put("shader.attributelayout_name_" + to_string(i), attribute.name);
1149 vkShaderCache.put("shader.attributelayout_type_" + to_string(i), attribute.type);
1150 vkShaderCache.put("shader.attributelayout_location_" + to_string(i), to_string(attribute.location));
1151 i++;
1152 }
1153 }
1154 // uniforms
1155 {
1156 auto i = 0;
1157 for (auto& uniformIt: shader->uniforms) {
1158 auto uniform = uniformIt.second;
1159 vkShaderCache.put("shader.uniform_name_" + to_string(i), uniform->name);
1160 vkShaderCache.put("shader.uniform_newname_" + to_string(i), uniform->newName);
1161 vkShaderCache.put("shader.uniform_type_" + to_string(i), to_string(uniform->type));
1162 vkShaderCache.put("shader.uniform_position_" + to_string(i), to_string(uniform->position));
1163 vkShaderCache.put("shader.uniform_size_" + to_string(i), to_string(uniform->size));
1164 i++;
1165 }
1166 }
1167
1168 // store glsl
1169 FileSystem::getInstance()->setContentFromString("shader/vk", shader->cacheId + ".glsl", shader->source);
1170
1171 // store SPIRV
1172 {
1173 vector<uint8_t> spirv8;
1174 for (auto v: shader->spirv) {
1175 spirv8.push_back((static_cast<uint32_t>(v)) & 0xff);
1176 spirv8.push_back((static_cast<uint32_t>(v) >> 8) & 0xff);
1177 spirv8.push_back((static_cast<uint32_t>(v) >> 16) & 0xff);
1178 spirv8.push_back((static_cast<uint32_t>(v) >> 24) & 0xff);
1179 }
1180 FileSystem::getInstance()->setContent("shader/vk", shader->cacheId + ".spirv", spirv8);
1181 }
1182
1183 // store definitions
1184 FileSystem::getInstance()->setContentFromString("shader/vk", shader->cacheId + ".definitions", shader->definitions);
1185
1186 // done
1187 vkShaderCache.store("shader/vk", shader->cacheId + ".properties");
1188 }
1189 }
1190
1191 //
1192 return true;
1193}
#define SAMPLER_HASH_MAX
Definition: VKRenderer.h:74
static bool addToShaderUniformBufferObject(VKRenderer::shader_type &shader, const unordered_map< string, string > &definitionValues, const unordered_map< string, vector< string > > &structs, const vector< string > &uniforms, const string &prefix, unordered_set< string > &uniformStructsArrays, string &uniformsBlock)
Add shader uniform buffer object.
static void loadShader(VKRenderer::shader_type &shader, int32_t type, const string &pathName, const string &fileName, const string &definitions=string(), const string &functions=string())
Loads a shader.
static bool linkProgram(VKRenderer::program_type &program)
Links attached shaders to a program.
static EShLanguage shaderFindLanguage(const VkShaderStageFlagBits shaderType)
Shader VK type to language converter.
static void shaderInitResources(TBuiltInResource &resources)
Set up shader constraints/resources.
static int determineAlignment(const unordered_map< string, vector< string > > &structs, const vector< string > &uniforms)
Determine alignment.
Standard math functions.
Definition: Math.h:21
File system singleton class.
Definition: FileSystem.h:14
Console class.
Definition: Console.h:26
Integer class.
Definition: Integer.h:26
Properties class, which helps out with storeing or loading key value pairs from/to property files.
Definition: Properties.h:23
void put(const string &key, const string &value)
Add property.
Definition: Properties.h:58
const string & get(const string &key, const string &defaultValue) const
Get property value by key.
Definition: Properties.h:46
void load(const string &pathName, const string &fileName, FileSystemInterface *fileSystem=nullptr)
Load property file.
Definition: Properties.cpp:26
void store(const string &pathName, const string &fileName, FileSystemInterface *fileSystem=nullptr) const
Store property file.
Definition: Properties.cpp:43
SHA256 hash class.
Definition: SHA256.h:16
String tokenizer class.
void tokenize(const string &str, const string &delimiters)
Tokenize.
String tools class.
Definition: StringTools.h:20
unordered_map< string, uniform_type * > uniforms
Definition: VKRenderer.h:170