text_to_binary.memory_test.cpp 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412
  1. // Copyright (c) 2015-2016 The Khronos Group Inc.
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. // Assembler tests for instructions in the "Memory Instructions" section of
  15. // the SPIR-V spec.
  16. #include <sstream>
  17. #include <string>
  18. #include <vector>
  19. #include "gmock/gmock.h"
  20. #include "test/test_fixture.h"
  21. #include "test/unit_spirv.h"
  22. namespace spvtools {
  23. namespace {
  24. using spvtest::EnumCase;
  25. using spvtest::MakeInstruction;
  26. using spvtest::TextToBinaryTest;
  27. using ::testing::Eq;
  28. using ::testing::HasSubstr;
  29. // Test assembly of Memory Access masks
  30. using MemoryAccessTest = spvtest::TextToBinaryTestBase<
  31. ::testing::TestWithParam<EnumCase<spv::MemoryAccessMask>>>;
  32. TEST_P(MemoryAccessTest, AnySingleMemoryAccessMask) {
  33. std::stringstream input;
  34. input << "OpStore %ptr %value " << GetParam().name();
  35. for (auto operand : GetParam().operands()) input << " " << operand;
  36. EXPECT_THAT(
  37. CompiledInstructions(input.str()),
  38. Eq(MakeInstruction(spv::Op::OpStore, {1, 2, (uint32_t)GetParam().value()},
  39. GetParam().operands())));
  40. }
  41. INSTANTIATE_TEST_SUITE_P(
  42. TextToBinaryMemoryAccessTest, MemoryAccessTest,
  43. ::testing::ValuesIn(std::vector<EnumCase<spv::MemoryAccessMask>>{
  44. {spv::MemoryAccessMask::MaskNone, "None", {}},
  45. {spv::MemoryAccessMask::Volatile, "Volatile", {}},
  46. {spv::MemoryAccessMask::Aligned, "Aligned", {16}},
  47. {spv::MemoryAccessMask::Nontemporal, "Nontemporal", {}},
  48. }));
  49. TEST_F(TextToBinaryTest, CombinedMemoryAccessMask) {
  50. const std::string input = "OpStore %ptr %value Volatile|Aligned 16";
  51. const uint32_t expected_mask = uint32_t(spv::MemoryAccessMask::Volatile |
  52. spv::MemoryAccessMask::Aligned);
  53. EXPECT_THAT(expected_mask, Eq(3u));
  54. EXPECT_THAT(CompiledInstructions(input),
  55. Eq(MakeInstruction(spv::Op::OpStore, {1, 2, expected_mask, 16})));
  56. }
  57. // Test Storage Class enum values
  58. using StorageClassTest = spvtest::TextToBinaryTestBase<
  59. ::testing::TestWithParam<EnumCase<spv::StorageClass>>>;
  60. TEST_P(StorageClassTest, AnyStorageClass) {
  61. const std::string input = "%1 = OpVariable %2 " + GetParam().name();
  62. EXPECT_THAT(CompiledInstructions(input),
  63. Eq(MakeInstruction(spv::Op::OpVariable,
  64. {1, 2, (uint32_t)GetParam().value()})));
  65. }
  66. // clang-format off
  67. #define CASE(NAME) { spv::StorageClass::NAME, #NAME, {} }
  68. INSTANTIATE_TEST_SUITE_P(
  69. TextToBinaryStorageClassTest, StorageClassTest,
  70. ::testing::ValuesIn(std::vector<EnumCase<spv::StorageClass>>{
  71. CASE(UniformConstant),
  72. CASE(Input),
  73. CASE(Uniform),
  74. CASE(Output),
  75. CASE(Workgroup),
  76. CASE(CrossWorkgroup),
  77. CASE(Private),
  78. CASE(Function),
  79. CASE(Generic),
  80. CASE(PushConstant),
  81. CASE(AtomicCounter),
  82. CASE(Image),
  83. }));
  84. #undef CASE
  85. // clang-format on
  86. using MemoryRoundTripTest = RoundTripTest;
  87. // OpPtrEqual appeared in SPIR-V 1.4
  88. TEST_F(MemoryRoundTripTest, OpPtrEqualGood) {
  89. std::string spirv = "%2 = OpPtrEqual %1 %3 %4\n";
  90. EXPECT_THAT(CompiledInstructions(spirv, SPV_ENV_UNIVERSAL_1_4),
  91. Eq(MakeInstruction(spv::Op::OpPtrEqual, {1, 2, 3, 4})));
  92. std::string disassembly = EncodeAndDecodeSuccessfully(
  93. spirv, SPV_BINARY_TO_TEXT_OPTION_NONE, SPV_TEXT_TO_BINARY_OPTION_NONE,
  94. SPV_ENV_UNIVERSAL_1_4);
  95. EXPECT_THAT(disassembly, Eq(spirv));
  96. }
  97. TEST_F(MemoryRoundTripTest, OpPtrEqualV13Bad) {
  98. std::string spirv = "%2 = OpPtrEqual %1 %3 %4\n";
  99. std::string err = CompileFailure(spirv, SPV_ENV_UNIVERSAL_1_3);
  100. EXPECT_THAT(err, HasSubstr("Invalid Opcode name 'OpPtrEqual'"));
  101. }
  102. // OpPtrNotEqual appeared in SPIR-V 1.4
  103. TEST_F(MemoryRoundTripTest, OpPtrNotEqualGood) {
  104. std::string spirv = "%2 = OpPtrNotEqual %1 %3 %4\n";
  105. EXPECT_THAT(CompiledInstructions(spirv, SPV_ENV_UNIVERSAL_1_4),
  106. Eq(MakeInstruction(spv::Op::OpPtrNotEqual, {1, 2, 3, 4})));
  107. std::string disassembly = EncodeAndDecodeSuccessfully(
  108. spirv, SPV_BINARY_TO_TEXT_OPTION_NONE, SPV_TEXT_TO_BINARY_OPTION_NONE,
  109. SPV_ENV_UNIVERSAL_1_4);
  110. EXPECT_THAT(disassembly, Eq(spirv));
  111. }
  112. TEST_F(MemoryRoundTripTest, OpPtrNotEqualV13Bad) {
  113. std::string spirv = "%2 = OpPtrNotEqual %1 %3 %4\n";
  114. std::string err = CompileFailure(spirv, SPV_ENV_UNIVERSAL_1_3);
  115. EXPECT_THAT(err, HasSubstr("Invalid Opcode name 'OpPtrNotEqual'"));
  116. }
  117. // OpPtrDiff appeared in SPIR-V 1.4
  118. TEST_F(MemoryRoundTripTest, OpPtrDiffGood) {
  119. std::string spirv = "%2 = OpPtrDiff %1 %3 %4\n";
  120. EXPECT_THAT(CompiledInstructions(spirv, SPV_ENV_UNIVERSAL_1_4),
  121. Eq(MakeInstruction(spv::Op::OpPtrDiff, {1, 2, 3, 4})));
  122. std::string disassembly = EncodeAndDecodeSuccessfully(
  123. spirv, SPV_BINARY_TO_TEXT_OPTION_NONE, SPV_TEXT_TO_BINARY_OPTION_NONE,
  124. SPV_ENV_UNIVERSAL_1_4);
  125. EXPECT_THAT(disassembly, Eq(spirv));
  126. }
  127. TEST_F(MemoryRoundTripTest, OpPtrDiffV13Good) {
  128. // OpPtrDiff is enabled by a capability as well, so we can assemble
  129. // it even in older SPIR-V environments. We do that so we can
  130. // write tests.
  131. std::string spirv = "%2 = OpPtrDiff %1 %3 %4\n";
  132. std::string disassembly = EncodeAndDecodeSuccessfully(
  133. spirv, SPV_BINARY_TO_TEXT_OPTION_NONE, SPV_TEXT_TO_BINARY_OPTION_NONE,
  134. SPV_ENV_UNIVERSAL_1_4);
  135. }
  136. // OpCopyMemory
  137. TEST_F(MemoryRoundTripTest, OpCopyMemoryNoMemAccessGood) {
  138. std::string spirv = "OpCopyMemory %1 %2\n";
  139. EXPECT_THAT(CompiledInstructions(spirv),
  140. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2})));
  141. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  142. EXPECT_THAT(disassembly, Eq(spirv));
  143. }
  144. TEST_F(MemoryRoundTripTest, OpCopyMemoryTooFewArgsBad) {
  145. std::string spirv = "OpCopyMemory %1\n";
  146. std::string err = CompileFailure(spirv);
  147. EXPECT_THAT(err, HasSubstr("Expected operand for OpCopyMemory instruction, "
  148. "but found the end of the stream."));
  149. }
  150. TEST_F(MemoryRoundTripTest, OpCopyMemoryTooManyArgsBad) {
  151. std::string spirv = "OpCopyMemory %1 %2 %3\n";
  152. std::string err = CompileFailure(spirv);
  153. EXPECT_THAT(err, HasSubstr("Invalid memory access operand '%3'"));
  154. }
  155. TEST_F(MemoryRoundTripTest, OpCopyMemoryAccessNoneGood) {
  156. std::string spirv = "OpCopyMemory %1 %2 None\n";
  157. EXPECT_THAT(CompiledInstructions(spirv),
  158. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 0})));
  159. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  160. EXPECT_THAT(disassembly, Eq(spirv));
  161. }
  162. TEST_F(MemoryRoundTripTest, OpCopyMemoryAccessVolatileGood) {
  163. std::string spirv = "OpCopyMemory %1 %2 Volatile\n";
  164. EXPECT_THAT(CompiledInstructions(spirv),
  165. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 1})));
  166. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  167. EXPECT_THAT(disassembly, Eq(spirv));
  168. }
  169. TEST_F(MemoryRoundTripTest, OpCopyMemoryAccessAligned8Good) {
  170. std::string spirv = "OpCopyMemory %1 %2 Aligned 8\n";
  171. EXPECT_THAT(CompiledInstructions(spirv),
  172. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 2, 8})));
  173. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  174. EXPECT_THAT(disassembly, Eq(spirv));
  175. }
  176. TEST_F(MemoryRoundTripTest, OpCopyMemoryAccessNontemporalGood) {
  177. std::string spirv = "OpCopyMemory %1 %2 Nontemporal\n";
  178. EXPECT_THAT(CompiledInstructions(spirv),
  179. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 4})));
  180. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  181. EXPECT_THAT(disassembly, Eq(spirv));
  182. }
  183. TEST_F(MemoryRoundTripTest, OpCopyMemoryAccessAvGood) {
  184. std::string spirv = "OpCopyMemory %1 %2 MakePointerAvailable %3\n";
  185. EXPECT_THAT(CompiledInstructions(spirv),
  186. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 8, 3})));
  187. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  188. EXPECT_THAT(disassembly, Eq(spirv));
  189. }
  190. TEST_F(MemoryRoundTripTest, OpCopyMemoryAccessVisGood) {
  191. std::string spirv = "OpCopyMemory %1 %2 MakePointerVisible %3\n";
  192. EXPECT_THAT(CompiledInstructions(spirv),
  193. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 16, 3})));
  194. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  195. EXPECT_THAT(disassembly, Eq(spirv));
  196. }
  197. TEST_F(MemoryRoundTripTest, OpCopyMemoryAccessNonPrivateGood) {
  198. std::string spirv = "OpCopyMemory %1 %2 NonPrivatePointer\n";
  199. EXPECT_THAT(CompiledInstructions(spirv),
  200. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 32})));
  201. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  202. EXPECT_THAT(disassembly, Eq(spirv));
  203. }
  204. TEST_F(MemoryRoundTripTest, OpCopyMemoryAccessMixedGood) {
  205. std::string spirv =
  206. "OpCopyMemory %1 %2 "
  207. "Volatile|Aligned|Nontemporal|MakePointerAvailable|"
  208. "MakePointerVisible|NonPrivatePointer 16 %3 %4\n";
  209. EXPECT_THAT(CompiledInstructions(spirv),
  210. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 63, 16, 3, 4})));
  211. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  212. EXPECT_THAT(disassembly, Eq(spirv));
  213. }
  214. TEST_F(MemoryRoundTripTest, OpCopyMemoryTwoAccessV13Good) {
  215. std::string spirv = "OpCopyMemory %1 %2 Volatile Volatile\n";
  216. // Note: This will assemble but should not validate for SPIR-V 1.3
  217. EXPECT_THAT(CompiledInstructions(spirv, SPV_ENV_UNIVERSAL_1_3),
  218. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 1, 1})));
  219. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  220. EXPECT_THAT(disassembly, Eq(spirv));
  221. }
  222. TEST_F(MemoryRoundTripTest, OpCopyMemoryTwoAccessV14Good) {
  223. std::string spirv = "OpCopyMemory %1 %2 Volatile Volatile\n";
  224. EXPECT_THAT(CompiledInstructions(spirv, SPV_ENV_UNIVERSAL_1_4),
  225. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 1, 1})));
  226. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  227. EXPECT_THAT(disassembly, Eq(spirv));
  228. }
  229. TEST_F(MemoryRoundTripTest, OpCopyMemoryTwoAccessMixedV14Good) {
  230. std::string spirv =
  231. "OpCopyMemory %1 %2 Volatile|Nontemporal|"
  232. "MakePointerVisible %3 "
  233. "Aligned|MakePointerAvailable|NonPrivatePointer 16 %4\n";
  234. EXPECT_THAT(
  235. CompiledInstructions(spirv),
  236. Eq(MakeInstruction(spv::Op::OpCopyMemory, {1, 2, 21, 3, 42, 16, 4})));
  237. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  238. EXPECT_THAT(disassembly, Eq(spirv));
  239. }
  240. // OpCopyMemorySized
  241. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedNoMemAccessGood) {
  242. std::string spirv = "OpCopyMemorySized %1 %2 %3\n";
  243. EXPECT_THAT(CompiledInstructions(spirv),
  244. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3})));
  245. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  246. EXPECT_THAT(disassembly, Eq(spirv));
  247. }
  248. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedTooFewArgsBad) {
  249. std::string spirv = "OpCopyMemorySized %1 %2\n";
  250. std::string err = CompileFailure(spirv);
  251. EXPECT_THAT(err, HasSubstr("Expected operand for OpCopyMemorySized "
  252. "instruction, but found the end of the stream."));
  253. }
  254. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedTooManyArgsBad) {
  255. std::string spirv = "OpCopyMemorySized %1 %2 %3 %4\n";
  256. std::string err = CompileFailure(spirv);
  257. EXPECT_THAT(err, HasSubstr("Invalid memory access operand '%4'"));
  258. }
  259. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedAccessNoneGood) {
  260. std::string spirv = "OpCopyMemorySized %1 %2 %3 None\n";
  261. EXPECT_THAT(CompiledInstructions(spirv),
  262. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 0})));
  263. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  264. EXPECT_THAT(disassembly, Eq(spirv));
  265. }
  266. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedAccessVolatileGood) {
  267. std::string spirv = "OpCopyMemorySized %1 %2 %3 Volatile\n";
  268. EXPECT_THAT(CompiledInstructions(spirv),
  269. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 1})));
  270. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  271. EXPECT_THAT(disassembly, Eq(spirv));
  272. }
  273. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedAccessAligned8Good) {
  274. std::string spirv = "OpCopyMemorySized %1 %2 %3 Aligned 8\n";
  275. EXPECT_THAT(CompiledInstructions(spirv),
  276. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 2, 8})));
  277. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  278. EXPECT_THAT(disassembly, Eq(spirv));
  279. }
  280. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedAccessNontemporalGood) {
  281. std::string spirv = "OpCopyMemorySized %1 %2 %3 Nontemporal\n";
  282. EXPECT_THAT(CompiledInstructions(spirv),
  283. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 4})));
  284. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  285. EXPECT_THAT(disassembly, Eq(spirv));
  286. }
  287. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedAccessAvGood) {
  288. std::string spirv = "OpCopyMemorySized %1 %2 %3 MakePointerAvailable %4\n";
  289. EXPECT_THAT(CompiledInstructions(spirv),
  290. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 8, 4})));
  291. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  292. EXPECT_THAT(disassembly, Eq(spirv));
  293. }
  294. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedAccessVisGood) {
  295. std::string spirv = "OpCopyMemorySized %1 %2 %3 MakePointerVisible %4\n";
  296. EXPECT_THAT(
  297. CompiledInstructions(spirv),
  298. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 16, 4})));
  299. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  300. EXPECT_THAT(disassembly, Eq(spirv));
  301. }
  302. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedAccessNonPrivateGood) {
  303. std::string spirv = "OpCopyMemorySized %1 %2 %3 NonPrivatePointer\n";
  304. EXPECT_THAT(CompiledInstructions(spirv),
  305. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 32})));
  306. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  307. EXPECT_THAT(disassembly, Eq(spirv));
  308. }
  309. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedAccessMixedGood) {
  310. std::string spirv =
  311. "OpCopyMemorySized %1 %2 %3 "
  312. "Volatile|Aligned|Nontemporal|MakePointerAvailable|"
  313. "MakePointerVisible|NonPrivatePointer 16 %4 %5\n";
  314. EXPECT_THAT(
  315. CompiledInstructions(spirv),
  316. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 63, 16, 4, 5})));
  317. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  318. EXPECT_THAT(disassembly, Eq(spirv));
  319. }
  320. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedTwoAccessV13Good) {
  321. std::string spirv = "OpCopyMemorySized %1 %2 %3 Volatile Volatile\n";
  322. // Note: This will assemble but should not validate for SPIR-V 1.3
  323. EXPECT_THAT(CompiledInstructions(spirv, SPV_ENV_UNIVERSAL_1_3),
  324. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 1, 1})));
  325. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  326. EXPECT_THAT(disassembly, Eq(spirv));
  327. }
  328. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedTwoAccessV14Good) {
  329. std::string spirv = "OpCopyMemorySized %1 %2 %3 Volatile Volatile\n";
  330. EXPECT_THAT(CompiledInstructions(spirv, SPV_ENV_UNIVERSAL_1_4),
  331. Eq(MakeInstruction(spv::Op::OpCopyMemorySized, {1, 2, 3, 1, 1})));
  332. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  333. EXPECT_THAT(disassembly, Eq(spirv));
  334. }
  335. TEST_F(MemoryRoundTripTest, OpCopyMemorySizedTwoAccessMixedV14Good) {
  336. std::string spirv =
  337. "OpCopyMemorySized %1 %2 %3 Volatile|Nontemporal|"
  338. "MakePointerVisible %4 "
  339. "Aligned|MakePointerAvailable|NonPrivatePointer 16 %5\n";
  340. EXPECT_THAT(CompiledInstructions(spirv),
  341. Eq(MakeInstruction(spv::Op::OpCopyMemorySized,
  342. {1, 2, 3, 21, 4, 42, 16, 5})));
  343. std::string disassembly = EncodeAndDecodeSuccessfully(spirv);
  344. EXPECT_THAT(disassembly, Eq(spirv));
  345. }
  346. // TODO(dneto): OpVariable with initializers
  347. // TODO(dneto): OpImageTexelPointer
  348. // TODO(dneto): OpLoad
  349. // TODO(dneto): OpStore
  350. // TODO(dneto): OpAccessChain
  351. // TODO(dneto): OpInBoundsAccessChain
  352. // TODO(dneto): OpPtrAccessChain
  353. // TODO(dneto): OpArrayLength
  354. // TODO(dneto): OpGenercPtrMemSemantics
  355. } // namespace
  356. } // namespace spvtools