mirror of
https://bitbucket.org/CPMADevs/cnq3
synced 2025-02-18 18:01:22 +00:00
388 lines
11 KiB
C++
388 lines
11 KiB
C++
/*
|
|
===========================================================================
|
|
Copyright (C) 2022-2024 Gian 'myT' Schellenbaum
|
|
|
|
This file is part of Challenge Quake 3 (CNQ3).
|
|
|
|
Challenge Quake 3 is free software; you can redistribute it
|
|
and/or modify it under the terms of the GNU General Public License as
|
|
published by the Free Software Foundation; either version 2 of the License,
|
|
or (at your option) any later version.
|
|
|
|
Challenge Quake 3 is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Challenge Quake 3. If not, see <https://www.gnu.org/licenses/>.
|
|
===========================================================================
|
|
*/
|
|
// compiles core shaders as byte code to be embedded into the CNQ3 client
|
|
|
|
|
|
#include <Windows.h>
|
|
#include <stdarg.h>
|
|
#include <stdio.h>
|
|
#include <assert.h>
|
|
#include <shlwapi.h>
|
|
#include <sal.h>
|
|
|
|
#include "../renderer/grp_uber_shaders.h"
|
|
|
|
|
|
char repoPath[MAX_PATH];
|
|
char outputPath[MAX_PATH];
|
|
char bin2headerPath[MAX_PATH];
|
|
char dxcPath[MAX_PATH];
|
|
|
|
const char* targetVS = "vs_6_0";
|
|
const char* targetPS = "ps_6_0";
|
|
const char* targetCS = "cs_6_0";
|
|
|
|
|
|
#define PS(Data) #Data,
|
|
const char* uberShaderPixelStates[] =
|
|
{
|
|
UBER_SHADER_PS_LIST(PS)
|
|
};
|
|
#undef PS
|
|
|
|
|
|
// -Zi embeds debug info
|
|
// -Qembed_debug embeds debug info in shader container
|
|
// -Vn header variable name
|
|
// -WX warnings as errors
|
|
// -O3 or -O0 optimization level
|
|
// -Wno-warning disables the warning
|
|
|
|
|
|
const char* va(_Printf_format_string_ const char* format, ...)
|
|
{
|
|
static char string[64][32000];
|
|
static int index = 0;
|
|
char* buf = string[index++ & 63];
|
|
va_list argptr;
|
|
|
|
va_start(argptr, format);
|
|
vsprintf(buf, format, argptr);
|
|
va_end(argptr);
|
|
|
|
return buf;
|
|
}
|
|
|
|
|
|
struct ShaderArgs
|
|
{
|
|
const char* headerPath;
|
|
const char* shaderPath;
|
|
const char* entryPoint;
|
|
const char* targetProfile;
|
|
};
|
|
|
|
void CompileShader(const ShaderArgs& args, int extraCount = 0, const char** extras = NULL)
|
|
{
|
|
static char temp[4096];
|
|
|
|
const char* headerPath = va("%s\\%s", outputPath, args.headerPath);
|
|
|
|
// -Ges: Enable strict mode
|
|
// -Gis: Force IEEE strictness
|
|
// -Zi: Embed debug info
|
|
// -Qembed_debug: Embed debug info in shader container
|
|
strcpy(temp, va("%s -Fh %s -E %s -T %s -WX -Ges -Gis -Zi -Qembed_debug",
|
|
dxcPath, headerPath, args.entryPoint, args.targetProfile));
|
|
|
|
for(int i = 0; i < extraCount; ++i)
|
|
{
|
|
strcat(temp, " ");
|
|
strcat(temp, extras[i]);
|
|
}
|
|
|
|
strcat(temp, " ");
|
|
strcat(temp, args.shaderPath);
|
|
|
|
printf("%s\n", temp);
|
|
system(temp);
|
|
}
|
|
|
|
struct SMAAArgs
|
|
{
|
|
const char* headerPath;
|
|
const char* shaderPath;
|
|
const char* presetMacro;
|
|
const char* variableName;
|
|
bool vertexShader;
|
|
};
|
|
|
|
void CompileSMAAShader(const SMAAArgs& smaaArgs)
|
|
{
|
|
const char* extras[] =
|
|
{
|
|
"-Vn", smaaArgs.variableName,
|
|
smaaArgs.presetMacro,
|
|
smaaArgs.vertexShader ? "-D SMAA_INCLUDE_VS=1" : "-D SMAA_INCLUDE_PS=1",
|
|
"-D SMAA_HLSL_5_1=1",
|
|
"-D SMAA_RT_METRICS=rtMetrics"
|
|
};
|
|
|
|
ShaderArgs args;
|
|
args.entryPoint = smaaArgs.vertexShader ? "vs" : "ps";
|
|
args.headerPath = smaaArgs.headerPath;
|
|
args.shaderPath = smaaArgs.shaderPath;
|
|
args.targetProfile = smaaArgs.vertexShader ? targetVS : targetPS;
|
|
CompileShader(args, _countof(extras), extras);
|
|
}
|
|
|
|
void ProcessSMAAShadersForPreset(const char* presetName, const char* presetMacro)
|
|
{
|
|
SMAAArgs args;
|
|
args.presetMacro = presetMacro;
|
|
for(int pass = 0; pass < 3; ++pass)
|
|
{
|
|
for(int ps = 0; ps < 2; ++ps)
|
|
{
|
|
args.headerPath = va("smaa_%s_%d_%s.h", presetName, pass + 1, ps ? "ps" : "vs");
|
|
args.shaderPath = va("smaa_%d.hlsl", pass + 1);
|
|
args.variableName = va("%s_%d_%s", presetName, pass + 1, ps ? "ps" : "vs");
|
|
args.vertexShader = ps == 0;
|
|
CompileSMAAShader(args);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CompileSMAAShaders()
|
|
{
|
|
ProcessSMAAShadersForPreset("low", "-D SMAA_PRESET_LOW=1");
|
|
ProcessSMAAShadersForPreset("medium", "-D SMAA_PRESET_MEDIUM=1");
|
|
ProcessSMAAShadersForPreset("high", "-D SMAA_PRESET_HIGH=1");
|
|
ProcessSMAAShadersForPreset("ultra", "-D SMAA_PRESET_ULTRA=1");
|
|
}
|
|
|
|
void CompileVS(const char* headerPath, const char* shaderPath)
|
|
{
|
|
const char* extras[] = { "-D VERTEX_SHADER=1" };
|
|
|
|
ShaderArgs args;
|
|
args.entryPoint = "vs";
|
|
args.headerPath = headerPath;
|
|
args.shaderPath = shaderPath;
|
|
args.targetProfile = targetVS;
|
|
CompileShader(args, _countof(extras), extras);
|
|
}
|
|
|
|
void CompilePS(const char* headerPath, const char* shaderPath)
|
|
{
|
|
const char* extras[] = { "-D PIXEL_SHADER=1" };
|
|
|
|
ShaderArgs args;
|
|
args.entryPoint = "ps";
|
|
args.headerPath = headerPath;
|
|
args.shaderPath = shaderPath;
|
|
args.targetProfile = targetPS;
|
|
CompileShader(args, _countof(extras), extras);
|
|
}
|
|
|
|
void CompileCS(const char* headerPath, const char* shaderPath)
|
|
{
|
|
const char* extras[] = { "-D COMPUTE_SHADER=1" };
|
|
|
|
ShaderArgs args;
|
|
args.entryPoint = "cs";
|
|
args.headerPath = headerPath;
|
|
args.shaderPath = shaderPath;
|
|
args.targetProfile = targetCS;
|
|
CompileShader(args, _countof(extras), extras);
|
|
}
|
|
|
|
void CompileVSAndPS(const char* headerPathPrefix, const char* shaderPath)
|
|
{
|
|
CompileVS(va("%s_vs.h", headerPathPrefix), shaderPath);
|
|
CompilePS(va("%s_ps.h", headerPathPrefix), shaderPath);
|
|
}
|
|
|
|
void CompileUberVS(const char* headerPath, const char* shaderPath, int stageCount)
|
|
{
|
|
const char* extras[] =
|
|
{
|
|
"-D VERTEX_SHADER=1",
|
|
va("-D STAGE_COUNT=%d", stageCount),
|
|
va("-Vn g_vs_%d", stageCount)
|
|
};
|
|
|
|
ShaderArgs args;
|
|
args.entryPoint = "vs";
|
|
args.headerPath = headerPath;
|
|
args.shaderPath = shaderPath;
|
|
args.targetProfile = targetVS;
|
|
CompileShader(args, _countof(extras), extras);
|
|
}
|
|
|
|
void CompileUberPS(const char* stateString)
|
|
{
|
|
UberPixelShaderState state;
|
|
if(!ParseUberPixelShaderState(state, stateString))
|
|
{
|
|
fprintf(stderr, "ParseUberPixelShaderState failed!\n");
|
|
exit(666);
|
|
}
|
|
|
|
const char* extras[16];
|
|
int extraCount = 0;
|
|
extras[extraCount++] = va("-Vn g_ps_%s", stateString);
|
|
extras[extraCount++] = "-D USE_INCLUDES=1";
|
|
extras[extraCount++] = "-D PIXEL_SHADER=1";
|
|
if(state.globalState & UBERPS_DITHER_BIT)
|
|
{
|
|
extras[extraCount++] = "-D DITHER=1";
|
|
}
|
|
if(state.globalState & UBERPS_DEPTHFADE_BIT)
|
|
{
|
|
extras[extraCount++] = "-D DEPTH_FADE=1";
|
|
}
|
|
extras[extraCount++] = va("-D STAGE_COUNT=%d", state.stageCount);
|
|
for(int s = 0; s < state.stageCount; ++s)
|
|
{
|
|
extras[extraCount++] = va("-D STAGE%d_BITS=0x%X", s, state.stageStates[s]);
|
|
}
|
|
|
|
ShaderArgs args;
|
|
args.entryPoint = "ps";
|
|
args.headerPath = va("uber_shader_ps_%s.h", stateString);
|
|
args.shaderPath = "uber_shader.hlsl";
|
|
args.targetProfile = targetPS;
|
|
CompileShader(args, extraCount, extras);
|
|
}
|
|
|
|
const char* Canonicalize(const char* path)
|
|
{
|
|
static char canonPath[MAX_PATH];
|
|
|
|
PathCanonicalizeA(canonPath, path);
|
|
|
|
return canonPath;
|
|
}
|
|
|
|
void InitDirectory(const char* dirName)
|
|
{
|
|
const char* rendererPath = va("%s\\code\\renderer", repoPath);
|
|
const char* cd = Canonicalize(va("%s\\shaders\\%s", rendererPath, dirName));
|
|
SetCurrentDirectoryA(cd);
|
|
const char* out = Canonicalize(va("%s\\compshaders\\%s", rendererPath, dirName));
|
|
strcpy(outputPath, out);
|
|
CreateDirectoryA(outputPath, NULL);
|
|
|
|
system(va("del %s\\*.h", outputPath));
|
|
system(va("del %s\\*.temp", outputPath));
|
|
}
|
|
|
|
void ProcessGRP()
|
|
{
|
|
InitDirectory("grp");
|
|
targetVS = "vs_6_0";
|
|
targetPS = "ps_6_0";
|
|
targetCS = "cs_6_0";
|
|
|
|
CompileVSAndPS("post_gamma", "post_gamma.hlsl");
|
|
CompileVSAndPS("post_inverse_gamma", "post_inverse_gamma.hlsl");
|
|
CompileVSAndPS("imgui", "imgui.hlsl");
|
|
CompileVSAndPS("nuklear", "nuklear.hlsl");
|
|
CompileVSAndPS("ui", "ui.hlsl");
|
|
CompileVSAndPS("depth_pre_pass", "depth_pre_pass.hlsl");
|
|
CompileVSAndPS("dynamic_light", "dynamic_light.hlsl");
|
|
CompileVS("fog_vs.h", "fog_inside.hlsl");
|
|
CompilePS("fog_inside_ps.h", "fog_inside.hlsl");
|
|
CompilePS("fog_outside_ps.h", "fog_outside.hlsl");
|
|
CompileCS("mip_1_cs.h", "mip_1.hlsl");
|
|
CompileCS("mip_2_cs.h", "mip_2.hlsl");
|
|
CompileCS("mip_3_cs.h", "mip_3.hlsl");
|
|
|
|
CompileSMAAShaders();
|
|
system(va("type %s\\smaa*.h > %s\\complete_smaa.h", outputPath, outputPath));
|
|
|
|
// type combines all files into one
|
|
system(va("type ..\\common\\state_bits.h.hlsli ..\\common\\blend.hlsli shared.hlsli uber_shader.hlsl > %s\\uber_shader.temp", outputPath));
|
|
system(va("%s --output %s\\uber_shader.h --hname uber_shader_string %s\\uber_shader.temp", bin2headerPath, outputPath, outputPath));
|
|
system(va("del %s\\uber_shader.temp", outputPath));
|
|
|
|
for(int i = 0; i < 8; ++i)
|
|
{
|
|
CompileUberVS(va("uber_shader_vs_%i.h", i + 1), "uber_shader.hlsl", i + 1);
|
|
}
|
|
system(va("type %s\\uber_shader_vs_*.h > %s\\complete_uber_vs.h", outputPath, outputPath));
|
|
system(va("del %s\\uber_shader_vs_*.h", outputPath));
|
|
|
|
for(int i = 0; i < _countof(uberShaderPixelStates); ++i)
|
|
{
|
|
CompileUberPS(uberShaderPixelStates[i]);
|
|
}
|
|
system(va("type %s\\uber_shader_ps_*.h > %s\\complete_uber_ps.h", outputPath, outputPath));
|
|
system(va("del %s\\uber_shader_ps_*.h", outputPath));
|
|
}
|
|
|
|
void ProcessCRP()
|
|
{
|
|
InitDirectory("crp");
|
|
targetVS = "vs_6_6";
|
|
targetPS = "ps_6_6";
|
|
targetCS = "cs_6_6";
|
|
|
|
CompileVSAndPS("blit", "blit.hlsl");
|
|
CompileVSAndPS("ui", "ui.hlsl");
|
|
CompileVSAndPS("imgui", "imgui.hlsl");
|
|
CompileVSAndPS("nuklear", "nuklear.hlsl");
|
|
CompileCS("mip_1_cs.h", "mip_1.hlsl");
|
|
CompileCS("mip_2_cs.h", "mip_2.hlsl");
|
|
CompileCS("mip_3_cs.h", "mip_3.hlsl");
|
|
|
|
CompileVSAndPS("opaque", "opaque.hlsl");
|
|
CompileVSAndPS("transp_draw", "transp_draw.hlsl");
|
|
CompileVSAndPS("transp_resolve", "transp_resolve.hlsl");
|
|
CompileVSAndPS("tone_map", "tone_map.hlsl");
|
|
CompileVSAndPS("tone_map_inverse", "tone_map_inverse.hlsl");
|
|
CompileVSAndPS("accumdof_accum", "accumdof_accum.hlsl");
|
|
CompileVSAndPS("accumdof_norm", "accumdof_norm.hlsl");
|
|
CompileVSAndPS("accumdof_debug", "accumdof_debug.hlsl");
|
|
CompileCS("gatherdof_split.h", "gatherdof_split.hlsl");
|
|
CompileCS("gatherdof_coc_tile_gen.h", "gatherdof_coc_tile_gen.hlsl");
|
|
CompileCS("gatherdof_coc_tile_max.h", "gatherdof_coc_tile_max.hlsl");
|
|
CompileCS("gatherdof_blur.h", "gatherdof_blur.hlsl");
|
|
CompileCS("gatherdof_fill.h", "gatherdof_fill.hlsl");
|
|
CompileVSAndPS("gatherdof_combine", "gatherdof_combine.hlsl");
|
|
CompileVSAndPS("gatherdof_debug", "gatherdof_debug.hlsl");
|
|
CompileVSAndPS("fog_inside", "fog_inside.hlsl");
|
|
CompileVSAndPS("fog_outside", "fog_outside.hlsl");
|
|
}
|
|
|
|
int main(int /*argc*/, const char** argv)
|
|
{
|
|
char dirPath[MAX_PATH];
|
|
strcpy(dirPath, argv[0]);
|
|
int l = strlen(dirPath);
|
|
while(l-- > 0)
|
|
{
|
|
if(dirPath[l] == '/' || dirPath[l] == '\\')
|
|
{
|
|
dirPath[l] = '\0';
|
|
break;
|
|
}
|
|
}
|
|
strcpy(repoPath, Canonicalize(va("%s\\..\\..", dirPath)));
|
|
strcpy(bin2headerPath, Canonicalize(va("%s\\tools\\bin2header.exe", repoPath)));
|
|
|
|
char* path = getenv("DXCPATH");
|
|
if(path != NULL)
|
|
{
|
|
strcpy(dxcPath, path);
|
|
}
|
|
else
|
|
{
|
|
strcpy(dxcPath, "dxc.exe");
|
|
}
|
|
system(va("%s --version", dxcPath));
|
|
|
|
ProcessGRP();
|
|
ProcessCRP();
|
|
|
|
return 0;
|
|
}
|