| // Copyright (c) 2015-2016 The Khronos Group 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. | 
 |  | 
 | #ifndef SOURCE_OPERAND_H_ | 
 | #define SOURCE_OPERAND_H_ | 
 |  | 
 | #include <functional> | 
 | #include <vector> | 
 |  | 
 | #include "source/table.h" | 
 | #include "spirv-tools/libspirv.h" | 
 |  | 
 | // A sequence of operand types. | 
 | // | 
 | // A SPIR-V parser uses an operand pattern to describe what is expected | 
 | // next on the input. | 
 | // | 
 | // As we parse an instruction in text or binary form from left to right, | 
 | // we pop and push at the end of the pattern vector. Symbols later in the | 
 | // pattern vector are matched against the input before symbols earlier in the | 
 | // pattern vector are matched. | 
 |  | 
 | // Using a vector in this way reduces memory traffic, which is good for | 
 | // performance. | 
 | using spv_operand_pattern_t = std::vector<spv_operand_type_t>; | 
 |  | 
 | // Finds the named operand in the table. The type parameter specifies the | 
 | // operand's group. A handle of the operand table entry for this operand will | 
 | // be written into *entry. | 
 | spv_result_t spvOperandTableNameLookup(spv_target_env, | 
 |                                        const spv_operand_table table, | 
 |                                        const spv_operand_type_t type, | 
 |                                        const char* name, | 
 |                                        const size_t name_length, | 
 |                                        spv_operand_desc* entry); | 
 |  | 
 | // Finds the operand with value in the table. The type parameter specifies the | 
 | // operand's group. A handle of the operand table entry for this operand will | 
 | // be written into *entry. | 
 | spv_result_t spvOperandTableValueLookup(spv_target_env, | 
 |                                         const spv_operand_table table, | 
 |                                         const spv_operand_type_t type, | 
 |                                         const uint32_t value, | 
 |                                         spv_operand_desc* entry); | 
 |  | 
 | // Gets the name string of the non-variable operand type. | 
 | const char* spvOperandTypeStr(spv_operand_type_t type); | 
 |  | 
 | // Returns true if the given type is concrete. | 
 | bool spvOperandIsConcrete(spv_operand_type_t type); | 
 |  | 
 | // Returns true if the given type is concrete and also a mask. | 
 | bool spvOperandIsConcreteMask(spv_operand_type_t type); | 
 |  | 
 | // Returns true if an operand of the given type is optional. | 
 | bool spvOperandIsOptional(spv_operand_type_t type); | 
 |  | 
 | // Returns true if an operand type represents zero or more logical operands. | 
 | // | 
 | // Note that a single logical operand may still be a variable number of words. | 
 | // For example, a literal string may be many words, but is just one logical | 
 | // operand. | 
 | bool spvOperandIsVariable(spv_operand_type_t type); | 
 |  | 
 | // Append a list of operand types to the end of the pattern vector. | 
 | // The types parameter specifies the source array of types, ending with | 
 | // SPV_OPERAND_TYPE_NONE. | 
 | void spvPushOperandTypes(const spv_operand_type_t* types, | 
 |                          spv_operand_pattern_t* pattern); | 
 |  | 
 | // Appends the operands expected after the given typed mask onto the | 
 | // end of the given pattern. | 
 | // | 
 | // Each set bit in the mask represents zero or more operand types that should | 
 | // be appended onto the pattern.  Operands for a less significant bit always | 
 | // appear after operands for a more significant bit. | 
 | // | 
 | // If a set bit is unknown, then we assume it has no operands. | 
 | void spvPushOperandTypesForMask(spv_target_env, | 
 |                                 const spv_operand_table operand_table, | 
 |                                 const spv_operand_type_t mask_type, | 
 |                                 const uint32_t mask, | 
 |                                 spv_operand_pattern_t* pattern); | 
 |  | 
 | // Expands an operand type representing zero or more logical operands, | 
 | // exactly once. | 
 | // | 
 | // If the given type represents potentially several logical operands, | 
 | // then prepend the given pattern with the first expansion of the logical | 
 | // operands, followed by original type.  Otherwise, don't modify the pattern. | 
 | // | 
 | // For example, the SPV_OPERAND_TYPE_VARIABLE_ID represents zero or more | 
 | // IDs.  In that case we would prepend the pattern with SPV_OPERAND_TYPE_ID | 
 | // followed by SPV_OPERAND_TYPE_VARIABLE_ID again. | 
 | // | 
 | // This also applies to zero or more tuples of logical operands.  In that case | 
 | // we prepend pattern with for the members of the tuple, followed by the | 
 | // original type argument.  The pattern must encode the fact that if any part | 
 | // of the tuple is present, then all tuple members should be.  So the first | 
 | // member of the tuple must be optional, and the remaining members | 
 | // non-optional. | 
 | // | 
 | // Returns true if we modified the pattern. | 
 | bool spvExpandOperandSequenceOnce(spv_operand_type_t type, | 
 |                                   spv_operand_pattern_t* pattern); | 
 |  | 
 | // Expands the first element in the pattern until it is a matchable operand | 
 | // type, then pops it off the front and returns it.  The pattern must not be | 
 | // empty. | 
 | // | 
 | // A matchable operand type is anything other than a zero-or-more-items | 
 | // operand type. | 
 | spv_operand_type_t spvTakeFirstMatchableOperand(spv_operand_pattern_t* pattern); | 
 |  | 
 | // Calculates the corresponding post-immediate alternate pattern, which allows | 
 | // a limited set of operand types. | 
 | spv_operand_pattern_t spvAlternatePatternFollowingImmediate( | 
 |     const spv_operand_pattern_t& pattern); | 
 |  | 
 | // Is the operand an ID? | 
 | bool spvIsIdType(spv_operand_type_t type); | 
 |  | 
 | // Is the operand an input ID? | 
 | bool spvIsInIdType(spv_operand_type_t type); | 
 |  | 
 | // Takes the opcode of an instruction and returns | 
 | // a function object that will return true if the index | 
 | // of the operand can be forward declared. This function will | 
 | // used in the SSA validation stage of the pipeline | 
 | std::function<bool(unsigned)> spvOperandCanBeForwardDeclaredFunction( | 
 |     SpvOp opcode); | 
 |  | 
 | #endif  // SOURCE_OPERAND_H_ |