mirror of
https://github.com/libretro/RetroArch
synced 2025-04-06 19:21:37 +00:00
Merge pull request #3312 from Themaister/master
Vulkan: Make glslang shaders debuggable.
This commit is contained in:
commit
210db93237
@ -28,12 +28,14 @@
|
|||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
static bool read_shader_file(const char *path, vector<string> *output)
|
static bool read_shader_file(const char *path, vector<string> *output, bool root_file)
|
||||||
{
|
{
|
||||||
char *buf = nullptr;
|
char *buf = nullptr;
|
||||||
ssize_t len = 0;
|
ssize_t len = 0;
|
||||||
struct string_list *list = NULL;
|
const char *basename = path_basename(path);
|
||||||
char include_path[PATH_MAX];
|
char include_path[PATH_MAX];
|
||||||
|
char tmp[PATH_MAX];
|
||||||
|
vector<const char *> lines;
|
||||||
|
|
||||||
if (!filestream_read_file(path, (void**)&buf, &len))
|
if (!filestream_read_file(path, (void**)&buf, &len))
|
||||||
{
|
{
|
||||||
@ -41,30 +43,57 @@ static bool read_shader_file(const char *path, vector<string> *output)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
list = string_split(buf, "\n");
|
// Cannot use string_split since it removes blank lines (strtok).
|
||||||
|
char *ptr = buf;
|
||||||
|
while (ptr && *ptr)
|
||||||
|
{
|
||||||
|
lines.push_back(ptr);
|
||||||
|
|
||||||
if (!list)
|
char *next_ptr = strchr(ptr, '\n');
|
||||||
|
if (next_ptr)
|
||||||
|
{
|
||||||
|
ptr = next_ptr + 1;
|
||||||
|
*next_ptr = '\0';
|
||||||
|
}
|
||||||
|
else
|
||||||
|
ptr = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (lines.empty())
|
||||||
{
|
{
|
||||||
free(buf);
|
free(buf);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (list->size == 0)
|
if (root_file)
|
||||||
{
|
{
|
||||||
free(buf);
|
if (strstr(lines[0], "#version ") != lines[0])
|
||||||
string_list_free(list);
|
{
|
||||||
return false;
|
RARCH_ERR("First line of the shader must contain a valid #version string.\n");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
output->push_back(lines[0]);
|
||||||
|
// Allows us to use #line to make dealing with shader errors easier.
|
||||||
|
// This is supported by glslang, but since we always use glslang statically,
|
||||||
|
// this is fine.
|
||||||
|
output->push_back("#extension GL_GOOGLE_cpp_style_line_directive : require");
|
||||||
}
|
}
|
||||||
|
|
||||||
for (size_t i = 0; i < list->size; i++)
|
// At least VIM treats the first line as line #1, so offset everything by one.
|
||||||
|
snprintf(tmp, sizeof(tmp), "#line %u \"%s\"", root_file ? 2 : 1, basename);
|
||||||
|
output->push_back(tmp);
|
||||||
|
|
||||||
|
for (size_t i = root_file ? 1 : 0; i < lines.size(); i++)
|
||||||
{
|
{
|
||||||
const char *line = list->elems[i].data;
|
const char *line = lines[i];
|
||||||
if (strstr(line, "#include ") == line)
|
if (strstr(line, "#include ") == line)
|
||||||
{
|
{
|
||||||
char *c = (char*)strchr(line, '"');
|
char *c = (char*)strchr(line, '"');
|
||||||
if (!c)
|
if (!c)
|
||||||
{
|
{
|
||||||
RARCH_ERR("Invalid include statement \"%s\".\n", line);
|
RARCH_ERR("Invalid include statement \"%s\".\n", line);
|
||||||
|
free(buf);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
c++;
|
c++;
|
||||||
@ -72,19 +101,36 @@ static bool read_shader_file(const char *path, vector<string> *output)
|
|||||||
if (!closing)
|
if (!closing)
|
||||||
{
|
{
|
||||||
RARCH_ERR("Invalid include statement \"%s\".\n", line);
|
RARCH_ERR("Invalid include statement \"%s\".\n", line);
|
||||||
|
free(buf);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
*closing = '\0';
|
*closing = '\0';
|
||||||
fill_pathname_resolve_relative(include_path, path, c, sizeof(include_path));
|
fill_pathname_resolve_relative(include_path, path, c, sizeof(include_path));
|
||||||
|
|
||||||
if (!read_shader_file(include_path, output))
|
if (!read_shader_file(include_path, output, false))
|
||||||
|
{
|
||||||
|
free(buf);
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// After including a file, use line directive to pull it back to current file.
|
||||||
|
snprintf(tmp, sizeof(tmp), "#line %u \"%s\"", unsigned(i + 1), basename);
|
||||||
|
output->push_back(tmp);
|
||||||
|
}
|
||||||
|
else if (strstr(line, "#endif") || strstr(line, "#pragma"))
|
||||||
|
{
|
||||||
|
// #line seems to be ignored if preprocessor tests fail,
|
||||||
|
// so we should reapply #line after each #endif.
|
||||||
|
// Add extra offset here since we're setting #line for the line after this one.
|
||||||
|
snprintf(tmp, sizeof(tmp), "#line %u \"%s\"", unsigned(i + 2), basename);
|
||||||
|
output->push_back(line);
|
||||||
|
output->push_back(tmp);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
output->push_back(line);
|
output->push_back(line);
|
||||||
}
|
}
|
||||||
|
|
||||||
string_list_free(list);
|
free(buf);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,14 +152,6 @@ static string build_stage_source(const vector<string> &lines, const char *stage)
|
|||||||
auto expected = string("#pragma stage ") + stage;
|
auto expected = string("#pragma stage ") + stage;
|
||||||
active = itr->find(expected) != string::npos;
|
active = itr->find(expected) != string::npos;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Improve debuggability.
|
|
||||||
if (active)
|
|
||||||
{
|
|
||||||
str << "#line ";
|
|
||||||
str << (itr - begin(lines)) + 2;
|
|
||||||
str << '\n';
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (itr->find("#pragma name ") == 0 ||
|
else if (itr->find("#pragma name ") == 0 ||
|
||||||
itr->find("#pragma format ") == 0)
|
itr->find("#pragma format ") == 0)
|
||||||
@ -258,19 +296,12 @@ bool glslang_compile_shader(const char *shader_path, glslang_output *output)
|
|||||||
vector<string> lines;
|
vector<string> lines;
|
||||||
|
|
||||||
RARCH_LOG("[slang]: Compiling shader \"%s\".\n", shader_path);
|
RARCH_LOG("[slang]: Compiling shader \"%s\".\n", shader_path);
|
||||||
if (!read_shader_file(shader_path, &lines))
|
if (!read_shader_file(shader_path, &lines, true))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (!glslang_parse_meta(lines, &output->meta))
|
if (!glslang_parse_meta(lines, &output->meta))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
auto &header = lines.front();
|
|
||||||
if (header.find_first_of("#version ") != 0)
|
|
||||||
{
|
|
||||||
RARCH_ERR("First line of the shader must contain a valid #version string.\n");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!glslang::compile_spirv(build_stage_source(lines, "vertex"),
|
if (!glslang::compile_spirv(build_stage_source(lines, "vertex"),
|
||||||
glslang::StageVertex, &output->vertex))
|
glslang::StageVertex, &output->vertex))
|
||||||
{
|
{
|
||||||
|
@ -289,7 +289,6 @@ static bool add_active_buffer_ranges(const Compiler &compiler, const Resource &r
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// TODO: Handle invalid semantics as user defined.
|
|
||||||
RARCH_ERR("[slang]: Unknown semantic found.\n");
|
RARCH_ERR("[slang]: Unknown semantic found.\n");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user