| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428 |
- // Copyright (c) 2019 Google LLC
- //
- // 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.
- // This file is specifically named spvtools_fuzz.proto so that the string
- // 'spvtools_fuzz' appears in the names of global-scope symbols that protoc
- // generates when targeting C++. This is to reduce the potential for name
- // clashes with other globally-scoped symbols.
- syntax = "proto3";
- package spvtools.fuzz.protobufs;
- message UInt32Pair {
- // A pair of uint32s; useful for defining mappings.
- uint32 first = 1;
- uint32 second = 2;
- }
- message InstructionDescriptor {
- // Describes an instruction in some block of a function with respect to a
- // base instruction.
- // The id of an instruction after which the instruction being described is
- // believed to be located. It might be the using instruction itself.
- uint32 base_instruction_result_id = 1;
- // The opcode for the instruction being described.
- uint32 target_instruction_opcode = 2;
- // The number of matching opcodes to skip over when searching from the base
- // instruction to the instruction being described.
- uint32 num_opcodes_to_ignore = 3;
- }
- message IdUseDescriptor {
- // Describes a use of an id as an input operand to an instruction in some
- // block of a function.
- // Example:
- // - id_of_interest = 42
- // - enclosing_instruction = (
- // base_instruction_result_id = 50,
- // target_instruction_opcode = OpStore
- // num_opcodes_to_ignore = 7
- // )
- // - in_operand_index = 1
- // represents a use of id 42 as input operand 1 to an OpStore instruction,
- // such that the OpStore instruction can be found in the same basic block as
- // the instruction with result id 50, and in particular is the 8th OpStore
- // instruction found from instruction 50 onwards (i.e. 7 OpStore
- // instructions are skipped).
- // An id that we would like to be able to find a use of.
- uint32 id_of_interest = 1;
- // The input operand index at which the use is expected.
- InstructionDescriptor enclosing_instruction = 2;
- uint32 in_operand_index = 3;
- }
- message DataDescriptor {
- // Represents a data element that can be accessed from an id, by walking the
- // type hierarchy via a sequence of 0 or more indices.
- //
- // Very similar to a UniformBufferElementDescriptor, except that a
- // DataDescriptor is rooted at the id of a scalar or composite.
- // The object being accessed - a scalar or composite
- uint32 object = 1;
- // 0 or more indices, used to index into a composite object
- repeated uint32 index = 2;
- }
- message UniformBufferElementDescriptor {
- // Represents a data element inside a uniform buffer. The element is
- // specified via (a) the result id of a uniform variable in which the element
- // is contained, and (b) a series of indices that need to be followed to get
- // to the element (via fields and array/vector indices).
- //
- // Example: suppose there is a uniform variable with descriptor set 7 and
- // binding 9, and that the uniform variable has the following type (using
- // GLSL-like syntax):
- //
- // struct S {
- // float f;
- // vec3 g;
- // int4 h[10];
- // };
- //
- // Then:
- // - (7, 9, [0]) describes the 'f' field.
- // - (7, 9, [1,1]) describes the y component of the 'g' field.
- // - (7, 9, [2,7,3]) describes the w component of element 7 of the 'h' field
- // The descriptor set and binding associated with a uniform variable.
- uint32 descriptor_set = 1;
- uint32 binding = 2;
- // An ordered sequence of indices through composite structures in the
- // uniform buffer.
- repeated uint32 index = 3;
- }
- message InstructionOperand {
- // Represents an operand to a SPIR-V instruction.
- // The type of the operand.
- uint32 operand_type = 1;
- // The data associated with the operand. For most operands (e.g. ids,
- // storage classes and literals) this will be a single word.
- repeated uint32 operand_data = 2;
- }
- message Instruction {
- // Represents a SPIR-V instruction.
- // The instruction's opcode (e.g. OpLabel).
- uint32 opcode = 1;
- // The id of the instruction's result type; 0 if there is no result type.
- uint32 result_type_id = 2;
- // The id of the instruction's result; 0 if there is no result.
- uint32 result_id = 3;
- // Zero or more input operands.
- repeated InstructionOperand input_operand = 4;
- }
- message FactSequence {
- repeated Fact fact = 1;
- }
- message Fact {
- oneof fact {
- // Order the fact options by numeric id (rather than alphabetically).
- FactConstantUniform constant_uniform_fact = 1;
- FactDataSynonym data_synonym_fact = 2;
- FactBlockIsDead block_is_dead_fact = 3;
- FactFunctionIsLivesafe function_is_livesafe_fact = 4;
- FactPointeeValueIsIrrelevant pointee_value_is_irrelevant_fact = 5;
- FactIdEquation id_equation_fact = 6;
- FactIdIsIrrelevant id_is_irrelevant = 7;
- }
- }
- // Keep fact message types in alphabetical order:
- message FactBlockIsDead {
- // Records the fact that a block is guaranteed to be dynamically unreachable.
- // This is useful because it informs the fuzzer that rather arbitrary changes
- // can be made to this block.
- uint32 block_id = 1;
- }
- message FactConstantUniform {
- // Records the fact that a uniform buffer element is guaranteed to be equal
- // to a particular constant value. spirv-fuzz can use such guarantees to
- // obfuscate code, e.g. to manufacture an expression that will (due to the
- // guarantee) evaluate to a particular value at runtime but in a manner that
- // cannot be predicted at compile-time.
- // An element of a uniform buffer
- UniformBufferElementDescriptor uniform_buffer_element_descriptor = 1;
- // The words of the associated constant
- repeated uint32 constant_word = 2;
- }
- message FactDataSynonym {
- // Records the fact that the data held in two data descriptors are guaranteed
- // to be equal. spirv-fuzz can use this to replace uses of one piece of data
- // with a known-to-be-equal piece of data.
- // Data descriptors guaranteed to hold identical data.
- DataDescriptor data1 = 1;
- DataDescriptor data2 = 2;
- }
- message FactFunctionIsLivesafe {
- // Records the fact that a function is guaranteed to be "livesafe", meaning
- // that it will not make out-of-bounds accesses, does not contain reachable
- // OpKill or OpUnreachable instructions, does not contain loops that will
- // execute for large numbers of iterations, and only invokes other livesafe
- // functions.
- uint32 function_id = 1;
- }
- message FactIdEquation {
- // Records the fact that the equation:
- //
- // lhs_id = opcode rhs_id[0] rhs_id[1] ... rhs_id[N-1]
- //
- // holds; e.g. that the equation:
- //
- // %12 = OpIAdd %13 %14
- //
- // holds in the case where lhs_id is 12, rhs_id is [13, 14], and the opcode is
- // OpIAdd.
- // The left-hand-side of the equation.
- uint32 lhs_id = 1;
- // A SPIR-V opcode, from a restricted set of instructions for which equation
- // facts make sense.
- uint32 opcode = 2;
- // The operands to the right-hand-side of the equation.
- repeated uint32 rhs_id = 3;
- }
- message FactIdIsIrrelevant {
- // Records a fact that |result_id| is irrelevant (i.e. it's usage doesn't
- // change the semantics of the module). This implies that a use of this id
- // can later be replaced with some other id of the same type, or the
- // definition of |result_id| can be changed so that it yields a different value.
- // An irrelevant id.
- uint32 result_id = 1;
- }
- message FactPointeeValueIsIrrelevant {
- // Records the fact that value of the data pointed to by a pointer id does
- // not influence the observable behaviour of the module. This means that
- // arbitrary stores can be made through the pointer, and that nothing can be
- // guaranteed about the values that are loaded via the pointer.
- // A result id of pointer type
- uint32 pointer_id = 1;
- }
- message AccessChainClampingInfo {
- // When making a function livesafe it is necessary to clamp the indices that
- // occur as operands to access chain instructions so that they are guaranteed
- // to be in bounds. This message type allows an access chain instruction to
- // have an associated sequence of ids that are reserved for comparing an
- // access chain index with a bound (e.g. an array size), and selecting
- // between the access chain index (if it is within bounds) and the bound (if
- // it is not).
- //
- // This allows turning an instruction of the form:
- //
- // %result = OpAccessChain %type %object ... %index ...
- //
- // into:
- //
- // %t1 = OpULessThanEqual %bool %index %bound_minus_one
- // %t2 = OpSelect %int_type %t1 %index %bound_minus_one
- // %result = OpAccessChain %type %object ... %t2 ...
- // The result id of an OpAccessChain or OpInBoundsAccessChain instruction.
- uint32 access_chain_id = 1;
- // A series of pairs of fresh ids, one per access chain index, for the results
- // of a compare instruction and a select instruction, serving the roles of %t1
- // and %t2 in the above example.
- repeated UInt32Pair compare_and_select_ids = 2;
- }
- message SideEffectWrapperInfo {
- // When flattening a conditional branch, it is necessary to enclose
- // instructions that have side effects inside conditionals, so that
- // they are only executed if the condition holds. Otherwise, there
- // might be unintended changes in memory, or crashes that would not
- // originally happen.
- // For example, the instruction %id = OpLoad %type %ptr, found in
- // the true branch of the conditional, will be enclosed in a new
- // conditional (assuming that the block containing it can be split
- // around it) as follows:
- //
- // [previous instructions in the block]
- // OpSelectionMerge %merge_block_id None
- // OpBranchConditional %cond %execute_block_id %alternative_block_id
- // %execute_block_id = OpLabel
- // %actual_result_id = OpLoad %type %ptr
- // OpBranch %merge_block_id
- // %alternative_block_id = OpLabel
- // %placeholder_result_id = OpCopyObject %type %value_to_copy_id
- // OpBranch %merge_block_id
- // %merge_block_id = OpLabel
- // %id = OpPhi %type %actual_result_id %execute_block_id %placeholder_result_id %alternative_block_id
- // [following instructions from the original block]
- //
- // If the instruction does not have a result id, this is simplified.
- // For example, OpStore %ptr %value, found in the true branch of a
- // conditional, is enclosed as follows:
- //
- // [previous instructions in the block]
- // OpSelectionMerge %merge_block None
- // OpBranchConditional %cond %execute_block_id %merge_block_id
- // %execute_block_id = OpLabel
- // OpStore %ptr %value
- // OpBranch %merge_block_id
- // %merge_block_id = OpLabel
- // [following instructions from the original block]
- //
- // The same happens if the instruction is found in the false branch
- // of the conditional being flattened, except that the label ids in
- // the OpBranchConditional are swapped.
- // An instruction descriptor for identifying the instruction to be
- // enclosed inside a conditional. An instruction descriptor is
- // necessary because the instruction might not have a result id.
- InstructionDescriptor instruction = 1;
- // A fresh id for the new merge block.
- uint32 merge_block_id = 2;
- // A fresh id for the new block where the actual instruction is
- // executed.
- uint32 execute_block_id = 3;
- // The following fields are only needed if the original instruction has a
- // result id. They can be set to 0 if not needed.
- // A fresh id for the result id of the instruction (the original
- // one is used by the OpPhi instruction).
- uint32 actual_result_id = 4;
- // A fresh id for the new block where the placeholder instruction
- // is placed.
- uint32 alternative_block_id = 5;
- // A fresh id for the placeholder instruction.
- uint32 placeholder_result_id = 6;
- // An id present in the module, available to use at this point in
- // the program and with the same type as the original instruction,
- // that can be used to create a placeholder OpCopyObject
- // instruction.
- uint32 value_to_copy_id = 7;
- }
- message ReturnMergingInfo {
- // TransformationMergeFunctionReturns needs to modify each merge block of
- // loops containing return instructions, by:
- // - adding instructions to decide whether the function is returning
- // - adding instructions to pass on the return value of the function,
- // if it is returning
- // - changing the branch instruction (which must be an unconditional branch)
- // to a conditional branch that, if the function is returning, branches to
- // the merge block of the innermost loop that contains this merge block
- // (which can be the new merge block introduced by the transformation).
- //
- // One such merge block of the form:
- // %block = OpLabel
- // %phi1 = OpPhi %type1 %val1_1 %pred1 %val1_2 %pred2
- // %phi2 = OpPhi %type2 %val2_1 %pred1 %val2_2 %pred2
- // OpBranch %next
- //
- // is transformed into:
- // %block = OpLabel
- // %is_returning_id = OpPhi %bool %false %pred1 %false %pred2 %true %ret_bb1 %is_bb2_returning %mer_bb2
- // %maybe_return_val_id = OpPhi %return_type %any_returnable_val %pred1 %any_returnable_val %pred2
- // %ret_val1 %ret_bb1 %ret_val2 %mer_bb2
- // %phi1 = OpPhi %type1 %val1_1 %pred1 %val1_2 %pred2
- // %any_suitable_id_1 %ret_bb1 %any_suitable_id_1 %mer_bb2
- // %phi2 = OpPhi %type2 %val2_1 %pred1 %val2_2 %pred2
- // %any_suitable_id_1 %ret_bb1 %any_suitable_id_1 %mer_bb2
- // OpBranchConditional %is_returning_id %innermost_loop_merge %next
- //
- // where %ret_bb1 is a block that originally contains a return instruction and %mer_bb2 is the merge block of an inner
- // loop, from where the function might be returning.
- //
- // Note that the block is required to only have OpLabel, OpPhi or OpBranch instructions.
- // The id of the merge block that needs to be modified.
- uint32 merge_block_id = 1;
- // A fresh id for a boolean OpPhi whose value will be true iff the function
- // is returning. This will be used to decide whether to break out of the loop
- // or to use the original branch of the function. This value will also be
- // used by the merge block of the enclosing loop (if there is one) if the
- // function is returning from this block.
- uint32 is_returning_id = 2;
- // A fresh id that will get the value being returned, if the function is
- // returning. If the function return type is void, this is ignored.
- uint32 maybe_return_val_id = 3;
- // A mapping from each existing OpPhi id to a suitable id of the same type
- // available to use before the instruction.
- repeated UInt32Pair opphi_to_suitable_id = 4;
- }
- message LoopLimiterInfo {
- // Structure capturing the information required to manipulate a loop limiter
- // at a loop header.
- // The header for the loop.
- uint32 loop_header_id = 1;
- // A fresh id into which the loop limiter's current value can be loaded.
- uint32 load_id = 2;
- // A fresh id that can be used to increment the loaded value by 1.
- uint32 increment_id = 3;
- // A fresh id that can be used to compare the loaded value with the loop
- // limit.
- uint32 compare_id = 4;
- // A fresh id that can be used to compute the conjunction or disjunction of
- // an original loop exit condition with |compare_id|, if the loop's back edge
- // block can conditionally exit the loop.
- uint32 logical_op_id = 5;
- // A sequence of ids suitable for extending OpPhi instructions of the loop
- // merge block if it did not previously have an incoming edge from the loop
- // back edge block.
- repeated uint32 phi_id = 6;
- }
- message TransformationSequence {
- repeated Transformation transformation = 1;
- }
- message Transformation {
- oneof transformation {
- // Order the transformation options by numeric id (rather than
- // alphabetically).
- TransformationMoveBlockDown move_block_down = 1;
- TransformationSplitBlock split_block = 2;
- TransformationAddConstantBoolean add_constant_boolean = 3;
- TransformationAddConstantScalar add_constant_scalar = 4;
- TransformationAddTypeBoolean add_type_boolean = 5;
- TransformationAddTypeFloat add_type_float = 6;
- TransformationAddTypeInt add_type_int = 7;
- TransformationAddDeadBreak add_dead_break = 8;
- TransformationReplaceBooleanConstantWithConstantBinary
- replace_boolean_constant_with_constant_binary = 9;
- TransformationAddTypePointer add_type_pointer = 10;
- TransformationReplaceConstantWithUniform replace_constant_with_uniform = 11;
- TransformationAddDeadContinue add_dead_continue = 12;
- TransformationReplaceIdWithSynonym replace_id_with_synonym = 13;
- TransformationSetSelectionControl set_selection_control = 14;
- TransformationCompositeConstruct composite_construct = 15;
- TransformationSetLoopControl set_loop_control = 16;
- TransformationSetFunctionControl set_function_control = 17;
- TransformationAddNoContractionDecoration add_no_contraction_decoration = 18;
- TransformationSetMemoryOperandsMask set_memory_operands_mask = 19;
- TransformationCompositeExtract composite_extract = 20;
- TransformationVectorShuffle vector_shuffle = 21;
- TransformationOutlineFunction outline_function = 22;
- TransformationMergeBlocks merge_blocks = 23;
- TransformationAddTypeVector add_type_vector = 24;
- TransformationAddTypeArray add_type_array = 25;
- TransformationAddTypeMatrix add_type_matrix = 26;
- TransformationAddTypeStruct add_type_struct = 27;
- TransformationAddTypeFunction add_type_function = 28;
- TransformationAddConstantComposite add_constant_composite = 29;
- TransformationAddGlobalVariable add_global_variable = 30;
- TransformationAddGlobalUndef add_global_undef = 31;
- TransformationAddFunction add_function = 32;
- TransformationAddDeadBlock add_dead_block = 33;
- TransformationAddLocalVariable add_local_variable = 34;
- TransformationLoad load = 35;
- TransformationStore store = 36;
- TransformationFunctionCall function_call = 37;
- TransformationAccessChain access_chain = 38;
- TransformationEquationInstruction equation_instruction = 39;
- TransformationSwapCommutableOperands swap_commutable_operands = 40;
- TransformationPermuteFunctionParameters permute_function_parameters = 41;
- TransformationToggleAccessChainInstruction toggle_access_chain_instruction = 42;
- TransformationAddConstantNull add_constant_null = 43;
- TransformationComputeDataSynonymFactClosure compute_data_synonym_fact_closure = 44;
- TransformationAdjustBranchWeights adjust_branch_weights = 45;
- TransformationPushIdThroughVariable push_id_through_variable = 46;
- TransformationAddSpecConstantOp add_spec_constant_op = 47;
- TransformationReplaceLinearAlgebraInstruction replace_linear_algebra_instruction = 48;
- TransformationSwapConditionalBranchOperands swap_conditional_branch_operands = 49;
- TransformationPermutePhiOperands permute_phi_operands = 50;
- TransformationAddParameter add_parameter = 51;
- TransformationAddCopyMemory add_copy_memory = 52;
- TransformationInvertComparisonOperator invert_comparison_operator = 53;
- TransformationAddImageSampleUnusedComponents add_image_sample_unused_components = 54;
- TransformationReplaceParameterWithGlobal replace_parameter_with_global = 55;
- TransformationRecordSynonymousConstants record_synonymous_constants = 56;
- TransformationAddSynonym add_synonym = 57;
- TransformationAddRelaxedDecoration add_relaxed_decoration = 58;
- TransformationReplaceParamsWithStruct replace_params_with_struct = 59;
- TransformationReplaceCopyObjectWithStoreLoad replace_copy_object_with_store_load = 60;
- TransformationReplaceCopyMemoryWithLoadStore replace_copy_memory_with_load_store = 61;
- TransformationReplaceLoadStoreWithCopyMemory replace_load_store_with_copy_memory = 62;
- TransformationAddLoopPreheader add_loop_preheader = 63;
- TransformationMoveInstructionDown move_instruction_down = 64;
- TransformationMakeVectorOperationDynamic make_vector_operation_dynamic = 65;
- TransformationReplaceAddSubMulWithCarryingExtended replace_add_sub_mul_with_carrying_extended = 66;
- TransformationPropagateInstructionUp propagate_instruction_up = 67;
- TransformationCompositeInsert composite_insert = 68;
- TransformationInlineFunction inline_function = 69;
- TransformationAddOpPhiSynonym add_opphi_synonym = 70;
- TransformationMutatePointer mutate_pointer = 71;
- TransformationReplaceIrrelevantId replace_irrelevant_id = 72;
- TransformationReplaceOpPhiIdFromDeadPredecessor replace_opphi_id_from_dead_predecessor = 73;
- TransformationReplaceOpSelectWithConditionalBranch replace_opselect_with_conditional_branch = 74;
- TransformationDuplicateRegionWithSelection duplicate_region_with_selection = 75;
- TransformationFlattenConditionalBranch flatten_conditional_branch = 76;
- TransformationAddBitInstructionSynonym add_bit_instruction_synonym = 77;
- TransformationAddLoopToCreateIntConstantSynonym add_loop_to_create_int_constant_synonym = 78;
- TransformationWrapRegionInSelection wrap_region_in_selection = 79;
- TransformationAddEarlyTerminatorWrapper add_early_terminator_wrapper = 80;
- TransformationPropagateInstructionDown propagate_instruction_down = 81;
- TransformationReplaceBranchFromDeadBlockWithExit replace_branch_from_dead_block_with_exit = 82;
- TransformationWrapEarlyTerminatorInFunction wrap_early_terminator_in_function = 83;
- TransformationMergeFunctionReturns merge_function_returns = 84;
- TransformationExpandVectorReduction expand_vector_reduction = 85;
- TransformationSwapFunctionVariables swap_function_variables = 86;
- TransformationSwapTwoFunctions swap_two_functions = 87;
- TransformationWrapVectorSynonym wrap_vector_synonym = 88;
- // Add additional option using the next available number.
- }
- }
- // Keep transformation message types in alphabetical order:
- message TransformationAccessChain {
- // Adds an access chain instruction based on a given pointer and indices.
- // When accessing a struct, the corresponding indices must be 32-bit integer constants.
- // For any other composite, the indices can be any 32-bit integer, and the transformation
- // adds two instructions for each such index to clamp it to the bound, as follows:
- //
- // %t1 = OpULessThanEqual %bool %index %bound_minus_one
- // %t2 = OpSelect %int_type %t1 %index %bound_minus_one
- // Result id for the access chain
- uint32 fresh_id = 1;
- // The pointer from which the access chain starts
- uint32 pointer_id = 2;
- // Zero or more access chain indices
- repeated uint32 index_id = 3;
- // A descriptor for an instruction in a block before which the new
- // OpAccessChain instruction should be inserted
- InstructionDescriptor instruction_to_insert_before = 4;
- // Additional fresh ids, required to clamp index variables. A pair is needed
- // for each access to a non-struct composite.
- repeated UInt32Pair fresh_ids_for_clamping = 5;
- }
- message TransformationAddBitInstructionSynonym {
- // A transformation that adds synonyms for bit instructions by evaluating
- // each bit with the corresponding operation. There is a SPIR-V code example in the
- // header file of the transformation class that can help understand the transformation.
- // This transformation is only applicable if the described instruction has one of the following opcodes.
- // Supported:
- // OpBitwiseOr
- // OpBitwiseXor
- // OpBitwiseAnd
- // OpNot
- // To be supported in the future:
- // OpShiftRightLogical
- // OpShiftRightArithmetic
- // OpShiftLeftLogical
- // OpBitReverse
- // OpBitCount
- // The bit instruction result id.
- uint32 instruction_result_id = 1;
- // The fresh ids required to apply the transformation.
- repeated uint32 fresh_ids = 2;
- }
- message TransformationAddConstantBoolean {
- // Supports adding the constants true and false to a module, which may be
- // necessary in order to enable other transformations if they are not present.
- // Also, creates an IdIsIrrelevant fact about |fresh_id| if |is_irrelevant| is true.
- uint32 fresh_id = 1;
- bool is_true = 2;
- // If the constant should be marked as irrelevant.
- bool is_irrelevant = 3;
- }
- message TransformationAddConstantComposite {
- // Adds a constant of the given composite type to the module.
- // Also, creates an IdIsIrrelevant fact about |fresh_id| if
- // |is_irrelevant| is true.
- // Fresh id for the composite
- uint32 fresh_id = 1;
- // A composite type id
- uint32 type_id = 2;
- // Constituent ids for the composite
- repeated uint32 constituent_id = 3;
- // If the constant should be marked as irrelevant.
- bool is_irrelevant = 4;
- }
- message TransformationAddConstantNull {
- // Adds a null constant.
- // Id for the constant
- uint32 fresh_id = 1;
- // Type of the constant
- uint32 type_id = 2;
- }
- message TransformationAddConstantScalar {
- // Adds a constant of the given scalar type.
- // Also, creates an IdIsIrrelevant fact about
- // |fresh_id| if |is_irrelevant| is true.
- // Id for the constant
- uint32 fresh_id = 1;
- // Id for the scalar type of the constant
- uint32 type_id = 2;
- // Value of the constant
- repeated uint32 word = 3;
- // If the constant should be marked as irrelevant.
- bool is_irrelevant = 4;
- }
- message TransformationAddCopyMemory {
- // Adds an OpCopyMemory instruction into the module.
- // Creates either a global or a local variable (based on
- // |storage_class| field) to copy the target into.
- // OpCopyMemory will be inserted before this instruction.
- InstructionDescriptor instruction_descriptor = 1;
- // Fresh id to copy memory into.
- uint32 fresh_id = 2;
- // Source to copy memory from.
- uint32 source_id = 3;
- // Storage class for the target variable. Can be either Function or Private.
- uint32 storage_class = 4;
- // Result id for the variable's initializer operand. Its type must be equal to
- // variable's pointee type.
- uint32 initializer_id = 5;
- }
- message TransformationAddDeadBlock {
- // Adds a new block to the module that is statically reachable from an
- // existing block, but dynamically unreachable.
- // Fresh id for the dead block
- uint32 fresh_id = 1;
- // Id of an existing block terminated with OpBranch, such that this OpBranch
- // can be replaced with an OpBranchConditional to its exiting successor or
- // the dead block
- uint32 existing_block = 2;
- // Determines whether the condition associated with the OpBranchConditional
- // is true or false
- bool condition_value = 3;
- }
- message TransformationAddDeadBreak {
- // A transformation that turns a basic block that unconditionally branches to
- // its successor into a block that potentially breaks out of a structured
- // control flow construct, but in such a manner that the break cannot actually
- // be taken.
- // The block to break from
- uint32 from_block = 1;
- // The merge block to break to
- uint32 to_block = 2;
- // Determines whether the break condition is true or false
- bool break_condition_value = 3;
- // A sequence of ids suitable for extending OpPhi instructions as a result of
- // the new break edge
- repeated uint32 phi_id = 4;
- }
- message TransformationAddDeadContinue {
- // A transformation that turns a basic block appearing in a loop and that
- // unconditionally branches to its successor into a block that potentially
- // branches to the continue target of the loop, but in such a manner that the
- // continue branch cannot actually be taken.
- // The block to continue from
- uint32 from_block = 1;
- // Determines whether the continue condition is true or false
- bool continue_condition_value = 2;
- // A sequence of ids suitable for extending OpPhi instructions as a result of
- // the new break edge
- repeated uint32 phi_id = 3;
- }
- message TransformationAddEarlyTerminatorWrapper {
- // Adds a function to the module containing a single block with a single non-
- // label instruction that is either OpKill, OpUnreachable, or
- // OpTerminateInvocation. The purpose of this is to allow such instructions
- // to be subsequently replaced with wrapper functions, which can then enable
- // transformations (such as inlining) that are hard in the direct presence
- // of these instructions.
- // Fresh id for the function.
- uint32 function_fresh_id = 1;
- // Fresh id for the single basic block in the function.
- uint32 label_fresh_id = 2;
- // One of OpKill, OpUnreachable, OpTerminateInvocation. If additional early
- // termination instructions are added to SPIR-V they should also be handled
- // here.
- uint32 opcode = 3;
- }
- message TransformationAddFunction {
- // Adds a SPIR-V function to the module.
- // The series of instructions that comprise the function.
- repeated Instruction instruction = 1;
- // True if and only if the given function should be made livesafe (see
- // FactFunctionIsLivesafe for definition).
- bool is_livesafe = 2;
- // Fresh id for a new variable that will serve as a "loop limiter" for the
- // function; only relevant if |is_livesafe| holds.
- uint32 loop_limiter_variable_id = 3;
- // Id of an existing unsigned integer constant providing the maximum value
- // that the loop limiter can reach before the loop is broken from; only
- // relevant if |is_livesafe| holds.
- uint32 loop_limit_constant_id = 4;
- // Fresh ids for each loop in the function that allow the loop limiter to be
- // manipulated; only relevant if |is_livesafe| holds.
- repeated LoopLimiterInfo loop_limiter_info = 5;
- // Id of an existing global value with the same return type as the function
- // that can be used to replace OpKill and OpReachable instructions with
- // ReturnValue instructions. Ignored if the function has void return type.
- // Only relevant if |is_livesafe| holds.
- uint32 kill_unreachable_return_value_id = 6;
- // A mapping (represented as a sequence) from every access chain result id in
- // the function to the ids required to clamp its indices to ensure they are in
- // bounds; only relevant if |is_livesafe| holds.
- repeated AccessChainClampingInfo access_chain_clamping_info = 7;
- }
- message TransformationAddGlobalUndef {
- // Adds an undefined value of a given type to the module at global scope.
- // Fresh id for the undefined value
- uint32 fresh_id = 1;
- // The type of the undefined value
- uint32 type_id = 2;
- }
- message TransformationAddGlobalVariable {
- // Adds a global variable of the given type to the module, with Private or
- // Workgroup storage class, and optionally (for the Private case) with an
- // initializer.
- // Fresh id for the global variable
- uint32 fresh_id = 1;
- // The type of the global variable
- uint32 type_id = 2;
- uint32 storage_class = 3;
- // Initial value of the variable
- uint32 initializer_id = 4;
- // True if and only if the behaviour of the module should not depend on the
- // value of the variable, in which case stores to the variable can be
- // performed in an arbitrary fashion.
- bool value_is_irrelevant = 5;
- }
- message TransformationAddImageSampleUnusedComponents {
- // A transformation that adds unused components to an image sample coordinate.
- // An vector id with the original coordinate and the unused components.
- uint32 coordinate_with_unused_components_id = 1;
- // A descriptor for an image sample instruction.
- InstructionDescriptor instruction_descriptor = 2;
- }
- message TransformationAddLocalVariable {
- // Adds a local variable of the given type (which must be a pointer with
- // Function storage class) to the given function, initialized to the given
- // id.
- // Fresh id for the local variable
- uint32 fresh_id = 1;
- // The type of the local variable
- uint32 type_id = 2;
- // The id of the function to which the local variable should be added
- uint32 function_id = 3;
- // Initial value of the variable
- uint32 initializer_id = 4;
- // True if and only if the behaviour of the module should not depend on the
- // value of the variable, in which case stores to the variable can be
- // performed in an arbitrary fashion.
- bool value_is_irrelevant = 5;
- }
- message TransformationAddLoopPreheader {
- // A transformation that adds a loop preheader block before the given loop header.
- // The id of the loop header block
- uint32 loop_header_block = 1;
- // A fresh id for the preheader block
- uint32 fresh_id = 2;
- // Fresh ids for splitting the OpPhi instructions in the header.
- // A new OpPhi instruction in the preheader is needed for each OpPhi instruction in the header,
- // if the header has more than one predecessor outside of the loop.
- // This allows turning instructions of the form:
- //
- // %loop_header_block = OpLabel
- // %id1 = OpPhi %type %val1 %pred1_id %val2 %pred2_id %val3 %backedge_block_id
- //
- // into:
- // %fresh_id = OpLabel
- // %phi_id1 = OpPhi %type %val1 %pred1_id %val2 %pred2_id
- // OpBranch %header_id
- // %loop_header_block = OpLabel
- // %id1 = OpPhi %type %phi_id1 %fresh_id %val3 %backedge_block_id
- repeated uint32 phi_id = 3;
- }
- message TransformationAddLoopToCreateIntConstantSynonym {
- // A transformation that uses a loop to create a synonym for an integer
- // constant C (scalar or vector) using an initial value I, a step value S and
- // a number of iterations N such that C = I - N * S. For each iteration, S is
- // subtracted from the total.
- // The loop can be made up of one or two blocks, and it is inserted before a
- // block with a single predecessor. In the one-block case, it is of the form:
- //
- // %loop_id = OpLabel
- // %ctr_id = OpPhi %int %int_0 %pred %incremented_ctr_id %loop_id
- // %temp_id = OpPhi %type_of_I %I %pred %eventual_syn_id %loop_id
- // %eventual_syn_id = OpISub %type_of_I %temp_id %step_val_id
- // %incremented_ctr_id = OpIAdd %int %ctr_id %int_1
- // %cond_id = OpSLessThan %bool %incremented_ctr_id %num_iterations_id
- // OpLoopMerge %block_after_loop_id %loop_id None
- // OpBranchConditional %cond_id %loop_id %block_after_loop_id
- //
- // A new OpPhi instruction is then added to %block_after_loop_id, as follows:
- //
- // %block_after_loop_id = OpLabel
- // %syn_id = OpPhi %type_of_I %eventual_syn_id %loop_id
- //
- // This can be translated, assuming that N > 0, to:
- // int syn = I;
- // for (int ctr = 0; ctr < N; ctr++) syn = syn - S;
- //
- // All existing OpPhi instructions in %block_after_loop_id are also updated
- // to reflect the fact that its predecessor is now %loop_id.
- // The following are existing ids.
- // The id of the integer constant C that we want a synonym of.
- uint32 constant_id = 1;
- // The id of the initial value integer constant I.
- uint32 initial_val_id = 2;
- // The id of the step value integer constant S.
- uint32 step_val_id = 3;
- // The id of the integer scalar constant, its value being the number of
- // iterations N.
- uint32 num_iterations_id = 4;
- // The label id of the block before which the loop must be inserted.
- uint32 block_after_loop_id = 5;
- // The following are fresh ids.
- // A fresh id for the synonym.
- uint32 syn_id = 6;
- // A fresh id for the label of the loop,
- uint32 loop_id = 7;
- // A fresh id for the counter.
- uint32 ctr_id = 8;
- // A fresh id taking the value I - S * ctr at the ctr-th iteration.
- uint32 temp_id = 9;
- // A fresh id taking the value I - S * (ctr + 1) at the ctr-th iteration, and
- // thus I - S * N at the last iteration.
- uint32 eventual_syn_id = 10;
- // A fresh id for the incremented counter.
- uint32 incremented_ctr_id = 11;
- // A fresh id for the loop condition.
- uint32 cond_id = 12;
- // The instructions in the loop can also be laid out in two basic blocks, as follows:
- //
- // %loop_id = OpLabel
- // %ctr_id = OpPhi %int %int_0 %pred %incremented_ctr_id %loop_id
- // %temp_id = OpPhi %type_of_I %I %pred %eventual_syn_id %loop_id
- // OpLoopMerge %block_after_loop_id %additional_block_id None
- // OpBranch %additional_block_id
- //
- // %additional_block_id = OpLabel
- // %eventual_syn_id = OpISub %type_of_I %temp_id %step_val_id
- // %incremented_ctr_id = OpIAdd %int %ctr_id %int_1
- // %cond_id = OpSLessThan %bool %incremented_ctr_id %num_iterations_id
- // OpBranchConditional %cond_id %loop_id %block_after_loop_id
- // A fresh id for the additional block. If this is 0, it means that only one
- // block is to be created.
- uint32 additional_block_id = 13;
- }
- message TransformationAddNoContractionDecoration {
- // Applies OpDecorate NoContraction to the given result id
- // Result id to be decorated
- uint32 result_id = 1;
- }
- message TransformationAddOpPhiSynonym {
- // Adds an OpPhi instruction at the start of a block with n predecessors (pred_1, pred_2, ..., pred_n)
- // and n related ids (id_1, id_2, ..., id_n) which are pairwise synonymous.
- // The instruction will be of the form:
- // %fresh_id = OpPhi %type %id_1 %pred_1 %id_2 %pred_2 ... %id_n %pred_n
- // and fresh_id will be recorded as being synonymous with all the other ids.
- // Label id of the block
- uint32 block_id = 1;
- // Pairs (pred_i, id_i)
- repeated UInt32Pair pred_to_id = 2;
- // Fresh id for the new instruction
- uint32 fresh_id = 3;
- }
- message TransformationAddParameter {
- // Adds a new parameter into the function.
- // Result id of the function to add parameters to.
- uint32 function_id = 1;
- // Fresh id for a new parameter.
- uint32 parameter_fresh_id = 2;
- // Type id for a new parameter.
- uint32 parameter_type_id = 3;
- // A map that maps from the OpFunctionCall id to the id that will be passed as the new
- // parameter at that call site. It must have the same type as that of the new parameter.
- repeated UInt32Pair call_parameter_ids = 4;
- // A fresh id for a new function type. This might not be used
- // if a required function type already exists or if we can change
- // the old function type.
- uint32 function_type_fresh_id = 5;
- }
- message TransformationAddRelaxedDecoration {
- // Applies OpDecorate RelaxedPrecision to the given result id
- // Result id to be decorated
- uint32 result_id = 1;
- }
- message TransformationAddSpecConstantOp {
- // Adds OpSpecConstantOp into the module.
- // Result id for the new instruction.
- uint32 fresh_id = 1;
- // Type id for the new instruction.
- uint32 type_id = 2;
- // Opcode operand of the OpSpecConstantOp instruction.
- uint32 opcode = 3;
- // Operands of the |opcode| instruction.
- repeated InstructionOperand operand = 4;
- }
- message TransformationAddSynonym {
- // Adds a |synonymous_instruction| before |insert_before| instruction with
- // and creates a fact that |result_id| and the result id of |synonymous_instruction|
- // are synonymous.
- // Result id of the first synonym.
- uint32 result_id = 1;
- // Type of the synonym to apply. Some types might produce instructions
- // with commutative operands. Such types do not specify the order of the
- // operands since we have a special transformation to swap commutable operands.
- //
- // TODO(https://github.com/KhronosGroup/SPIRV-Tools/issues/3499):
- // Consider adding more types here.
- enum SynonymType {
- // New synonym is derived by adding zero to the |result_id|.
- ADD_ZERO = 0;
- // New synonym is derived by subtracting zero from the |result_id|.
- SUB_ZERO = 1;
- // New synonym is derived by multiplying |result_id| by one.
- MUL_ONE = 2;
- // New synonym is derived by applying OpCopyObject instruction to |result_id|.
- COPY_OBJECT = 3;
- // New synonym is derived by applying OpLogicalOr to |result_id| with the second
- // operand being 'false'.
- LOGICAL_OR = 4;
- // New synonym is derived by applying OpLogicalAnd to |result_id| with the second
- // operand being 'true'.
- LOGICAL_AND = 5;
- // New synonym is derived by applying OpBitwiseOr to |result_id| with the second
- // operand being 0 taken with the same bit length as |result_id|
- BITWISE_OR = 6;
- // New synonym is derived by applying OpBitwiseXor to |result_id| with the second
- // operand being 0 taken with the same bit length as |result_id|
- BITWISE_XOR = 7;
- }
- // Type of the synonym to create. See SynonymType for more details.
- SynonymType synonym_type = 2;
- // Fresh result id for a created synonym.
- uint32 synonym_fresh_id = 3;
- // An instruction to insert a new synonym before.
- InstructionDescriptor insert_before = 4;
- }
- message TransformationAddTypeArray {
- // Adds an array type of the given element type and size to the module
- // Fresh id for the array type
- uint32 fresh_id = 1;
- // The array's element type
- uint32 element_type_id = 2;
- // The array's size
- uint32 size_id = 3;
- }
- message TransformationAddTypeBoolean {
- // Adds OpTypeBool to the module
- // Id to be used for the type
- uint32 fresh_id = 1;
- }
- message TransformationAddTypeFloat {
- // Adds OpTypeFloat to the module with the given width
- // Id to be used for the type
- uint32 fresh_id = 1;
- // Floating-point width
- uint32 width = 2;
- }
- message TransformationAddTypeFunction {
- // Adds a function type to the module
- // Fresh id for the function type
- uint32 fresh_id = 1;
- // The function's return type
- uint32 return_type_id = 2;
- // The function's argument types
- repeated uint32 argument_type_id = 3;
- }
- message TransformationAddTypeInt {
- // Adds OpTypeInt to the module with the given width and signedness
- // Id to be used for the type
- uint32 fresh_id = 1;
- // Integer width
- uint32 width = 2;
- // True if and only if this is a signed type
- bool is_signed = 3;
- }
- message TransformationAddTypeMatrix {
- // Adds a matrix type to the module
- // Fresh id for the matrix type
- uint32 fresh_id = 1;
- // The matrix's column type, which must be a floating-point vector (as per
- // the "data rules" in the SPIR-V specification).
- uint32 column_type_id = 2;
- // The matrix's column count
- uint32 column_count = 3;
- }
- message TransformationAddTypePointer {
- // Adds OpTypePointer to the module, with the given storage class and base
- // type
- // Id to be used for the type
- uint32 fresh_id = 1;
- // Pointer storage class
- uint32 storage_class = 2;
- // Id of the base type for the pointer
- uint32 base_type_id = 3;
- }
- message TransformationAddTypeStruct {
- // Adds a struct type to the module
- // Fresh id for the struct type
- uint32 fresh_id = 1;
- // The struct's member types
- repeated uint32 member_type_id = 3;
- }
- message TransformationAddTypeVector {
- // Adds a vector type to the module
- // Fresh id for the vector type
- uint32 fresh_id = 1;
- // The vector's component type
- uint32 component_type_id = 2;
- // The vector's component count
- uint32 component_count = 3;
- }
- message TransformationAdjustBranchWeights {
- // A transformation that adjusts the branch weights
- // of a branch conditional instruction.
- // A descriptor for a branch conditional instruction.
- InstructionDescriptor instruction_descriptor = 1;
- // Branch weights of a branch conditional instruction.
- UInt32Pair branch_weights = 2;
- }
- message TransformationCompositeConstruct {
- // A transformation that introduces an OpCompositeConstruct instruction to
- // make a composite object.
- // Id of the type of the composite that is to be constructed
- uint32 composite_type_id = 1;
- // Ids of the objects that will form the components of the composite
- repeated uint32 component = 2;
- // A descriptor for an instruction in a block before which the new
- // OpCompositeConstruct instruction should be inserted
- InstructionDescriptor instruction_to_insert_before = 3;
- // A fresh id for the composite object
- uint32 fresh_id = 4;
- }
- message TransformationCompositeExtract {
- // A transformation that adds an instruction to extract an element from a
- // composite.
- // A descriptor for an instruction in a block before which the new
- // OpCompositeExtract instruction should be inserted
- InstructionDescriptor instruction_to_insert_before = 1;
- // Result id for the extract operation.
- uint32 fresh_id = 2;
- // Id of the composite from which data is to be extracted.
- uint32 composite_id = 3;
- // Indices that indicate which part of the composite should be extracted.
- repeated uint32 index = 4;
- }
- message TransformationCompositeInsert {
- // A transformation that adds an instruction OpCompositeInsert which creates
- // a new composite from an existing composite, with an element inserted.
- // A descriptor for an instruction before which the new instruction
- // OpCompositeInsert should be inserted.
- InstructionDescriptor instruction_to_insert_before = 1;
- // Result id of the inserted OpCompositeInsert instruction.
- uint32 fresh_id = 2;
- // Id of the composite used as the basis for the insertion.
- uint32 composite_id = 3;
- // Id of the object to be inserted.
- uint32 object_id = 4;
- // Indices that indicate which part of the composite should be inserted into.
- repeated uint32 index = 5;
- }
- message TransformationComputeDataSynonymFactClosure {
- // A transformation that impacts the fact manager only, forcing a computation
- // of the closure of data synonym facts, so that e.g. if the components of
- // vectors v and w are known to be pairwise synonymous, it is deduced that v
- // and w are themselves synonymous.
- // When searching equivalence classes for implied facts, equivalence classes
- // larger than this size will be skipped.
- uint32 maximum_equivalence_class_size = 1;
- }
- message TransformationDuplicateRegionWithSelection {
- // A transformation that inserts a conditional statement with a boolean expression
- // of arbitrary value and duplicates a given single-entry, single-exit region, so
- // that it is present in each conditional branch and will be executed regardless
- // of which branch will be taken.
- // Fresh id for a label of the new entry block.
- uint32 new_entry_fresh_id = 1;
- // Id for a boolean expression.
- uint32 condition_id = 2;
- // Fresh id for a label of the merge block of the conditional.
- uint32 merge_label_fresh_id = 3;
- // Block id of the entry block of the original region.
- uint32 entry_block_id = 4;
- // Block id of the exit block of the original region.
- uint32 exit_block_id = 5;
- // Map that maps from a label in the original region to the corresponding label
- // in the duplicated region.
- repeated UInt32Pair original_label_to_duplicate_label = 6;
- // Map that maps from a result id in the original region to the corresponding
- // result id in the duplicated region.
- repeated UInt32Pair original_id_to_duplicate_id = 7;
- // Map that maps from a result id in the original region to the result id of the
- // corresponding OpPhi instruction.
- repeated UInt32Pair original_id_to_phi_id = 8;
- }
- message TransformationEquationInstruction {
- // A transformation that adds an instruction to the module that defines an
- // equation between its result id and input operand ids, such that the
- // equation is guaranteed to hold at any program point where all ids involved
- // are available (i.e. at any program point dominated by the instruction).
- // The result id of the new instruction
- uint32 fresh_id = 1;
- // The instruction's opcode
- uint32 opcode = 2;
- // The input operands to the instruction
- repeated uint32 in_operand_id = 3;
- // A descriptor for an instruction in a block before which the new
- // instruction should be inserted
- InstructionDescriptor instruction_to_insert_before = 4;
- }
- message TransformationExpandVectorReduction {
- // A transformation that adds synonyms for OpAny and OpAll instructions by
- // evaluating each vector component with the corresponding logical operation.
- // There is a SPIR-V code example in the header file of the transformation
- // class that can help understand the transformation.
- // The OpAny or OpAll instruction result id.
- uint32 instruction_result_id = 1;
- // The fresh ids required to apply the transformation.
- repeated uint32 fresh_ids = 2;
- }
- message TransformationFlattenConditionalBranch {
- // A transformation that takes a selection construct with a header
- // containing an OpBranchConditional instruction and flattens it.
- // For example, something of the form:
- //
- // %1 = OpLabel
- // [header instructions]
- // OpSelectionMerge %4 None
- // OpBranchConditional %cond %2 %3
- // %2 = OpLabel
- // [true branch instructions]
- // OpBranch %4
- // %3 = OpLabel
- // [false branch instructions]
- // OpBranch %4
- // %4 = OpLabel
- // ...
- //
- // becomes:
- //
- // %1 = OpLabel
- // [header instructions]
- // OpBranch %2
- // %2 = OpLabel
- // [true branch instructions]
- // OpBranch %3
- // %3 = OpLabel
- // [false branch instructions]
- // OpBranch %4
- // %4 = OpLabel
- // ...
- //
- // If all of the instructions in the true or false branches have
- // no side effects, this is semantics-preserving.
- // Side-effecting instructions will instead be enclosed by smaller
- // conditionals. For more details, look at the definition for the
- // SideEffectWrapperInfo message.
- //
- // Nested conditionals or loops are not supported. The false branch
- // could also be executed before the true branch, depending on the
- // |true_branch_first| field.
- // The label id of the header block
- uint32 header_block_id = 1;
- // A boolean field deciding the order in which the original branches
- // will be laid out: the true branch will be laid out first iff this
- // field is true.
- bool true_branch_first = 2;
- // If the convergence block contains an OpPhi with bvec2 result type, it may
- // be necessary to introduce a bvec2 with the selection construct's condition
- // in both components in order to turn the OpPhi into an OpSelect. This
- // this field provides a fresh id for an OpCompositeConstruct instruction for
- // this purpose. It should be set to 0 if no such instruction is required.
- uint32 fresh_id_for_bvec2_selector = 3;
- // The same as |fresh_id_for_bvec2_selector| but for the bvec3 case.
- uint32 fresh_id_for_bvec3_selector = 4;
- // The same as |fresh_id_for_bvec2_selector| but for the bvec4 case.
- uint32 fresh_id_for_bvec4_selector = 5;
- // A list of instructions with side effects, which must be enclosed
- // inside smaller conditionals before flattening the main one, and
- // the corresponding fresh ids and module ids needed.
- repeated SideEffectWrapperInfo side_effect_wrapper_info = 6;
- }
- message TransformationFunctionCall {
- // A transformation that introduces an OpFunctionCall instruction. The call
- // must not make the module's call graph cyclic. Beyond that, if the call
- // is in a dead block it can be to any function with arbitrary suitably-typed
- // arguments; otherwise it must be to a livesafe function, with injected
- // variables as pointer arguments and arbitrary non-pointer arguments.
- // A fresh id for the result of the call
- uint32 fresh_id = 1;
- // Id of the function to be called
- uint32 callee_id = 2;
- // Ids for arguments to the function
- repeated uint32 argument_id = 3;
- // A descriptor for an instruction in a block before which the new
- // OpFunctionCall instruction should be inserted
- InstructionDescriptor instruction_to_insert_before = 4;
- }
- message TransformationInlineFunction {
- // This transformation inlines a function by mapping the function instructions to fresh ids.
- // Result id of the function call instruction.
- uint32 function_call_id = 1;
- // For each result id defined by the called function,
- // this map provides an associated fresh id that can
- // be used in the inlined version of the function call.
- repeated UInt32Pair result_id_map = 2;
- }
- message TransformationInvertComparisonOperator {
- // For some instruction with result id |operator_id| that
- // represents a binary comparison operator (e.g. <, >, <=), this transformation
- // will replace that instruction's result id with |fresh_id|,
- // invert the opcode (< will become >=) and insert OpLogicalNot
- // instruction with result id |operator_id| below.
- // Result id of the instruction to invert.
- uint32 operator_id = 1;
- // Fresh id that will be used by the operator after the inversion.
- uint32 fresh_id = 2;
- }
- message TransformationLoad {
- // Transformation that adds an OpLoad or OpAtomicLoad instruction from a pointer into an id.
- // The result of the load instruction.
- uint32 fresh_id = 1;
- // The pointer to be loaded from.
- uint32 pointer_id = 2;
- // True if and only if the load should be atomic.
- bool is_atomic = 3;
- // The memory scope for the atomic load. Ignored unless |is_atomic| is true.
- uint32 memory_scope_id = 4;
- // The memory semantics for the atomic load. Ignored unless |is_atomic| is true.
- uint32 memory_semantics_id = 5;
- // A descriptor for an instruction in a block before which the new OpLoad
- // instruction should be inserted.
- InstructionDescriptor instruction_to_insert_before = 6;
- }
- message TransformationMakeVectorOperationDynamic {
- // A transformation that replaces the OpCompositeExtract and OpCompositeInsert
- // instructions with the OpVectorExtractDynamic and OpVectorInsertDynamic instructions.
- // The composite instruction result id.
- uint32 instruction_result_id = 1;
- // The OpCompositeExtract/Insert instructions accept integer literals as indices to the composite object.
- // However, the OpVectorInsert/ExtractDynamic instructions require its single index to be an integer instruction.
- // This is the result id of the integer instruction.
- uint32 constant_index_id = 2;
- }
- message TransformationMergeBlocks {
- // A transformation that merges a block with its predecessor.
- // The id of the block that is to be merged with its predecessor; the merged
- // block will have the *predecessor's* id.
- uint32 block_id = 1;
- }
- message TransformationMergeFunctionReturns {
- // A transformation that modifies a function so that it does not return early,
- // so it only has one return statement (ignoring unreachable blocks).
- //
- // The function is enclosed inside an outer loop, that is only executed once,
- // and whose merge block is the new return block of the function.
- //
- // Each return instruction is replaced by:
- // OpBranch %innermost_loop_merge
- // where %innermost_loop_merge is the innermost loop containing the return
- // instruction.
- //
- // Each merge block whose associated loop contains return instructions is
- // changed so that it branches to the merge block of the loop containing it,
- // as explained in the comments to the ReturnMergingInfo message.
- //
- // The new return block (the merge block of the new outer loop) will be of
- // the following form (if the return type is not void):
- // %outer_return_id = OpLabel
- // %return_val_id = OpPhi %return_type %val1 %block_1 %val2 %block_2 ...
- // OpReturnValue %return_val_id
- // where %block_k is either a return block that, in the original function, is
- // outside of any loops, or the merge block of a loop that contains return
- // instructions and is not, originally, nested inside another loop, and
- // %block_k is the corresponding return value.
- // If the function has void type, there will be no OpPhi instruction and the
- // last instruction will be OpReturn.
- // The id of the function to which the transformation is being applied.
- uint32 function_id = 1;
- // A fresh id for the header of the new outer loop.
- uint32 outer_header_id = 2;
- // A fresh id for an unreachable continue construct for the new outer loop.
- uint32 unreachable_continue_id = 7;
- // A fresh id for the new return block of the function,
- // i.e. the merge block of the new outer loop.
- uint32 outer_return_id = 3;
- // A fresh id for the value that will be returned.
- // This is ignored if the function has void return type.
- uint32 return_val_id = 4;
- // An existing id of the same type as the return value, which is
- // available to use at the end of the entry block.
- // This is ignored if the function has void return type or if no
- // loops in the function contain a return instruction.
- // If the function is not void, the transformation will add an
- // OpPhi instruction to each merge block whose associated loop
- // contains at least a return instruction. The value associated
- // with existing predecessors from which the function cannot be
- // returning will be this id, used as a placeholder.
- uint32 any_returnable_val_id = 5;
- // The information needed to modify the merge blocks of
- // loops containing return instructions.
- repeated ReturnMergingInfo return_merging_info = 6;
- }
- message TransformationMoveBlockDown {
- // A transformation that moves a basic block to be one position lower in
- // program order.
- // The id of the block to move down.
- uint32 block_id = 1;
- }
- message TransformationMoveInstructionDown {
- // Swaps |instruction| with the next instruction in the block.
- // The instruction to move down.
- InstructionDescriptor instruction = 1;
- }
- message TransformationMutatePointer {
- // Backs up value of the pointer, writes into the pointer and
- // restores the original value.
- // Result id of the pointer instruction to mutate.
- uint32 pointer_id = 1;
- // Fresh id for the OpLoad instruction.
- uint32 fresh_id = 2;
- // Instruction to insert backup, mutation and restoration code before.
- InstructionDescriptor insert_before = 3;
- }
- message TransformationOutlineFunction {
- // A transformation that outlines a single-entry single-exit region of a
- // control flow graph into a separate function, and replaces the region with
- // a call to that function.
- // Id of the entry block of the single-entry single-exit region to be outlined
- uint32 entry_block = 1;
- // Id of the exit block of the single-entry single-exit region to be outlined
- uint32 exit_block = 2;
- // Id of a struct that will store the return values of the new function
- uint32 new_function_struct_return_type_id = 3;
- // A fresh id for the type of the outlined function
- uint32 new_function_type_id = 4;
- // A fresh id for the outlined function itself
- uint32 new_function_id = 5;
- // A fresh id to represent the block in the outlined function that represents
- // the first block of the outlined region.
- uint32 new_function_region_entry_block = 6;
- // A fresh id for the result of the OpFunctionCall instruction that will call
- // the outlined function
- uint32 new_caller_result_id = 7;
- // A fresh id to capture the return value of the outlined function - the
- // argument to OpReturn
- uint32 new_callee_result_id = 8;
- // Ids defined outside the region and used inside the region will become
- // parameters to the outlined function. This is a mapping from used ids to
- // fresh parameter ids.
- repeated UInt32Pair input_id_to_fresh_id = 9;
- // Ids defined inside the region and used outside the region will become
- // fresh ids defined by the outlined function, which get copied into the
- // function's struct return value and then copied into their destination ids
- // by the caller. This is a mapping from original ids to corresponding fresh
- // ids.
- repeated UInt32Pair output_id_to_fresh_id = 10;
- }
- message TransformationPermuteFunctionParameters {
- // A transformation that, given a non-entry-point function taking n
- // parameters and a permutation of the set [0, n-1]:
- // - Introduces a new function type that is the same as the original
- // function's type but with the order of arguments permuted
- // (only if it doesn't already exist)
- // - Changes the type of the function to this type
- // - Adjusts all calls to the function so that their arguments are permuted
- // Function, whose parameters will be permuted
- uint32 function_id = 1;
- // Fresh id for a new type of the function. This might not be used
- // if a required function type already exists or if we can change
- // the old function type.
- uint32 function_type_fresh_id = 2;
- // An array of size |n|, where |n| is a number of arguments to a function
- // with |function_id|. For each i: 0 <= permutation[i] < n.
- //
- // i-th element of this array contains a position for an i-th
- // function's argument (i.e. i-th argument will be permutation[i]-th
- // after running this transformation)
- repeated uint32 permutation = 3;
- }
- message TransformationPermutePhiOperands {
- // Permutes operands of some OpPhi instruction.
- // Result id of the instruction to apply the transformation to.
- uint32 result_id = 1;
- // A sequence of numbers in the range [0, n/2 - 1] where |n| is the number
- // of operands of the OpPhi instruction with |result_id|.
- repeated uint32 permutation = 2;
- }
- message TransformationPropagateInstructionDown {
- // Propagates an instruction from |block_id| into its successors.
- // Concretely, the transformation clones the propagated instruction
- // into some of the successors of |block_id| and removes the original
- // instruction. Additionally, an OpPhi instruction may be added to make sure
- // that the transformation can be applied in various scenarios.
- //
- // Note that the instruction might not be propagated down into every successor
- // of |block_id| since it might make the module invalid.
- // Id of the block to propagate an instruction from. The decision on what
- // instruction to propagate is made based on whether the instruction interacts
- // with memory, whether that instruction is used in its block etc (see the
- // transformation class for more details).
- uint32 block_id = 1;
- // A fresh id for an OpPhi instruction. This might not be used by the
- // transformation since an OpPhi instruction is created only if needed
- // (e.g. an instruction is propagated into divergent blocks).
- uint32 phi_fresh_id = 2;
- // A map from the id of some successor of the |block_id| to the fresh id.
- // The map contains a fresh id for at least every successor of the |block_id|.
- // Every fresh id in the map corresponds to the result id of the clone,
- // propagated into the corresponding successor block. This transformation
- // might use overflow ids if they are available and this field doesn't account
- // for every successor of |block_id|.
- repeated UInt32Pair successor_id_to_fresh_id = 3;
- }
- message TransformationPropagateInstructionUp {
- // Propagates an instruction in the block into the block's predecessors.
- // Concretely, this transformation clones some particular instruction from
- // the |block_id| into every block's predecessor and replaces the original
- // instruction with OpPhi. Take a look at the transformation class to learn
- // more about how we choose what instruction to propagate.
- // Id of the block to propagate an instruction from.
- uint32 block_id = 1;
- // A map from the id of some predecessor of the |block_id| to the fresh id.
- // The map contains a fresh id for at least every predecessor of the |block_id|.
- // The instruction is propagated by creating a number of clones - one clone for
- // each predecessor. Fresh ids from this field are used as result ids of cloned
- // instructions.
- repeated UInt32Pair predecessor_id_to_fresh_id = 2;
- }
- message TransformationPushIdThroughVariable {
- // A transformation that makes |value_synonym_id| and |value_id| to be
- // synonymous by storing |value_id| into |variable_id| and
- // loading |variable_id| to |value_synonym_id|.
- // The value to be stored.
- uint32 value_id = 1;
- // A fresh id for the result of the load instruction.
- uint32 value_synonym_id = 2;
- // A fresh id for the variable to be stored to.
- uint32 variable_id = 3;
- // Constant to initialize the variable from.
- uint32 initializer_id = 4;
- // The variable storage class (global or local).
- uint32 variable_storage_class = 5;
- // A descriptor for an instruction which the new OpStore
- // and OpLoad instructions might be inserted before.
- InstructionDescriptor instruction_descriptor = 6;
- }
- message TransformationRecordSynonymousConstants {
- // A transformation that, given the IDs to two synonymous constants,
- // records the fact that they are synonymous. The module is not changed.
- // Two constants are synonymous if:
- // - they have the same type (ignoring the presence of integer sign)
- // - they have the same opcode (one of OpConstant, OpConstantTrue,
- // OpConstantFalse, OpConstantNull)
- // - they have the same value
- // If the types are the same, OpConstantNull is equivalent to
- // OpConstantFalse or OpConstant with value zero.
- // The id of a constant
- uint32 constant1_id = 1;
- // The id of the synonym
- uint32 constant2_id = 2;
- }
- message TransformationReplaceAddSubMulWithCarryingExtended {
- // Replaces OpIAdd with OpIAddCarry, OpISub with OpISubBorrow, OpIMul
- // with OpUMulExtended or OpSMulExtended (depending on the signedness
- // of the operands) and stores the result into a |struct_fresh_id|.
- // In the original instruction the result type id and the type ids of
- // the operands must be the same. Then the transformation extracts
- // the first element of the result into the original |result_id|.
- // This value is the same as the result of the original instruction.
- // The fresh id of the intermediate result.
- uint32 struct_fresh_id = 1;
- // The result id of the original instruction.
- uint32 result_id = 2;
- }
- message TransformationReplaceBranchFromDeadBlockWithExit {
- // Given a dead block that ends with OpBranch, replaces OpBranch with an
- // "exit" instruction; one of OpReturn/OpReturnValue, OpKill (in a fragment
- // shader) or OpUnreachable.
- // The dead block whose terminator is to be replaced.
- uint32 block_id = 1;
- // The opcode of the new terminator.
- uint32 opcode = 2;
- // Ignored unless opcode is OpReturnValue, in which case this field provides
- // a suitable result id to be returned.
- uint32 return_value_id = 3;
- }
- message TransformationReplaceParameterWithGlobal {
- // Removes parameter with result id |parameter_id| from its function
- // and creates a global variable to pass its value to the function instead.
- // Fresh id for a new function type. This might not be used if a required
- // function type already exists or if we can change the old function type.
- uint32 function_type_fresh_id = 2;
- // Result id of the OpFunctionParameter instruction to remove.
- uint32 parameter_id = 3;
- // Fresh id of a global variable used to pass parameter's value to the function.
- uint32 global_variable_fresh_id = 4;
- }
- message TransformationReplaceBooleanConstantWithConstantBinary {
- // A transformation to capture replacing a use of a boolean constant with
- // binary operation on two constant values
- // A descriptor for the boolean constant id we would like to replace
- IdUseDescriptor id_use_descriptor = 1;
- // Id for the constant to be used on the LHS of the comparison
- uint32 lhs_id = 2;
- // Id for the constant to be used on the RHS of the comparison
- uint32 rhs_id = 3;
- // Opcode for binary operator
- uint32 opcode = 4;
- // Id that will store the result of the binary operation instruction
- uint32 fresh_id_for_binary_operation = 5;
- }
- message TransformationReplaceConstantWithUniform {
- // Replaces a use of a constant id with the result of a load from an
- // element of uniform buffer known to hold the same value as the constant
- // A descriptor for the id we would like to replace
- IdUseDescriptor id_use_descriptor = 1;
- // Uniform descriptor to identify which uniform value to choose
- UniformBufferElementDescriptor uniform_descriptor = 2;
- // Id that will store the result of an access chain
- uint32 fresh_id_for_access_chain = 3;
- // Id that will store the result of a load
- uint32 fresh_id_for_load = 4;
- }
- message TransformationReplaceCopyMemoryWithLoadStore {
- // A transformation that replaces instructions OpCopyMemory with loading
- // the source variable to an intermediate value and storing this value into the
- // target variable of the original OpCopyMemory instruction.
- // The intermediate value.
- uint32 fresh_id = 1;
- // The instruction descriptor to OpCopyMemory. It is necessary, because
- // OpCopyMemory doesn't have a result id.
- InstructionDescriptor copy_memory_instruction_descriptor = 2;
- }
- message TransformationReplaceCopyObjectWithStoreLoad {
- // A transformation that replaces instruction OpCopyObject with
- // storing into a new variable and immediately loading from this
- // variable to |result_id| of the original OpCopyObject instruction.
- // The result id of initial OpCopyObject instruction
- uint32 copy_object_result_id = 1;
- // A fresh id for the variable to be stored to.
- uint32 fresh_variable_id = 2;
- // The variable storage class (Function or Private).
- uint32 variable_storage_class = 3;
- // Constant to initialize the variable with.
- uint32 variable_initializer_id = 4;
- }
- message TransformationReplaceIdWithSynonym {
- // Replaces a use of an id with an id that is known to be synonymous, e.g.
- // because it was obtained via applying OpCopyObject
- // The id use that is to be replaced
- IdUseDescriptor id_use_descriptor = 1;
- // The synonymous id
- uint32 synonymous_id = 2;
- }
- message TransformationReplaceIrrelevantId {
- // Replaces an irrelevant id with another id of the same type.
- // The id use that is to be replaced
- IdUseDescriptor id_use_descriptor = 1;
- // The replacement id
- uint32 replacement_id = 2;
- }
- message TransformationReplaceLinearAlgebraInstruction {
- // Replaces a linear algebra instruction with its
- // mathematical definition.
- // The fresh ids needed to apply the transformation.
- repeated uint32 fresh_ids = 1;
- // A descriptor for a linear algebra instruction.
- InstructionDescriptor instruction_descriptor = 2;
- }
- message TransformationReplaceLoadStoreWithCopyMemory {
- // A transformation that takes a pair of instruction descriptors
- // to OpLoad and OpStore that have the same intermediate value
- // and replaces the OpStore with an equivalent OpCopyMemory.
- // The instruction descriptor to OpLoad
- InstructionDescriptor load_instruction_descriptor = 1;
- // The instruction descriptor to OpStore
- InstructionDescriptor store_instruction_descriptor = 2;
- }
- message TransformationReplaceOpPhiIdFromDeadPredecessor {
- // Replaces one of the ids used by an OpPhi instruction, when
- // the corresponding predecessor is dead, with any available id
- // of the correct type.
- // The result id of the OpPhi instruction.
- uint32 opphi_id = 1;
- // The label id of one of the predecessors of the block containing
- // the OpPhi instruction, corresponding to the id that we want to
- // replace.
- uint32 pred_label_id = 2;
- // The id that, after the transformation, will be associated with
- // the given predecessor.
- uint32 replacement_id = 3;
- }
- message TransformationReplaceOpSelectWithConditionalBranch {
- // A transformation that takes an OpSelect instruction with a
- // scalar boolean condition and replaces it with a conditional
- // branch and an OpPhi instruction.
- // The OpSelect instruction must be the first instruction in its
- // block, which must have a unique predecessor. The block will
- // become the merge block of a new construct, while its predecessor
- // will become the header.
- // Given the original OpSelect instruction:
- // %id = OpSelect %type %cond %then %else
- // The branching instruction of the header will be:
- // OpBranchConditional %cond %true_block_id %false_block_id
- // and the OpSelect instruction will be turned into:
- // %id = OpPhi %type %then %true_block_id %else %false_block_id
- // At most one of |true_block_id| and |false_block_id| can be zero. In
- // that case, there will be no such block and all references to it
- // will be replaced by %merge_block (where %merge_block is the
- // block containing the OpSelect instruction).
- // The result id of the OpSelect instruction.
- uint32 select_id = 1;
- // A fresh id for the new block that the predecessor of the block
- // containing |select_id| will branch to if the condition holds.
- uint32 true_block_id = 2;
- // A fresh id for the new block that the predecessor of the block
- // containing |select_id| will branch to if the condition does not
- // hold.
- uint32 false_block_id = 3;
- }
- message TransformationReplaceParamsWithStruct {
- // Replaces parameters of the function with a struct containing
- // values of those parameters.
- // Result ids of parameters to replace.
- repeated uint32 parameter_id = 1;
- // Fresh id for a new function type. This might be unused if the required type
- // already exists in the module or if we can change the old type.
- uint32 fresh_function_type_id = 2;
- // Fresh id for a new struct function parameter to be used as a replacement.
- uint32 fresh_parameter_id = 3;
- // Fresh ids for struct objects containing values of replaced parameters.
- // This field contains a fresh id for at least every result id of a relevant
- // OpFunctionCall instruction.
- repeated UInt32Pair caller_id_to_fresh_composite_id = 4;
- }
- message TransformationSetFunctionControl {
- // A transformation that sets the function control operand of an OpFunction
- // instruction.
- // The result id of an OpFunction instruction
- uint32 function_id = 1;
- // The value to which the 'function control' operand should be set.
- uint32 function_control = 2;
- }
- message TransformationSetLoopControl {
- // A transformation that sets the loop control operand of an OpLoopMerge
- // instruction.
- // The id of a basic block that should contain OpLoopMerge
- uint32 block_id = 1;
- // The value to which the 'loop control' operand should be set.
- // This must be a legal loop control mask.
- uint32 loop_control = 2;
- // Provides a peel count value for the loop. Used if and only if the
- // PeelCount bit is set. Must be zero if the PeelCount bit is not set (can
- // still be zero if this bit is set).
- uint32 peel_count = 3;
- // Provides a partial count value for the loop. Used if and only if the
- // PartialCount bit is set. Must be zero if the PartialCount bit is not set
- // (can still be zero if this bit is set).
- uint32 partial_count = 4;
- }
- message TransformationSetMemoryOperandsMask {
- // A transformation that sets the memory operands mask of a memory access
- // instruction.
- // A descriptor for a memory access instruction, e.g. an OpLoad
- InstructionDescriptor memory_access_instruction = 1;
- // A mask of memory operands to be applied to the instruction. It must be the
- // same as the original mask, except that Volatile can be added, and
- // Nontemporal can be added or removed.
- uint32 memory_operands_mask = 2;
- // Some memory access instructions allow more than one mask to be specified;
- // this field indicates which mask should be set
- uint32 memory_operands_mask_index = 3;
- }
- message TransformationSetSelectionControl {
- // A transformation that sets the selection control operand of an
- // OpSelectionMerge instruction.
- // The id of a basic block that should contain OpSelectionMerge
- uint32 block_id = 1;
- // The value to which the 'selection control' operand should be set.
- // Although technically 'selection control' is a literal mask that can be
- // some combination of 'None', 'Flatten' and 'DontFlatten', the combination
- // 'Flatten | DontFlatten' does not make sense and is not allowed here.
- uint32 selection_control = 2;
- }
- message TransformationSplitBlock {
- // A transformation that splits a basic block into two basic blocks
- // A descriptor for an instruction such that the block containing the
- // described instruction should be split right before the instruction.
- InstructionDescriptor instruction_to_split_before = 1;
- // An id that must not yet be used by the module to which this transformation
- // is applied. Rather than having the transformation choose a suitable id on
- // application, we require the id to be given upfront in order to facilitate
- // reducing fuzzed shaders by removing transformations. The reason is that
- // future transformations may refer to the fresh id introduced by this
- // transformation, and if we end up changing what that id is, due to removing
- // earlier transformations, it may inhibit later transformations from
- // applying.
- uint32 fresh_id = 2;
- }
- message TransformationStore {
- // Transformation that adds an OpStore or OpAtomicStore instruction of an id to a pointer.
- // The pointer to be stored to.
- uint32 pointer_id = 1;
- // True if and only if the load should be atomic.
- bool is_atomic = 2;
- // The memory scope for the atomic load. Ignored unless |is_atomic| is true.
- uint32 memory_scope_id = 3;
- // The memory semantics for the atomic load. Ignored unless |is_atomic| is true.
- uint32 memory_semantics_id = 4;
- // The value to be stored.
- uint32 value_id = 5;
- // A descriptor for an instruction in a block before which the new OpStore
- // instruction should be inserted.
- InstructionDescriptor instruction_to_insert_before = 6;
- }
- message TransformationSwapCommutableOperands {
- // A transformation that swaps the operands of a commutative instruction.
- // A descriptor for a commutative instruction
- InstructionDescriptor instruction_descriptor = 1;
- }
- message TransformationSwapConditionalBranchOperands {
- // Swaps label ids in OpBranchConditional instruction.
- // Additionally, inverts the guard and swaps branch weights
- // if present.
- // Descriptor of the instruction to swap operands of.
- InstructionDescriptor instruction_descriptor = 1;
- // Fresh result id for the OpLogicalNot instruction, used
- // to invert the guard.
- uint32 fresh_id = 2;
- }
- message TransformationSwapFunctionVariables {
- // A transformation that swaps function variables
- // Result id of the first variable.
- uint32 result_id1 = 1;
- // Result id of the second variable.
- uint32 result_id2 = 2;
- }
- message TransformationSwapTwoFunctions {
- // A transformation that swaps the position of two functions within the same module.
- // the IDs for the two functions that are swapped.
- uint32 function_id1 = 1;
- uint32 function_id2 = 2;
- }
- message TransformationToggleAccessChainInstruction {
- // A transformation that toggles an access chain instruction.
- // A descriptor for an access chain instruction
- InstructionDescriptor instruction_descriptor = 1;
- }
- message TransformationVectorShuffle {
- // A transformation that adds a vector shuffle instruction.
- // A descriptor for an instruction in a block before which the new
- // OpVectorShuffle instruction should be inserted
- InstructionDescriptor instruction_to_insert_before = 1;
- // Result id for the shuffle operation.
- uint32 fresh_id = 2;
- // Id of the first vector operand.
- uint32 vector1 = 3;
- // Id of the second vector operand.
- uint32 vector2 = 4;
- // Indices that indicate which components of the input vectors should be used.
- repeated uint32 component = 5;
- }
- message TransformationWrapEarlyTerminatorInFunction {
- // Replaces an early terminator - OpKill, OpReachable or OpTerminateInvocation
- // - with a call to a wrapper function for the terminator.
- // A fresh id for a new OpFunctionCall instruction.
- uint32 fresh_id = 1;
- // A descriptor for an OpKill, OpUnreachable or OpTerminateInvocation
- // instruction.
- InstructionDescriptor early_terminator_instruction = 2;
- // An id with the same type as the enclosing function's return type that is
- // available at the early terminator. This is used to change the terminator
- // to OpReturnValue. Ignored if the enclosing function has void return type,
- // in which case OpReturn can be used as the new terminator.
- uint32 returned_value_id = 3;
- }
- message TransformationWrapRegionInSelection {
- // Transforms a single-entry-single-exit region R into
- // if (|branch_condition|) { R } else { R }
- // The entry block for R becomes a selection header and
- // the exit block - a selection merge.
- //
- // Note that the region R is not duplicated. Thus, the effect of
- // this transformation can be represented as follows:
- // entry
- // entry / \
- // | \ /
- // R --> R
- // | |
- // exit exit
- // This behaviour is different from TransformationDuplicateRegionWithSelection
- // that copies the blocks in R.
- // The entry block for the region R.
- uint32 region_entry_block_id = 1;
- // The exit block for the region R.
- uint32 region_exit_block_id = 2;
- // Boolean value for the condition expression.
- bool branch_condition = 3;
- }
- message TransformationWrapVectorSynonym {
- // A transformation that wraps an arithmetic operation into a vector operation
- // and get the result of the original operation from the corresponding index.
- // For instance, for this transformation, an scalar operation between two scalars:
- // define op ∈ {+, -, *}
- // c = a op b
- //
- // requires the availability of two vectors:
- //
- // va = vector(..., a, ...)
- // vb = vector(..., b, ...)
- //
- // where a and b are in the same position i in each of their corresponding vector
- // and a is synonymous with va[i] and b is synonymous with vb[i].
- //
- // The transformation then add an instruction vc = va op vb where c is synonymous
- // with vc[i].
- // The result if of the original scalar operation instruction.
- uint32 instruction_id = 1;
- // The result id for the first vector that contains the first value of the scalar operation.
- uint32 vector_operand1 = 2;
- // The result id for the second vector that contains the second value of the scalar operation.
- uint32 vector_operand2 = 3;
- // A fresh id for the resulted vector from the addition of the first and second vector.
- uint32 fresh_id = 4;
- // The position in the vector where the value of original instruction is located. Must be in
- // the corresponding vector range.
- uint32 scalar_position = 5;
- }
|