mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-04 02:41:19 +00:00
7d1cd11ba9
To avoid duplicated compilation for openmw translation units. Link openmw-lib to openmw-tests instead.
235 lines
8.2 KiB
C++
235 lines
8.2 KiB
C++
#include <components/files/conversion.hpp>
|
|
#include <components/shader/shadermanager.hpp>
|
|
#include <components/testing/util.hpp>
|
|
|
|
#include <fstream>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
namespace
|
|
{
|
|
using namespace testing;
|
|
using namespace Shader;
|
|
|
|
struct ShaderManagerTest : Test
|
|
{
|
|
ShaderManager mManager;
|
|
ShaderManager::DefineMap mDefines;
|
|
|
|
ShaderManagerTest() { mManager.setShaderPath("tests_output"); }
|
|
|
|
template <class F>
|
|
void withShaderFile(const std::string& content, F&& f)
|
|
{
|
|
withShaderFile("", content, std::forward<F>(f));
|
|
}
|
|
|
|
template <class F>
|
|
void withShaderFile(const std::string& suffix, const std::string& content, F&& f)
|
|
{
|
|
auto subdir = std::filesystem::path("lib")
|
|
/ std::filesystem::path(
|
|
std::string(UnitTest::GetInstance()->current_test_info()->name()) + suffix + ".vert");
|
|
auto path = TestingOpenMW::outputFilePathWithSubDir(subdir);
|
|
{
|
|
std::ofstream stream(path);
|
|
stream << content;
|
|
stream.close();
|
|
}
|
|
|
|
f(subdir);
|
|
}
|
|
};
|
|
|
|
TEST_F(ShaderManagerTest, get_shader_with_empty_content_should_succeed)
|
|
{
|
|
const std::string content;
|
|
|
|
withShaderFile(content, [this](const std::filesystem::path& templateName) {
|
|
EXPECT_TRUE(mManager.getShader(Files::pathToUnicodeString(templateName)));
|
|
});
|
|
}
|
|
|
|
TEST_F(ShaderManagerTest, get_shader_should_not_change_source_without_template_parameters)
|
|
{
|
|
const std::string content
|
|
= "#version 120\n"
|
|
"void main() {}\n";
|
|
|
|
withShaderFile(content, [&](const std::filesystem::path& templateName) {
|
|
const auto shader = mManager.getShader(Files::pathToUnicodeString(templateName), mDefines);
|
|
ASSERT_TRUE(shader);
|
|
EXPECT_EQ(shader->getShaderSource(), content);
|
|
});
|
|
}
|
|
|
|
TEST_F(ShaderManagerTest, get_shader_should_replace_includes_with_content)
|
|
{
|
|
const std::string content0 = "void foo() {}\n";
|
|
|
|
withShaderFile("_0", content0, [&](const std::filesystem::path& templateName0) {
|
|
const std::string content1 =
|
|
"#include \"" + Files::pathToUnicodeString(templateName0) + "\"\n"
|
|
"void bar() { foo() }\n";
|
|
|
|
withShaderFile("_1", content1, [&](const std::filesystem::path& templateName1) {
|
|
const std::string content2 =
|
|
"#version 120\n"
|
|
"#include \"" + Files::pathToUnicodeString(templateName1) + "\"\n"
|
|
"void main() { bar() }\n";
|
|
|
|
withShaderFile(content2, [&](const std::filesystem::path& templateName2) {
|
|
const auto shader = mManager.getShader(Files::pathToUnicodeString(templateName2), mDefines);
|
|
ASSERT_TRUE(shader);
|
|
const std::string expected
|
|
= "#version 120\n"
|
|
"#line 0 1\n"
|
|
"#line 0 2\n"
|
|
"void foo() {}\n"
|
|
"\n"
|
|
"#line 0 0\n"
|
|
"\n"
|
|
"void bar() { foo() }\n"
|
|
"\n"
|
|
"#line 1 0\n"
|
|
"\n"
|
|
"void main() { bar() }\n";
|
|
EXPECT_EQ(shader->getShaderSource(), expected);
|
|
});
|
|
});
|
|
});
|
|
}
|
|
|
|
TEST_F(ShaderManagerTest, get_shader_should_replace_defines)
|
|
{
|
|
const std::string content
|
|
= "#version 120\n"
|
|
"#define FLAG @flag\n"
|
|
"void main() {}\n";
|
|
|
|
withShaderFile(content, [&](const std::filesystem::path& templateName) {
|
|
mDefines["flag"] = "1";
|
|
const auto shader = mManager.getShader(Files::pathToUnicodeString(templateName), mDefines);
|
|
ASSERT_TRUE(shader);
|
|
const std::string expected
|
|
= "#version 120\n"
|
|
"#define FLAG 1\n"
|
|
"void main() {}\n";
|
|
EXPECT_EQ(shader->getShaderSource(), expected);
|
|
});
|
|
}
|
|
|
|
TEST_F(ShaderManagerTest, get_shader_should_expand_loop)
|
|
{
|
|
const std::string content
|
|
= "#version 120\n"
|
|
"@foreach index @list\n"
|
|
" varying vec4 foo@index;\n"
|
|
"@endforeach\n"
|
|
"void main() {}\n";
|
|
|
|
withShaderFile(content, [&](const std::filesystem::path& templateName) {
|
|
mDefines["list"] = "1,2,3";
|
|
const auto shader = mManager.getShader(Files::pathToUnicodeString(templateName), mDefines);
|
|
ASSERT_TRUE(shader);
|
|
const std::string expected
|
|
= "#version 120\n"
|
|
" varying vec4 foo1;\n"
|
|
" varying vec4 foo2;\n"
|
|
" varying vec4 foo3;\n"
|
|
"\n"
|
|
"#line 5\n"
|
|
"void main() {}\n";
|
|
EXPECT_EQ(shader->getShaderSource(), expected);
|
|
});
|
|
}
|
|
|
|
TEST_F(ShaderManagerTest, get_shader_should_replace_loops_with_conditions)
|
|
{
|
|
const std::string content
|
|
= "#version 120\n"
|
|
"@foreach index @list\n"
|
|
" varying vec4 foo@index;\n"
|
|
"@endforeach\n"
|
|
"void main()\n"
|
|
"{\n"
|
|
"#ifdef BAR\n"
|
|
"@foreach index @list\n"
|
|
" foo@index = vec4(1.0);\n"
|
|
"@endforeach\n"
|
|
"#elif BAZ\n"
|
|
"@foreach index @list\n"
|
|
" foo@index = vec4(2.0);\n"
|
|
"@endforeach\n"
|
|
"#else\n"
|
|
"@foreach index @list\n"
|
|
" foo@index = vec4(3.0);\n"
|
|
"@endforeach\n"
|
|
"#endif\n"
|
|
"}\n";
|
|
|
|
withShaderFile(content, [&](const std::filesystem::path& templateName) {
|
|
mDefines["list"] = "1,2,3";
|
|
const auto shader = mManager.getShader(Files::pathToUnicodeString(templateName), mDefines);
|
|
ASSERT_TRUE(shader);
|
|
const std::string expected
|
|
= "#version 120\n"
|
|
" varying vec4 foo1;\n"
|
|
" varying vec4 foo2;\n"
|
|
" varying vec4 foo3;\n"
|
|
"\n"
|
|
"#line 5\n"
|
|
"void main()\n"
|
|
"{\n"
|
|
"#ifdef BAR\n"
|
|
" foo1 = vec4(1.0);\n"
|
|
" foo2 = vec4(1.0);\n"
|
|
" foo3 = vec4(1.0);\n"
|
|
"\n"
|
|
"#line 11\n"
|
|
"#elif BAZ\n"
|
|
"#line 12\n"
|
|
" foo1 = vec4(2.0);\n"
|
|
" foo2 = vec4(2.0);\n"
|
|
" foo3 = vec4(2.0);\n"
|
|
"\n"
|
|
"#line 15\n"
|
|
"#else\n"
|
|
"#line 16\n"
|
|
" foo1 = vec4(3.0);\n"
|
|
" foo2 = vec4(3.0);\n"
|
|
" foo3 = vec4(3.0);\n"
|
|
"\n"
|
|
"#line 19\n"
|
|
"#endif\n"
|
|
"#line 20\n"
|
|
"}\n";
|
|
EXPECT_EQ(shader->getShaderSource(), expected);
|
|
});
|
|
}
|
|
|
|
TEST_F(ShaderManagerTest, get_shader_should_fail_on_absent_template_parameters_in_single_line_comments)
|
|
{
|
|
const std::string content
|
|
= "#version 120\n"
|
|
"// #define FLAG @flag\n"
|
|
"void main() {}\n";
|
|
|
|
withShaderFile(content, [&](const std::filesystem::path& templateName) {
|
|
EXPECT_FALSE(mManager.getShader(Files::pathToUnicodeString(templateName), mDefines));
|
|
});
|
|
}
|
|
|
|
TEST_F(ShaderManagerTest, get_shader_should_fail_on_absent_template_parameter_in_multi_line_comments)
|
|
{
|
|
const std::string content
|
|
= "#version 120\n"
|
|
"/* #define FLAG @flag */\n"
|
|
"void main() {}\n";
|
|
|
|
withShaderFile(content, [&](const std::filesystem::path& templateName) {
|
|
EXPECT_FALSE(mManager.getShader(Files::pathToUnicodeString(templateName), mDefines));
|
|
});
|
|
}
|
|
}
|