gtk/gsk/gpu/shaders/generate-header.py
Benjamin Otte b481fd854f gpu: Encode number of textures use in every shader
Just define GSK_N_TEXTURES in every glsl file, extract that #define in
the python parser and emit a static const uint variable
"{shader_name}_n_textures" in the generated header.
2024-07-22 18:37:07 +02:00

338 lines
13 KiB
Python

#!/usr/bin/env python3
import sys
import re
import os
name = os.path.splitext(os.path.splitext(os.path.basename(sys.argv[1]))[0])[0][6:]
var_name = "gsk_gpu_" + name.replace('-', '_')
struct_name = "GskGpu" + name.title().replace('-', '') + "Instance"
n_textures = -1
filename = sys.argv[1]
with open(filename) as f:
lines = f.readlines()
matches = []
for pos, line in enumerate (lines):
match = re.search(r"^#define GSK_N_TEXTURES ([0-9]+)$", line)
if match:
n_textures = int(match.group(1))
match = re.search(r"^IN\(([0-9]+)\) ([a-z0-9]+) ([a-zA-Z0-9_]+);$", line)
if not match:
if re.search(r"layout.*\sin\s.*", line):
raise Exception(f'''{filename}:{pos}: Failed to parse file''')
continue
if not match.group(3).startswith('in'):
raise Exception(f'''{filename}:{pos}: Variable doesn't start with "in"''')
matches.append({'name': ''.join('_' + char.lower() if char.isupper() else char for char in match.group(3))[3:],
'attrib_name': match.group(3),
'location': int(match.group(1)),
'type': match.group(2)})
if n_textures < 0:
raise Exception(f'''{filename}: GSK_N_TEXTURES not defined''')
if n_textures > 2:
raise Exception(f'''{filename}: GSK_N_TEXTURES must be <= 2''')
print(f'''/* This file is auto-generated; any change will not be preserved */
#pragma once
#define {var_name}_n_textures {n_textures}
typedef struct _{struct_name} {struct_name};
struct _{struct_name} {{''')
expected = 0
for match in matches:
if expected != int(match['location']):
raise Exception(f"{filename}: Should be layout location {expected} but is {match['location']}") # noqa
if match['type'] == 'float':
print(f" float {match['name']};")
expected += 1
elif match['type'] == 'int':
print(f" gint32 {match['name']};")
expected += 1
elif match['type'] == 'uint':
print(f" guint32 {match['name']};")
expected += 1
elif match['type'] == 'uvec2':
print(f" guint32 {match['name']}[2];")
expected += 1
elif match['type'] == 'vec2':
print(f" float {match['name']}[2];")
expected += 1
elif match['type'] == 'vec3':
print(f" float {match['name']}[3];")
expected += 1
elif match['type'] == 'vec4':
print(f" float {match['name']}[4];")
expected += 1
elif match['type'] == 'mat3x4':
print(f" float {match['name']}[12];")
expected += 3
elif match['type'] == 'mat4':
print(f" float {match['name']}[16];")
expected += 4
else:
raise Exception(f"{filename}: Don't know what a {match['type']} is")
print(f'''}};
''')
print(f'''static inline void
{var_name}_setup_vao (gsize offset)
{{''')
for i, match in enumerate(matches):
if match['type'] == 'float':
print(f''' glEnableVertexAttribArray ({match['location']});
glVertexAttribDivisor ({match['location']}, 1);
glVertexAttribPointer ({match['location']},
1,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']})));''');
elif match['type'] == 'uint':
print(f''' glEnableVertexAttribArray ({match['location']});
glVertexAttribDivisor ({match['location']}, 1);
glVertexAttribIPointer ({match['location']},
1,
GL_UNSIGNED_INT,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']})));''');
elif match['type'] == 'uvec2':
print(f''' glEnableVertexAttribArray ({match['location']});
glVertexAttribDivisor ({match['location']}, 1);
glVertexAttribIPointer ({match['location']},
2,
GL_UNSIGNED_INT,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']})));''');
elif match['type'] == 'vec2':
print(f''' glEnableVertexAttribArray ({match['location']});
glVertexAttribDivisor ({match['location']}, 1);
glVertexAttribPointer ({match['location']},
2,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']})));''');
elif match['type'] == 'vec3':
print(f''' glEnableVertexAttribArray ({match['location']});
glVertexAttribDivisor ({match['location']}, 1);
glVertexAttribPointer ({match['location']},
3,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']})));''');
elif match['type'] == 'vec4':
print(f''' glEnableVertexAttribArray ({match['location']});
glVertexAttribDivisor ({match['location']}, 1);
glVertexAttribPointer ({match['location']},
4,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']})));''');
elif match['type'] == 'mat3x4':
print(f''' glEnableVertexAttribArray ({match['location']});
glVertexAttribDivisor ({match['location']}, 1);
glVertexAttribPointer ({match['location']},
4,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']})));
glEnableVertexAttribArray ({int(match['location'] + 1)});
glVertexAttribDivisor ({int(match['location'] + 1)}, 1);
glVertexAttribPointer ({int(match['location']) + 1},
4,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 4));
glEnableVertexAttribArray ({int(match['location'] + 2)});
glVertexAttribDivisor ({int(match['location'] + 2)}, 1);
glVertexAttribPointer ({int(match['location']) + 2},
4,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 8));''')
elif match['type'] == 'mat4':
print(f''' glEnableVertexAttribArray ({match['location']});
glVertexAttribDivisor ({match['location']}, 1);
glVertexAttribPointer ({match['location']},
4,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']})));
glEnableVertexAttribArray ({int(match['location'] + 1)});
glVertexAttribDivisor ({int(match['location'] + 1)}, 1);
glVertexAttribPointer ({int(match['location']) + 1},
4,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 4));
glEnableVertexAttribArray ({int(match['location'] + 2)});
glVertexAttribDivisor ({int(match['location'] + 2)}, 1);
glVertexAttribPointer ({int(match['location']) + 2},
4,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 8));
glEnableVertexAttribArray ({int(match['location'] + 3)});
glVertexAttribDivisor ({int(match['location'] + 3)}, 1);
glVertexAttribPointer ({int(match['location']) + 3},
4,
GL_FLOAT,
GL_FALSE,
sizeof ({struct_name}),
GSIZE_TO_POINTER (offset + G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 12));''')
else:
raise Exception(f"{filename}: Don't know what a {match['type']} is")
print(f'''}}
''');
print(f'''static void
{var_name}_setup_attrib_locations (GLuint program)
{{''')
for match in matches:
print(f''' glBindAttribLocation (program, {match['location']}, "{match['attrib_name']}");''')
print(f'''}}
''');
print(f'''#ifdef GDK_RENDERING_VULKAN
static const VkPipelineVertexInputStateCreateInfo {var_name}_info = {{
.sType = VK_STRUCTURE_TYPE_PIPELINE_VERTEX_INPUT_STATE_CREATE_INFO,
.vertexBindingDescriptionCount = 1,
.pVertexBindingDescriptions = (VkVertexInputBindingDescription[1]) {{
{{
.binding = 0,
.stride = sizeof ({struct_name}),
.inputRate = VK_VERTEX_INPUT_RATE_INSTANCE
}}
}},
.vertexAttributeDescriptionCount = {expected},
.pVertexAttributeDescriptions = (VkVertexInputAttributeDescription[{expected}]) {{''')
for match in matches:
if match['type'] == 'float':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},''')
elif match['type'] == 'int':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32_SINT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},''')
elif match['type'] == 'uint':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32_UINT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},''')
elif match['type'] == 'uvec2':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32G32_UINT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},''')
elif match['type'] == 'vec2':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32G32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},''')
elif match['type'] == 'vec3':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32G32B32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},''')
elif match['type'] == 'vec4':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32G32B32A32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},''')
elif match['type'] == 'mat3x4':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32G32B32A32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},
{{
.location = {int(match['location']) + 1},
.binding = 0,
.format = VK_FORMAT_R32G32B32A32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 4,
}},
{{
.location = {int(match['location']) + 2},
.binding = 0,
.format = VK_FORMAT_R32G32B32A32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 8,
}},''')
elif match['type'] == 'mat4':
print(f''' {{
.location = {match['location']},
.binding = 0,
.format = VK_FORMAT_R32G32B32A32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}),
}},
{{
.location = {int(match['location']) + 1},
.binding = 0,
.format = VK_FORMAT_R32G32B32A32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 4,
}},
{{
.location = {int(match['location']) + 2},
.binding = 0,
.format = VK_FORMAT_R32G32B32A32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 8,
}},
{{
.location = {int(match['location']) + 3},
.binding = 0,
.format = VK_FORMAT_R32G32B32A32_SFLOAT,
.offset = G_STRUCT_OFFSET({struct_name}, {match['name']}) + sizeof (float) * 12,
}},''')
else:
raise Exception(f"{filename}: Don't know what a {match['type']} is")
print(f''' }},
}};
#endif
''');