1 // Copyright (c) 2016 Google Inc.
2 //
3 // Licensed under the Apache License, Version 2.0 (the "License");
4 // you may not use this file except in compliance with the License.
5 // You may obtain a copy of the License at
6 //
7 //     http://www.apache.org/licenses/LICENSE-2.0
8 //
9 // Unless required by applicable law or agreed to in writing, software
10 // distributed under the License is distributed on an "AS IS" BASIS,
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 // See the License for the specific language governing permissions and
13 // limitations under the License.
14 
15 #ifndef TEST_OPT_ASSEMBLY_BUILDER_H_
16 #define TEST_OPT_ASSEMBLY_BUILDER_H_
17 
18 #include <algorithm>
19 #include <cstdint>
20 #include <sstream>
21 #include <string>
22 #include <unordered_set>
23 #include <vector>
24 
25 namespace spvtools {
26 namespace opt {
27 
28 // A simple SPIR-V assembly code builder for test uses. It builds an SPIR-V
29 // assembly module from vectors of assembly strings. It allows users to add
30 // instructions to the main function and the type-constants-globals section
31 // directly. It relies on OpName instructions and friendly-name disassembling
32 // to keep the ID names unchanged after assembling.
33 //
34 // An assembly module is divided into several sections, matching with the
35 // SPIR-V Logical Layout:
36 //  Global Preamble:
37 //    OpCapability instructions;
38 //    OpExtension instructions and OpExtInstImport instructions;
39 //    OpMemoryModel instruction;
40 //    OpEntryPoint and OpExecutionMode instruction;
41 //    OpString, OpSourceExtension, OpSource and OpSourceContinued instructions.
42 //  Names:
43 //    OpName instructions.
44 //  Annotations:
45 //    OpDecorate, OpMemberDecorate, OpGroupDecorate, OpGroupMemberDecorate and
46 //    OpDecorationGroup.
47 //  Types, Constants and Global variables:
48 //    Types, constants and global variables declaration instructions.
49 //  Main Function:
50 //    Main function instructions.
51 //  Main Function Postamble:
52 //    The return and function end instructions.
53 //
54 // The assembly code is built by concatenating all the strings in the above
55 // sections.
56 //
57 // Users define the contents in section <Type, Constants and Global Variables>
58 // and <Main Function>. The <Names> section is to hold the names for IDs to
59 // keep them unchanged before and after assembling. All defined IDs to be added
60 // to this code builder will be assigned with a global name through OpName
61 // instruction. The name is extracted from the definition instruction.
62 //  E.g. adding instruction: %var_a = OpConstant %int 2, will also add an
63 //  instruction: OpName %var_a, "var_a".
64 //
65 // Note that the name must not be used on more than one defined IDs and
66 // friendly-name disassembling must be enabled so that OpName instructions will
67 // be respected.
68 class AssemblyBuilder {
69   // The base ID value for spec constants.
70   static const uint32_t SPEC_ID_BASE = 200;
71 
72  public:
73   // Initalize a minimal SPIR-V assembly code as the template. The minimal
74   // module contains an empty main function and some predefined names for the
75   // main function.
AssemblyBuilder()76   AssemblyBuilder()
77       : spec_id_counter_(SPEC_ID_BASE),
78         global_preamble_({
79             // clang-format off
80                   "OpCapability Shader",
81                   "OpCapability Float64",
82              "%1 = OpExtInstImport \"GLSL.std.450\"",
83                   "OpMemoryModel Logical GLSL450",
84                   "OpEntryPoint Vertex %main \"main\"",
85             // clang-format on
86         }),
87         names_(),
88         annotations_(),
89         types_consts_globals_(),
90         main_func_(),
91         main_func_postamble_({
92             "OpReturn",
93             "OpFunctionEnd",
94         }) {
95     AppendTypesConstantsGlobals({
96         "%void = OpTypeVoid",
97         "%main_func_type = OpTypeFunction %void",
98     });
99     AppendInMain({
100         "%main = OpFunction %void None %main_func_type",
101         "%main_func_entry_block = OpLabel",
102     });
103   }
104 
105   // Appends OpName instructions to this builder. Instrcution strings that do
106   // not start with 'OpName ' will be skipped. Returns the references of this
107   // assembly builder.
AppendNames(const std::vector<std::string> & vec_asm_code)108   AssemblyBuilder& AppendNames(const std::vector<std::string>& vec_asm_code) {
109     for (auto& inst_str : vec_asm_code) {
110       if (inst_str.find("OpName ") == 0) {
111         names_.push_back(inst_str);
112       }
113     }
114     return *this;
115   }
116 
117   // Appends instructions to the types-constants-globals section and returns
118   // the reference of this assembly builder. IDs defined in the given code will
119   // be added to the Names section and then be registered with OpName
120   // instruction. Corresponding decoration instruction will be added for spec
121   // constants defined with opcode: 'OpSpecConstant'.
AppendTypesConstantsGlobals(const std::vector<std::string> & vec_asm_code)122   AssemblyBuilder& AppendTypesConstantsGlobals(
123       const std::vector<std::string>& vec_asm_code) {
124     AddNamesForResultIDsIn(vec_asm_code);
125     // Check spec constants defined with OpSpecConstant.
126     for (auto& inst_str : vec_asm_code) {
127       if (inst_str.find("= OpSpecConstant ") != std::string::npos ||
128           inst_str.find("= OpSpecConstantTrue ") != std::string::npos ||
129           inst_str.find("= OpSpecConstantFalse ") != std::string::npos) {
130         AddSpecIDFor(GetResultIDName(inst_str));
131       }
132     }
133     types_consts_globals_.insert(types_consts_globals_.end(),
134                                  vec_asm_code.begin(), vec_asm_code.end());
135     return *this;
136   }
137 
138   // Appends instructions to the main function block, which is already labelled
139   // with "main_func_entry_block". Returns the reference of this assembly
140   // builder. IDs defined in the given code will be added to the Names section
141   // and then be registered with OpName instruction.
AppendInMain(const std::vector<std::string> & vec_asm_code)142   AssemblyBuilder& AppendInMain(const std::vector<std::string>& vec_asm_code) {
143     AddNamesForResultIDsIn(vec_asm_code);
144     main_func_.insert(main_func_.end(), vec_asm_code.begin(),
145                       vec_asm_code.end());
146     return *this;
147   }
148 
149   // Appends annotation instructions to the annotation section, and returns the
150   // reference of this assembly builder.
AppendAnnotations(const std::vector<std::string> & vec_annotations)151   AssemblyBuilder& AppendAnnotations(
152       const std::vector<std::string>& vec_annotations) {
153     annotations_.insert(annotations_.end(), vec_annotations.begin(),
154                         vec_annotations.end());
155     return *this;
156   }
157 
158   // Pre-pends string to the preamble of the module. Useful for EFFCEE checks.
PrependPreamble(const std::vector<std::string> & preamble)159   AssemblyBuilder& PrependPreamble(const std::vector<std::string>& preamble) {
160     preamble_.insert(preamble_.end(), preamble.begin(), preamble.end());
161     return *this;
162   }
163 
164   // Get the SPIR-V assembly code as string.
GetCode()165   std::string GetCode() const {
166     std::ostringstream ss;
167     for (const auto& line : preamble_) {
168       ss << line << std::endl;
169     }
170     for (const auto& line : global_preamble_) {
171       ss << line << std::endl;
172     }
173     for (const auto& line : names_) {
174       ss << line << std::endl;
175     }
176     for (const auto& line : annotations_) {
177       ss << line << std::endl;
178     }
179     for (const auto& line : types_consts_globals_) {
180       ss << line << std::endl;
181     }
182     for (const auto& line : main_func_) {
183       ss << line << std::endl;
184     }
185     for (const auto& line : main_func_postamble_) {
186       ss << line << std::endl;
187     }
188     return ss.str();
189   }
190 
191  private:
192   // Adds a given name to the Name section with OpName. If the given name has
193   // been added before, does nothing.
AddOpNameIfNotExist(const std::string & id_name)194   void AddOpNameIfNotExist(const std::string& id_name) {
195     if (!used_names_.count(id_name)) {
196       std::stringstream opname_inst;
197       opname_inst << "OpName "
198                   << "%" << id_name << " \"" << id_name << "\"";
199       names_.emplace_back(opname_inst.str());
200       used_names_.insert(id_name);
201     }
202   }
203 
204   // Adds the names in a vector of assembly code strings to the Names section.
205   // If a '=' sign is found in an instruction, this instruction will be treated
206   // as an ID defining instruction. The ID name used in the instruction will be
207   // extracted and added to the Names section.
AddNamesForResultIDsIn(const std::vector<std::string> & vec_asm_code)208   void AddNamesForResultIDsIn(const std::vector<std::string>& vec_asm_code) {
209     for (const auto& line : vec_asm_code) {
210       std::string name = GetResultIDName(line);
211       if (!name.empty()) {
212         AddOpNameIfNotExist(name);
213       }
214     }
215   }
216 
217   // Adds an OpDecorate SpecId instruction for the given ID name.
AddSpecIDFor(const std::string & id_name)218   void AddSpecIDFor(const std::string& id_name) {
219     std::stringstream decorate_inst;
220     decorate_inst << "OpDecorate "
221                   << "%" << id_name << " SpecId " << spec_id_counter_;
222     spec_id_counter_ += 1;
223     annotations_.emplace_back(decorate_inst.str());
224   }
225 
226   // Extracts the ID name from a SPIR-V assembly instruction string. If the
227   // instruction is an ID-defining instruction (has result ID), returns the
228   // name of the result ID in string. If the instruction does not have result
229   // ID, returns an empty string.
GetResultIDName(const std::string inst_str)230   std::string GetResultIDName(const std::string inst_str) {
231     std::string name;
232     if (inst_str.find('=') != std::string::npos) {
233       size_t assign_sign = inst_str.find('=');
234       name = inst_str.substr(0, assign_sign);
235       name.erase(remove_if(name.begin(), name.end(),
236                            [](char c) { return c == ' ' || c == '%'; }),
237                  name.end());
238     }
239     return name;
240   }
241 
242   uint32_t spec_id_counter_;
243   // User-defined preamble.
244   std::vector<std::string> preamble_;
245   // The vector that contains common preambles shared across all test SPIR-V
246   // code.
247   std::vector<std::string> global_preamble_;
248   // The vector that contains OpName instructions.
249   std::vector<std::string> names_;
250   // The vector that contains annotation instructions.
251   std::vector<std::string> annotations_;
252   // The vector that contains the code to declare types, constants and global
253   // variables (aka. the Types-Constants-Globals section).
254   std::vector<std::string> types_consts_globals_;
255   // The vector that contains the code in main function's entry block.
256   std::vector<std::string> main_func_;
257   // The vector that contains the postamble of main function body.
258   std::vector<std::string> main_func_postamble_;
259   // All of the defined variable names.
260   std::unordered_set<std::string> used_names_;
261 };
262 
263 }  // namespace opt
264 }  // namespace spvtools
265 
266 #endif  // TEST_OPT_ASSEMBLY_BUILDER_H_
267