Shader structure done.

This commit is contained in:
2024-12-29 10:09:20 -06:00
parent ba305de596
commit f574b60856
19 changed files with 382 additions and 262 deletions

View File

@ -1,9 +1,18 @@
uniform float4x4 u_Projection;
uniform float4x4 u_View;
uniform float4x4 u_Model;
uniform float4 u_Color;
uniform bool u_HasTexture;
uniform Sampler2D u_Texture;
struct Transforms {
float4x4 projection;
float4x4 view;
float4x4 model;
}
struct Colors {
bool hasTexture;
float4 colors[4];
int somethingElse;
Sampler2D texture;
}
uniform Transforms transforms;
uniform Colors colors;
struct AssembledVertex {
float3 position : POSITION;
@ -19,10 +28,6 @@ struct VertexStageOutput {
float4 sv_position : SV_Position;
};
float4 someFunction(float4 color) {
return color * float4(0.5, 0.5, 0.5, 1.0);
}
[shader("vertex")]
VertexStageOutput vertexMain(
AssembledVertex assembledVertex
@ -35,7 +40,7 @@ VertexStageOutput vertexMain(
output.sv_position = mul(
float4(position, 1.0),
mul(u_Model, mul(u_View, u_Projection))
mul(transforms.model, mul(transforms.view, transforms.projection))
);
return output;
@ -46,10 +51,10 @@ Fragment fragmentMain(
float2 uv: UV
) : SV_Target {
Fragment output;
if(u_HasTexture) {
output.color = u_Texture.Sample(uv) * u_Color;
if (colors.hasTexture) {
output.color = colors.texture.Sample(uv) * colors.colors[0];
} else {
output.color = someFunction(u_Color);
output.color = colors.colors[0] - float4(colors.somethingElse, 0, 0, 0);
}
return output;
}

View File

@ -8,10 +8,6 @@
#include "asset/AssetManager.hpp"
#include "game/Game.hpp"
#if DAWN_DEBUG_SHADERS
#include <fstream>
#endif
using namespace Dawn;
const std::string ShaderLoader::ASSET_TYPE = "shader";
@ -39,93 +35,13 @@ void ShaderLoader::updateSync() {
// Load the shader string
Slang::ComPtr<IBlob> diagnostics;
module = sm->session->loadModule(
auto module = sm->session->loadModule(
this->name.c_str(),
diagnostics.writeRef()
);
// Get list of entry points and create components
int32_t definedEntryPointCount = module->getDefinedEntryPointCount();
IComponentType** components = new IComponentType*[definedEntryPointCount + 1];
int32_t componentCount = 0;
components[componentCount++] = module;// First component is module.
// Get the entry point info and append to components list.
for(auto i = 0; i < definedEntryPointCount; i++) {
Slang::ComPtr<IEntryPoint> ep;
auto result = module->getDefinedEntryPoint(i, ep.writeRef());
if(result != SLANG_OK) assertUnreachable("Failed to get entry point.");
components[componentCount++] = ep;
}
// Create the composite component type
sm->session->createCompositeComponentType(
components,
componentCount,
program.writeRef(),
diagnostics.writeRef()
);
if(diagnostics) {
assertUnreachable("%s\n", (const char*) diagnostics->getBufferPointer());
return;
}
// Link the program.
auto result = program->link(linkedProgram.writeRef(), diagnostics.writeRef());
if(diagnostics) {
assertUnreachable("%s\n", (const char*) diagnostics->getBufferPointer());
return;
}
// Create the shader program.
Slang::ComPtr<IBlob> blob;
slang::ProgramLayout* layout = program->getLayout();
std::vector<std::shared_ptr<ShaderStage>> shaderStages;
for(auto i = 0; i < definedEntryPointCount; i++) {
// Get the code
auto result = linkedProgram->getEntryPointCode(
i,
0,
blob.writeRef(),
diagnostics.writeRef()
);
if(diagnostics) {
assertUnreachable("%s\n", (const char*) diagnostics->getBufferPointer());
}
// Get the stage information
auto entryPointReflection = layout->getEntryPointByIndex(i);
auto stage = entryPointReflection->getStage();
// Write out to file for debugging
#if DAWN_DEBUG_SHADERS
std::filesystem::path filePath =
"debug/shaders/" + this->name + "/" + std::to_string(i) + ".glsl"
;
std::filesystem::create_directories(filePath.parent_path());
std::cout << "Writing shader to " << filePath << std::endl;
std::ofstream file(filePath);
file << (const char*)blob->getBufferPointer();
file.close();
#endif
// Create the shader entry
auto shaderStage = std::make_shared<ShaderStage>();
shaderStage->init(
stage,
std::string((const char*)blob->getBufferPointer())
);
// Add to the list
shaderStages.push_back(shaderStage);
}
// Create the shader program.
shader->init(shaderStages);
shader->init(module, sm->session);
// Finished loading.
delete [] components;
this->state = ShaderLoaderState::LOADED;
this->loaded = true;
}
@ -141,9 +57,4 @@ std::shared_ptr<ShaderProgram> ShaderLoader::getShader() {
ShaderLoader::~ShaderLoader() {
shader = nullptr;
if(linkedProgram) {
linkedProgram->release();
linkedProgram = nullptr;
}
}

View File

@ -24,11 +24,6 @@ namespace Dawn {
public:
const static std::string ASSET_TYPE;
Slang::ComPtr<IComponentType> linkedProgram;
Slang::ComPtr<IComponentType> program;
IModule* module;
ShaderLoader(
const std::shared_ptr<AssetManager> assetManager,
const std::string name

View File

@ -28,21 +28,22 @@ void SimpleTexturedMaterial::load(std::shared_ptr<SceneLoadContext> ctx) {
}
struct Color SimpleTexturedMaterial::getColor() {
return this->data.color;
// return this->data.color;
return COLOR_WHITE;
}
std::shared_ptr<Texture> SimpleTexturedMaterial::getTexture() {
return this->data.texture;
return this->texture;
}
void SimpleTexturedMaterial::setTexture(
const std::shared_ptr<Texture> texture
) {
this->data.texture = texture;
this->texture = texture;
}
void SimpleTexturedMaterial::setColor(const struct Color color) {
this->data.color = color;
// this->data.color = color;
}
std::vector<std::shared_ptr<RenderPass>> SimpleTexturedMaterial::getPasses(

View File

@ -5,36 +5,13 @@
#pragma once
#include "component/display/material/Material.hpp"
#include "display/shader/ShaderData.hpp"
namespace Dawn {
struct SimpleTexturedMaterialShaderData : public ShaderData {
protected:
void writeData() override {
this->beginStruct();
// this->writeMat4(this->projection);
// this->writeMat4(this->view);
// this->writeMat4(this->model);
// this->writeColor(this->color);
// this->writeBoolean(this->hasTexture);
// this->writeTexture(this->texture);
this->endStruct();
}
public:
struct Color color;
glm::mat4 model;
glm::mat4 projection;
glm::mat4 view;
bool hasTexture;
std::shared_ptr<Texture> texture;
};
class SimpleTexturedMaterial : public Material {
private:
struct SimpleTexturedMaterialShaderData data;
std::shared_ptr<ShaderProgram> shader;
std::shared_ptr<Texture> texture;
protected:
void initShaderPrograms() override;

View File

@ -30,7 +30,7 @@ RenderPass::RenderPass(
}
void RenderPass::draw() {
// mesh->draw(drawMode, indiceStart, indiceCount);
mesh->draw(drawMode, indiceStart, indiceCount);
}
RenderPass::~RenderPass() {

View File

@ -11,4 +11,5 @@ target_sources(${DAWN_TARGET_NAME}
IShaderStage.cpp
IShaderProgram.cpp
IShaderData.cpp
ShaderStructure.cpp
)

View File

@ -4,17 +4,18 @@
// https://opensource.org/licenses/MIT
#include "IShaderData.hpp"
#include "assert/assert.hpp"
using namespace Dawn;
void IShaderData::write(std::shared_ptr<ShaderProgram> shader) {
assertNotNull(shader, "Shader cannot be null.");
this->shader = shader;
this->writeData();
this->shader = nullptr;
IShaderData::IShaderData(
const std::shared_ptr<ShaderStructure> &structure
) : structure(structure) {
data = (void *)malloc(structure->size);
}
std::shared_ptr<ShaderProgram> IShaderData::getShader() {
return this->shader;
IShaderData::~IShaderData() {
if(data != nullptr) {
free(data);
data = nullptr;
}
}

View File

@ -8,35 +8,16 @@
#include "display/shader/ShaderProgram.hpp"
namespace Dawn {
class IShaderData {
private:
std::shared_ptr<ShaderProgram> shader;
struct IShaderData {
protected:
virtual void writeData() = 0;
virtual void beginStruct() = 0;
virtual void writeMat4(const glm::mat4 &mat) = 0;
virtual void writeVec3(const glm::vec3 &vec) = 0;
virtual void writeVec4(const glm::vec4 &vec) = 0;
virtual void writeColor(const struct Color &color) = 0;
virtual void writeBoolean(const bool &value) = 0;
virtual void writeTexture(std::shared_ptr<Texture> &texture) = 0;
virtual void endStruct() = 0;
std::shared_ptr<ShaderStructure> structure;
void *data = nullptr;
public:
/**
* Writes the data to the shader.
*
* @param shader The shader to write to.
*/
void write(std::shared_ptr<ShaderProgram> shader);
IShaderData(
const std::shared_ptr<ShaderStructure> &structure
);
/**
* Returns the shader that was written to.
*
* @return The shader that was written to.
*/
std::shared_ptr<ShaderProgram> getShader();
virtual ~IShaderData();
};
}

View File

@ -4,14 +4,134 @@
// https://opensource.org/licenses/MIT
#include "IShaderProgram.hpp"
#include "assert/assert.hpp"
#if DAWN_DEBUG_SHADERS
#include <fstream>
#endif
using namespace Dawn;
void IShaderProgram::init(
const std::vector<std::shared_ptr<ShaderStage>> &stages
IModule *module,
Slang::ComPtr<ISession> session
) {
this->stages = stages;
assertNotNull(module, "Module cannot be null.");
this->module = module;
// Get list of entry points and create components
int32_t definedEntryPointCount = module->getDefinedEntryPointCount();
// +1 for module
auto components = new IComponentType*[definedEntryPointCount + 1];
int32_t componentCount = 0;
components[componentCount++] = module;// First component is module.
// Get the entry point info and append to components list.
for(auto i = 0; i < definedEntryPointCount; i++) {
auto result = module->getDefinedEntryPoint(
i,
(IEntryPoint**)(&components[componentCount++])
);
if(result != SLANG_OK) assertUnreachable("Failed to get entry point.");
}
// Create the composite component type
Slang::ComPtr<IBlob> diagnostics;
session->createCompositeComponentType(
components,
componentCount,
program.writeRef(),
diagnostics.writeRef()
);
if(diagnostics) {
assertUnreachable("%s\n", (const char*) diagnostics->getBufferPointer());
return;
}
// Link the program.
auto result = program->link(linkedProgram.writeRef(), diagnostics.writeRef());
if(diagnostics) {
assertUnreachable("%s\n", (const char*) diagnostics->getBufferPointer());
return;
}
// Get the layout
layout = program->getLayout();
// Create the shader program.
Slang::ComPtr<IBlob> blob;
for(auto i = 0; i < definedEntryPointCount; i++) {
// Get the code
auto result = linkedProgram->getEntryPointCode(
i,
0,
blob.writeRef(),
diagnostics.writeRef()
);
if(diagnostics) {
assertUnreachable("%s\n", (const char*) diagnostics->getBufferPointer());
}
// Get the stage information
auto entryPointReflection = layout->getEntryPointByIndex(i);
auto stage = entryPointReflection->getStage();
// Write out to file for debugging
#if DAWN_DEBUG_SHADERS
std::filesystem::path filePath = (
"debug/" +
std::string(module->getName()) +
"/" +
std::string(entryPointReflection->getName()) +
".glsl"
);
std::filesystem::create_directories(filePath.parent_path());
std::cout << "Writing shader to " << filePath << std::endl;
std::ofstream file(filePath);
file << (const char*)blob->getBufferPointer();
file.close();
#endif
// Create the shader entry
auto shaderStage = std::make_shared<ShaderStage>();
shaderStage->init(
stage,
std::string((const char*)blob->getBufferPointer())
);
// Add to the list
stages.push_back(shaderStage);
}
// Cleanup components
delete [] components;
// Reflect out the data
auto rootLayout = layout->getGlobalParamsVarLayout();
auto rootLayoutTypeLayout = rootLayout->getTypeLayout();
assertTrue(
rootLayoutTypeLayout->getKind() == TypeReflection::Kind::ConstantBuffer,
"Root layout, should and must be a constant buffer."
);
auto realRootLayout = rootLayoutTypeLayout->getElementVarLayout();
auto realRootLayoutTypeLayout = realRootLayout->getTypeLayout();
assertTrue(
realRootLayoutTypeLayout->getKind() == TypeReflection::Kind::Struct,
"Real root layout should and must be a struct."
);
// Now we can really begin parsing out the data.
structure = std::make_shared<struct ShaderStructure>(
realRootLayout
);
assertTrue(structure->name == "", "Root structure should not have a name.");
}
IShaderProgram::~IShaderProgram() {
// Release the linked program
if(linkedProgram) {
linkedProgram->release();
linkedProgram = nullptr;
}
}

View File

@ -5,12 +5,18 @@
#pragma once
#include "display/shader/ShaderStage.hpp"
#include "display/shader/ShaderStructure.hpp"
namespace Dawn {
class IShaderProgram {
protected:
IModule *module;
std::vector<std::shared_ptr<ShaderStage>> stages;
ComPtr<IComponentType> program;
ComPtr<IComponentType> linkedProgram;
ProgramLayout *layout;
std::shared_ptr<struct ShaderStructure> structure;
public:
/**
@ -25,10 +31,12 @@ namespace Dawn {
* loading the shader code and "initializing" the stages before we create
* and initialize the program which will use and link the stages.
*
* @param stages The stages to use in this program.
* @param module The SLANG module that was loaded.
* @param session The SLANG session that was used to load the module.
*/
virtual void init(
const std::vector<std::shared_ptr<ShaderStage>> &stages
IModule *module,
ComPtr<ISession> session
);
/**

View File

@ -6,8 +6,10 @@
#pragma once
#include "dawn.hpp"
#include "slang.h"
#include "slang-com-ptr.h"
using namespace slang;
using namespace Slang;
namespace Dawn {
class IShaderStage {

View File

@ -0,0 +1,59 @@
// Copyright (c) 2024 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "ShaderStructure.hpp"
#include "display/shader/ShaderData.hpp"
#include "assert/assert.hpp"
using namespace Dawn;
ShaderStructure::ShaderStructure(VariableLayoutReflection *reflection) {
auto typeLayout = reflection->getTypeLayout();
auto cName = reflection->getName();
this->name = cName ? std::string(cName) : "";
this->size = typeLayout->getSize();
this->alignment = typeLayout->getAlignment();
switch(typeLayout->getKind()) {
case TypeReflection::Kind::Struct: {
this->type = ShaderStructureType::STRUCT;
auto count = typeLayout->getFieldCount();
size_t offset = this->start;
for(auto i = 0; i < count; i++) {
auto field = typeLayout->getFieldByIndex(i);
ShaderStructure member(field);
member.start = offset;
offset += member.size;
this->members.push_back(member);
}
assertTrue(offset == this->size, "Struct size/offset mismatch?");
break;
}
case TypeReflection::Kind::Array: {
this->type = ShaderStructureType::ARRAY;
this->length = typeLayout->getElementCount();
break;
}
case TypeReflection::Kind::Matrix:
case TypeReflection::Kind::Vector:
case TypeReflection::Kind::Scalar:
this->type = ShaderStructureType::VARIABLE;
break;
case TypeReflection::Kind::Resource:
this->type = ShaderStructureType::RESOURCE;
break;
default: {
assertUnreachable("Unknown type layout kind: %d", typeLayout->getKind());
}
}
}
std::shared_ptr<ShaderData> ShaderStructure::createData() {
return std::make_shared<ShaderData>(shared_from_this());
}

View File

@ -0,0 +1,57 @@
// Copyright (c) 2024 Dominic Masters
//
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#pragma once
#include "dawn.hpp"
#include "slang.h"
#include "slang-com-ptr.h"
using namespace slang;
using namespace Slang;
namespace Dawn {
class ShaderData;
enum class ShaderStructureType {
STRUCT,
ARRAY,
VARIABLE,
RESOURCE
};
struct ShaderStructure :
public std::enable_shared_from_this<ShaderStructure>
{
public:
std::string name;
ShaderStructureType type;
// Shared properties
size_t start = 0;
size_t alignment;
size_t size;
// Struct properties
std::vector<struct ShaderStructure> members;
// Array properties
size_t length;
/**
* Constructs the ShaderStructure object
*
* @param reflection Reflection data to construct the structure from.
*/
ShaderStructure(VariableLayoutReflection *reflection);
/**
* Creates data for a shader that matches this structure.
*
* @param name Name of the member to get.
* @return The member structure.
*/
std::shared_ptr<ShaderData> createData();
};
}

View File

@ -3,61 +3,19 @@
// This software is released under the MIT License.
// https://opensource.org/licenses/MIT
#include "ShaderData.hpp"
#include "assert/assert.hpp"
#include "assert/assertgl.hpp"
#include "display/shader/ShaderData.hpp"
using namespace Dawn;
void ShaderData::beginStruct() {
// Do nothing
ShaderData::ShaderData(
const std::shared_ptr<ShaderStructure> &structure
) : IShaderData(structure) {
glGenBuffers(1, &this->buffer);
}
void ShaderData::writeMat4(const glm::mat4 &mat) {
char_t name[1024];
GLsizei length;
GLint size;
GLenum type;
glGetActiveUniform(getShader()->id, i, sizeof(name), &length, &size, &type, name);
assertNoGLError();
std::cout << "Uniform: " << name << std::endl;
glUniformMatrix4fv(this->i, 1, GL_FALSE, glm::value_ptr(mat));
assertNoGLError();
this->i++;
}
void ShaderData::writeVec3(const glm::vec3 &vec) {
assertUnreachable("Not implemented");
}
void ShaderData::writeVec4(const glm::vec4 &vec) {
assertUnreachable("Not implemented");
}
void ShaderData::writeColor(const struct Color &color) {
glUniform4fv(this->i, 1, &color.r);
assertNoGLError();
this->i++;
}
void ShaderData::writeBoolean(const bool &value) {
glUniform1i(this->i, value ? 1 : 0);
assertNoGLError();
this->i++;
}
void ShaderData::writeTexture(std::shared_ptr<Texture> &texture) {
if(texture == nullptr) {
this->i++;
return;
ShaderData::~ShaderData() {
if(this->buffer != -1) {
glDeleteBuffers(1, &this->buffer);
this->buffer = -1;
}
texture->bind(0);
glUniform1i(this->i, 0);
assertNoGLError();
this->i++;
}
void ShaderData::endStruct() {
// Do nothing
}

View File

@ -5,20 +5,18 @@
#pragma once
#include "display/shader/IShaderData.hpp"
#include "dawnopengl.hpp"
namespace Dawn {
class ShaderData : public IShaderData {
private:
int32_t i = 0;
protected:
void beginStruct() override;
void writeMat4(const glm::mat4 &mat) override;
void writeVec3(const glm::vec3 &vec) override;
void writeVec4(const glm::vec4 &vec) override;
void writeColor(const struct Color &color) override;
void writeBoolean(const bool &value) override;
void writeTexture(std::shared_ptr<Texture> &texture) override;
void endStruct() override;
GLuint buffer = -1;
public:
ShaderData(
const std::shared_ptr<ShaderStructure> &structure
);
~ShaderData() override;
};
}

View File

@ -11,12 +11,22 @@
using namespace Dawn;
struct TestData {
glm::mat4 projection;
glm::mat4 view;
glm::mat4 model;
int hasTexture;
int padding[3];
struct Color color;
int test;
};
void ShaderProgram::init(
const std::vector<std::shared_ptr<ShaderStage>> &stages
IModule *module,
Slang::ComPtr<ISession> session
) {
assertTrue(this->id == -1, "ShaderProgram already initialized?");
IShaderProgram::init(stages);
IShaderProgram::init(module, session);
// Create the program
this->id = glCreateProgram();
@ -54,40 +64,75 @@ void ShaderProgram::init(
// GLint numUniforms = 0;
// glGetProgramiv(this->id, GL_ACTIVE_UNIFORMS, &numUniforms);
// assertNoGLError();
// for(GLint i = 0; i < numUniforms; ++i) {
// char name[1024];
// GLsizei length;
// GLint size;
// GLenum type;
// glGetActiveUniform(this->id, i, sizeof(name), &length, &size, &type, name);
// assertNoGLError();
// std::cout << "Uniform: " << i << ": " << name << " has size " << size << " and length " << length << std::endl;
// }
// GLint numUniformBlocks = 0;
// glGetProgramiv(this->id, GL_ACTIVE_UNIFORM_BLOCKS, &numUniformBlocks);
// for(GLint i = 0; i < numUniformBlocks; ++i) {
// GLint size;
// glGetActiveUniformBlockiv(this->id, i, GL_UNIFORM_BLOCK_DATA_SIZE, &size);
// assertNoGLError();
// std::cout << "Uniform Block: " << i << " has size " << size << std::endl;
// }
// auto data = std::make_shared<SimpleTexturedMaterialShaderData>();
// data->color = COLOR_WHITE;
// data->model = glm::mat4(1.0f);
// data->projection = glm::perspective(
// glm::radians(45.0f), 800.0f / 600.0f, 0.1f, 100.0f
// );
// data->view = glm::lookAt(
// glm::vec3(4,3,3), glm::vec3(0,0,0), glm::vec3(0,1,0)
// );
glUseProgram(this->id);
assertNoGLError();
GLuint ubo;
glGenBuffers(1, &ubo);
assertNoGLError();
auto data = std::make_shared<SimpleTexturedMaterialShaderData>();
data->color = COLOR_WHITE;
data->model = glm::mat4(1.0f);
data->projection = glm::perspective(
glm::radians(45.0f), 800.0f / 600.0f, 0.1f, 100.0f
struct TestData data;
data.color = COLOR_WHITE;
data.model = glm::mat4(1.0f);
data.projection = glm::perspective(
glm::radians(45.0f), 800.0f / 600.0f, 0.1f, 10000.0f
);
data->view = glm::lookAt(
glm::vec3(4,3,3), glm::vec3(0,0,0), glm::vec3(0,1,0)
data.view = glm::lookAt(
glm::vec3(300,300,300), glm::vec3(0,0,0), glm::vec3(0,1,0)
);
data->write(shared_from_this());
data.hasTexture = false;
data.test = 1;
glBindBuffer(GL_UNIFORM_BUFFER, ubo);
assertNoGLError();
glBindBufferBase(GL_UNIFORM_BUFFER, 0, ubo);
assertNoGLError();
glBufferData(GL_UNIFORM_BUFFER, sizeof(struct TestData), &data, GL_STATIC_DRAW);
assertNoGLError();
// testTexture = std::make_shared<Texture>();
// testTexture->setSize(2, 2, TextureFormat::RGBA, TextureDataFormat::UNSIGNED_BYTE);
// uint8_t colors[4 * 4];
// colors[0] = 255; colors[1] = 0; colors[2] = 0; colors[3] = 255;
// colors[4] = 0; colors[5] = 255; colors[6] = 0; colors[7] = 255;
// colors[8] = 0; colors[9] = 0; colors[10] = 255; colors[11] = 255;
// colors[12] = 255; colors[13] = 255; colors[14] = 255; colors[15] = 255;
// testTexture->buffer(colors);
// testTexture->bind(0);
// int index = glGetUniformLocation(this->id, "globalParams_uniforms_texture_0");
// assertNoGLError();
// std::cout << "Index: " << index << std::endl;
// glUniform1i(index, 0);
// assertNoGLError();
// data->write(shared_from_this());
}
ShaderProgram::~ShaderProgram() {

View File

@ -6,23 +6,25 @@
#pragma once
#include "display/shader/IShaderProgram.hpp"
#include "dawnopengl.hpp"
#include "display/Texture.hpp"
#include "slang.h"
using namespace slang;
namespace Dawn {
class ShaderData;
class ShaderProgram :
public IShaderProgram,
public std::enable_shared_from_this<ShaderProgram>
{
private:
GLuint id = -1;
std::shared_ptr<Texture> testTexture;
public:
void init(
const std::vector<std::shared_ptr<ShaderStage>> &stages
) override;
~ShaderProgram();
friend class ShaderData;
IModule *module,
Slang::ComPtr<ISession> session
);
~ShaderProgram() override;
};
}

View File

@ -48,7 +48,6 @@ function(tool_texture target file)
${DAWN_ASSETS_SOURCE_DIR}/${file}
${Python3_EXECUTABLE}
USES_TERMINAL
)
add_dependencies(dawnassets ${target}_texture)
endfunction()