8#define GLFW_INCLUDE_VULKAN
11#include <ext/vulkan/glslang/Public/ShaderLang.h>
12#include <ext/vulkan/spirv/GlslangToSpv.h>
13#include <ext/vulkan/OGLCompilersDLL/InitializeDll.h>
18#include <unordered_map>
19#include <unordered_set>
41using std::unordered_map;
42using std::unordered_set;
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;
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;
177 return EShLangVertex;
183 auto alignmentMax = 0;
184 for (
auto uniform: uniforms) {
188 auto isArray =
false;
192 if (uniformName.find(
'[') != -1 && uniformName.find(
']') != -1) {
193 uniformName = StringTools::substring(uniformName, 0, uniformName.find(
'['));
195 if (uniformType ==
"int") {
196 uint32_t size =
sizeof(int32_t);
197 alignmentMax = Math::max(alignmentMax, size);
199 if (uniformType ==
"float") {
200 uint32_t size =
sizeof(float);
201 alignmentMax = Math::max(alignmentMax, size);
203 if (uniformType ==
"vec2") {
204 uint32_t size =
sizeof(float) * 2;
205 alignmentMax = Math::max(alignmentMax, size);
207 if (uniformType ==
"vec3") {
208 uint32_t size =
sizeof(float) * 4;
209 alignmentMax = Math::max(alignmentMax, size);
211 if (uniformType ==
"vec4") {
212 uint32_t size =
sizeof(float) * 4;
213 alignmentMax = Math::max(alignmentMax, size);
215 if (uniformType ==
"mat3") {
216 uint32_t size =
sizeof(float) * 4;
217 alignmentMax = Math::max(alignmentMax, size);
219 if (uniformType ==
"mat4") {
220 uint32_t size =
sizeof(float) * 4;
221 alignmentMax = Math::max(alignmentMax, size);
223 if (uniformType ==
"sampler2D") {
226 if (uniformType ==
"samplerCube") {
229 if (structs.find(uniformType) != structs.end()) {
231 alignmentMax = Math::max(alignmentMax, size);
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) {
246 auto isArray =
false;
250 if (uniformName.find(
'[') != -1 && uniformName.find(
']') != string::npos) {
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);
257 arraySize = Integer::parse(arraySizeString);
258 uniformName = StringTools::substring(uniformName, 0, uniformName.find(
'['));
259 if (uniformType !=
"sampler2D" && uniformType !=
"samplerCube") uniformStructsArrays.insert(uniformName);
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);
269 .
name = prefix + uniformName + suffix,
270 .newName = prefix + uniformName + suffix,
272 .position = position,
276 shader.uboSize = position + size;
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);
287 .
name = prefix + uniformName + suffix,
288 .newName = prefix + uniformName + suffix,
290 .position = position,
294 shader.uboSize = position + size;
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);
305 .
name = prefix + uniformName + suffix,
306 .newName = prefix + uniformName + suffix,
308 .position = position,
312 shader.uboSize = position + size;
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);
323 .
name = prefix + uniformName + suffix,
324 .newName = prefix + uniformName + suffix,
326 .position = position,
330 shader.uboSize = position + size;
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);
341 .
name = prefix + uniformName + suffix,
342 .newName = prefix + uniformName + suffix,
344 .position = position,
348 shader.uboSize = position + size;
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);
359 .
name = prefix + uniformName + suffix,
360 .newName = prefix + uniformName + suffix,
362 .position = position,
366 shader.uboSize = position + size;
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);
377 .
name = prefix + uniformName + suffix,
378 .newName = prefix + uniformName + suffix,
380 .position = position,
384 shader.uboSize = position + size;
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):
"";
393 .
name = prefix + uniformName + suffix,
394 .newName = prefix + uniformName + newSuffix,
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):
"";
409 .
name = prefix + uniformName + suffix,
410 .newName = prefix + uniformName + newSuffix,
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);
431 Console::println(
"VKGL3CoreShaderProgram::" +
string(__FUNCTION__) +
"(): Unknown uniform type: " + uniformType +
"@" + prefix + uniform);
435 uniformsBlock+= uniform +
"\n";
442 if (
VERBOSE ==
true) Console::println(
"VKGL3CoreShaderProgram::" +
string(__FUNCTION__) +
"(): INIT: " + pathName +
"/" + fileName +
": " + definitions);
445 shader.
type = (VkShaderStageFlagBits)type;
446 shader.
file = pathName +
"/" + fileName;
449 shader.
cacheId =
"shader-" + to_string(shader.
id) +
"-" + StringTools::replace(shader.
file,
"/",
"_");
450 shader.
hash = SHA256::encode(shader.
cacheId +
"." + to_string(type) + definitions + functions);
453 if (FileSystem::getInstance()->fileExists(
"shader/vk/" + shader.
cacheId +
".properties") ==
true) {
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;
464 auto shaderSource = StringTools::replace(
465 StringTools::replace(
466 FileSystem::getInstance()->getContentAsString(pathName, fileName),
468 "#define HAVE_VULKAN\n\n" + definitions +
"\n\n"
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");
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;
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;
504 if (multiLineComment ==
true) {
505 if (StringTools::endsWith(line,
"*/") ==
true) multiLineComment =
false;
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;
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()));
516 definition = StringTools::trim(StringTools::substring(line,
string(inverted ==
false?
"#ifdef ":
"#ifndef ").size()));
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) {
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);
533 if (StringTools::startsWith(line,
"#elif defined(") ==
true ||
534 StringTools::startsWith(line,
"#elif !defined(") ==
true) {
535 auto inverted = StringTools::startsWith(line,
"#elif !defined(") ==
true;
537 if (testedDefinitions.size() == 0) Console::println(
"VKGL3CoreShaderProgram::" +
string(__FUNCTION__) +
"(): Have preprocessor else end: invalid depth");
else {
538 testedDefinitions.pop();
539 matchedDefinitions.pop_back();
541 newShaderSourceLines.push_back(
"// " + line);
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) {
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;
558 if (StringTools::startsWith(line,
"#define ") ==
true) {
559 auto definition = StringTools::trim(StringTools::substring(line,
string(
"#define ").size()));
560 if (definition.find(
' ') != string::npos) {
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);
569 definitions.push_back(definition);
570 newShaderSourceLines.push_back((matchedAllDefinitions ==
true?
"":
"// ") + line);
571 if (
VERBOSE ==
true) Console::println(
"VKGL3CoreShaderProgram::" +
string(__FUNCTION__) +
"(): Have define: " + definition);
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;
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();
588 newShaderSourceLines.push_back(
"// " + line);
590 if (matchedAllDefinitions ==
true) {
591 if (StringTools::startsWith(line,
"struct ") ==
true) {
592 currentStruct = StringTools::trim(StringTools::substring(line,
string(
"struct ").size(), line.find(
'{')));
594 if (currentStruct.size() > 0) {
596 currentStruct.clear();
598 structs[currentStruct].push_back(line);
601 if (currentStruct.size() > 0) {
604 if ((StringTools::startsWith(line,
"uniform ")) ==
true) {
606 if (line.find(
"sampler2D") != string::npos) {
607 uniform = StringTools::substring(line,
string(
"uniform").size() + 1);
613 auto isArray =
false;
615 if (uniformName.find(
'[') != string::npos && uniformName.find(
']') != string::npos) {
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(
'['));
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 +
";");
628 if (line.find(
"samplerCube") != string::npos) {
629 uniform = StringTools::substring(line,
string(
"uniform").size() + 1);
635 auto isArray =
false;
637 if (uniformName.find(
'[') != string::npos && uniformName.find(
']') != string::npos) {
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(
'['));
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 +
";");
650 uniform = StringTools::substring(line,
string(
"uniform").size() + 1);
653 uniforms.push_back(uniform);
654 newShaderSourceLines.push_back(
"// " + line);
655 if (
VERBOSE ==
true) Console::println(
"VKGL3CoreShaderProgram::" +
string(__FUNCTION__) +
"(): Have uniform: " + uniform);
657 if (StringTools::startsWith(line,
"out ") ==
true || StringTools::startsWith(line,
"flat out ") ==
true) {
662 if (inOutType ==
"flat") {
668 auto outLocation = 0;
670 if (attributeLayout.type ==
"mat3") {
673 if (attributeLayout.type ==
"mat4") {
683 .location =
static_cast<uint8_t
>(outLocation)
688 "inOutType: " + inOutType +
" / " +
689 "outType: " + outType +
" / " +
690 "outName: " + outName +
" / " +
691 "location: " + to_string(outLocation)
694 newShaderSourceLines.push_back(
"layout (location = " + to_string(outLocation) +
") " + line);
697 newShaderSourceLines.push_back(
"layout (location = 0) " + line);
700 if (StringTools::startsWith(line,
"in ") ==
true || StringTools::startsWith(line,
"flat in ") ==
true) {
705 if (inOutType ==
"flat") {
712 Console::println(
"layout (location = {$IN_ATTRIBUTE_LOCATION_" + inName +
"_IDX}) " + line);
714 newShaderSourceLines.push_back(
"layout (location = {$IN_ATTRIBUTE_LOCATION_" + inName +
"_IDX}) " + line);
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);
728 newShaderSourceLines.push_back(line);
730 newShaderSourceLines.push_back(line);
733 newShaderSourceLines.push_back(
"// " + line);
738 unordered_set<string> uniformStructsArrays;
739 string uniformsBlock =
"";
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";
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));
754 shaderSource.clear();
755 auto injectedUniformsAt = -1;
756 auto injectedYFlip =
false;
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;
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;
776 if (StringTools::startsWith(line,
"//") ==
true) {
777 shaderSource = line + shaderSource;
780 for (
auto& uniformIt: shader.
uniforms) {
782 if (uniformIt.second->name != uniformIt.second->newName) {
783 line = StringTools::replace(
785 uniformIt.second->name,
786 uniformIt.second->newName
791 if (uniformIt.second->name != uniformIt.second->newName) {
792 line = StringTools::replace(
794 uniformIt.second->name,
795 uniformIt.second->newName
799 auto uniformName = uniformIt.second->name;
800 line = StringTools::regexReplace(
802 "(\\b)" + uniformName +
"(\\b)",
803 "$1ubo_generated." + uniformName +
"$2"
806 line = StringTools::replace(line,
"ubo_generated.ubo_generated.",
"ubo_generated.");
810 for (
auto& uniformName: uniformStructsArrays) {
811 line = StringTools::regexReplace(
813 "(\\b)" + uniformName +
"(\\b)",
814 "$1ubo_generated." + uniformName +
"$2"
817 line = StringTools::replace(line,
"ubo_generated.ubo_generated.",
"ubo_generated.");
820 if (type ==
SHADER_VERTEX_SHADER && injectedYFlip ==
false && StringTools::startsWith(line,
"}") ==
true) {
822 "gl_Position.y*= -1.0;\ngl_Position.z = (gl_Position.z + gl_Position.w) / 2.0;\n" +
825 injectedYFlip =
true;
827 shaderSource = line + shaderSource;
832 Console::println(
"VKGL3CoreShaderProgram::" +
string(__FUNCTION__) +
"(): Could not inject OpenGL GL like Y and Z correction math");
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));
842 shader.
source = shaderSource;
846 map<string, int32_t> uniformsByName;
847 auto useCache =
false;
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");
858 if (FileSystem::getInstance()->fileExists(
"shader/vk/program-" + to_string(program.
id) +
".properties") ==
true) {
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");
869 program.
layoutBindings = Integer::parse(vkProgramCache.
get(
"program.layout_bindings",
"-1"));
873 for (
auto shader: program.
shaders) {
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++) {
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);
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");
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"));
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(
911 .location =
static_cast<uint8_t
>(outLocation)
916 shader->attributeLayouts.shrink_to_fit();
921 shader->uboSize = Integer::parse(vkShaderCache.
get(
"shader.ubo_size",
"-1"));
922 shader->uboBindingIdx = Integer::parse(vkShaderCache.
get(
"shader.ubo_bindingidx",
"-1"));
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),
"");
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"));
935 .position = position,
948 for (
auto shader: program.
shaders) {
949 for (
auto& uniformIt: shader->uniforms) {
950 auto& uniform = *uniformIt.second;
951 uniformsByName[uniform.name] = uniformIdx++;
958 for (
auto shader: program.
shaders) {
960 bindingIdx = Math::max(shader->maxBindings + 1, bindingIdx);
964 for (
auto shader: program.
shaders) {
966 if (shader->uboSize > 0) {
968 shader->uboBindingIdx = bindingIdx;
969 shader->source = StringTools::replace(shader->source,
"{$UBO_BINDING_IDX}", to_string(bindingIdx));
977 for (
auto shader: program.
shaders) {
979 for (
auto& uniformIt: shader->uniforms) {
980 auto& uniform = *uniformIt.second;
983 shader->
source = StringTools::replace(shader->source,
"{$SAMPLER2D_BINDING_" + uniform.newName +
"_IDX}", to_string(bindingIdx));
984 uniform.position = bindingIdx++;
987 shader->source = StringTools::replace(shader->source,
"{$SAMPLERCUBE_BINDING_" + uniform.newName +
"_IDX}", to_string(bindingIdx));
988 uniform.position = bindingIdx++;
990 uniformsByName[uniform.name] = uniformIdx++;
994 if (shaderLast !=
nullptr) {
996 shader->source = StringTools::replace(shader->source,
"{$IN_ATTRIBUTE_LOCATION_" + attributeLayout.name +
"_IDX}", to_string(attributeLayout.location));
1002 glslang::TShader glslShader(stage);
1003 glslang::TProgram glslProgram;
1004 const char *shaderStrings[1];
1005 TBuiltInResource resources;
1009 EShMessages messages = (EShMessages)(EShMsgSpvRules | EShMsgVulkanRules);
1011 shaderStrings[0] = shader->source.c_str();
1012 glslShader.setStrings(shaderStrings, 1);
1014 if (!glslShader.parse(&resources, 100,
false, messages)) {
1018 string(
"VKGL3CoreShaderProgram::") +
1019 string(__FUNCTION__) +
1021 to_string(shader->id) +
1023 string(
": parsing failed: ") +
1024 shader->file +
": " +
1025 glslShader.getInfoLog() +
": " +
1026 glslShader.getInfoDebugLog()
1029 Console::println(shader->source);
1033 glslProgram.addShader(&glslShader);
1034 if (glslProgram.link(messages) ==
false) {
1038 string(
"VKGL3CoreShaderProgram::") +
1039 string(__FUNCTION__) +
1041 to_string(shader->id) +
1043 string(
": linking failed: ") +
1044 shader->file +
": " +
1045 glslShader.getInfoLog() +
": " +
1046 glslShader.getInfoDebugLog()
1049 Console::println(shader->source);
1053 glslang::GlslangToSpv(*glslProgram.getIntermediate(stage), shader->spirv);
1056 shaderLast = shader;
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;
1073 for (
auto shader: program.
shaders) {
1074 shader->uniformList.resize(uniformMaxId + 1);
1077 auto uniformId = it.first;
1078 auto uniformName = it.second;
1081 for (
auto shader: program.
shaders) {
1082 auto shaderUniformIt = shader->uniforms.find(uniformName);
1083 if (shaderUniformIt == shader->uniforms.end()) {
1086 auto uniform = shaderUniformIt->second;
1087 shader->uniformList[uniformId] = uniform;
1094 if (program.
type == 1) {
1095 for (
auto shader: program.
shaders) {
1098 string(
"VKGL3CoreShaderProgram::") +
1099 string(__FUNCTION__) +
1101 to_string(shader->id) +
1103 string(
": warning: more than ") + to_string(
SAMPLER_HASH_MAX) +
string(
" samplers @ ") +
1106 for (
auto samplerUniform: shader->samplerUniformList) {
1107 Console::println(
"\t" + samplerUniform->name);
1114 for (
auto shader: program.
shaders) {
1115 shader->samplerUniformList.shrink_to_fit();
1120 if (useCache ==
false) {
1124 vkProgramCache.
put(
"program.id", to_string(program.
id));
1125 vkProgramCache.
put(
"program.layout_bindings", to_string(program.
layoutBindings));
1127 for (
auto& shader: program.
shaders) {
1128 vkProgramCache.
put(
"program.shader_" + to_string(i) +
"_cacheid", shader->cacheId);
1131 vkProgramCache.
store(
"shader/vk",
"program-" + to_string(program.
id) +
".properties");
1135 for (
auto shader: program.
shaders) {
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));
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));
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));
1169 FileSystem::getInstance()->setContentFromString(
"shader/vk", shader->cacheId +
".glsl", shader->source);
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);
1180 FileSystem::getInstance()->setContent(
"shader/vk", shader->cacheId +
".spirv", spirv8);
1184 FileSystem::getInstance()->setContentFromString(
"shader/vk", shader->cacheId +
".definitions", shader->definitions);
1187 vkShaderCache.
store(
"shader/vk", shader->cacheId +
".properties");
GL3/Core -> Vulkan shader program.
static constexpr uint32_t SHADER_VERTEX_SHADER
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 int align(int alignment, int offset)
Align.
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 constexpr bool VERBOSE
static EShLanguage shaderFindLanguage(const VkShaderStageFlagBits shaderType)
Shader VK type to language converter.
static void shaderInitResources(TBuiltInResource &resources)
Set up shader constraints/resources.
static constexpr uint32_t SHADER_FRAGMENT_SHADER
static int determineAlignment(const unordered_map< string, vector< string > > &structs, const vector< string > &uniforms)
Determine alignment.
File system singleton class.
Properties class, which helps out with storeing or loading key value pairs from/to property files.
void put(const string &key, const string &value)
Add property.
const string & get(const string &key, const string &defaultValue) const
Get property value by key.
void load(const string &pathName, const string &fileName, FileSystemInterface *fileSystem=nullptr)
Load property file.
void store(const string &pathName, const string &fileName, FileSystemInterface *fileSystem=nullptr) const
Store property file.
const string & nextToken()
void tokenize(const string &str, const string &delimiters)
Tokenize.
unordered_map< int32_t, string > uniforms
vector< shader_type * > shaders
unordered_map< string, uniform_type * > uniforms
vector< attribute_layout > attributeLayouts
VkShaderStageFlagBits type