/* * This file is part of libplacebo. * * libplacebo 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. * * libplacebo 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 libplacebo. If not, see . */ #include "hash.h" #include "spirv.h" #include "utils.h" #include "glsl/glslang.h" // This header contains only preprocessor definitions #include // This is awkward, but we cannot use upstream macro, it was fixed in 11.11.0 #define PL_GLSLANG_VERSION_GREATER_THAN(major, minor, patch) \ ((GLSLANG_VERSION_MAJOR) > (major) || ((major) == GLSLANG_VERSION_MAJOR && \ ((GLSLANG_VERSION_MINOR) > (minor) || ((minor) == GLSLANG_VERSION_MINOR && \ (GLSLANG_VERSION_PATCH) > (patch))))) #if PL_GLSLANG_VERSION_GREATER_THAN(11, 8, 0) #define GLSLANG_SPV_MAX PL_SPV_VERSION(1, 6) #elif PL_GLSLANG_VERSION_GREATER_THAN(7, 13, 3496) #define GLSLANG_SPV_MAX PL_SPV_VERSION(1, 5) #elif PL_GLSLANG_VERSION_GREATER_THAN(6, 2, 2596) #define GLSLANG_SPV_MAX PL_SPV_VERSION(1, 3) #else #define GLSLANG_SPV_MAX PL_SPV_VERSION(1, 0) #endif const struct spirv_compiler pl_spirv_glslang; static void glslang_destroy(pl_spirv spirv) { pl_glslang_uninit(); pl_free((void *) spirv); } static pl_spirv glslang_create(pl_log log, struct pl_spirv_version spirv_ver) { if (!pl_glslang_init()) { pl_fatal(log, "Failed initializing glslang SPIR-V compiler!"); return NULL; } struct pl_spirv_t *spirv = pl_alloc_ptr(NULL, spirv); *spirv = (struct pl_spirv_t) { .signature = pl_str0_hash(pl_spirv_glslang.name), .impl = &pl_spirv_glslang, .version = spirv_ver, .log = log, }; PL_INFO(spirv, "glslang version: %d.%d.%d", GLSLANG_VERSION_MAJOR, GLSLANG_VERSION_MINOR, GLSLANG_VERSION_PATCH); // Clamp to supported version by glslang if (GLSLANG_SPV_MAX < spirv->version.spv_version) { spirv->version.spv_version = GLSLANG_SPV_MAX; spirv->version.env_version = pl_spirv_version_to_vulkan(GLSLANG_SPV_MAX); } pl_hash_merge(&spirv->signature, (uint64_t) spirv->version.spv_version << 32 | spirv->version.env_version); pl_hash_merge(&spirv->signature, (GLSLANG_VERSION_MAJOR & 0xFF) << 24 | (GLSLANG_VERSION_MINOR & 0xFF) << 16 | (GLSLANG_VERSION_PATCH & 0xFFFF)); return spirv; } static pl_str glslang_compile(pl_spirv spirv, void *alloc, struct pl_glsl_version glsl_ver, enum glsl_shader_stage stage, const char *shader) { struct pl_glslang_res *res; res = pl_glslang_compile(glsl_ver, spirv->version, stage, shader); if (!res || !res->success) { PL_ERR(spirv, "glslang failed: %s", res ? res->error_msg : "(null)"); pl_free(res); return (struct pl_str) {0}; } struct pl_str ret = { .buf = pl_steal(alloc, res->data), .len = res->size, }; pl_free(res); return ret; } const struct spirv_compiler pl_spirv_glslang = { .name = "glslang", .destroy = glslang_destroy, .create = glslang_create, .compile = glslang_compile, };