| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970 |
- // Copyright (c) 2017 Google Inc.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- #include <sstream>
- #include <string>
- #include "gmock/gmock.h"
- #include "test/unit_spirv.h"
- #include "test/val/val_fixtures.h"
- namespace spvtools {
- namespace val {
- namespace {
- using ::testing::HasSubstr;
- using ::testing::Not;
- using ValidateAtomics = spvtest::ValidateBase<bool>;
- std::string GenerateShaderCodeImpl(
- const std::string& body, const std::string& capabilities_and_extensions,
- const std::string& definitions, const std::string& memory_model,
- const std::string& execution) {
- std::ostringstream ss;
- ss << R"(
- OpCapability Shader
- )";
- ss << capabilities_and_extensions;
- ss << "OpMemoryModel Logical " << memory_model << "\n";
- ss << execution;
- ss << R"(
- %void = OpTypeVoid
- %func = OpTypeFunction %void
- %bool = OpTypeBool
- %f32 = OpTypeFloat 32
- %u32 = OpTypeInt 32 0
- %f32vec4 = OpTypeVector %f32 4
- %f32_0 = OpConstant %f32 0
- %f32_1 = OpConstant %f32 1
- %u32_0 = OpConstant %u32 0
- %u32_1 = OpConstant %u32 1
- %f32vec4_0000 = OpConstantComposite %f32vec4 %f32_0 %f32_0 %f32_0 %f32_0
- %cross_device = OpConstant %u32 0
- %device = OpConstant %u32 1
- %workgroup = OpConstant %u32 2
- %subgroup = OpConstant %u32 3
- %invocation = OpConstant %u32 4
- %queuefamily = OpConstant %u32 5
- %relaxed = OpConstant %u32 0
- %acquire = OpConstant %u32 2
- %release = OpConstant %u32 4
- %acquire_release = OpConstant %u32 8
- %acquire_and_release = OpConstant %u32 6
- %sequentially_consistent = OpConstant %u32 16
- %acquire_release_uniform_workgroup = OpConstant %u32 328
- %f32_ptr = OpTypePointer Workgroup %f32
- %f32_var = OpVariable %f32_ptr Workgroup
- %u32_ptr = OpTypePointer Workgroup %u32
- %u32_var = OpVariable %u32_ptr Workgroup
- %f32vec4_ptr = OpTypePointer Workgroup %f32vec4
- %f32vec4_var = OpVariable %f32vec4_ptr Workgroup
- %f32_ptr_function = OpTypePointer Function %f32
- )";
- ss << definitions;
- ss << R"(
- %main = OpFunction %void None %func
- %main_entry = OpLabel
- )";
- ss << body;
- ss << R"(
- OpReturn
- OpFunctionEnd)";
- return ss.str();
- }
- std::string GenerateShaderCode(
- const std::string& body,
- const std::string& capabilities_and_extensions = "",
- const std::string& extra_defs = "",
- const std::string& memory_model = "GLSL450") {
- const std::string execution = R"(
- OpEntryPoint Fragment %main "main"
- OpExecutionMode %main OriginUpperLeft
- )";
- const std::string definitions = R"(
- %u64 = OpTypeInt 64 0
- %s64 = OpTypeInt 64 1
- %u64_1 = OpConstant %u64 1
- %s64_1 = OpConstant %s64 1
- %u64_ptr = OpTypePointer Workgroup %u64
- %s64_ptr = OpTypePointer Workgroup %s64
- %u64_var = OpVariable %u64_ptr Workgroup
- %s64_var = OpVariable %s64_ptr Workgroup
- )";
- return GenerateShaderCodeImpl(
- body, "OpCapability Int64\n" + capabilities_and_extensions,
- definitions + extra_defs, memory_model, execution);
- }
- std::string GenerateShaderComputeCode(
- const std::string& body,
- const std::string& capabilities_and_extensions = "",
- const std::string& extra_defs = "",
- const std::string& memory_model = "GLSL450") {
- const std::string execution = R"(
- OpEntryPoint GLCompute %main "main"
- OpExecutionMode %main LocalSize 32 1 1
- )";
- const std::string definitions = R"(
- %u64 = OpTypeInt 64 0
- %s64 = OpTypeInt 64 1
- %u64_1 = OpConstant %u64 1
- %s64_1 = OpConstant %s64 1
- %u64_ptr = OpTypePointer Workgroup %u64
- %s64_ptr = OpTypePointer Workgroup %s64
- %u64_var = OpVariable %u64_ptr Workgroup
- %s64_var = OpVariable %s64_ptr Workgroup
- )";
- return GenerateShaderCodeImpl(
- body, "OpCapability Int64\n" + capabilities_and_extensions,
- definitions + extra_defs, memory_model, execution);
- }
- std::string GenerateKernelCode(
- const std::string& body,
- const std::string& capabilities_and_extensions = "") {
- std::ostringstream ss;
- ss << R"(
- OpCapability Addresses
- OpCapability Kernel
- OpCapability Linkage
- OpCapability Int64
- )";
- ss << capabilities_and_extensions;
- ss << R"(
- OpMemoryModel Physical32 OpenCL
- %void = OpTypeVoid
- %func = OpTypeFunction %void
- %bool = OpTypeBool
- %f32 = OpTypeFloat 32
- %u32 = OpTypeInt 32 0
- %u64 = OpTypeInt 64 0
- %f32vec4 = OpTypeVector %f32 4
- %f32_0 = OpConstant %f32 0
- %f32_1 = OpConstant %f32 1
- %u32_0 = OpConstant %u32 0
- %u32_1 = OpConstant %u32 1
- %u64_1 = OpConstant %u64 1
- %f32vec4_0000 = OpConstantComposite %f32vec4 %f32_0 %f32_0 %f32_0 %f32_0
- %cross_device = OpConstant %u32 0
- %device = OpConstant %u32 1
- %workgroup = OpConstant %u32 2
- %subgroup = OpConstant %u32 3
- %invocation = OpConstant %u32 4
- %relaxed = OpConstant %u32 0
- %acquire = OpConstant %u32 2
- %release = OpConstant %u32 4
- %acquire_release = OpConstant %u32 8
- %acquire_and_release = OpConstant %u32 6
- %sequentially_consistent = OpConstant %u32 16
- %acquire_release_uniform_workgroup = OpConstant %u32 328
- %acquire_release_atomic_counter_workgroup = OpConstant %u32 1288
- %f32_ptr = OpTypePointer Workgroup %f32
- %f32_var = OpVariable %f32_ptr Workgroup
- %u32_ptr = OpTypePointer Workgroup %u32
- %u32_var = OpVariable %u32_ptr Workgroup
- %u64_ptr = OpTypePointer Workgroup %u64
- %u64_var = OpVariable %u64_ptr Workgroup
- %f32vec4_ptr = OpTypePointer Workgroup %f32vec4
- %f32vec4_var = OpVariable %f32vec4_ptr Workgroup
- %f32_ptr_function = OpTypePointer Function %f32
- %f32_ptr_uniformconstant = OpTypePointer UniformConstant %f32
- %f32_uc_var = OpVariable %f32_ptr_uniformconstant UniformConstant
- %f32_ptr_image = OpTypePointer Image %f32
- %f32_im_var = OpVariable %f32_ptr_image Image
- %main = OpFunction %void None %func
- %main_entry = OpLabel
- )";
- ss << body;
- ss << R"(
- OpReturn
- OpFunctionEnd)";
- return ss.str();
- }
- TEST_F(ValidateAtomics, AtomicLoadShaderSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %u32_var %device %relaxed
- %val2 = OpAtomicLoad %u32 %u32_var %workgroup %acquire
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicLoadKernelSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %f32 %f32_var %device %relaxed
- %val2 = OpAtomicLoad %u32 %u32_var %workgroup %sequentially_consistent
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicLoadInt64ShaderSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u64 %u64_var %subgroup %sequentially_consistent
- )";
- CompileSuccessfully(GenerateShaderCode(body, "OpCapability Int64Atomics\n"));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicLoadInt64KernelSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u64 %u64_var %subgroup %acquire
- )";
- CompileSuccessfully(GenerateKernelCode(body, "OpCapability Int64Atomics\n"));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicLoadInt32VulkanSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %u32_var %device %relaxed
- %val2 = OpAtomicLoad %u32 %u32_var %workgroup %acquire
- %val3 = OpAtomicLoad %u32 %u32_var %invocation %relaxed
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicLoadVulkanWrongStorageClass) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %u32_var %device %relaxed
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-None-04645"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("in Vulkan environment, Workgroup Storage Class is limited to "
- "MeshNV, TaskNV, and GLCompute execution model"));
- }
- TEST_F(ValidateAtomics, AtomicAddIntVulkanWrongType1) {
- const std::string body = R"(
- %val1 = OpAtomicIAdd %f32 %f32_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicIAdd: "
- "expected Result Type to be integer scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicAddIntVulkanWrongType2) {
- const std::string body = R"(
- %val1 = OpAtomicIAdd %f32vec4 %f32vec4_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicIAdd: "
- "expected Result Type to be integer scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicAddFloatVulkan) {
- const std::string body = R"(
- %val1 = OpAtomicFAddEXT %f32 %f32_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_CAPABILITY, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Opcode AtomicFAddEXT requires one of these capabilities: "
- "AtomicFloat16VectorNV AtomicFloat32AddEXT AtomicFloat64AddEXT "
- "AtomicFloat16AddEXT"));
- }
- TEST_F(ValidateAtomics, AtomicMinFloatVulkan) {
- const std::string body = R"(
- %val1 = OpAtomicFMinEXT %f32 %f32_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_CAPABILITY, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Opcode AtomicFMinEXT requires one of these capabilities: "
- "AtomicFloat16VectorNV AtomicFloat32MinMaxEXT "
- "AtomicFloat64MinMaxEXT AtomicFloat16MinMaxEXT"));
- }
- TEST_F(ValidateAtomics, AtomicMaxFloatVulkan) {
- const std::string body = R"(
- %val1 = OpAtomicFMaxEXT %f32 %f32_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_CAPABILITY, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "Opcode AtomicFMaxEXT requires one of these capabilities: "
- "AtomicFloat16VectorNV AtomicFloat32MinMaxEXT AtomicFloat64MinMaxEXT "
- "AtomicFloat16MinMaxEXT"));
- }
- TEST_F(ValidateAtomics, AtomicAddFloatVulkanWrongType1) {
- const std::string body = R"(
- %val1 = OpAtomicFAddEXT %f32vec4 %f32vec4_var %device %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32AddEXT
- OpExtension "SPV_EXT_shader_atomic_float_add"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFAddEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicMinFloatVulkanWrongType1) {
- const std::string body = R"(
- %val1 = OpAtomicFMinEXT %f32vec4 %f32vec4_var %device %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFMinEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicMaxFloatVulkanWrongType1) {
- const std::string body = R"(
- %val1 = OpAtomicFMaxEXT %f32vec4 %f32vec4_var %device %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFMaxEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicAddFloatVulkanWrongType2) {
- const std::string body = R"(
- %val1 = OpAtomicFAddEXT %u32 %u32_var %device %relaxed %u32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32AddEXT
- OpExtension "SPV_EXT_shader_atomic_float_add"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFAddEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicMinFloatVulkanWrongType2) {
- const std::string body = R"(
- %val1 = OpAtomicFMinEXT %u32 %u32_var %device %relaxed %u32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFMinEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicMaxFloatVulkanWrongType2) {
- const std::string body = R"(
- %val1 = OpAtomicFMaxEXT %u32 %u32_var %device %relaxed %u32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFMaxEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicAddFloatVulkanWrongType3) {
- const std::string body = R"(
- %val1 = OpAtomicFAddEXT %u64 %u64_var %device %relaxed %u64_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32AddEXT
- OpExtension "SPV_EXT_shader_atomic_float_add"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFAddEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicMinFloatVulkanWrongType3) {
- const std::string body = R"(
- %val1 = OpAtomicFMinEXT %u64 %u64_var %device %relaxed %u64_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFMinEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicMaxFloatVulkanWrongType3) {
- const std::string body = R"(
- %val1 = OpAtomicFMaxEXT %u64 %u64_var %device %relaxed %u64_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFMaxEXT: "
- "expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicAddFloatVulkanWrongCapability) {
- const std::string body = R"(
- %val1 = OpAtomicFAddEXT %f32 %f32_var %device %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat64AddEXT
- OpExtension "SPV_EXT_shader_atomic_float_add"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFAddEXT: float add atomics "
- "require the AtomicFloat32AddEXT capability"));
- }
- TEST_F(ValidateAtomics, AtomicMinFloatVulkanWrongCapability) {
- const std::string body = R"(
- %val1 = OpAtomicFMinEXT %f32 %f32_var %device %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat64MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFMinEXT: float min/max atomics "
- "require the AtomicFloat32MinMaxEXT capability"));
- }
- TEST_F(ValidateAtomics, AtomicMaxFloatVulkanWrongCapability) {
- const std::string body = R"(
- %val1 = OpAtomicFMaxEXT %f32 %f32_var %device %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat64MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFMaxEXT: float min/max atomics "
- "require the AtomicFloat32MinMaxEXT capability"));
- }
- TEST_F(ValidateAtomics, AtomicAddFloat16VulkanSuccess) {
- const std::string defs = R"(
- %f16 = OpTypeFloat 16
- %f16_1 = OpConstant %f16 1
- %f16_ptr = OpTypePointer Workgroup %f16
- %f16_var = OpVariable %f16_ptr Workgroup
- )";
- const std::string body = R"(
- %val1 = OpAtomicFAddEXT %f16 %f16_var %device %relaxed %f16_1
- )";
- const std::string extra = R"(
- OpCapability Float16
- OpCapability AtomicFloat16AddEXT
- OpExtension "SPV_EXT_shader_atomic_float16_add"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra, defs),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicAddFloatVulkanSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicFAddEXT %f32 %f32_var %device %relaxed %f32_1
- %val2 = OpAtomicFAddEXT %f32 %f32_var %invocation %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32AddEXT
- OpExtension "SPV_EXT_shader_atomic_float_add"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicMinFloat16VulkanSuccess) {
- const std::string defs = R"(
- %f16 = OpTypeFloat 16
- %f16_1 = OpConstant %f16 1
- %f16_ptr = OpTypePointer Workgroup %f16
- %f16_var = OpVariable %f16_ptr Workgroup
- )";
- const std::string body = R"(
- %val1 = OpAtomicFMinEXT %f16 %f16_var %device %relaxed %f16_1
- )";
- const std::string extra = R"(
- OpCapability Float16
- OpCapability AtomicFloat16MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra, defs),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicMaxFloat16VulkanSuccess) {
- const std::string defs = R"(
- %f16 = OpTypeFloat 16
- %f16_1 = OpConstant %f16 1
- %f16_ptr = OpTypePointer Workgroup %f16
- %f16_var = OpVariable %f16_ptr Workgroup
- )";
- const std::string body = R"(
- %val1 = OpAtomicFMaxEXT %f16 %f16_var %device %relaxed %f16_1
- )";
- const std::string extra = R"(
- OpCapability Float16
- OpCapability AtomicFloat16MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra, defs),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicMinFloat32VulkanSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicFMinEXT %f32 %f32_var %device %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicMaxFloat32VulkanSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicFMaxEXT %f32 %f32_var %device %relaxed %f32_1
- )";
- const std::string extra = R"(
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicMinFloat64VulkanSuccess) {
- const std::string defs = R"(
- %f64 = OpTypeFloat 64
- %f64_1 = OpConstant %f64 1
- %f64_ptr = OpTypePointer Workgroup %f64
- %f64_var = OpVariable %f64_ptr Workgroup
- )";
- const std::string body = R"(
- %val1 = OpAtomicFMinEXT %f64 %f64_var %device %relaxed %f64_1
- )";
- const std::string extra = R"(
- OpCapability Float64
- OpCapability AtomicFloat64MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra, defs),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicMaxFloat64VulkanSuccess) {
- const std::string defs = R"(
- %f64 = OpTypeFloat 64
- %f64_1 = OpConstant %f64 1
- %f64_ptr = OpTypePointer Workgroup %f64
- %f64_var = OpVariable %f64_ptr Workgroup
- )";
- const std::string body = R"(
- %val1 = OpAtomicFMaxEXT %f64 %f64_var %device %relaxed %f64_1
- )";
- const std::string extra = R"(
- OpCapability Float64
- OpCapability AtomicFloat64MinMaxEXT
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra, defs),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicLoadFloatVulkan) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %f32 %f32_var %device %relaxed
- %val2 = OpAtomicLoad %f32 %f32_var %workgroup %acquire
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicStoreVulkanWrongStorageClass) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-None-04645"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("in Vulkan environment, Workgroup Storage Class is limited to "
- "MeshNV, TaskNV, and GLCompute execution model"));
- }
- TEST_F(ValidateAtomics, AtomicStoreFloatVulkan) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicExchangeFloatVulkan) {
- const std::string body = R"(
- %val2 = OpAtomicExchange %f32 %f32_var %device %relaxed %f32_0
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicLoadInt64WithCapabilityVulkanSuccess) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u64 %u64_var %device %relaxed
- %val2 = OpAtomicLoad %u64 %u64_var %workgroup %acquire
- %val3 = OpAtomicLoad %u64 %u64_var %invocation %relaxed
- )";
- CompileSuccessfully(
- GenerateShaderComputeCode(body, "OpCapability Int64Atomics\n"),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicLoadInt64WithoutCapabilityVulkan) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u64 %u64_var %device %relaxed
- %val2 = OpAtomicLoad %u64 %u64_var %workgroup %acquire
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("64-bit atomics require the Int64Atomics capability"));
- }
- TEST_F(ValidateAtomics, AtomicStoreOpenCLFunctionPointerStorageTypeSuccess) {
- const std::string body = R"(
- %f32_var_function = OpVariable %f32_ptr_function Function
- OpAtomicStore %f32_var_function %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_OPENCL_1_2);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_OPENCL_1_2));
- }
- TEST_F(ValidateAtomics, AtomicStoreVulkanFunctionPointerStorageType) {
- const std::string body = R"(
- %f32_var_function = OpVariable %f32_ptr_function Function
- OpAtomicStore %f32_var_function %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-None-04686"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicStore: Vulkan spec only allows storage classes for "
- "atomic to be: Uniform, Workgroup, Image, StorageBuffer, "
- "PhysicalStorageBuffer or TaskPayloadWorkgroupEXT."));
- }
- TEST_F(ValidateAtomics, AtomicStoreFunctionPointerStorageType) {
- const std::string body = R"(
- %f32_var_function = OpVariable %f32_ptr_function Function
- OpAtomicStore %f32_var_function %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicStore: Function storage class forbidden when "
- "the Shader capability is declared."));
- }
- // TODO([email protected]): the corresponding check fails Vulkan CTS,
- // reenable once fixed.
- TEST_F(ValidateAtomics, DISABLED_AtomicLoadVulkanSubgroup) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %u32_var %subgroup %acquire
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicLoad: in Vulkan environment memory scope is "
- "limited to Device, Workgroup and Invocation"));
- }
- TEST_F(ValidateAtomics, AtomicLoadVulkanRelease) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %u32_var %workgroup %release
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-OpAtomicLoad-04731"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Vulkan spec disallows OpAtomicLoad with Memory Semantics "
- "Release, AcquireRelease and SequentiallyConsistent"));
- }
- TEST_F(ValidateAtomics, AtomicLoadVulkanAcquireRelease) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %u32_var %workgroup %acquire_release
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-OpAtomicLoad-04731"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Vulkan spec disallows OpAtomicLoad with Memory Semantics "
- "Release, AcquireRelease and SequentiallyConsistent"));
- }
- TEST_F(ValidateAtomics, AtomicLoadVulkanSequentiallyConsistent) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %u32_var %workgroup %sequentially_consistent
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-OpAtomicLoad-04731"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Vulkan spec disallows OpAtomicLoad with Memory Semantics "
- "Release, AcquireRelease and SequentiallyConsistent"));
- }
- TEST_F(ValidateAtomics, AtomicLoadVulkanInvocationSemantics) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %u32_var %invocation %acquire
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-None-04641"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicLoad: Vulkan specification requires Memory Semantics to "
- "be None if used with Invocation Memory Scope"));
- }
- TEST_F(ValidateAtomics, AtomicLoadShaderFloat) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %f32 %f32_var %device %relaxed
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicLoadVulkanInt64) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u64 %u64_var %device %relaxed
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicLoad: 64-bit atomics require the Int64Atomics capability"));
- }
- TEST_F(ValidateAtomics, AtomicLoadKernelInt64) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u64 %u64_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicLoad: 64-bit atomics require the Int64Atomics capability"));
- }
- TEST_F(ValidateAtomics, AtomicStoreVulkanInt64) {
- const std::string body = R"(
- OpAtomicStore %u64_var %device %relaxed %u64_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicStore: 64-bit atomics require the Int64Atomics capability"));
- }
- TEST_F(ValidateAtomics, AtomicStoreKernelInt64) {
- const std::string body = R"(
- OpAtomicStore %u64_var %device %relaxed %u64_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicStore: 64-bit atomics require the Int64Atomics capability"));
- }
- TEST_F(ValidateAtomics, VK_KHR_shader_atomic_int64Success) {
- const std::string body = R"(
- %val1 = OpAtomicUMin %u64 %u64_var %device %relaxed %u64_1
- %val2 = OpAtomicUMax %u64 %u64_var %device %relaxed %u64_1
- %val3 = OpAtomicSMin %u64 %u64_var %device %relaxed %u64_1
- %val4 = OpAtomicSMax %u64 %u64_var %device %relaxed %u64_1
- %val5 = OpAtomicAnd %u64 %u64_var %device %relaxed %u64_1
- %val6 = OpAtomicOr %u64 %u64_var %device %relaxed %u64_1
- %val7 = OpAtomicXor %u64 %u64_var %device %relaxed %u64_1
- %val8 = OpAtomicIAdd %u64 %u64_var %device %relaxed %u64_1
- %val9 = OpAtomicExchange %u64 %u64_var %device %relaxed %u64_1
- %val10 = OpAtomicCompareExchange %u64 %u64_var %device %relaxed %relaxed %u64_1 %u64_1
- %val11 = OpAtomicUMin %s64 %s64_var %device %relaxed %s64_1
- %val12 = OpAtomicUMax %s64 %s64_var %device %relaxed %s64_1
- %val13 = OpAtomicSMin %s64 %s64_var %device %relaxed %s64_1
- %val14 = OpAtomicSMax %s64 %s64_var %device %relaxed %s64_1
- %val15 = OpAtomicAnd %s64 %s64_var %device %relaxed %s64_1
- %val16 = OpAtomicOr %s64 %s64_var %device %relaxed %s64_1
- %val17 = OpAtomicXor %s64 %s64_var %device %relaxed %s64_1
- %val18 = OpAtomicIAdd %s64 %s64_var %device %relaxed %s64_1
- %val19 = OpAtomicExchange %s64 %s64_var %device %relaxed %s64_1
- %val20 = OpAtomicCompareExchange %s64 %s64_var %device %relaxed %relaxed %s64_1 %s64_1
- %val21 = OpAtomicLoad %u64 %u64_var %device %relaxed
- %val22 = OpAtomicLoad %s64 %s64_var %device %relaxed
- OpAtomicStore %u64_var %device %relaxed %u64_1
- OpAtomicStore %s64_var %device %relaxed %s64_1
- )";
- CompileSuccessfully(
- GenerateShaderComputeCode(body, "OpCapability Int64Atomics\n"),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, VK_KHR_shader_atomic_int64MissingCapability) {
- const std::string body = R"(
- %val1 = OpAtomicUMin %u64 %u64_var %device %relaxed %u64_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicUMin: 64-bit atomics require the Int64Atomics capability"));
- }
- TEST_F(ValidateAtomics, AtomicLoadWrongResultType) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %f32vec4 %f32vec4_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicLoad: "
- "expected Result Type to be integer or float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicLoadWrongPointerType) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %f32 %f32_ptr %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Operand '27[%_ptr_Workgroup_float]' cannot be a type"));
- }
- TEST_F(ValidateAtomics, AtomicLoadWrongPointerDataType) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %u32 %f32_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicLoad: "
- "expected Pointer to point to a value of type Result Type"));
- }
- TEST_F(ValidateAtomics, AtomicLoadWrongScopeType) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %f32 %f32_var %f32_1 %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicLoad: expected scope to be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicLoadWrongMemorySemanticsType) {
- const std::string body = R"(
- %val1 = OpAtomicLoad %f32 %f32_var %device %u64_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicLoad: expected Memory Semantics to be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicStoreKernelSuccess) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- OpAtomicStore %u32_var %subgroup %release %u32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicStoreShaderSuccess) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %release %u32_1
- OpAtomicStore %u32_var %subgroup %sequentially_consistent %u32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicStoreVulkanSuccess) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %release %u32_1
- OpAtomicStore %u32_var %invocation %relaxed %u32_1
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- TEST_F(ValidateAtomics, AtomicStoreVulkanAcquire) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %acquire %u32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-OpAtomicStore-04730"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Vulkan spec disallows OpAtomicStore with Memory Semantics "
- "Acquire, AcquireRelease and SequentiallyConsistent"));
- }
- TEST_F(ValidateAtomics, AtomicStoreVulkanAcquireRelease) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %acquire_release %u32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-OpAtomicStore-04730"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Vulkan spec disallows OpAtomicStore with Memory Semantics "
- "Acquire, AcquireRelease and SequentiallyConsistent"));
- }
- TEST_F(ValidateAtomics, AtomicStoreVulkanSequentiallyConsistent) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %sequentially_consistent %u32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-OpAtomicStore-04730"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Vulkan spec disallows OpAtomicStore with Memory Semantics "
- "Acquire, AcquireRelease and SequentiallyConsistent"));
- }
- TEST_F(ValidateAtomics, AtomicStoreVulkanInvocationSemantics) {
- const std::string body = R"(
- OpAtomicStore %u32_var %invocation %acquire %u32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-None-04641"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicStore: Vulkan specification requires Memory Semantics "
- "to be None if used with Invocation Memory Scope"));
- }
- TEST_F(ValidateAtomics, AtomicStoreWrongPointerType) {
- const std::string body = R"(
- OpAtomicStore %f32_1 %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicStore: expected Pointer to be a pointer type"));
- }
- TEST_F(ValidateAtomics, AtomicStoreWrongPointerDataType) {
- const std::string body = R"(
- OpAtomicStore %f32vec4_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicStore: "
- "expected Pointer to be a pointer to integer or float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicStoreWrongPointerStorageTypeForOpenCL) {
- const std::string body = R"(
- OpAtomicStore %f32_im_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_OPENCL_1_2));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicStore: storage class must be Function, Workgroup, "
- "CrossWorkGroup or Generic in the OpenCL environment."));
- }
- TEST_F(ValidateAtomics, AtomicStoreWrongPointerStorageType) {
- const std::string body = R"(
- OpAtomicStore %f32_uc_var %device %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicStore: storage class forbidden by universal "
- "validation rules."));
- }
- TEST_F(ValidateAtomics, AtomicStoreWrongScopeType) {
- const std::string body = R"(
- OpAtomicStore %f32_var %f32_1 %relaxed %f32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicStore: expected scope to be a 32-bit int\n "
- "OpAtomicStore %28 %float_1 %uint_0_1 %float_1\n"));
- }
- TEST_F(ValidateAtomics, AtomicStoreWrongMemorySemanticsType) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %f32_1 %f32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicStore: expected Memory Semantics to be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicStoreWrongValueType) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %u32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicStore: "
- "expected Value type and the type pointed to by Pointer to "
- "be the same"));
- }
- TEST_F(ValidateAtomics, AtomicExchangeShaderSuccess) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicExchange %u32 %u32_var %device %relaxed %u32_0
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicExchangeKernelSuccess) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- %val2 = OpAtomicExchange %f32 %f32_var %device %relaxed %f32_0
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val4 = OpAtomicExchange %u32 %u32_var %device %relaxed %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicExchangeShaderFloat) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- %val2 = OpAtomicExchange %f32 %f32_var %device %relaxed %f32_0
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicExchangeWrongResultType) {
- const std::string body = R"(
- OpStore %f32vec4_var %f32vec4_0000
- %val2 = OpAtomicExchange %f32vec4 %f32vec4_var %device %relaxed %f32vec4_0000
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicExchange: "
- "expected Result Type to be integer or float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicExchangeWrongPointerType) {
- const std::string body = R"(
- %val2 = OpAtomicExchange %f32 %f32vec4_ptr %device %relaxed %f32vec4_0000
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("Operand '33[%_ptr_Workgroup_v4float]' cannot be a "
- "type"));
- }
- TEST_F(ValidateAtomics, AtomicExchangeWrongPointerDataType) {
- const std::string body = R"(
- OpStore %f32vec4_var %f32vec4_0000
- %val2 = OpAtomicExchange %f32 %f32vec4_var %device %relaxed %f32vec4_0000
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicExchange: "
- "expected Pointer to point to a value of type Result Type"));
- }
- TEST_F(ValidateAtomics, AtomicExchangeWrongScopeType) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- %val2 = OpAtomicExchange %f32 %f32_var %f32_1 %relaxed %f32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicExchange: expected scope to be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicExchangeWrongMemorySemanticsType) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- %val2 = OpAtomicExchange %f32 %f32_var %device %f32_1 %f32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicExchange: expected Memory Semantics to be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicExchangeWrongValueType) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- %val2 = OpAtomicExchange %f32 %f32_var %device %relaxed %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicExchange: "
- "expected Value to be of type Result Type"));
- }
- TEST_F(ValidateAtomics, AtomicExchangeVulkanInvocationSemantics) {
- const std::string body = R"(
- OpAtomicStore %u32_var %invocation %relaxed %u32_1
- %val2 = OpAtomicExchange %u32 %u32_var %invocation %acquire %u32_0
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-None-04641"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicExchange: Vulkan specification requires Memory "
- "Semantics to be None if used with Invocation Memory Scope"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeShaderSuccess) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %device %relaxed %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeKernelSuccess) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %device %relaxed %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeShaderFloat) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- %val1 = OpAtomicCompareExchange %f32 %f32_var %device %relaxed %relaxed %f32_0 %f32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: "
- "expected Result Type to be integer scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWrongResultType) {
- const std::string body = R"(
- OpStore %f32vec4_var %f32vec4_0000
- %val2 = OpAtomicCompareExchange %f32vec4 %f32vec4_var %device %relaxed %relaxed %f32vec4_0000 %f32vec4_0000
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: "
- "expected Result Type to be integer scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWrongPointerType) {
- const std::string body = R"(
- %val2 = OpAtomicCompareExchange %f32 %f32vec4_ptr %device %relaxed %relaxed %f32vec4_0000 %f32vec4_0000
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("Operand '33[%_ptr_Workgroup_v4float]' cannot be a "
- "type"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWrongPointerDataType) {
- const std::string body = R"(
- OpStore %f32vec4_var %f32vec4_0000
- %val2 = OpAtomicCompareExchange %u32 %f32vec4_var %device %relaxed %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: "
- "expected Pointer to point to a value of type Result Type"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWrongScopeType) {
- const std::string body = R"(
- OpAtomicStore %u64_var %device %relaxed %u64_1
- %val2 = OpAtomicCompareExchange %u64 %u64_var %u64_1 %relaxed %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body, "OpCapability Int64Atomics\n"));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: expected scope to be a 32-bit "
- "int"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWrongMemorySemanticsType1) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %device %f32_1 %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: expected Memory Semantics to "
- "be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWrongMemorySemanticsType2) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %device %relaxed %f32_1 %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: expected Memory Semantics to "
- "be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeUnequalRelease) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %device %relaxed %release %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: Memory Semantics Release and "
- "AcquireRelease cannot be used for operand Unequal"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWrongValueType) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %device %relaxed %relaxed %f32_1 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: "
- "expected Value to be of type Result Type"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWrongComparatorType) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %device %relaxed %relaxed %u32_0 %f32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: "
- "expected Comparator to be of type Result Type"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWeakSuccess) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val4 = OpAtomicCompareExchangeWeak %u32 %u32_var %device %relaxed %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeWeakWrongResultType) {
- const std::string body = R"(
- OpAtomicStore %f32_var %device %relaxed %f32_1
- %val2 = OpAtomicCompareExchangeWeak %f32 %f32_var %device %relaxed %relaxed %f32_0 %f32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicCompareExchangeWeak: "
- "expected Result Type to be integer scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeVulkanInvocationSemanticsEqual) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %invocation %release %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-None-04641"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: Vulkan specification requires Memory "
- "Semantics to be None if used with Invocation Memory Scope"));
- }
- TEST_F(ValidateAtomics, AtomicCompareExchangeVulkanInvocationSemanticsUnequal) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val2 = OpAtomicCompareExchange %u32 %u32_var %invocation %relaxed %acquire %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- EXPECT_THAT(getDiagnosticString(),
- AnyVUID("VUID-StandaloneSpirv-None-04641"));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicCompareExchange: Vulkan specification requires Memory "
- "Semantics to be None if used with Invocation Memory Scope"));
- }
- TEST_F(ValidateAtomics, AtomicArithmeticsSuccess) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val1 = OpAtomicIIncrement %u32 %u32_var %device %acquire_release
- %val2 = OpAtomicIDecrement %u32 %u32_var %device %acquire_release
- %val3 = OpAtomicIAdd %u32 %u32_var %device %acquire_release %u32_1
- %val4 = OpAtomicISub %u32 %u32_var %device %acquire_release %u32_1
- %val5 = OpAtomicUMin %u32 %u32_var %device %acquire_release %u32_1
- %val6 = OpAtomicUMax %u32 %u32_var %device %acquire_release %u32_1
- %val7 = OpAtomicSMin %u32 %u32_var %device %sequentially_consistent %u32_1
- %val8 = OpAtomicSMax %u32 %u32_var %device %sequentially_consistent %u32_1
- %val9 = OpAtomicAnd %u32 %u32_var %device %sequentially_consistent %u32_1
- %val10 = OpAtomicOr %u32 %u32_var %device %sequentially_consistent %u32_1
- %val11 = OpAtomicXor %u32 %u32_var %device %sequentially_consistent %u32_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicFlagsSuccess) {
- const std::string body = R"(
- OpAtomicFlagClear %u32_var %device %release
- %val1 = OpAtomicFlagTestAndSet %bool %u32_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicFlagTestAndSetWrongResultType) {
- const std::string body = R"(
- %val1 = OpAtomicFlagTestAndSet %u32 %u32_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFlagTestAndSet: "
- "expected Result Type to be bool scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicFlagTestAndSetNotPointer) {
- const std::string body = R"(
- %val1 = OpAtomicFlagTestAndSet %bool %u32_1 %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFlagTestAndSet: "
- "expected Pointer to be a pointer type"));
- }
- TEST_F(ValidateAtomics, AtomicFlagTestAndSetNotIntPointer) {
- const std::string body = R"(
- %val1 = OpAtomicFlagTestAndSet %bool %f32_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicFlagTestAndSet: "
- "expected Pointer to point to a value of 32-bit integer type"));
- }
- TEST_F(ValidateAtomics, AtomicFlagTestAndSetNotInt32Pointer) {
- const std::string body = R"(
- %val1 = OpAtomicFlagTestAndSet %bool %u64_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body, "OpCapability Int64Atomics\n"));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicFlagTestAndSet: "
- "expected Pointer to point to a value of 32-bit integer type"));
- }
- TEST_F(ValidateAtomics, AtomicFlagTestAndSetWrongScopeType) {
- const std::string body = R"(
- %val1 = OpAtomicFlagTestAndSet %bool %u32_var %u64_1 %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicFlagTestAndSet: expected scope to be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicFlagTestAndSetWrongMemorySemanticsType) {
- const std::string body = R"(
- %val1 = OpAtomicFlagTestAndSet %bool %u32_var %device %u64_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFlagTestAndSet: "
- "expected Memory Semantics to be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicFlagClearAcquire) {
- const std::string body = R"(
- OpAtomicFlagClear %u32_var %device %acquire
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("Memory Semantics Acquire and AcquireRelease cannot be "
- "used with AtomicFlagClear"));
- }
- TEST_F(ValidateAtomics, AtomicFlagClearNotPointer) {
- const std::string body = R"(
- OpAtomicFlagClear %u32_1 %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFlagClear: "
- "expected Pointer to be a pointer type"));
- }
- TEST_F(ValidateAtomics, AtomicFlagClearNotIntPointer) {
- const std::string body = R"(
- OpAtomicFlagClear %f32_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicFlagClear: "
- "expected Pointer to point to a value of 32-bit integer type"));
- }
- TEST_F(ValidateAtomics, AtomicFlagClearNotInt32Pointer) {
- const std::string body = R"(
- OpAtomicFlagClear %u64_var %device %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body, "OpCapability Int64Atomics\n"));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicFlagClear: "
- "expected Pointer to point to a value of 32-bit integer type"));
- }
- TEST_F(ValidateAtomics, AtomicFlagClearWrongScopeType) {
- const std::string body = R"(
- OpAtomicFlagClear %u32_var %u64_1 %relaxed
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicFlagClear: expected scope to be a 32-bit "
- "int\n OpAtomicFlagClear %30 %ulong_1 %uint_0_1\n"));
- }
- TEST_F(ValidateAtomics, AtomicFlagClearWrongMemorySemanticsType) {
- const std::string body = R"(
- OpAtomicFlagClear %u32_var %device %u64_1
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicFlagClear: expected Memory Semantics to be a 32-bit int"));
- }
- TEST_F(ValidateAtomics, AtomicIIncrementAcquireAndRelease) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val1 = OpAtomicIIncrement %u32 %u32_var %device %acquire_and_release
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicIIncrement: Memory Semantics can have at most "
- "one of the following bits set: Acquire, Release, "
- "AcquireRelease or SequentiallyConsistent"));
- }
- TEST_F(ValidateAtomics, AtomicUniformMemorySemanticsShader) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val1 = OpAtomicIIncrement %u32 %u32_var %device %acquire_release_uniform_workgroup
- )";
- CompileSuccessfully(GenerateShaderCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicUniformMemorySemanticsKernel) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val1 = OpAtomicIIncrement %u32 %u32_var %device %acquire_release_uniform_workgroup
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicIIncrement: Memory Semantics UniformMemory "
- "requires capability Shader"));
- }
- // Lack of the AtomicStorage capability is intentionally ignored, see
- // https://github.com/KhronosGroup/glslang/issues/1618 for the reasoning why.
- TEST_F(ValidateAtomics, AtomicCounterMemorySemanticsNoCapability) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val1 = OpAtomicIIncrement %u32 %u32_var %device
- %acquire_release_atomic_counter_workgroup
- )";
- CompileSuccessfully(GenerateKernelCode(body));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, AtomicCounterMemorySemanticsWithCapability) {
- const std::string body = R"(
- OpAtomicStore %u32_var %device %relaxed %u32_1
- %val1 = OpAtomicIIncrement %u32 %u32_var %device %acquire_release_atomic_counter_workgroup
- )";
- CompileSuccessfully(GenerateKernelCode(body, "OpCapability AtomicStorage\n"));
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicLoad) {
- const std::string body = R"(
- %ld = OpAtomicLoad %u32 %u32_var %workgroup %sequentially_consistent
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicStore) {
- const std::string body = R"(
- OpAtomicStore %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics,
- VulkanMemoryModelBanSequentiallyConsistentAtomicExchange) {
- const std::string body = R"(
- %ex = OpAtomicExchange %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics,
- VulkanMemoryModelBanSequentiallyConsistentAtomicCompareExchangeEqual) {
- const std::string body = R"(
- %ex = OpAtomicCompareExchange %u32 %u32_var %workgroup %sequentially_consistent %relaxed %u32_0 %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics,
- VulkanMemoryModelBanSequentiallyConsistentAtomicCompareExchangeUnequal) {
- const std::string body = R"(
- %ex = OpAtomicCompareExchange %u32 %u32_var %workgroup %relaxed %sequentially_consistent %u32_0 %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics,
- VulkanMemoryModelBanSequentiallyConsistentAtomicIIncrement) {
- const std::string body = R"(
- %inc = OpAtomicIIncrement %u32 %u32_var %workgroup %sequentially_consistent
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics,
- VulkanMemoryModelBanSequentiallyConsistentAtomicIDecrement) {
- const std::string body = R"(
- %dec = OpAtomicIDecrement %u32 %u32_var %workgroup %sequentially_consistent
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicIAdd) {
- const std::string body = R"(
- %add = OpAtomicIAdd %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicISub) {
- const std::string body = R"(
- %sub = OpAtomicISub %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicSMin) {
- const std::string body = R"(
- %min = OpAtomicSMin %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicUMin) {
- const std::string body = R"(
- %min = OpAtomicUMin %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicFMinEXT) {
- const std::string body = R"(
- %max = OpAtomicFMinEXT %f32 %f32_var %workgroup %sequentially_consistent %f32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicSMax) {
- const std::string body = R"(
- %max = OpAtomicSMax %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicUMax) {
- const std::string body = R"(
- %max = OpAtomicUMax %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicFMaxEXT) {
- const std::string body = R"(
- %max = OpAtomicFMaxEXT %f32 %f32_var %workgroup %sequentially_consistent %f32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpCapability AtomicFloat32MinMaxEXT
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpExtension "SPV_EXT_shader_atomic_float_min_max"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicAnd) {
- const std::string body = R"(
- %and = OpAtomicAnd %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicOr) {
- const std::string body = R"(
- %or = OpAtomicOr %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelBanSequentiallyConsistentAtomicXor) {
- const std::string body = R"(
- %xor = OpAtomicXor %u32 %u32_var %workgroup %sequentially_consistent %u32_0
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("SequentiallyConsistent memory semantics cannot be "
- "used with the VulkanKHR memory model."));
- }
- TEST_F(ValidateAtomics, OutputMemoryKHRRequiresVulkanMemoryModelKHR) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %1 "func"
- OpExecutionMode %1 OriginUpperLeft
- %2 = OpTypeVoid
- %3 = OpTypeInt 32 0
- %semantics = OpConstant %3 4100
- %5 = OpTypeFunction %2
- %workgroup = OpConstant %3 2
- %ptr = OpTypePointer Workgroup %3
- %var = OpVariable %ptr Workgroup
- %1 = OpFunction %2 None %5
- %7 = OpLabel
- OpAtomicStore %var %workgroup %semantics %workgroup
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(text);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicStore: Memory Semantics OutputMemoryKHR "
- "requires capability VulkanMemoryModelKHR"));
- }
- TEST_F(ValidateAtomics, MakeAvailableKHRRequiresVulkanMemoryModelKHR) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %1 "func"
- OpExecutionMode %1 OriginUpperLeft
- %2 = OpTypeVoid
- %3 = OpTypeInt 32 0
- %semantics = OpConstant %3 8196
- %5 = OpTypeFunction %2
- %workgroup = OpConstant %3 2
- %ptr = OpTypePointer Workgroup %3
- %var = OpVariable %ptr Workgroup
- %1 = OpFunction %2 None %5
- %7 = OpLabel
- OpAtomicStore %var %workgroup %semantics %workgroup
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(text);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicStore: Memory Semantics MakeAvailableKHR "
- "requires capability VulkanMemoryModelKHR"));
- }
- TEST_F(ValidateAtomics, MakeVisibleKHRRequiresVulkanMemoryModelKHR) {
- const std::string text = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %1 "func"
- OpExecutionMode %1 OriginUpperLeft
- %2 = OpTypeVoid
- %3 = OpTypeInt 32 0
- %semantics = OpConstant %3 16386
- %5 = OpTypeFunction %2
- %workgroup = OpConstant %3 2
- %ptr = OpTypePointer Workgroup %3
- %var = OpVariable %ptr Workgroup
- %1 = OpFunction %2 None %5
- %7 = OpLabel
- %ld = OpAtomicLoad %3 %var %workgroup %semantics
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(text);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicLoad: Memory Semantics MakeVisibleKHR requires "
- "capability VulkanMemoryModelKHR"));
- }
- TEST_F(ValidateAtomics, MakeAvailableKHRRequiresReleaseSemantics) {
- const std::string text = R"(
- OpCapability Shader
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpMemoryModel Logical VulkanKHR
- OpEntryPoint Fragment %1 "func"
- OpExecutionMode %1 OriginUpperLeft
- %2 = OpTypeVoid
- %3 = OpTypeInt 32 0
- %semantics = OpConstant %3 8448
- %5 = OpTypeFunction %2
- %workgroup = OpConstant %3 2
- %ptr = OpTypePointer Workgroup %3
- %var = OpVariable %ptr Workgroup
- %1 = OpFunction %2 None %5
- %7 = OpLabel
- OpAtomicStore %var %workgroup %semantics %workgroup
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicStore: MakeAvailableKHR Memory Semantics also requires "
- "either Release or AcquireRelease Memory Semantics"));
- }
- TEST_F(ValidateAtomics, MakeVisibleKHRRequiresAcquireSemantics) {
- const std::string text = R"(
- OpCapability Shader
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpMemoryModel Logical VulkanKHR
- OpEntryPoint Fragment %1 "func"
- OpExecutionMode %1 OriginUpperLeft
- %2 = OpTypeVoid
- %3 = OpTypeInt 32 0
- %semantics = OpConstant %3 16640
- %5 = OpTypeFunction %2
- %workgroup = OpConstant %3 2
- %ptr = OpTypePointer Workgroup %3
- %var = OpVariable %ptr Workgroup
- %1 = OpFunction %2 None %5
- %7 = OpLabel
- %ld = OpAtomicLoad %3 %var %workgroup %semantics
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicLoad: MakeVisibleKHR Memory Semantics also requires "
- "either Acquire or AcquireRelease Memory Semantics"));
- }
- TEST_F(ValidateAtomics, MakeAvailableKHRRequiresStorageSemantics) {
- const std::string text = R"(
- OpCapability Shader
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpMemoryModel Logical VulkanKHR
- OpEntryPoint Fragment %1 "func"
- OpExecutionMode %1 OriginUpperLeft
- %2 = OpTypeVoid
- %3 = OpTypeInt 32 0
- %semantics = OpConstant %3 8196
- %5 = OpTypeFunction %2
- %workgroup = OpConstant %3 2
- %ptr = OpTypePointer Workgroup %3
- %var = OpVariable %ptr Workgroup
- %1 = OpFunction %2 None %5
- %7 = OpLabel
- OpAtomicStore %var %workgroup %semantics %workgroup
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicStore: expected Memory Semantics to include a storage class"));
- }
- TEST_F(ValidateAtomics, MakeVisibleKHRRequiresStorageSemantics) {
- const std::string text = R"(
- OpCapability Shader
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpMemoryModel Logical VulkanKHR
- OpEntryPoint Fragment %1 "func"
- OpExecutionMode %1 OriginUpperLeft
- %2 = OpTypeVoid
- %3 = OpTypeInt 32 0
- %semantics = OpConstant %3 16386
- %5 = OpTypeFunction %2
- %workgroup = OpConstant %3 2
- %ptr = OpTypePointer Workgroup %3
- %var = OpVariable %ptr Workgroup
- %1 = OpFunction %2 None %5
- %7 = OpLabel
- %ld = OpAtomicLoad %3 %var %workgroup %semantics
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(text, SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicLoad: expected Memory Semantics to include a storage class"));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelAllowsQueueFamilyKHR) {
- const std::string body = R"(
- %val = OpAtomicAnd %u32 %u32_var %queuefamily %relaxed %u32_1
- )";
- const std::string extra = R"(
- OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderComputeCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_VULKAN_1_1);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_1));
- }
- TEST_F(ValidateAtomics, NonVulkanMemoryModelDisallowsQueueFamilyKHR) {
- const std::string body = R"(
- %val = OpAtomicAnd %u32 %u32_var %queuefamily %relaxed %u32_1
- )";
- CompileSuccessfully(GenerateShaderCode(body), SPV_ENV_VULKAN_1_1);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions(SPV_ENV_VULKAN_1_1));
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicAnd: Memory Scope QueueFamilyKHR requires "
- "capability VulkanMemoryModelKHR\n %42 = OpAtomicAnd "
- "%uint %29 %uint_5 %uint_0_1 %uint_1\n"));
- }
- TEST_F(ValidateAtomics, SemanticsSpecConstantShader) {
- const std::string spirv = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %func "func"
- OpExecutionMode %func OriginUpperLeft
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %spec_const = OpSpecConstant %int 0
- %workgroup = OpConstant %int 2
- %ptr_int_workgroup = OpTypePointer Workgroup %int
- %var = OpVariable %ptr_int_workgroup Workgroup
- %voidfn = OpTypeFunction %void
- %func = OpFunction %void None %voidfn
- %entry = OpLabel
- %ld = OpAtomicLoad %int %var %workgroup %spec_const
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("Memory Semantics ids must be OpConstant when Shader "
- "capability is present"));
- }
- TEST_F(ValidateAtomics, SemanticsSpecConstantKernel) {
- const std::string spirv = R"(
- OpCapability Kernel
- OpCapability Linkage
- OpMemoryModel Logical OpenCL
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %spec_const = OpSpecConstant %int 0
- %workgroup = OpConstant %int 2
- %ptr_int_workgroup = OpTypePointer Workgroup %int
- %var = OpVariable %ptr_int_workgroup Workgroup
- %voidfn = OpTypeFunction %void
- %func = OpFunction %void None %voidfn
- %entry = OpLabel
- %ld = OpAtomicLoad %int %var %workgroup %spec_const
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, ScopeSpecConstantShader) {
- const std::string spirv = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- OpEntryPoint Fragment %func "func"
- OpExecutionMode %func OriginUpperLeft
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %spec_const = OpSpecConstant %int 0
- %relaxed = OpConstant %int 0
- %ptr_int_workgroup = OpTypePointer Workgroup %int
- %var = OpVariable %ptr_int_workgroup Workgroup
- %voidfn = OpTypeFunction %void
- %func = OpFunction %void None %voidfn
- %entry = OpLabel
- %ld = OpAtomicLoad %int %var %spec_const %relaxed
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "Scope ids must be OpConstant when Shader capability is present"));
- }
- TEST_F(ValidateAtomics, ScopeSpecConstantKernel) {
- const std::string spirv = R"(
- OpCapability Kernel
- OpCapability Linkage
- OpMemoryModel Logical OpenCL
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %spec_const = OpSpecConstant %int 0
- %relaxed = OpConstant %int 0
- %ptr_int_workgroup = OpTypePointer Workgroup %int
- %var = OpVariable %ptr_int_workgroup Workgroup
- %voidfn = OpTypeFunction %void
- %func = OpFunction %void None %voidfn
- %entry = OpLabel
- %ld = OpAtomicLoad %int %var %spec_const %relaxed
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelDeviceScopeBad) {
- const std::string body = R"(
- %val = OpAtomicAnd %u32 %u32_var %device %relaxed %u32_1
- )";
- const std::string extra = R"(OpCapability VulkanMemoryModelKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("Use of device scope with VulkanKHR memory model requires the "
- "VulkanMemoryModelDeviceScopeKHR capability"));
- }
- TEST_F(ValidateAtomics, VulkanMemoryModelDeviceScopeGood) {
- const std::string body = R"(
- %val = OpAtomicAnd %u32 %u32_var %device %relaxed %u32_1
- )";
- const std::string extra = R"(OpCapability VulkanMemoryModelKHR
- OpCapability VulkanMemoryModelDeviceScopeKHR
- OpExtension "SPV_KHR_vulkan_memory_model"
- )";
- CompileSuccessfully(GenerateShaderCode(body, extra, "", "VulkanKHR"),
- SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_3));
- }
- TEST_F(ValidateAtomics, CompareExchangeWeakV13ValV14Good) {
- const std::string body = R"(
- %val1 = OpAtomicCompareExchangeWeak %u32 %u32_var %device %relaxed %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_3);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_4));
- }
- TEST_F(ValidateAtomics, CompareExchangeWeakV14Bad) {
- const std::string body = R"(
- %val1 = OpAtomicCompareExchangeWeak %u32 %u32_var %device %relaxed %relaxed %u32_0 %u32_0
- )";
- CompileSuccessfully(GenerateKernelCode(body), SPV_ENV_UNIVERSAL_1_4);
- EXPECT_EQ(SPV_ERROR_WRONG_VERSION,
- ValidateInstructions(SPV_ENV_UNIVERSAL_1_4));
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "AtomicCompareExchangeWeak requires SPIR-V version 1.3 or earlier"));
- }
- TEST_F(ValidateAtomics, CompareExchangeVolatileMatch) {
- const std::string spirv = R"(
- OpCapability Shader
- OpCapability VulkanMemoryModelKHR
- OpCapability Linkage
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpMemoryModel Logical VulkanKHR
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %workgroup = OpConstant %int 2
- %volatile = OpConstant %int 32768
- %ptr_wg_int = OpTypePointer Workgroup %int
- %wg_var = OpVariable %ptr_wg_int Workgroup
- %void_fn = OpTypeFunction %void
- %func = OpFunction %void None %void_fn
- %entry = OpLabel
- %cmp_ex = OpAtomicCompareExchange %int %wg_var %workgroup %volatile %volatile %int_0 %int_1
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, CompareExchangeVolatileMismatch) {
- const std::string spirv = R"(
- OpCapability Shader
- OpCapability VulkanMemoryModelKHR
- OpCapability Linkage
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpMemoryModel Logical VulkanKHR
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %workgroup = OpConstant %int 2
- %volatile = OpConstant %int 32768
- %non_volatile = OpConstant %int 0
- %ptr_wg_int = OpTypePointer Workgroup %int
- %wg_var = OpVariable %ptr_wg_int Workgroup
- %void_fn = OpTypeFunction %void
- %func = OpFunction %void None %void_fn
- %entry = OpLabel
- %cmp_ex = OpAtomicCompareExchange %int %wg_var %workgroup %non_volatile %volatile %int_0 %int_1
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("Volatile mask setting must match for Equal and "
- "Unequal memory semantics"));
- }
- TEST_F(ValidateAtomics, CompareExchangeVolatileMismatchCooperativeMatrix) {
- const std::string spirv = R"(
- OpCapability Shader
- OpCapability VulkanMemoryModelKHR
- OpCapability Linkage
- OpCapability CooperativeMatrixNV
- OpExtension "SPV_KHR_vulkan_memory_model"
- OpExtension "SPV_NV_cooperative_matrix"
- OpMemoryModel Logical VulkanKHR
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %workgroup = OpConstant %int 2
- %volatile = OpSpecConstant %int 32768
- %non_volatile = OpSpecConstant %int 32768
- %ptr_wg_int = OpTypePointer Workgroup %int
- %wg_var = OpVariable %ptr_wg_int Workgroup
- %void_fn = OpTypeFunction %void
- %func = OpFunction %void None %void_fn
- %entry = OpLabel
- %cmp_ex = OpAtomicCompareExchange %int %wg_var %workgroup %volatile %non_volatile %int_0 %int_1
- OpReturn
- OpFunctionEnd
- )";
- // This is ok because we cannot evaluate the spec constant defaults.
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions());
- }
- TEST_F(ValidateAtomics, VolatileRequiresVulkanMemoryModel) {
- const std::string spirv = R"(
- OpCapability Shader
- OpCapability Linkage
- OpMemoryModel Logical GLSL450
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %workgroup = OpConstant %int 2
- %volatile = OpConstant %int 32768
- %ptr_wg_int = OpTypePointer Workgroup %int
- %wg_var = OpVariable %ptr_wg_int Workgroup
- %void_fn = OpTypeFunction %void
- %func = OpFunction %void None %void_fn
- %entry = OpLabel
- %ld = OpAtomicLoad %int %wg_var %workgroup %volatile
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("Memory Semantics Volatile requires capability "
- "VulkanMemoryModelKHR"));
- }
- TEST_F(ValidateAtomics, CooperativeMatrixSemanticsMustBeConstant) {
- const std::string spirv = R"(
- OpCapability Shader
- OpCapability Linkage
- OpCapability CooperativeMatrixNV
- OpExtension "SPV_NV_cooperative_matrix"
- OpMemoryModel Logical GLSL450
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %workgroup = OpConstant %int 2
- %undef = OpUndef %int
- %ptr_wg_int = OpTypePointer Workgroup %int
- %wg_var = OpVariable %ptr_wg_int Workgroup
- %void_fn = OpTypeFunction %void
- %func = OpFunction %void None %void_fn
- %entry = OpLabel
- %ld = OpAtomicLoad %int %wg_var %workgroup %undef
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("Memory Semantics must be a constant instruction when "
- "CooperativeMatrixNV capability is present"));
- }
- TEST_F(ValidateAtomics, IIncrementBadPointerDataType) {
- const std::string spirv = R"(
- OpCapability Shader
- OpMemoryModel Logical GLSL450
- %uint = OpTypeInt 32 0
- %_ptr_Input_uint = OpTypePointer Input %uint
- %v3uint = OpTypeVector %uint 3
- %_ptr_Input_v3uint = OpTypePointer Input %v3uint
- %void = OpTypeVoid
- %16 = OpTypeFunction %void
- %uint_538976288 = OpConstant %uint 538976288
- %int = OpTypeInt 32 1
- %_runtimearr_int = OpTypeRuntimeArray %int
- %_struct_5 = OpTypeStruct %_runtimearr_int
- %_ptr_Uniform__struct_5 = OpTypePointer Uniform %_struct_5
- %3 = OpVariable %_ptr_Input_v3uint Input
- %7 = OpVariable %_ptr_Uniform__struct_5 Uniform
- %8224 = OpFunction %void None %16
- %65312 = OpLabel
- %25 = OpAccessChain %_ptr_Input_uint %3 %uint_538976288
- %26 = OpLoad %uint %25
- %2097184 = OpAtomicIIncrement %int %7 %uint_538976288 %26
- OpUnreachable
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicIIncrement: expected Pointer to point to a "
- "value of type Result Type"));
- }
- TEST_F(ValidateAtomics, AtomicFloat16VectorSuccess) {
- const std::string definitions = R"(
- %f16 = OpTypeFloat 16
- %f16vec2 = OpTypeVector %f16 2
- %f16vec4 = OpTypeVector %f16 4
- %f16_1 = OpConstant %f16 1
- %f16vec2_1 = OpConstantComposite %f16vec2 %f16_1 %f16_1
- %f16vec4_1 = OpConstantComposite %f16vec4 %f16_1 %f16_1 %f16_1 %f16_1
- %f16vec2_ptr = OpTypePointer Workgroup %f16vec2
- %f16vec4_ptr = OpTypePointer Workgroup %f16vec4
- %f16vec2_var = OpVariable %f16vec2_ptr Workgroup
- %f16vec4_var = OpVariable %f16vec4_ptr Workgroup
- )";
- const std::string body = R"(
- %val3 = OpAtomicFMinEXT %f16vec2 %f16vec2_var %device %relaxed %f16vec2_1
- %val4 = OpAtomicFMaxEXT %f16vec2 %f16vec2_var %device %relaxed %f16vec2_1
- %val8 = OpAtomicFAddEXT %f16vec2 %f16vec2_var %device %relaxed %f16vec2_1
- %val9 = OpAtomicExchange %f16vec2 %f16vec2_var %device %relaxed %f16vec2_1
- %val11 = OpAtomicFMinEXT %f16vec4 %f16vec4_var %device %relaxed %f16vec4_1
- %val12 = OpAtomicFMaxEXT %f16vec4 %f16vec4_var %device %relaxed %f16vec4_1
- %val18 = OpAtomicFAddEXT %f16vec4 %f16vec4_var %device %relaxed %f16vec4_1
- %val19 = OpAtomicExchange %f16vec4 %f16vec4_var %device %relaxed %f16vec4_1
- )";
- CompileSuccessfully(GenerateShaderComputeCode(
- body,
- "OpCapability Float16\n"
- "OpCapability AtomicFloat16VectorNV\n"
- "OpExtension \"SPV_NV_shader_atomic_fp16_vector\"\n",
- definitions),
- SPV_ENV_VULKAN_1_0);
- ASSERT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_VULKAN_1_0));
- }
- static constexpr char Float16Vector3Defs[] = R"(
- %f16 = OpTypeFloat 16
- %f16vec3 = OpTypeVector %f16 3
- %f16_1 = OpConstant %f16 1
- %f16vec3_1 = OpConstantComposite %f16vec3 %f16_1 %f16_1 %f16_1
- %f16vec3_ptr = OpTypePointer Workgroup %f16vec3
- %f16vec3_var = OpVariable %f16vec3_ptr Workgroup
- )";
- TEST_F(ValidateAtomics, AtomicFloat16Vector3MinFail) {
- const std::string definitions = Float16Vector3Defs;
- const std::string body = R"(
- %val11 = OpAtomicFMinEXT %f16vec3 %f16vec3_var %device %relaxed %f16vec3_1
- )";
- CompileSuccessfully(GenerateShaderComputeCode(
- body,
- "OpCapability Float16\n"
- "OpCapability AtomicFloat16VectorNV\n"
- "OpExtension \"SPV_NV_shader_atomic_fp16_vector\"\n",
- definitions),
- SPV_ENV_VULKAN_1_0);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicFMinEXT: expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicFloat16Vector3MaxFail) {
- const std::string definitions = Float16Vector3Defs;
- const std::string body = R"(
- %val12 = OpAtomicFMaxEXT %f16vec3 %f16vec3_var %device %relaxed %f16vec3_1
- )";
- CompileSuccessfully(GenerateShaderComputeCode(
- body,
- "OpCapability Float16\n"
- "OpCapability AtomicFloat16VectorNV\n"
- "OpExtension \"SPV_NV_shader_atomic_fp16_vector\"\n",
- definitions),
- SPV_ENV_VULKAN_1_0);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicFMaxEXT: expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicFloat16Vector3AddFail) {
- const std::string definitions = Float16Vector3Defs;
- const std::string body = R"(
- %val18 = OpAtomicFAddEXT %f16vec3 %f16vec3_var %device %relaxed %f16vec3_1
- )";
- CompileSuccessfully(GenerateShaderComputeCode(
- body,
- "OpCapability Float16\n"
- "OpCapability AtomicFloat16VectorNV\n"
- "OpExtension \"SPV_NV_shader_atomic_fp16_vector\"\n",
- definitions),
- SPV_ENV_VULKAN_1_0);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr("AtomicFAddEXT: expected Result Type to be float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicFloat16Vector3ExchangeFail) {
- const std::string definitions = Float16Vector3Defs;
- const std::string body = R"(
- %val19 = OpAtomicExchange %f16vec3 %f16vec3_var %device %relaxed %f16vec3_1
- )";
- CompileSuccessfully(GenerateShaderComputeCode(
- body,
- "OpCapability Float16\n"
- "OpCapability AtomicFloat16VectorNV\n"
- "OpExtension \"SPV_NV_shader_atomic_fp16_vector\"\n",
- definitions),
- SPV_ENV_VULKAN_1_0);
- EXPECT_EQ(SPV_ERROR_INVALID_DATA, ValidateInstructions());
- EXPECT_THAT(getDiagnosticString(),
- HasSubstr("AtomicExchange: expected Result Type to be integer or "
- "float scalar type"));
- }
- TEST_F(ValidateAtomics, AtomicLoadUntypedPointer) {
- const std::string spirv = R"(
- OpCapability Shader
- OpCapability UntypedPointersKHR
- OpCapability WorkgroupMemoryExplicitLayoutKHR
- OpExtension "SPV_KHR_workgroup_memory_explicit_layout"
- OpExtension "SPV_KHR_untyped_pointers"
- OpMemoryModel Logical GLSL450
- OpEntryPoint GLCompute %main "main" %var
- OpDecorate %struct Block
- OpMemberDecorate %struct 0 Offset 0
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %struct = OpTypeStruct %int
- %ptr = OpTypeUntypedPointerKHR Workgroup
- %var = OpUntypedVariableKHR %ptr Workgroup %struct
- %void_fn = OpTypeFunction %void
- %main = OpFunction %void None %void_fn
- %entry = OpLabel
- %load = OpAtomicLoad %int %var %int_1 %int_0
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv, SPV_ENV_UNIVERSAL_1_4);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_4));
- }
- TEST_F(ValidateAtomics, AtomicStoreUntypedPointer) {
- const std::string spirv = R"(
- OpCapability Shader
- OpCapability UntypedPointersKHR
- OpCapability WorkgroupMemoryExplicitLayoutKHR
- OpExtension "SPV_KHR_workgroup_memory_explicit_layout"
- OpExtension "SPV_KHR_untyped_pointers"
- OpMemoryModel Logical GLSL450
- OpEntryPoint GLCompute %main "main" %var
- OpDecorate %struct Block
- OpMemberDecorate %struct 0 Offset 0
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %struct = OpTypeStruct %int
- %ptr = OpTypeUntypedPointerKHR Workgroup
- %var = OpUntypedVariableKHR %ptr Workgroup %struct
- %void_fn = OpTypeFunction %void
- %main = OpFunction %void None %void_fn
- %entry = OpLabel
- OpAtomicStore %var %int_1 %int_0 %int_0
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv, SPV_ENV_UNIVERSAL_1_4);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_4));
- }
- TEST_F(ValidateAtomics, AtomicExchangeUntypedPointer) {
- const std::string spirv = R"(
- OpCapability Shader
- OpCapability UntypedPointersKHR
- OpCapability WorkgroupMemoryExplicitLayoutKHR
- OpExtension "SPV_KHR_workgroup_memory_explicit_layout"
- OpExtension "SPV_KHR_untyped_pointers"
- OpMemoryModel Logical GLSL450
- OpEntryPoint GLCompute %main "main" %var
- OpDecorate %struct Block
- OpMemberDecorate %struct 0 Offset 0
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %struct = OpTypeStruct %int
- %ptr = OpTypeUntypedPointerKHR Workgroup
- %var = OpUntypedVariableKHR %ptr Workgroup %struct
- %void_fn = OpTypeFunction %void
- %main = OpFunction %void None %void_fn
- %entry = OpLabel
- %ex = OpAtomicExchange %int %var %int_1 %int_0 %int_0
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv, SPV_ENV_UNIVERSAL_1_4);
- EXPECT_EQ(SPV_SUCCESS, ValidateInstructions(SPV_ENV_UNIVERSAL_1_4));
- }
- TEST_F(ValidateAtomics, AtomicFlagClearUntypedPointer) {
- const std::string spirv = R"(
- OpCapability Kernel
- OpCapability Linkage
- OpCapability UntypedPointersKHR
- OpExtension "SPV_KHR_untyped_pointers"
- OpMemoryModel Logical OpenCL
- %void = OpTypeVoid
- %int = OpTypeInt 32 0
- %int_0 = OpConstant %int 0
- %int_1 = OpConstant %int 1
- %ptr = OpTypeUntypedPointerKHR Workgroup
- %var = OpUntypedVariableKHR %ptr Workgroup %int
- %void_fn = OpTypeFunction %void
- %main = OpFunction %void None %void_fn
- %entry = OpLabel
- OpAtomicFlagClear %var %int_1 %int_0
- OpReturn
- OpFunctionEnd
- )";
- CompileSuccessfully(spirv);
- EXPECT_EQ(SPV_ERROR_INVALID_ID, ValidateInstructions());
- EXPECT_THAT(
- getDiagnosticString(),
- HasSubstr(
- "Untyped pointers are not supported by atomic flag instructions"));
- }
- } // namespace
- } // namespace val
- } // namespace spvtools
|