summaryrefslogtreecommitdiffstats
path: root/video/out/gpu/spirv_shaderc.c
blob: ee702053d52b091e2cc4714cb44e876bbacb5d02 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
#include "common/msg.h"

#include "context.h"
#include "spirv.h"

#include <shaderc/shaderc.h>

struct priv {
    shaderc_compiler_t compiler;
    shaderc_compile_options_t opts;
};

static void shaderc_uninit(struct ra_ctx *ctx)
{
    struct priv *p = ctx->spirv->priv;
    if (!p)
        return;

    shaderc_compile_options_release(p->opts);
    shaderc_compiler_release(p->compiler);
}

static bool shaderc_init(struct ra_ctx *ctx)
{
    struct priv *p = ctx->spirv->priv = talloc_zero(ctx->spirv, struct priv);

    p->compiler = shaderc_compiler_initialize();
    if (!p->compiler)
        goto error;
    p->opts = shaderc_compile_options_initialize();
    if (!p->opts)
        goto error;

    shaderc_compile_options_set_optimization_level(p->opts,
                                            shaderc_optimization_level_size);
    if (ctx->opts.debug)
        shaderc_compile_options_set_generate_debug_info(p->opts);

    int ver, rev;
    shaderc_get_spv_version(&ver, &rev);
    ctx->spirv->compiler_version = ver * 100 + rev; // forwards compatibility
    ctx->spirv->glsl_version = 450; // impossible to query?
    return true;

error:
    shaderc_uninit(ctx);
    return false;
}

static shaderc_compilation_result_t compile(struct priv *p,
                                            enum glsl_shader type,
                                            const char *glsl, bool debug)
{
    static const shaderc_shader_kind kinds[] = {
        [GLSL_SHADER_VERTEX]   = shaderc_glsl_vertex_shader,
        [GLSL_SHADER_FRAGMENT] = shaderc_glsl_fragment_shader,
        [GLSL_SHADER_COMPUTE]  = shaderc_glsl_compute_shader,
    };

    if (debug) {
        return shaderc_compile_into_spv_assembly(p->compiler, glsl, strlen(glsl),
                                        kinds[type], "input", "main", p->opts);
    } else {
        return shaderc_compile_into_spv(p->compiler, glsl, strlen(glsl),
                                        kinds[type], "input", "main", p->opts);
    }
}

static bool shaderc_compile(struct spirv_compiler *spirv, void *tactx,
                            enum glsl_shader type, const char *glsl,
                            struct bstr *out_spirv)
{
    struct priv *p = spirv->priv;

    shaderc_compilation_result_t res = compile(p, type, glsl, false);
    int errs = shaderc_result_get_num_errors(res),
        warn = shaderc_result_get_num_warnings(res),
        msgl = errs ? MSGL_ERR : warn ? MSGL_WARN : MSGL_V;

    const char *msg = shaderc_result_get_error_message(res);
    if (msg[0])
        MP_MSG(spirv, msgl, "shaderc output:\n%s", msg);

    int s = shaderc_result_get_compilation_status(res);
    bool success = s == shaderc_compilation_status_success;

    static const char *results[] = {
        [shaderc_compilation_status_success]            = "success",
        [shaderc_compilation_status_invalid_stage]      = "invalid stage",
        [shaderc_compilation_status_compilation_error]  = "error",
        [shaderc_compilation_status_internal_error]     = "internal error",
        [shaderc_compilation_status_null_result_object] = "no result",
        [shaderc_compilation_status_invalid_assembly]   = "invalid assembly",
    };

    const char *status = s < MP_ARRAY_SIZE(results) ? results[s] : "unknown";
    MP_MSG(spirv, msgl, "shaderc compile status '%s' (%d errors, %d warnings)\n",
           status, errs, warn);

    if (success) {
        void *bytes = (void *) shaderc_result_get_bytes(res);
        out_spirv->len = shaderc_result_get_length(res);
        out_spirv->start = talloc_memdup(tactx, bytes, out_spirv->len);
    }

    // Also print SPIR-V disassembly for debugging purposes. Unfortunately
    // there doesn't seem to be a way to get this except compiling the shader
    // a second time..
    if (mp_msg_test(spirv->log, MSGL_TRACE)) {
        shaderc_compilation_result_t dis = compile(p, type, glsl, true);
        MP_TRACE(spirv, "Generated SPIR-V:\n%.*s",
                 (int)shaderc_result_get_length(dis),
                 shaderc_result_get_bytes(dis));
        shaderc_result_release(dis);
    }

    shaderc_result_release(res);
    return success;
}

const struct spirv_compiler_fns spirv_shaderc = {
    .compile_glsl = shaderc_compile,
    .init = shaderc_init,
    .uninit = shaderc_uninit,
};