blob: cc3bae1cb55121c6973d04dbfccaae8e184b7867 [file] [log] [blame]
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +01001#include <libspirv/libspirv.h>
2
3#include <string.h>
4
Lei Zhang2d879b92015-08-27 11:22:14 -04005/// Generate a spv_ext_inst_desc_t literal for a GLSL std450 extended
6/// instruction with one/two/three <id> parameter(s).
7#define GLSL450Inst1(name) \
Lei Zhangdca65b32015-08-27 10:00:16 -04008 #name, GLSLstd450::GLSLstd450##name, { SPV_OPERAND_TYPE_ID }
Lei Zhang2d879b92015-08-27 11:22:14 -04009#define GLSL450Inst2(name) \
Lei Zhangdca65b32015-08-27 10:00:16 -040010 #name, GLSLstd450::GLSLstd450##name, { \
11 SPV_OPERAND_TYPE_ID, SPV_OPERAND_TYPE_ID \
12 }
Lei Zhang2d879b92015-08-27 11:22:14 -040013#define GLSL450Inst3(name) \
Lei Zhangdca65b32015-08-27 10:00:16 -040014 #name, GLSLstd450::GLSLstd450##name, { \
15 SPV_OPERAND_TYPE_ID, SPV_OPERAND_TYPE_ID, SPV_OPERAND_TYPE_ID \
16 }
Andrew Woloszyn1d2a87e2015-08-21 13:39:34 -040017
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010018static const spv_ext_inst_desc_t glslStd450Entries[] = {
Lei Zhang2d879b92015-08-27 11:22:14 -040019 {GLSL450Inst1(Round)},
20 {GLSL450Inst1(RoundEven)},
21 {GLSL450Inst1(Trunc)},
22 {GLSL450Inst1(FAbs)},
23 {GLSL450Inst1(SAbs)},
24 {GLSL450Inst1(FSign)},
25 {GLSL450Inst1(SSign)},
26 {GLSL450Inst1(Floor)},
27 {GLSL450Inst1(Ceil)},
28 {GLSL450Inst1(Fract)},
29 {GLSL450Inst1(Radians)},
30 {GLSL450Inst1(Degrees)},
31 {GLSL450Inst1(Sin)},
32 {GLSL450Inst1(Cos)},
33 {GLSL450Inst1(Tan)},
34 {GLSL450Inst1(Asin)},
35 {GLSL450Inst1(Acos)},
36 {GLSL450Inst1(Atan)},
37 {GLSL450Inst1(Sinh)},
38 {GLSL450Inst1(Cosh)},
39 {GLSL450Inst1(Tanh)},
40 {GLSL450Inst1(Asinh)},
41 {GLSL450Inst1(Acosh)},
42 {GLSL450Inst1(Atanh)},
43 {GLSL450Inst2(Atan2)},
44 {GLSL450Inst2(Pow)},
45 {GLSL450Inst1(Exp)},
46 {GLSL450Inst1(Log)},
47 {GLSL450Inst1(Exp2)},
48 {GLSL450Inst1(Log2)},
49 {GLSL450Inst1(Sqrt)},
Lei Zhangdca65b32015-08-27 10:00:16 -040050 // clang-format off
51 {"Inversesqrt", GLSLstd450::GLSLstd450InverseSqrt, {SPV_OPERAND_TYPE_ID}},
Lei Zhang2d879b92015-08-27 11:22:14 -040052 {GLSL450Inst1(Determinant)},
Lei Zhangdca65b32015-08-27 10:00:16 -040053 {"Inverse", GLSLstd450::GLSLstd450MatrixInverse, {SPV_OPERAND_TYPE_ID}},
54 // clang-format on
Lei Zhang2d879b92015-08-27 11:22:14 -040055 {GLSL450Inst2(Modf)},
56 {GLSL450Inst1(ModfStruct)},
57 {GLSL450Inst2(FMin)},
58 {GLSL450Inst2(UMin)},
59 {GLSL450Inst2(SMin)},
60 {GLSL450Inst2(FMax)},
61 {GLSL450Inst2(UMax)},
62 {GLSL450Inst2(SMax)},
63 {GLSL450Inst3(FClamp)},
64 {GLSL450Inst3(UClamp)},
65 {GLSL450Inst3(SClamp)},
66 {GLSL450Inst3(Mix)},
67 {GLSL450Inst2(Step)},
Lei Zhangdca65b32015-08-27 10:00:16 -040068 // clang-format off
69 {"Smoothstep", GLSLstd450::GLSLstd450SmoothStep, {SPV_OPERAND_TYPE_ID, SPV_OPERAND_TYPE_ID, SPV_OPERAND_TYPE_ID}},
70 // clang-format on
Lei Zhang2d879b92015-08-27 11:22:14 -040071 {GLSL450Inst3(Fma)},
72 {GLSL450Inst2(Frexp)},
73 {GLSL450Inst1(FrexpStruct)},
74 {GLSL450Inst2(Ldexp)},
Kenneth Benzie (Benie)83e5a292015-05-22 18:26:19 +010075 // TODO: Add remaining GLSL.std.450 instructions
76};
77
78static const spv_ext_inst_desc_t openclStd12Entries[] = {
79 {"placeholder", 0, {}},
80 // TODO: Add remaining OpenCL.std.12 instructions
81};
82
83static const spv_ext_inst_desc_t openclStd20Entries[] = {
84 {"placeholder", 0, {}},
85 // TODO: Add remaining OpenCL.std.20 instructions
86};
87
88static const spv_ext_inst_desc_t openclStd21Entries[] = {
89 {"placeholder", 0, {}},
90 // TODO: Add remaining OpenCL.std.21 instructions
91};
92
93spv_result_t spvExtInstTableGet(spv_ext_inst_table *pExtInstTable) {
94 spvCheck(!pExtInstTable, return SPV_ERROR_INVALID_POINTER);
95
96 static const spv_ext_inst_group_t groups[] = {
97 {SPV_EXT_INST_TYPE_GLSL_STD_450,
98 sizeof(glslStd450Entries) / sizeof(spv_ext_inst_desc_t),
99 glslStd450Entries},
100 {SPV_EXT_INST_TYPE_OPENCL_STD_12,
101 sizeof(openclStd12Entries) / sizeof(spv_ext_inst_desc_t),
102 openclStd12Entries},
103 {SPV_EXT_INST_TYPE_OPENCL_STD_20,
104 sizeof(openclStd20Entries) / sizeof(spv_ext_inst_desc_t),
105 openclStd20Entries},
106 {SPV_EXT_INST_TYPE_OPENCL_STD_21,
107 sizeof(openclStd21Entries) / sizeof(spv_ext_inst_desc_t),
108 openclStd21Entries},
109 };
110
111 static const spv_ext_inst_table_t table = {
112 sizeof(groups) / sizeof(spv_ext_inst_group_t), groups};
113
114 *pExtInstTable = &table;
115
116 return SPV_SUCCESS;
117}
118
119spv_ext_inst_type_t spvExtInstImportTypeGet(const char *name) {
120 if (!strcmp("GLSL.std.450", name)) {
121 return SPV_EXT_INST_TYPE_GLSL_STD_450;
122 }
123 if (!strcmp("OpenCL.std.12", name)) {
124 return SPV_EXT_INST_TYPE_OPENCL_STD_12;
125 }
126 if (!strcmp("OpenCL.std.20", name)) {
127 return SPV_EXT_INST_TYPE_OPENCL_STD_20;
128 }
129 if (!strcmp("OpenCL.std.21", name)) {
130 return SPV_EXT_INST_TYPE_OPENCL_STD_21;
131 }
132 return SPV_EXT_INST_TYPE_NONE;
133}
134
135spv_result_t spvExtInstTableNameLookup(const spv_ext_inst_table table,
136 const spv_ext_inst_type_t type,
137 const char *name,
138 spv_ext_inst_desc *pEntry) {
139 spvCheck(!table, return SPV_ERROR_INVALID_TABLE);
140 spvCheck(!pEntry, return SPV_ERROR_INVALID_POINTER);
141
142 for (uint32_t groupIndex = 0; groupIndex < table->count; groupIndex++) {
143 auto &group = table->groups[groupIndex];
144 if (type == group.type) {
145 for (uint32_t index = 0; index < group.count; index++) {
146 auto &entry = group.entries[index];
147 if (!strcmp(name, entry.name)) {
148 *pEntry = &table->groups[groupIndex].entries[index];
149 return SPV_SUCCESS;
150 }
151 }
152 }
153 }
154
155 return SPV_ERROR_INVALID_LOOKUP;
156}
157
158spv_result_t spvExtInstTableValueLookup(const spv_ext_inst_table table,
159 const spv_ext_inst_type_t type,
160 const uint32_t value,
161 spv_ext_inst_desc *pEntry) {
162 spvCheck(!table, return SPV_ERROR_INVALID_TABLE);
163 spvCheck(!pEntry, return SPV_ERROR_INVALID_POINTER);
164
165 for (uint32_t groupIndex = 0; groupIndex < table->count; groupIndex++) {
166 auto &group = table->groups[groupIndex];
167 if (type == group.type) {
168 for (uint32_t index = 0; index < group.count; index++) {
169 auto &entry = group.entries[index];
170 if (value == entry.ext_inst) {
171 *pEntry = &table->groups[groupIndex].entries[index];
172 return SPV_SUCCESS;
173 }
174 }
175 }
176 }
177
178 return SPV_ERROR_INVALID_LOOKUP;
179}