/* * Copyright 2018 Józef Kucia for CodeWeavers * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include "vkd3d_test.h" #include #include static void test_invalid_shaders(void) { struct vkd3d_shader_compile_info info; struct vkd3d_shader_code spirv; int rc; static const DWORD ps_break_code[] = { #if 0 ps_4_0 dcl_constantbuffer cb0[1], immediateIndexed dcl_output o0.xyzw if_z cb0[0].x mov o0.xyzw, l(1.000000,1.000000,1.000000,1.000000) break endif mov o0.xyzw, l(0,0,0,0) ret #endif 0x43425844, 0x1316702a, 0xb1a7ebfc, 0xf477753e, 0x72605647, 0x00000001, 0x000000f8, 0x00000003, 0x0000002c, 0x0000003c, 0x00000070, 0x4e475349, 0x00000008, 0x00000000, 0x00000008, 0x4e47534f, 0x0000002c, 0x00000001, 0x00000008, 0x00000020, 0x00000000, 0x00000000, 0x00000001, 0x00000000, 0x0000000f, 0x545f5653, 0x65677261, 0xabab0074, 0x52444853, 0x00000080, 0x00000040, 0x00000020, 0x04000059, 0x00208e46, 0x00000000, 0x00000001, 0x03000065, 0x001020f2, 0x00000000, 0x0400001f, 0x0020800a, 0x00000000, 0x00000000, 0x08000036, 0x001020f2, 0x00000000, 0x00004002, 0x3f800000, 0x3f800000, 0x3f800000, 0x3f800000, 0x01000002, 0x01000015, 0x08000036, 0x001020f2, 0x00000000, 0x00004002, 0x00000000, 0x00000000, 0x00000000, 0x00000000, 0x0100003e, }; static const struct vkd3d_shader_compile_option option = { .name = VKD3D_SHADER_COMPILE_OPTION_STRIP_DEBUG, .value = 1, }; info.type = VKD3D_SHADER_STRUCTURE_TYPE_COMPILE_INFO; info.next = NULL; info.source.code = ps_break_code; info.source.size = sizeof(ps_break_code); info.source_type = VKD3D_SHADER_SOURCE_DXBC_TPF; info.target_type = VKD3D_SHADER_TARGET_SPIRV_BINARY; info.options = &option; info.option_count = 1; info.log_level = VKD3D_SHADER_LOG_NONE; info.source_name = NULL; rc = vkd3d_shader_compile(&info, &spirv, NULL); ok(rc == VKD3D_ERROR_INVALID_SHADER, "Got unexpected error code %d.\n", rc); rc = vkd3d_shader_scan(&info, NULL); ok(rc == VKD3D_ERROR_INVALID_SHADER, "Got unexpected error code %d.\n", rc); } static void test_vkd3d_shader_pfns(void) { PFN_vkd3d_shader_get_supported_source_types pfn_vkd3d_shader_get_supported_source_types; PFN_vkd3d_shader_get_supported_target_types pfn_vkd3d_shader_get_supported_target_types; PFN_vkd3d_shader_free_scan_descriptor_info pfn_vkd3d_shader_free_scan_descriptor_info; PFN_vkd3d_shader_serialize_root_signature pfn_vkd3d_shader_serialize_root_signature; PFN_vkd3d_shader_find_signature_element pfn_vkd3d_shader_find_signature_element; PFN_vkd3d_shader_free_shader_signature pfn_vkd3d_shader_free_shader_signature; PFN_vkd3d_shader_parse_input_signature pfn_vkd3d_shader_parse_input_signature; PFN_vkd3d_shader_parse_root_signature pfn_vkd3d_shader_parse_root_signature; PFN_vkd3d_shader_free_root_signature pfn_vkd3d_shader_free_root_signature; PFN_vkd3d_shader_free_shader_code pfn_vkd3d_shader_free_shader_code; PFN_vkd3d_shader_get_version pfn_vkd3d_shader_get_version; PFN_vkd3d_shader_compile pfn_vkd3d_shader_compile; PFN_vkd3d_shader_scan pfn_vkd3d_shader_scan; struct vkd3d_shader_versioned_root_signature_desc root_signature_desc; unsigned int major, minor, expected_major, expected_minor; struct vkd3d_shader_scan_descriptor_info descriptor_info; const enum vkd3d_shader_source_type *source_types; const enum vkd3d_shader_target_type *target_types; struct vkd3d_shader_signature_element *element; struct vkd3d_shader_compile_info compile_info; unsigned int i, j, source_count, target_count; struct vkd3d_shader_signature signature; struct vkd3d_shader_code dxbc, spirv; const char *version, *p; bool b; int rc; static const struct vkd3d_shader_versioned_root_signature_desc empty_rs_desc = { .version = VKD3D_SHADER_ROOT_SIGNATURE_VERSION_1_0, }; static const DWORD vs_code[] = { #if 0 float4 main(int4 p : POSITION) : SV_Position { return p; } #endif 0x43425844, 0x3fd50ab1, 0x580a1d14, 0x28f5f602, 0xd1083e3a, 0x00000001, 0x000000d8, 0x00000003, 0x0000002c, 0x00000060, 0x00000094, 0x4e475349, 0x0000002c, 0x00000001, 0x00000008, 0x00000020, 0x00000000, 0x00000000, 0x00000002, 0x00000000, 0x00000f0f, 0x49534f50, 0x4e4f4954, 0xababab00, 0x4e47534f, 0x0000002c, 0x00000001, 0x00000008, 0x00000020, 0x00000000, 0x00000001, 0x00000003, 0x00000000, 0x0000000f, 0x505f5653, 0x7469736f, 0x006e6f69, 0x52444853, 0x0000003c, 0x00010040, 0x0000000f, 0x0300005f, 0x001010f2, 0x00000000, 0x04000067, 0x001020f2, 0x00000000, 0x00000001, 0x0500002b, 0x001020f2, 0x00000000, 0x00101e46, 0x00000000, 0x0100003e, }; static const struct vkd3d_shader_code vs = {vs_code, sizeof(vs_code)}; pfn_vkd3d_shader_get_supported_source_types = vkd3d_shader_get_supported_source_types; pfn_vkd3d_shader_get_supported_target_types = vkd3d_shader_get_supported_target_types; pfn_vkd3d_shader_free_scan_descriptor_info = vkd3d_shader_free_scan_descriptor_info; pfn_vkd3d_shader_serialize_root_signature = vkd3d_shader_serialize_root_signature; pfn_vkd3d_shader_find_signature_element = vkd3d_shader_find_signature_element; pfn_vkd3d_shader_free_shader_signature = vkd3d_shader_free_shader_signature; pfn_vkd3d_shader_parse_input_signature = vkd3d_shader_parse_input_signature; pfn_vkd3d_shader_parse_root_signature = vkd3d_shader_parse_root_signature; pfn_vkd3d_shader_free_root_signature = vkd3d_shader_free_root_signature; pfn_vkd3d_shader_free_shader_code = vkd3d_shader_free_shader_code; pfn_vkd3d_shader_get_version = vkd3d_shader_get_version; pfn_vkd3d_shader_compile = vkd3d_shader_compile; pfn_vkd3d_shader_scan = vkd3d_shader_scan; sscanf(PACKAGE_VERSION, "%d.%d", &expected_major, &expected_minor); version = pfn_vkd3d_shader_get_version(&major, &minor); p = strstr(version, "vkd3d-shader " PACKAGE_VERSION); ok(p == version, "Got unexpected version string \"%s\"\n", version); ok(major == expected_major, "Got unexpected major version %u.\n", major); ok(minor == expected_minor, "Got unexpected minor version %u.\n", minor); source_types = pfn_vkd3d_shader_get_supported_source_types(&source_count); ok(source_types, "Got unexpected source types array %p.\n", source_types); ok(source_count, "Got unexpected source type count %u.\n", source_count); b = false; for (i = 0; i < source_count; ++i) { target_types = pfn_vkd3d_shader_get_supported_target_types(source_types[i], &target_count); ok(target_types, "Got unexpected target types array %p.\n", target_types); ok(target_count, "Got unexpected target type count %u.\n", target_count); for (j = 0; j < target_count; ++j) { if (source_types[i] == VKD3D_SHADER_SOURCE_DXBC_TPF && target_types[j] == VKD3D_SHADER_TARGET_SPIRV_BINARY) b = true; } } ok(b, "The dxbc-tpf source type with spirv-binary target type is not supported.\n"); rc = pfn_vkd3d_shader_serialize_root_signature(&empty_rs_desc, &dxbc, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); rc = pfn_vkd3d_shader_parse_root_signature(&dxbc, &root_signature_desc, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); pfn_vkd3d_shader_free_root_signature(&root_signature_desc); pfn_vkd3d_shader_free_shader_code(&dxbc); rc = pfn_vkd3d_shader_parse_input_signature(&vs, &signature, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); element = pfn_vkd3d_shader_find_signature_element(&signature, "position", 0, 0); ok(element, "Could not find shader signature element.\n"); pfn_vkd3d_shader_free_shader_signature(&signature); compile_info.type = VKD3D_SHADER_STRUCTURE_TYPE_COMPILE_INFO; compile_info.next = NULL; compile_info.source = vs; compile_info.source_type = VKD3D_SHADER_SOURCE_DXBC_TPF; compile_info.target_type = VKD3D_SHADER_TARGET_SPIRV_BINARY; compile_info.options = NULL; compile_info.option_count = 0; compile_info.log_level = VKD3D_SHADER_LOG_NONE; compile_info.source_name = NULL; rc = pfn_vkd3d_shader_compile(&compile_info, &spirv, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); pfn_vkd3d_shader_free_shader_code(&spirv); memset(&descriptor_info, 0, sizeof(descriptor_info)); descriptor_info.type = VKD3D_SHADER_STRUCTURE_TYPE_SCAN_DESCRIPTOR_INFO; compile_info.next = &descriptor_info; rc = pfn_vkd3d_shader_scan(&compile_info, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); pfn_vkd3d_shader_free_scan_descriptor_info(&descriptor_info); } static void test_version(void) { unsigned int major, minor, expected_major, expected_minor; const char *version, *p; sscanf(PACKAGE_VERSION, "%d.%d", &expected_major, &expected_minor); version = vkd3d_shader_get_version(NULL, NULL); p = strstr(version, "vkd3d-shader " PACKAGE_VERSION); ok(p == version, "Got unexpected version string \"%s\"\n", version); major = ~0u; vkd3d_shader_get_version(&major, NULL); ok(major == expected_major, "Got unexpected major version %u.\n", major); minor = ~0u; vkd3d_shader_get_version(NULL, &minor); ok(minor == expected_minor, "Got unexpected minor version %u.\n", minor); major = minor = ~0u; vkd3d_shader_get_version(&major, &minor); ok(major == expected_major, "Got unexpected major version %u.\n", major); ok(minor == expected_minor, "Got unexpected minor version %u.\n", minor); } static void test_d3dbc(void) { struct vkd3d_shader_compile_info info; struct vkd3d_shader_code d3d_asm; int rc; static const uint32_t vs_minimal[] = { 0xfffe0100, /* vs_1_0 */ 0x0000ffff, /* end */ }; static const uint32_t invalid_type[] = { 0x00010100, /* _1_0 */ 0x0000ffff, /* end */ }; static const uint32_t invalid_version[] = { 0xfffe0400, /* vs_4_0 */ 0x0000ffff, /* end */ }; static const uint32_t ps[] = { 0xffff0101, /* ps_1_1 */ 0x00000001, 0x800f0000, 0x90e40000, /* mov r0, v0 */ 0x0000ffff, /* end */ }; static const char expected[] = "vs_1_0\n"; info.type = VKD3D_SHADER_STRUCTURE_TYPE_COMPILE_INFO; info.next = NULL; info.source.code = vs_minimal; info.source.size = sizeof(vs_minimal); info.source_type = VKD3D_SHADER_SOURCE_D3D_BYTECODE; info.target_type = VKD3D_SHADER_TARGET_D3D_ASM; info.options = NULL; info.option_count = 0; info.log_level = VKD3D_SHADER_LOG_NONE; info.source_name = NULL; rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); ok(d3d_asm.size == strlen(expected), "Got unexpected size %zu.\n", d3d_asm.size); ok(!memcmp(d3d_asm.code, expected, d3d_asm.size), "Got unexpected code \"%.*s\"\n", (int)d3d_asm.size, (char *)d3d_asm.code); vkd3d_shader_free_shader_code(&d3d_asm); rc = vkd3d_shader_scan(&info, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); info.source.size = sizeof(vs_minimal) + 1; rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); vkd3d_shader_free_shader_code(&d3d_asm); info.source.size = ~(size_t)0; rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); vkd3d_shader_free_shader_code(&d3d_asm); info.source.size = sizeof(vs_minimal) - 1; rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_ERROR_INVALID_SHADER, "Got unexpected error code %d.\n", rc); info.source.code = invalid_type; info.source.size = sizeof(invalid_type); rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_ERROR_INVALID_SHADER, "Got unexpected error code %d.\n", rc); info.source.code = invalid_version; info.source.size = sizeof(invalid_version); rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_ERROR_INVALID_SHADER, "Got unexpected error code %d.\n", rc); info.source.code = ps; info.source.size = sizeof(ps); rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_OK, "Got unexpected error code %d.\n", rc); vkd3d_shader_free_shader_code(&d3d_asm); /* Truncated before the destination parameter. */ info.source.size = sizeof(ps) - 3 * sizeof(*ps); rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_ERROR_INVALID_SHADER, "Got unexpected error code %d.\n", rc); /* Truncated before the source parameter. */ info.source.size = sizeof(ps) - 2 * sizeof(*ps); rc = vkd3d_shader_compile(&info, &d3d_asm, NULL); ok(rc == VKD3D_ERROR_INVALID_SHADER, "Got unexpected error code %d.\n", rc); } static void test_dxbc(void) { PFN_vkd3d_shader_serialize_dxbc pfn_vkd3d_shader_serialize_dxbc; PFN_vkd3d_shader_parse_dxbc pfn_vkd3d_shader_parse_dxbc; PFN_vkd3d_shader_free_dxbc pfn_vkd3d_shader_free_dxbc; struct vkd3d_shader_dxbc_desc dxbc_desc; const uint8_t *dxbc_start, *dxbc_end; struct vkd3d_shader_code dxbc; size_t expected_size; unsigned int i; int ret; static const uint32_t section_0[] = { 0x00000000, 0x00000001, 0x00000001, 0x00000002, 0x00000003, 0x00000005, 0x00000008, 0x0000000d, 0x00000015, 0x00000022, 0x00000037, 0x00000059, 0x00000090, 0x000000e9, 0x00000179, 0x00000262, }; static const uint8_t section_1[] = { 0x1, 0x4, 0x1, 0x5, 0x9, 0x2, 0x6, 0x5, 0x3, 0x5, 0x8, 0x9, 0x7, 0x9, 0x3, 0x2, 0x3, 0x8, 0x4, 0x6, 0x2, 0x6, 0x4, 0x3, 0x3, 0x8, 0x3, 0x2, 0x7, 0x9, 0x5, 0x0, 0x2, 0x8, 0x8, 0x4, 0x1, 0x9, 0x7, 0x1, 0x6, 0x9, 0x3, 0x9, 0x9, 0x3, 0x7, 0x5, 0x1, 0x0, 0x5, 0x8, 0x2, 0x0, 0x9, 0x7, 0x4, 0x9, 0x4, 0x4, 0x5, 0x9, 0x2, 0x3, }; static const struct vkd3d_shader_dxbc_section_desc sections[] = { {.tag = 0x00424946, .data = {.code = section_0, .size = sizeof(section_0)}}, {.tag = 0x49504950, .data = {.code = section_1, .size = sizeof(section_1)}}, }; static const uint32_t checksum[] = {0x7cfc687d, 0x7e8f4cff, 0x72a4739a, 0xd75c3703}; pfn_vkd3d_shader_serialize_dxbc = vkd3d_shader_serialize_dxbc; pfn_vkd3d_shader_parse_dxbc = vkd3d_shader_parse_dxbc; pfn_vkd3d_shader_free_dxbc = vkd3d_shader_free_dxbc; /* 8 * u32 for the DXBC header */ expected_size = 8 * sizeof(uint32_t); for (i = 0; i < ARRAY_SIZE(sections); ++i) { /* 1 u32 for the section offset + 2 u32 for the section header. */ expected_size += 3 * sizeof(uint32_t) + sections[i].data.size; } ret = pfn_vkd3d_shader_serialize_dxbc(ARRAY_SIZE(sections), sections, &dxbc, NULL); ok(ret == VKD3D_OK, "Got unexpected ret %d.\n", ret); ok(dxbc.size == expected_size, "Got unexpected size %zu, expected %zu.\n", dxbc_desc.size, expected_size); ret = pfn_vkd3d_shader_parse_dxbc(&dxbc, 0, &dxbc_desc, NULL); ok(ret == VKD3D_OK, "Got unexpected ret %d.\n", ret); ok(dxbc_desc.tag == 0x43425844, "Got unexpected tag 0x%08x.\n", dxbc_desc.tag); ok(!memcmp(dxbc_desc.checksum, checksum, sizeof(dxbc_desc.checksum)), "Got unexpected checksum 0x%08x 0x%08x 0x%08x 0x%08x.\n", dxbc_desc.checksum[0], dxbc_desc.checksum[1], dxbc_desc.checksum[2], dxbc_desc.checksum[3]); ok(dxbc_desc.version == 1, "Got unexpected version %#x.\n", dxbc_desc.version); ok(dxbc_desc.size == expected_size, "Got unexpected size %zu, expected %zu.\n", dxbc_desc.size, expected_size); ok(dxbc_desc.section_count == ARRAY_SIZE(sections), "Got unexpected section count %zu, expected %zu.\n", dxbc_desc.section_count, ARRAY_SIZE(sections)); dxbc_start = dxbc.code; dxbc_end = dxbc_start + dxbc.size; for (i = 0; i < dxbc_desc.section_count; ++i) { const struct vkd3d_shader_dxbc_section_desc *section = &dxbc_desc.sections[i]; const uint8_t *data = section->data.code; vkd3d_test_push_context("Section %u", i); ok(section->tag == sections[i].tag, "Got unexpected tag 0x%08x, expected 0x%08x.\n", section->tag, sections[i].tag); ok(section->data.size == sections[i].data.size, "Got unexpected size %zu, expected %zu.\n", section->data.size, sections[i].data.size); ok(!memcmp(data, sections[i].data.code, section->data.size), "Got unexpected section data.\n"); ok(data > dxbc_start && data <= dxbc_end - section->data.size, "Data {%p, %zu} is not contained within blob {%p, %zu}.\n", data, section->data.size, dxbc_start, dxbc_end - dxbc_start); vkd3d_test_pop_context(); } pfn_vkd3d_shader_free_dxbc(&dxbc_desc); vkd3d_shader_free_shader_code(&dxbc); } START_TEST(vkd3d_shader_api) { setlocale(LC_ALL, ""); run_test(test_invalid_shaders); run_test(test_vkd3d_shader_pfns); run_test(test_version); run_test(test_d3dbc); run_test(test_dxbc); }