summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/gpu/effects/shaders.cc104
-rw-r--r--src/tests/test_assets.cc4
-rw-r--r--src/tests/test_shader_assets.cc66
-rw-r--r--src/tests/test_shader_composer.cc43
4 files changed, 197 insertions, 20 deletions
diff --git a/src/gpu/effects/shaders.cc b/src/gpu/effects/shaders.cc
index 6b37869..cd516cd 100644
--- a/src/gpu/effects/shaders.cc
+++ b/src/gpu/effects/shaders.cc
@@ -2,39 +2,107 @@
// It defines WGSL shader code for various effects.
#include "../demo_effects.h"
+
+
+
+#if defined(USE_TEST_ASSETS)
+
+#include "test_assets.h"
+
+#else
+
#include "generated/assets.h"
+
+#endif
+
+
+
#include "gpu/effects/shader_composer.h"
+
#include "util/asset_manager.h"
+
+
void InitShaderComposer() {
+
auto& sc = ShaderComposer::Get();
- sc.RegisterSnippet("common_uniforms",
- (const char*)GetAsset(AssetId::ASSET_SHADER_COMMON_UNIFORMS));
- sc.RegisterSnippet("sdf_primitives",
- (const char*)GetAsset(AssetId::ASSET_SHADER_SDF_PRIMITIVES));
- sc.RegisterSnippet("lighting",
- (const char*)GetAsset(AssetId::ASSET_SHADER_LIGHTING));
- sc.RegisterSnippet("ray_box",
- (const char*)GetAsset(AssetId::ASSET_SHADER_RAY_BOX));
+
+
+ auto register_if_exists = [&](const char* name, AssetId id) {
+
+ size_t size;
+
+ const char* data = (const char*)GetAsset(id, &size);
+
+ if (data) {
+
+ sc.RegisterSnippet(name, std::string(data, size));
+
+ }
+
+ };
+
+
+
+ register_if_exists("common_uniforms", AssetId::ASSET_SHADER_COMMON_UNIFORMS);
+
+ register_if_exists("sdf_primitives", AssetId::ASSET_SHADER_SDF_PRIMITIVES);
+
+ register_if_exists("lighting", AssetId::ASSET_SHADER_LIGHTING);
+
+ register_if_exists("ray_box", AssetId::ASSET_SHADER_RAY_BOX);
+
}
-const char* main_shader_wgsl = (const char*)GetAsset(AssetId::ASSET_SHADER_MAIN);
+
+
+// Helper to get asset string or empty string
+
+static const char* SafeGetAsset(AssetId id) {
+
+ const uint8_t* data = GetAsset(id);
+
+ return data ? (const char*)data : "";
+
+}
+
+
+
+const char* main_shader_wgsl = SafeGetAsset(AssetId::ASSET_SHADER_MAIN);
+
const char* particle_compute_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_PARTICLE_COMPUTE);
+
+ SafeGetAsset(AssetId::ASSET_SHADER_PARTICLE_COMPUTE);
+
const char* particle_render_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_PARTICLE_RENDER);
+
+ SafeGetAsset(AssetId::ASSET_SHADER_PARTICLE_RENDER);
+
const char* passthrough_shader_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_PASSTHROUGH);
+
+ SafeGetAsset(AssetId::ASSET_SHADER_PASSTHROUGH);
+
const char* ellipse_shader_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_ELLIPSE);
+
+ SafeGetAsset(AssetId::ASSET_SHADER_ELLIPSE);
+
const char* particle_spray_compute_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_PARTICLE_SPRAY_COMPUTE);
+
+ SafeGetAsset(AssetId::ASSET_SHADER_PARTICLE_SPRAY_COMPUTE);
+
const char* gaussian_blur_shader_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_GAUSSIAN_BLUR);
+
+ SafeGetAsset(AssetId::ASSET_SHADER_GAUSSIAN_BLUR);
+
const char* solarize_shader_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_SOLARIZE);
+
+ SafeGetAsset(AssetId::ASSET_SHADER_SOLARIZE);
+
const char* distort_shader_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_DISTORT);
+
+ SafeGetAsset(AssetId::ASSET_SHADER_DISTORT);
+
const char* chroma_aberration_shader_wgsl =
- (const char*)GetAsset(AssetId::ASSET_SHADER_CHROMA_ABERRATION); \ No newline at end of file
+
+ SafeGetAsset(AssetId::ASSET_SHADER_CHROMA_ABERRATION);
diff --git a/src/tests/test_assets.cc b/src/tests/test_assets.cc
index 6f57d8f..7f26e71 100644
--- a/src/tests/test_assets.cc
+++ b/src/tests/test_assets.cc
@@ -16,7 +16,7 @@ int main() {
printf("Running AssetManager test...\n");
size_t size = 0;
- const uint8_t* data1 = GetAsset(AssetId::ASSET_TEST_ASSET, &size);
+ const uint8_t* data1 = GetAsset(AssetId::ASSET_TEST_ASSET_1, &size);
assert(data1 != nullptr);
assert(size > 0);
@@ -33,7 +33,7 @@ int main() {
// Test caching: request the same asset again and verify pointer is identical
size_t size2 = 0;
- const uint8_t* data2 = GetAsset(AssetId::ASSET_TEST_ASSET, &size2);
+ const uint8_t* data2 = GetAsset(AssetId::ASSET_TEST_ASSET_1, &size2);
assert(data2 != nullptr);
assert(size2 == size);
assert(data1 == data2); // Pointers should be the same for cached static asset
diff --git a/src/tests/test_shader_assets.cc b/src/tests/test_shader_assets.cc
new file mode 100644
index 0000000..42d1c4c
--- /dev/null
+++ b/src/tests/test_shader_assets.cc
@@ -0,0 +1,66 @@
+// This file is part of the 64k demo project.
+// It validates that WGSL shader assets are present and look like valid WGSL.
+
+#include "generated/assets.h"
+#include <cassert>
+#include <cstdio>
+#include <cstring>
+#include <string>
+#include <vector>
+
+bool validate_shader(AssetId id, const char* name, const std::vector<const char*>& expected_keywords) {
+ printf("Validating shader: %s...\n", name);
+ size_t size = 0;
+ const char* data = (const char*)GetAsset(id, &size);
+
+ if (data == nullptr || size == 0) {
+ printf("FAILED: Shader %s is missing or empty!\n", name);
+ return false;
+ }
+
+ std::string code(data, size);
+ for (const char* keyword : expected_keywords) {
+ if (code.find(keyword) == std::string::npos) {
+ printf("FAILED: Shader %s missing expected keyword '%s'!\n", name, keyword);
+ // printf("Code snippet:\n%.100s...\n", data);
+ return false;
+ }
+ }
+
+ printf("PASSED: %s (%zu bytes)\n", name, size);
+ return true;
+}
+
+int main() {
+ printf("--- RUNNING SHADER ASSET VALIDATION ---\n");
+
+ bool all_passed = true;
+
+ // Snippets
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_COMMON_UNIFORMS, "COMMON_UNIFORMS", {"struct", "GlobalUniforms"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_SDF_PRIMITIVES, "SDF_PRIMITIVES", {"fn", "sd"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_LIGHTING, "LIGHTING", {"fn", "calc"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_RAY_BOX, "RAY_BOX", {"fn", "intersect"});
+
+ // Full Shaders (Entry points)
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_RENDERER_3D, "RENDERER_3D", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_MAIN, "MAIN", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_PARTICLE_COMPUTE, "PARTICLE_COMPUTE", {"@compute", "main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_PARTICLE_RENDER, "PARTICLE_RENDER", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_PASSTHROUGH, "PASSTHROUGH", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_ELLIPSE, "ELLIPSE", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_PARTICLE_SPRAY_COMPUTE, "PARTICLE_SPRAY_COMPUTE", {"@compute", "main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_GAUSSIAN_BLUR, "GAUSSIAN_BLUR", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_SOLARIZE, "SOLARIZE", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_DISTORT, "DISTORT", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_CHROMA_ABERRATION, "CHROMA_ABERRATION", {"@vertex", "vs_main", "@fragment", "fs_main"});
+ all_passed &= validate_shader(AssetId::ASSET_SHADER_VISUAL_DEBUG, "VISUAL_DEBUG", {"@vertex", "vs_main", "@fragment", "fs_main"});
+
+ if (!all_passed) {
+ printf("--- SHADER ASSET VALIDATION FAILED ---\n");
+ return 1;
+ }
+
+ printf("--- ALL SHADER ASSETS VALIDATED ---\n");
+ return 0;
+}
diff --git a/src/tests/test_shader_composer.cc b/src/tests/test_shader_composer.cc
index cdb5c88..7efcd83 100644
--- a/src/tests/test_shader_composer.cc
+++ b/src/tests/test_shader_composer.cc
@@ -6,6 +6,16 @@
#include <iostream>
#include <string>
+#if defined(USE_TEST_ASSETS)
+#include "test_assets.h"
+#else
+#include "generated/assets.h"
+#endif
+
+// Forward declaration for asset loading
+const uint8_t* GetAsset(AssetId asset_id, size_t* out_size);
+
+
void test_composition() {
std::cout << "Testing Shader Composition..." << std::endl;
auto& sc = ShaderComposer::Get();
@@ -31,8 +41,41 @@ void test_composition() {
std::cout << "Composition logic verified." << std::endl;
}
+void test_asset_composition() {
+ std::cout << "Testing Asset-Based Shader Composition..." << std::endl;
+
+ // Use test assets
+ auto& sc = ShaderComposer::Get();
+
+ size_t snippet_a_size;
+ const char* snippet_a_code = (const char*)GetAsset(AssetId::ASSET_SHADER_SNIPPET_A, &snippet_a_size);
+ assert(snippet_a_code != nullptr);
+ sc.RegisterSnippet("SNIPPET_A", std::string(snippet_a_code, snippet_a_size));
+
+ size_t snippet_b_size;
+ const char* snippet_b_code = (const char*)GetAsset(AssetId::ASSET_SHADER_SNIPPET_B, &snippet_b_size);
+ sc.RegisterSnippet("SNIPPET_B", std::string(snippet_b_code, snippet_b_size));
+
+ std::string main_code = "fn main() -> f32 { return snippet_a() + snippet_b(); }";
+ std::string result = sc.Compose({"SNIPPET_A", "SNIPPET_B"}, main_code);
+
+ assert(result.find("fn snippet_a()") != std::string::npos);
+ assert(result.find("fn snippet_b()") != std::string::npos);
+ assert(result.find("fn main()") != std::string::npos);
+
+ size_t pos_a = result.find("snippet_a");
+ size_t pos_b = result.find("snippet_b");
+ size_t pos_main = result.find("main");
+
+ assert(pos_a < pos_b);
+ assert(pos_b < pos_main);
+
+ std::cout << "Asset-based composition logic verified." << std::endl;
+}
+
int main() {
test_composition();
+ test_asset_composition();
std::cout << "--- ALL SHADER COMPOSER TESTS PASSED ---" << std::endl;
return 0;
}