Merge pull request #3312 from Themaister/master

Vulkan: Make glslang shaders debuggable.
This commit is contained in:
Twinaphex 2016-08-01 14:36:17 +02:00 committed by GitHub
commit 210db93237
2 changed files with 59 additions and 29 deletions

View File

@ -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))
{ {

View File

@ -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;
} }