xash3d-fwgs/ref_vk/sebastian.py

709 lines
19 KiB
Python
Raw Normal View History

2022-11-14 00:32:51 +01:00
#!/usr/bin/env python3
import json
import argparse
import struct
import copy
from spirv import spv
import os
2022-11-14 00:07:36 +01:00
# import sys
# print(sys.argv, file=sys.stderr)
parser = argparse.ArgumentParser(description='Build pipeline descriptor')
parser.add_argument('--path', '-p', help='Directory where to look for .spv shader files')
2022-10-15 21:13:52 +02:00
parser.add_argument('--output', '-o', type=argparse.FileType('wb'), help='Compiled pipeline')
parser.add_argument('--depend', '-d', type=argparse.FileType('w'), help='Generate dependency file (json)')
parser.add_argument('pipelines', type=argparse.FileType('r'))
# TODO strip debug OpName OpLine etc
args = parser.parse_args()
spvOp = spv['Op']
spvOpNames = dict()
for name, n in spvOp.items():
spvOpNames[n] = name
2022-11-14 07:55:38 +01:00
#print("cwd", os.path.abspath('.'), file=sys.stderr)
2022-11-14 00:40:25 +01:00
src_dir = os.path.abspath(os.path.dirname(args.pipelines.name))
2022-11-14 07:55:38 +01:00
#print("src", src_dir, file=sys.stderr)
2022-11-14 00:40:25 +01:00
2022-11-14 00:07:36 +01:00
# #dst_dir = os.path.abspath(os.path.dirname(args.output.name))
# #print("dst", dst_dir, file=sys.stderr)
shaders_path = os.path.abspath(args.path if args.path else '.')
2022-11-14 07:55:38 +01:00
#print("shaders_path", shaders_path, file=sys.stderr)
2022-11-14 01:21:45 +01:00
def removeprefix(s, pre):
return s[len(pre):] if s.startswith(pre) else s
# remove comment lines and fix comma
def prepareJSON(path):
raw_json = buffer = result = ""
onecomment = blockcomment = 0
for char in path.read():
if (len(buffer) > 1):
buffer = buffer[1:]
buffer += char
if buffer == "*/":
blockcomment = 0
raw_json = raw_json[:-1]
elif blockcomment:
continue
elif buffer == "/*":
blockcomment = 1
elif char == "\n" or char == "\r":
buffer = ""
onecomment = 0
elif char == "\t" or char == " " or onecomment:
continue
elif buffer == "//":
raw_json = raw_json[:-1]
onecomment = 1
elif buffer != "":
raw_json += char
raw_json = raw_json.replace(",]","]")
raw_json = raw_json.replace(",}","}")
try:
result = json.loads(raw_json)
2022-10-29 22:25:00 +02:00
#print(json.dumps(result, sort_keys=False, indent=4))
except json.decoder.JSONDecodeError as exp:
print("Decoding JSON has failed")
print(raw_json)
raise
return result
2022-10-22 23:44:31 +02:00
class Serializer:
def __init__(self, file):
self.file = file
def write(self, v):
self.file.write(v)
def writeU32(self, v):
self.write(struct.pack('I', v))
def writeBytes(self, v):
self.writeU32(len(v))
self.write(v)
def writeString(self, v):
bs = v.encode('utf-8') + b'\x00'
rem = len(bs) % 4
if rem != 0:
bs += b'\x00' * (4 - rem)
self.writeBytes(bs)
def writeArray(self, v):
self.writeU32(len(v))
for i in v:
if isinstance(i, int):
self.writeU32(i)
else:
i.serialize(self)
class ImageFormat:
VK_FORMAT_UNDEFINED = 0
VK_FORMAT_R8G8B8A8_UNORM = 37
VK_FORMAT_R16G16B16A16_SFLOAT = 97
VK_FORMAT_R32G32B32A32_SFLOAT = 109
__map = {
'Unknown': VK_FORMAT_UNDEFINED,
'Rgba32f' : VK_FORMAT_R32G32B32A32_SFLOAT,
'Rgba16f' : VK_FORMAT_R16G16B16A16_SFLOAT,
'Rgba8' : VK_FORMAT_R8G8B8A8_UNORM,
# TODO map more
}
__revmap = None
def mapToVk(fmt):
if not ImageFormat.__revmap:
revmap = {}
formats = spv['ImageFormat']
for k, v in formats.items():
if k in ImageFormat.__map:
revmap[v] = ImageFormat.__map[k]
ImageFormat.__revmap = revmap
return ImageFormat.__revmap[fmt]
class TypeInfo:
2022-11-05 22:27:45 +01:00
TYPE_SAMPLER = 0
TYPE_COMBINED_IMAGE_SAMPLER = 1
TYPE_SAMPLED_IMAGE = 2
TYPE_STORAGE_IMAGE = 3
TYPE_UNIFORM_TEXEL_BUFFER = 4
TYPE_STORAGE_TEXEL_BUFFER = 5
TYPE_UNIFORM_BUFFER = 6
TYPE_STORAGE_BUFFER = 7
TYPE_UNIFORM_BUFFER_DYNAMIC = 8
TYPE_STORAGE_BUFFER_DYNAMIC = 9
TYPE_INPUT_ATTACHMENT = 10
TYPE_INLINE_UNIFORM_BLOCK = 1000138000
TYPE_ACCELERATION_STRUCTURE_KHR = 1000150000
TYPE_MUTABLE_VALVE = 1000351000
TYPE_SAMPLE_WEIGHT_IMAGE_QCOM = 1000440000
TYPE_BLOCK_MATCH_IMAGE_QCOM = 1000440001
def __init__(self, type=None, parent=None, count=1):
self.type = parent.type if parent else type
self.is_image = parent.is_image if parent else False
self.image_format = parent.image_format if parent else None
self.count = count
# TODO self.writable = None
# TODO self.readable = None
def __eq__(self, other):
if self.type != other.type:
return False
if self.count != other.count:
return False
assert(self.is_image == other.is_image)
if self.is_image:
assert(self.image_format != None)
assert(other.image_format != None)
if self.image_format != other.image_format and self.image_format != 0 and other.image_format != 0:
return False
return True
def serialize(self, out):
out.writeU32(self.type)
out.writeU32(self.count)
if self.is_image:
out.writeU32(ImageFormat.mapToVk(self.image_format))
class SpirvNode:
def __init__(self):
self.descriptor_set = None
self.binding = None
self.name = None
self.parent_type_node = None
2022-11-05 22:27:45 +01:00
self.type = None
self.storage_class = None
self.value = None
self.count = None
2022-11-05 22:27:45 +01:00
def getType(self):
node = self
while node:
#print(f"Checking node {node.name}, {node.storage_class}, {node.type}")
if node.type:
return node.type
if node.count:
return TypeInfo(parent=node.parent_type_node.getType(), count = node.count)
node = node.parent_type_node
raise Exception('Couldn\'t find type for node %s' % self.name)
2022-11-05 22:27:45 +01:00
def setStorageClass(self, storage_class):
if storage_class == spv['StorageClass']['Uniform']:
# TODO support older spirv shaders
# E.g. for the same
# layout (set = 0, binding = 7) readonly buffer SBOLights { LightsMetadata m; } lights;
# Newer, with --target-env=vulkan-1.2:
# OpName %lights "lights"
# OpDecorate %lights DescriptorSet 0
# OpDecorate %lights Binding 7
# %lights = OpVariable %_ptr_StorageBuffer_SBOLights StorageBuffer
#
# %_ptr_StorageBuffer_SBOLights = OpTypePointer StorageBuffer %SBOLights
#
# OpName %SBOLights "SBOLights"
# OpMemberName %SBOLights 0 "m"
# OpMemberDecorate %SBOLights 0 NonWritable
# OpMemberDecorate %SBOLights 0 Offset 0
# OpDecorate %SBOLights Block
# Older:
# OpName %lights "lights"
# OpDecorate %lights DescriptorSet 0
# OpDecorate %lights Binding 7
# %lights = OpVariable %_ptr_Uniform_SBOLights Uniform
#
# %_ptr_Uniform_SBOLights = OpTypePointer Uniform %SBOLights
#
# OpName %SBOLights "SBOLights"
# OpMemberName %SBOLights 0 "m"
# OpMemberDecorate %SBOLights 0 NonWritable
# OpMemberDecorate %SBOLights 0 Offset 0
# OpDecorate %SBOLights BufferBlock
# %SBOLights = OpTypeStruct %LightsMetadata
self.type = TypeInfo(TypeInfo.TYPE_UNIFORM_BUFFER)
elif storage_class == spv['StorageClass']['StorageBuffer']:
self.type = TypeInfo(TypeInfo.TYPE_STORAGE_BUFFER)
self.storage_class = storage_class
#print(f"Set node {self.name} storage class {storage_class}, type {self.type}")
class SpirvContext:
def __init__(self, nodes_count):
self.nodes = [SpirvNode() for i in range(0, nodes_count)]
#self.bindings = dict()
pass
def getNode(self, index):
return self.nodes[index]
2022-11-05 22:27:45 +01:00
def spvOpName(ctx, args):
index = args[0]
name = struct.pack(str(len(args)-1)+'I', *args[1:]).split(b'\x00')[0].decode('utf8')
ctx.getNode(index).name = name
#print('Name for', args[0], name, len(name))
2022-11-05 22:27:45 +01:00
def spvOpDecorate(ctx, args):
node = ctx.getNode(args[0])
decor = args[1]
if decor == spv['Decoration']['DescriptorSet']:
node.descriptor_set = args[2]
elif decor == spv['Decoration']['Binding']:
node.binding = args[2]
#else:
#print('Decor ', id, decor)
2022-11-05 22:27:45 +01:00
def spvOpVariable(ctx, args):
parent_type_node = ctx.getNode(args[0])
2022-11-05 22:27:45 +01:00
node = ctx.getNode(args[1])
storage_class = args[2]
node.parent_type_node = parent_type_node
node.setStorageClass(storage_class)
2022-11-05 22:27:45 +01:00
#node.op_type = 'OpVariable'
#print(node.name, "=(var)>", parent_type_node.name, args[0])
2022-11-05 22:27:45 +01:00
def spvOpTypePointer(ctx, args):
node = ctx.getNode(args[0])
storage_class = args[1]
parent_type_node = ctx.getNode(args[2])
2022-11-05 22:27:45 +01:00
node.parent_type_node = parent_type_node
node.setStorageClass(storage_class)
2022-11-05 22:27:45 +01:00
#node.op_type = 'OpTypePointer'
#print(node.name, "=(ptr)>", parent_type_node.name, args[2])
2022-11-05 22:27:45 +01:00
def spvOpTypeAccelerationStructureKHR(ctx, args):
node = ctx.getNode(args[0])
node.type = TypeInfo(TypeInfo.TYPE_ACCELERATION_STRUCTURE_KHR)
2022-11-05 22:27:45 +01:00
def spvOpTypeImage(ctx, args):
node = ctx.getNode(args[0])
sampled_type = args[1]
dim = args[2]
depth = args[3]
arrayed = args[4]
ms = args[5]
sampled = args[6]
image_format = args[7]
type = TypeInfo.TYPE_STORAGE_IMAGE if sampled == 0 or sampled == 2 else TypeInfo.TYPE_COMBINED_IMAGE_SAMPLER # FIXME ?
node.type = TypeInfo(type)
node.type.is_image = True
node.type.image_format = image_format
2022-11-05 22:27:45 +01:00
qualifier = None if len(args) < 9 else args[8]
#print(f"{args[0]}: Image(type={sampled_type}, dim={dim}, depth={depth}, arrayed={arrayed}, ms={ms}, sampled={sampled}, image_format={image_format}, qualifier={qualifier})")
def spvOpTypeSampledImage(ctx, args):
node = ctx.getNode(args[0])
image_type = ctx.getNode(args[1])
node.parent_type_node = image_type
node.type = TypeInfo(TypeInfo.TYPE_COMBINED_IMAGE_SAMPLER)
node.type.is_image = True
node.type.image_format = spv['ImageFormat']['Unknown']
2022-11-05 22:27:45 +01:00
def spvOpTypeArray(ctx, args):
node = ctx.getNode(args[0])
element_type = ctx.getNode(args[1])
length_node = args[2]
2022-11-05 22:27:45 +01:00
node.count = ctx.getNode(length_node).value
node.parent_type_node = element_type
#print(f"{args[0]}: Array(type={args[1]}, length={node.count})")
def spvOpSpecConstant(ctx, args):
type_node = ctx.getNode(args[0])
node = ctx.getNode(args[1])
value = args[2]
# TODO mind the type
node.value = value
2022-11-05 22:27:45 +01:00
spvOpHandlers = {
2022-11-05 22:27:45 +01:00
spvOp['OpName']: spvOpName,
spvOp['OpDecorate']: spvOpDecorate,
spvOp['OpVariable']: spvOpVariable,
spvOp['OpTypePointer']: spvOpTypePointer,
spvOp['OpTypeAccelerationStructureKHR']: spvOpTypeAccelerationStructureKHR,
spvOp['OpTypeImage']: spvOpTypeImage,
spvOp['OpTypeSampledImage']: spvOpTypeSampledImage,
spvOp['OpTypeArray']: spvOpTypeArray,
spvOp['OpSpecConstant']: spvOpSpecConstant,
}
def parseSpirv(raw_data):
if len(raw_data) % 4 != 0:
raise Exception('SPIR-V size should be divisible by 4')
size = len(raw_data) // 4
if size < 5:
raise Exception('SPIR-V data is too short')
data = struct.unpack(str(size) + 'I', raw_data)
if data[0] != spv['MagicNumber']:
raise Exception('Unexpected magic ' + str(data[0]))
nodes_count = data[3]
ctx = SpirvContext(nodes_count)
off = 5
while off < size:
op = data[off] & 0xffff
words = data[off] >> 16
args = data[off+1:off+words]
if op in spvOpHandlers:
spvOpHandlers[op](ctx, args)
#print(spvOpNames[op], args)
off += words
2022-10-15 20:24:56 +02:00
return ctx
2022-11-19 21:02:18 +01:00
class NameIndex:
def __init__(self):
self.__name_to_index = {}
self.__all = []
def getIndex(self, name):
return self.__name_to_index[name] if name in self.__name_to_index else -1
def getByName(self, name):
return self.__all[self.__name_to_index[name]] if name in self.__name_to_index else None
def getByIndex(self, index):
return self.__all[index]
def put(self, name, value):
if name in self.__name_to_index:
raise Exception('Already have a value for "%s"' % (name))
index = len(self.__all)
self.__all.append(value)
self.__name_to_index[name] = index
return index
def serialize(self, out):
out.writeArray(self.__all)
class Resources:
def __init__(self):
self.__storage = NameIndex()
def getIndex(self, name, node):
index = self.__storage.getIndex(name)
if index >= 0:
res = self.__storage.getByIndex(index)
res.checkSameType(node)
2022-11-19 21:02:18 +01:00
return index
return self.__storage.put(name, self.Resource(name, node))
2022-11-19 21:02:18 +01:00
def serialize(self, out):
self.__storage.serialize(out)
class Resource:
def __init__(self, name, node):
2022-11-19 21:02:18 +01:00
self.__name = name
self.__type = node.getType()
2022-11-19 21:02:18 +01:00
#TODO: count, etc
def checkSameType(self, node):
if self.__type != node.getType():
2022-11-19 21:02:18 +01:00
raise Exception('Conflicting types for resource "%s": %s != %s' % (self.__name, self.__type, type))
def serialize(self, out):
out.writeString(self.__name)
self.__type.serialize(out)
2022-11-19 21:02:18 +01:00
resources = Resources()
class Binding:
STAGE_VERTEX_BIT = 0x00000001
STAGE_TESSELLATION_CONTROL_BIT = 0x00000002
STAGE_TESSELLATION_EVALUATION_BIT = 0x00000004
STAGE_GEOMETRY_BIT = 0x00000008
STAGE_FRAGMENT_BIT = 0x00000010
STAGE_COMPUTE_BIT = 0x00000020
STAGE_ALL_GRAPHICS = 0x0000001F
STAGE_ALL = 0x7FFFFFFF
STAGE_RAYGEN_BIT_KHR = 0x00000100
STAGE_ANY_HIT_BIT_KHR = 0x00000200
STAGE_CLOSEST_HIT_BIT_KHR = 0x00000400
STAGE_MISS_BIT_KHR = 0x00000800
STAGE_INTERSECTION_BIT_KHR = 0x00001000
STAGE_CALLABLE_BIT_KHR = 0x00002000
STAGE_TASK_BIT_NV = 0x00000040
STAGE_MESH_BIT_NV = 0x00000080
STAGE_SUBPASS_SHADING_BIT_HUAWEI = 0x00004000
WRITE_BIT = 0x80000000
2022-11-05 22:27:45 +01:00
def __init__(self, node):
self.write = node.name.startswith('out_')
self.index = node.binding
self.descriptor_set = node.descriptor_set
self.stages = 0
2022-11-19 21:02:18 +01:00
resource_name = removeprefix(node.name, 'out_')
self.__resource_index = resources.getIndex(resource_name, node)
2022-11-05 22:27:45 +01:00
assert(self.descriptor_set >= 0)
assert(self.descriptor_set < 255)
assert(self.index >= 0)
assert(self.index < 255)
# For sorting READ-ONLY first, and WRITE later
def __lt__(self, right):
if self.write < right.write:
return True
elif self.write > right.write:
return False
if self.descriptor_set < right.descriptor_set:
return True
elif self.descriptor_set > right.descriptor_set:
return False
if self.index < right.index:
return True
elif self.index > right.index:
return False
return self.__resource_index < right.__resource_index
def __str__(self):
return f"Binding(resource_index={self.__resource_index}, ds={self.descriptor_set}, b={self.index}, write={self.write}, stages={self.stages})"
def __repr__(self):
return self.__str__()
def serialize(self, out):
header = (Binding.WRITE_BIT if self.write else 0) | (self.descriptor_set << 8) | self.index
out.writeU32(header)
2022-11-19 21:02:18 +01:00
out.writeU32(self.__resource_index)
out.writeU32(self.stages)
class Shader:
def __init__(self, name, fullpath):
self.name = name
self.__fullpath = fullpath
self.__raw_data = None
self.__bindings = None
2022-11-05 22:27:45 +01:00
#print(name, '=>', len(self.raw_data))
2022-10-15 20:24:56 +02:00
def __str__(self):
return self.name
# ret = ''
# for index, node in enumerate(self.__spirv.nodes):
# if node.descriptor_set is not None:
# ret += ('[%d:%d] (id=%d) %s\n' % (node.descriptor_set, node.binding, index, node.name))
# return ret
def getRawData(self):
if not self.__raw_data:
self.__raw_data = open(self.__fullpath, 'rb').read()
return self.__raw_data
2022-10-15 20:24:56 +02:00
def getBindings(self):
if self.__bindings:
return self.__bindings
#print("Parsing", self.name)
2022-11-19 21:02:18 +01:00
spirv = parseSpirv(self.getRawData())
bindings = []
for node in spirv.nodes:
if node.binding == None or node.descriptor_set == None:
continue
bindings.append(Binding(node))
self.__bindings = bindings
return self.__bindings
def getFilePath(self):
return self.__fullpath
2022-10-15 21:13:52 +02:00
class Shaders:
__suffixes = {
Binding.STAGE_COMPUTE_BIT: '.comp.spv',
Binding.STAGE_RAYGEN_BIT_KHR: '.rgen.spv',
Binding.STAGE_ANY_HIT_BIT_KHR: '.rahit.spv',
Binding.STAGE_CLOSEST_HIT_BIT_KHR: '.rchit.spv',
Binding.STAGE_MISS_BIT_KHR: '.rmiss.spv'
}
2022-10-15 21:13:52 +02:00
def __init__(self):
self.__map = dict()
self.__shaders = []
def load(self, name, stage):
name = name + self.__suffixes[stage]
fullpath = os.path.join(shaders_path, name)
2022-10-15 21:13:52 +02:00
if name in self.__map:
return self.__shaders[self.__map[name]]
shader = Shader(name, fullpath)
2022-10-15 21:13:52 +02:00
index = len(self.__shaders)
self.__shaders.append(shader)
self.__map[name] = index
return shader
2022-11-19 21:02:18 +01:00
def parse(self):
for s in self.__shaders:
s.getBindings()
def getIndex(self, shader):
return self.__map[shader.name]
2022-10-22 23:44:31 +02:00
def serialize(self, out):
out.writeU32(len(self.__shaders))
2022-10-15 21:13:52 +02:00
for shader in self.__shaders:
2022-10-22 23:44:31 +02:00
out.writeString(shader.name)
out.writeBytes(shader.getRawData())
def getAllFiles(self):
return [shader.getFilePath() for shader in self.__shaders]
2022-10-15 21:13:52 +02:00
shaders = Shaders()
2022-10-15 21:13:52 +02:00
PIPELINE_COMPUTE = 1
PIPELINE_RAYTRACING = 2
NO_SHADER = 0xffffffff
class Pipeline:
def __init__(self, name, type_id):
self.name = name
self.type = type_id
self.__shaders = []
def addShader(self, shader_name, stage):
shader = shaders.load(shader_name, stage)
self.__shaders.append((shader, stage))
return shader
def __mergeBindings(self):
bindings = {}
for shader, stage in self.__shaders:
for binding in shader.getBindings():
addr = (binding.descriptor_set, binding.index)
if not addr in bindings:
bindings[addr] = copy.deepcopy(binding)
bindings[addr].stages |= stage
return bindings
2022-10-15 21:13:52 +02:00
def serialize(self, out):
bindings = sorted(self.__mergeBindings().values())
2022-11-05 22:27:45 +01:00
#print(self.name)
#for binding in bindings:
#print(f" ds={binding.descriptor_set}, b={binding.index}, stages={binding.stages:#x}, write={binding.write}")
out.writeU32(self.type)
out.writeString(self.name)
out.writeArray(bindings)
class PipelineRayTracing(Pipeline):
__hit2stage = {
'closest': Binding.STAGE_CLOSEST_HIT_BIT_KHR,
'any': Binding.STAGE_ANY_HIT_BIT_KHR,
}
def __init__(self, name, desc):
super().__init__(name, PIPELINE_RAYTRACING)
self.rgen = self.addShader(desc['rgen'], Binding.STAGE_RAYGEN_BIT_KHR)
self.miss = [] if not 'miss' in desc else [self.addShader(s, Binding.STAGE_MISS_BIT_KHR) for s in desc['miss']]
self.hit = [] if not 'hit' in desc else [self.__loadHit(hit) for hit in desc['hit']]
2022-10-15 21:13:52 +02:00
def serialize(self, out):
super().serialize(out)
out.writeU32(shaders.getIndex(self.rgen))
out.writeArray([shaders.getIndex(s) for s in self.miss])
2022-10-15 21:13:52 +02:00
out.writeU32(len(self.hit))
2022-10-15 21:13:52 +02:00
for hit in self.hit:
out.writeU32(shaders.getIndex(hit['closest']) if 'closest' in hit else NO_SHADER)
out.writeU32(shaders.getIndex(hit['any']) if 'any' in hit else NO_SHADER)
def __loadHit(self, hit):
ret = dict()
for k, v in hit.items():
ret[k] = self.addShader(v, self.__hit2stage[k])
return ret
class PipelineCompute(Pipeline):
def __init__(self, name, desc):
super().__init__(name, PIPELINE_COMPUTE)
self.comp = self.addShader(desc['comp'], Binding.STAGE_COMPUTE_BIT)
2022-10-15 21:13:52 +02:00
def serialize(self, out):
super().serialize(out)
out.writeU32(shaders.getIndex(self.comp))
def parsePipeline(pipelines, name, desc):
if 'inherit' in desc:
inherit = pipelines[desc['inherit']]
for k, v in inherit.items():
if not k in desc:
desc[k] = v
if 'rgen' in desc:
return PipelineRayTracing(name, desc)
elif 'comp' in desc:
return PipelineCompute(name, desc)
def loadPipelines():
pipelines_desc = prepareJSON(args.pipelines)
pipelines = dict()
for k, v in pipelines_desc.items():
if 'template' in v and v['template']:
continue
pipelines[k] = parsePipeline(pipelines_desc, k, v)
return pipelines
2022-10-22 23:44:31 +02:00
def writeOutput(file, pipelines):
2022-10-15 21:13:52 +02:00
MAGIC = bytearray([ord(c) for c in 'MEAT'])
2022-10-22 23:44:31 +02:00
out = Serializer(file)
2022-10-15 21:13:52 +02:00
out.write(MAGIC)
2022-11-19 21:02:18 +01:00
resources.serialize(out)
2022-10-15 21:13:52 +02:00
shaders.serialize(out)
out.writeArray(pipelines.values())
2022-10-15 21:13:52 +02:00
pipelines = loadPipelines()
2022-10-15 21:13:52 +02:00
if args.depend:
json.dump([os.path.relpath(file) for file in shaders.getAllFiles()], args.depend)
2022-10-15 21:13:52 +02:00
if args.output:
2022-11-19 21:02:18 +01:00
shaders.parse()
2022-10-15 21:13:52 +02:00
writeOutput(args.output, pipelines)