1 // Copyright 2018 The Amber Authors.
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 #include "src/vkscript/command_parser.h"
16
17 #include <algorithm>
18 #include <cassert>
19 #include <cctype>
20 #include <limits>
21 #include <string>
22 #include <utility>
23
24 #include "src/command_data.h"
25 #include "src/make_unique.h"
26 #include "src/tokenizer.h"
27 #include "src/type_parser.h"
28 #include "src/vkscript/datum_type_parser.h"
29
30 namespace amber {
31 namespace vkscript {
32 namespace {
33
ShaderNameToType(const std::string & name)34 ShaderType ShaderNameToType(const std::string& name) {
35 if (name == "fragment")
36 return kShaderTypeFragment;
37 if (name == "compute")
38 return kShaderTypeCompute;
39 if (name == "geometry")
40 return kShaderTypeGeometry;
41 if (name == "tessellation evaluation")
42 return kShaderTypeTessellationEvaluation;
43 if (name == "tessellation control")
44 return kShaderTypeTessellationControl;
45
46 return kShaderTypeVertex;
47 }
48
49 } // namespace
50
CommandParser(Script * script,Pipeline * pipeline,size_t current_line,const std::string & data)51 CommandParser::CommandParser(Script* script,
52 Pipeline* pipeline,
53 size_t current_line,
54 const std::string& data)
55 : script_(script),
56 pipeline_(pipeline),
57 tokenizer_(MakeUnique<Tokenizer>(data)) {
58 tokenizer_->SetCurrentLine(current_line);
59 }
60
61 CommandParser::~CommandParser() = default;
62
make_error(const std::string & err)63 std::string CommandParser::make_error(const std::string& err) {
64 return std::to_string(tokenizer_->GetCurrentLine()) + ": " + err;
65 }
66
ParseBoolean(const std::string & str,bool * result)67 Result CommandParser::ParseBoolean(const std::string& str, bool* result) {
68 assert(result);
69
70 std::string tmp;
71 tmp.resize(str.size());
72 std::transform(str.begin(), str.end(), tmp.begin(),
73 [](unsigned char c) { return std::tolower(c); });
74
75 if (tmp == "true") {
76 *result = true;
77 return {};
78 }
79 if (tmp == "false") {
80 *result = false;
81 return {};
82 }
83 return Result("Invalid value passed as a boolean string: " + str);
84 }
85
Parse()86 Result CommandParser::Parse() {
87 for (auto token = tokenizer_->NextToken(); !token->IsEOS();
88 token = tokenizer_->NextToken()) {
89 if (token->IsEOL())
90 continue;
91
92 if (!token->IsIdentifier()) {
93 return Result(make_error(
94 "Command not recognized. Received something other then a string: " +
95 token->ToOriginalString()));
96 }
97
98 std::string cmd_name = token->AsString();
99 Result r;
100 if (cmd_name == "draw") {
101 token = tokenizer_->NextToken();
102 if (!token->IsIdentifier())
103 return Result(make_error("Invalid draw command in test: " +
104 token->ToOriginalString()));
105
106 cmd_name = token->AsString();
107 if (cmd_name == "rect")
108 r = ProcessDrawRect();
109 else if (cmd_name == "arrays")
110 r = ProcessDrawArrays();
111 else
112 r = Result("Unknown draw command: " + cmd_name);
113
114 } else if (cmd_name == "clear") {
115 r = ProcessClear();
116 } else if (cmd_name == "ssbo") {
117 r = ProcessSSBO();
118 } else if (cmd_name == "uniform") {
119 r = ProcessUniform();
120 } else if (cmd_name == "patch") {
121 r = ProcessPatch();
122 } else if (cmd_name == "probe") {
123 r = ProcessProbe(false);
124 } else if (cmd_name == "tolerance") {
125 r = ProcessTolerance();
126 } else if (cmd_name == "relative") {
127 token = tokenizer_->NextToken();
128 if (!token->IsIdentifier() || token->AsString() != "probe")
129 return Result(make_error("relative must be used with probe: " +
130 token->ToOriginalString()));
131
132 r = ProcessProbe(true);
133 } else if (cmd_name == "compute") {
134 r = ProcessCompute();
135 } else if (cmd_name == "vertex" || cmd_name == "fragment" ||
136 cmd_name == "geometry" || cmd_name == "tessellation") {
137 std::string shader_name = cmd_name;
138 if (cmd_name == "tessellation") {
139 token = tokenizer_->NextToken();
140 if (!token->IsIdentifier() || (token->AsString() != "control" &&
141 token->AsString() != "evaluation")) {
142 return Result(
143 make_error("Tessellation entrypoint must have "
144 "<evaluation|control> in name: " +
145 token->ToOriginalString()));
146 }
147 shader_name += " " + token->AsString();
148 }
149
150 token = tokenizer_->NextToken();
151 if (!token->IsIdentifier() || token->AsString() != "entrypoint")
152 return Result(make_error("Unknown command: " + shader_name));
153
154 r = ProcessEntryPoint(shader_name);
155
156 // Pipeline Commands
157 } else if (cmd_name == "primitiveRestartEnable") {
158 r = ProcessPrimitiveRestartEnable();
159 } else if (cmd_name == "depthClampEnable") {
160 r = ProcessDepthClampEnable();
161 } else if (cmd_name == "rasterizerDiscardEnable") {
162 r = ProcessRasterizerDiscardEnable();
163 } else if (cmd_name == "depthBiasEnable") {
164 r = ProcessDepthBiasEnable();
165 } else if (cmd_name == "logicOpEnable") {
166 r = ProcessLogicOpEnable();
167 } else if (cmd_name == "blendEnable") {
168 r = ProcessBlendEnable();
169 } else if (cmd_name == "depthTestEnable") {
170 r = ProcessDepthTestEnable();
171 } else if (cmd_name == "depthWriteEnable") {
172 r = ProcessDepthWriteEnable();
173 } else if (cmd_name == "depthBoundsTestEnable") {
174 r = ProcessDepthBoundsTestEnable();
175 } else if (cmd_name == "stencilTestEnable") {
176 r = ProcessStencilTestEnable();
177 } else if (cmd_name == "topology") {
178 r = ProcessTopology();
179 } else if (cmd_name == "polygonMode") {
180 r = ProcessPolygonMode();
181 } else if (cmd_name == "logicOp") {
182 r = ProcessLogicOp();
183 } else if (cmd_name == "frontFace") {
184 r = ProcessFrontFace();
185 } else if (cmd_name == "cullMode") {
186 r = ProcessCullMode();
187 } else if (cmd_name == "depthBiasConstantFactor") {
188 r = ProcessDepthBiasConstantFactor();
189 } else if (cmd_name == "depthBiasClamp") {
190 r = ProcessDepthBiasClamp();
191 } else if (cmd_name == "depthBiasSlopeFactor") {
192 r = ProcessDepthBiasSlopeFactor();
193 } else if (cmd_name == "lineWidth") {
194 r = ProcessLineWidth();
195 } else if (cmd_name == "minDepthBounds") {
196 r = ProcessMinDepthBounds();
197 } else if (cmd_name == "maxDepthBounds") {
198 r = ProcessMaxDepthBounds();
199 } else if (cmd_name == "srcColorBlendFactor") {
200 r = ProcessSrcColorBlendFactor();
201 } else if (cmd_name == "dstColorBlendFactor") {
202 r = ProcessDstColorBlendFactor();
203 } else if (cmd_name == "srcAlphaBlendFactor") {
204 r = ProcessSrcAlphaBlendFactor();
205 } else if (cmd_name == "dstAlphaBlendFactor") {
206 r = ProcessDstAlphaBlendFactor();
207 } else if (cmd_name == "colorBlendOp") {
208 r = ProcessColorBlendOp();
209 } else if (cmd_name == "alphaBlendOp") {
210 r = ProcessAlphaBlendOp();
211 } else if (cmd_name == "depthCompareOp") {
212 r = ProcessDepthCompareOp();
213 } else if (cmd_name == "front.compareOp") {
214 r = ProcessFrontCompareOp();
215 } else if (cmd_name == "back.compareOp") {
216 r = ProcessBackCompareOp();
217 } else if (cmd_name == "front.failOp") {
218 r = ProcessFrontFailOp();
219 } else if (cmd_name == "front.passOp") {
220 r = ProcessFrontPassOp();
221 } else if (cmd_name == "front.depthFailOp") {
222 r = ProcessFrontDepthFailOp();
223 } else if (cmd_name == "back.failOp") {
224 r = ProcessBackFailOp();
225 } else if (cmd_name == "back.passOp") {
226 r = ProcessBackPassOp();
227 } else if (cmd_name == "back.depthFailOp") {
228 r = ProcessBackDepthFailOp();
229 } else if (cmd_name == "front.compareMask") {
230 r = ProcessFrontCompareMask();
231 } else if (cmd_name == "front.writeMask") {
232 r = ProcessFrontWriteMask();
233 } else if (cmd_name == "back.compareMask") {
234 r = ProcessBackCompareMask();
235 } else if (cmd_name == "back.writeMask") {
236 r = ProcessBackWriteMask();
237 } else if (cmd_name == "front.reference") {
238 r = ProcessFrontReference();
239 } else if (cmd_name == "back.reference") {
240 r = ProcessBackReference();
241 } else if (cmd_name == "colorWriteMask") {
242 r = ProcessColorWriteMask();
243 } else {
244 r = Result("Unknown command: " + cmd_name);
245 }
246
247 if (!r.IsSuccess())
248 return Result(make_error(r.Error()));
249 }
250
251 return {};
252 }
253
ProcessDrawRect()254 Result CommandParser::ProcessDrawRect() {
255 auto cmd = MakeUnique<DrawRectCommand>(pipeline_, pipeline_data_);
256 cmd->SetLine(tokenizer_->GetCurrentLine());
257
258 if (pipeline_->GetVertexBuffers().size() > 1) {
259 return Result(
260 "draw rect command is not supported in a pipeline with more than one "
261 "vertex buffer attached");
262 }
263
264 auto token = tokenizer_->NextToken();
265 while (token->IsIdentifier()) {
266 std::string str = token->AsString();
267 if (str != "ortho" && str != "patch")
268 return Result("Unknown parameter to draw rect: " + str);
269
270 if (str == "ortho") {
271 cmd->EnableOrtho();
272 } else {
273 cmd->EnablePatch();
274 }
275 token = tokenizer_->NextToken();
276 }
277
278 Result r = token->ConvertToDouble();
279 if (!r.IsSuccess())
280 return r;
281 cmd->SetX(token->AsFloat());
282
283 token = tokenizer_->NextToken();
284 r = token->ConvertToDouble();
285 if (!r.IsSuccess())
286 return r;
287 cmd->SetY(token->AsFloat());
288
289 token = tokenizer_->NextToken();
290 r = token->ConvertToDouble();
291 if (!r.IsSuccess())
292 return r;
293 cmd->SetWidth(token->AsFloat());
294
295 token = tokenizer_->NextToken();
296 r = token->ConvertToDouble();
297 if (!r.IsSuccess())
298 return r;
299 cmd->SetHeight(token->AsFloat());
300
301 token = tokenizer_->NextToken();
302 if (!token->IsEOS() && !token->IsEOL())
303 return Result("Extra parameter to draw rect command: " +
304 token->ToOriginalString());
305
306 commands_.push_back(std::move(cmd));
307 return {};
308 }
309
ProcessDrawArrays()310 Result CommandParser::ProcessDrawArrays() {
311 auto cmd = MakeUnique<DrawArraysCommand>(pipeline_, pipeline_data_);
312 cmd->SetLine(tokenizer_->GetCurrentLine());
313 bool instanced = false;
314
315 auto token = tokenizer_->NextToken();
316 while (token->IsIdentifier()) {
317 std::string str = token->AsString();
318 if (str != "indexed" && str != "instanced") {
319 Topology topo = NameToTopology(token->AsString());
320 if (topo != Topology::kUnknown) {
321 cmd->SetTopology(topo);
322
323 // Advance token here so we're consistent with the non-topology case.
324 token = tokenizer_->NextToken();
325 break;
326 }
327 return Result("Unknown parameter to draw arrays: " + str);
328 }
329
330 if (str == "indexed") {
331 cmd->EnableIndexed();
332 } else {
333 instanced = true;
334 }
335 token = tokenizer_->NextToken();
336 }
337
338 if (cmd->GetTopology() == Topology::kUnknown)
339 return Result("Missing draw arrays topology");
340
341 if (!token->IsInteger())
342 return Result("Missing integer first vertex value for draw arrays: " +
343 token->ToOriginalString());
344 cmd->SetFirstVertexIndex(token->AsUint32());
345
346 token = tokenizer_->NextToken();
347 if (!token->IsInteger())
348 return Result("Missing integer vertex count value for draw arrays: " +
349 token->ToOriginalString());
350 cmd->SetVertexCount(token->AsUint32());
351
352 token = tokenizer_->NextToken();
353 if (instanced) {
354 if (!token->IsEOL() && !token->IsEOS()) {
355 if (!token->IsInteger())
356 return Result("Invalid instance count for draw arrays: " +
357 token->ToOriginalString());
358
359 cmd->SetInstanceCount(token->AsUint32());
360 }
361 token = tokenizer_->NextToken();
362 }
363
364 if (!token->IsEOL() && !token->IsEOS())
365 return Result("Extra parameter to draw arrays command: " +
366 token->ToOriginalString());
367
368 commands_.push_back(std::move(cmd));
369 return {};
370 }
371
ProcessCompute()372 Result CommandParser::ProcessCompute() {
373 auto cmd = MakeUnique<ComputeCommand>(pipeline_);
374 cmd->SetLine(tokenizer_->GetCurrentLine());
375
376 auto token = tokenizer_->NextToken();
377
378 // Compute can start a compute line or an entryp oint line ...
379 if (token->IsIdentifier() && token->AsString() == "entrypoint")
380 return ProcessEntryPoint("compute");
381
382 if (!token->IsInteger())
383 return Result("Missing integer value for compute X entry: " +
384 token->ToOriginalString());
385 cmd->SetX(token->AsUint32());
386
387 token = tokenizer_->NextToken();
388 if (!token->IsInteger())
389 return Result("Missing integer value for compute Y entry: " +
390 token->ToOriginalString());
391 cmd->SetY(token->AsUint32());
392
393 token = tokenizer_->NextToken();
394 if (!token->IsInteger())
395 return Result("Missing integer value for compute Z entry: " +
396 token->ToOriginalString());
397 cmd->SetZ(token->AsUint32());
398
399 token = tokenizer_->NextToken();
400 if (!token->IsEOS() && !token->IsEOL())
401 return Result("Extra parameter to compute command: " +
402 token->ToOriginalString());
403
404 commands_.push_back(std::move(cmd));
405 return {};
406 }
407
ProcessClear()408 Result CommandParser::ProcessClear() {
409 std::unique_ptr<Command> cmd;
410
411 auto token = tokenizer_->NextToken();
412 std::string cmd_suffix = "";
413 if (token->IsIdentifier()) {
414 std::string str = token->AsString();
415 cmd_suffix = str + " ";
416 if (str == "depth") {
417 cmd = MakeUnique<ClearDepthCommand>(pipeline_);
418 cmd->SetLine(tokenizer_->GetCurrentLine());
419
420 token = tokenizer_->NextToken();
421 Result r = token->ConvertToDouble();
422 if (!r.IsSuccess())
423 return r;
424
425 cmd->AsClearDepth()->SetValue(token->AsFloat());
426 } else if (str == "stencil") {
427 cmd = MakeUnique<ClearStencilCommand>(pipeline_);
428 cmd->SetLine(tokenizer_->GetCurrentLine());
429
430 token = tokenizer_->NextToken();
431 if (token->IsEOL() || token->IsEOS())
432 return Result("Missing stencil value for clear stencil command: " +
433 token->ToOriginalString());
434 if (!token->IsInteger())
435 return Result("Invalid stencil value for clear stencil command: " +
436 token->ToOriginalString());
437
438 cmd->AsClearStencil()->SetValue(token->AsUint32());
439 } else if (str == "color") {
440 cmd = MakeUnique<ClearColorCommand>(pipeline_);
441 cmd->SetLine(tokenizer_->GetCurrentLine());
442
443 token = tokenizer_->NextToken();
444 Result r = token->ConvertToDouble();
445 if (!r.IsSuccess())
446 return r;
447 cmd->AsClearColor()->SetR(token->AsFloat());
448
449 token = tokenizer_->NextToken();
450 r = token->ConvertToDouble();
451 if (!r.IsSuccess())
452 return r;
453 cmd->AsClearColor()->SetG(token->AsFloat());
454
455 token = tokenizer_->NextToken();
456 r = token->ConvertToDouble();
457 if (!r.IsSuccess())
458 return r;
459 cmd->AsClearColor()->SetB(token->AsFloat());
460
461 token = tokenizer_->NextToken();
462 r = token->ConvertToDouble();
463 if (!r.IsSuccess())
464 return r;
465 cmd->AsClearColor()->SetA(token->AsFloat());
466 } else {
467 return Result("Extra parameter to clear command: " +
468 token->ToOriginalString());
469 }
470
471 token = tokenizer_->NextToken();
472 } else {
473 cmd = MakeUnique<ClearCommand>(pipeline_);
474 cmd->SetLine(tokenizer_->GetCurrentLine());
475 }
476 if (!token->IsEOS() && !token->IsEOL())
477 return Result("Extra parameter to clear " + cmd_suffix +
478 "command: " + token->ToOriginalString());
479
480 commands_.push_back(std::move(cmd));
481 return {};
482 }
483
ParseValues(const std::string & name,Format * fmt,std::vector<Value> * values)484 Result CommandParser::ParseValues(const std::string& name,
485 Format* fmt,
486 std::vector<Value>* values) {
487 assert(values);
488
489 uint32_t row_index = 0;
490 auto token = tokenizer_->NextToken();
491 size_t seen = 0;
492 while (!token->IsEOL() && !token->IsEOS()) {
493 Value v;
494
495 if ((fmt->IsFloat32() || fmt->IsFloat64())) {
496 if (!token->IsInteger() && !token->IsDouble()) {
497 return Result(std::string("Invalid value provided to ") + name +
498 " command: " + token->ToOriginalString());
499 }
500
501 Result r = token->ConvertToDouble();
502 if (!r.IsSuccess())
503 return r;
504
505 v.SetDoubleValue(token->AsDouble());
506 } else {
507 if (!token->IsInteger()) {
508 return Result(std::string("Invalid value provided to ") + name +
509 " command: " + token->ToOriginalString());
510 }
511
512 v.SetIntValue(token->AsUint64());
513 }
514
515 values->push_back(v);
516 token = tokenizer_->NextToken();
517
518 ++row_index;
519 ++seen;
520 }
521
522 // This could overflow, but I don't really expect us to get command files
523 // that big ....
524 size_t num_per_row = fmt->GetType()->RowCount();
525 if (seen == 0 || (seen % num_per_row) != 0) {
526 return Result(std::string("Incorrect number of values provided to ") +
527 name + " command");
528 }
529
530 return {};
531 }
532
ProcessSSBO()533 Result CommandParser::ProcessSSBO() {
534 auto cmd =
535 MakeUnique<BufferCommand>(BufferCommand::BufferType::kSSBO, pipeline_);
536 cmd->SetLine(tokenizer_->GetCurrentLine());
537
538 auto token = tokenizer_->NextToken();
539 if (token->IsEOL() || token->IsEOS())
540 return Result("Missing binding and size values for ssbo command");
541 if (!token->IsInteger())
542 return Result("Invalid binding value for ssbo command");
543
544 uint32_t val = token->AsUint32();
545
546 token = tokenizer_->NextToken();
547 if (token->IsIdentifier() && token->AsString() != "subdata") {
548 auto& str = token->AsString();
549 if (str.size() >= 2 && str[0] == ':') {
550 cmd->SetDescriptorSet(val);
551
552 auto substr = str.substr(1, str.size());
553 uint64_t binding_val = strtoul(substr.c_str(), nullptr, 10);
554 if (binding_val > std::numeric_limits<uint32_t>::max())
555 return Result("binding value too large in ssbo command");
556
557 cmd->SetBinding(static_cast<uint32_t>(binding_val));
558 } else {
559 return Result("Invalid value for ssbo command: " +
560 token->ToOriginalString());
561 }
562
563 token = tokenizer_->NextToken();
564 } else {
565 cmd->SetBinding(val);
566 }
567
568 {
569 // Generate an internal buffer for this binding if needed.
570 auto set = cmd->GetDescriptorSet();
571 auto binding = cmd->GetBinding();
572
573 auto* buffer = pipeline_->GetBufferForBinding(set, binding);
574 if (!buffer) {
575 auto b = MakeUnique<Buffer>();
576 b->SetName("AutoBuf-" + std::to_string(script_->GetBuffers().size()));
577 buffer = b.get();
578 script_->AddBuffer(std::move(b));
579 pipeline_->ClearBuffers(set, binding);
580 pipeline_->AddBuffer(buffer, BufferType::kStorage, set, binding, 0, 0);
581 }
582 cmd->SetBuffer(buffer);
583 }
584
585 if (token->IsIdentifier() && token->AsString() == "subdata") {
586 cmd->SetIsSubdata();
587
588 token = tokenizer_->NextToken();
589 if (!token->IsIdentifier())
590 return Result("Invalid type for ssbo command: " +
591 token->ToOriginalString());
592
593 DatumTypeParser tp;
594 auto type = tp.Parse(token->AsString());
595 if (!type)
596 return Result("Invalid type provided: " + token->AsString());
597
598 auto fmt = MakeUnique<Format>(type.get());
599 auto* buf = cmd->GetBuffer();
600 if (buf->FormatIsDefault() || !buf->GetFormat()) {
601 buf->SetFormat(fmt.get());
602 script_->RegisterFormat(std::move(fmt));
603 script_->RegisterType(std::move(type));
604 } else if (!buf->GetFormat()->Equal(fmt.get())) {
605 return Result("probe ssbo format does not match buffer format");
606 }
607
608 token = tokenizer_->NextToken();
609 if (!token->IsInteger()) {
610 return Result("Invalid offset for ssbo command: " +
611 token->ToOriginalString());
612 }
613 if (token->AsInt32() < 0) {
614 return Result("offset for SSBO must be positive, got: " +
615 std::to_string(token->AsInt32()));
616 }
617 if ((token->AsUint32() % buf->GetFormat()->SizeInBytes()) != 0) {
618 return Result(
619 "offset for SSBO must be a multiple of the data size expected " +
620 std::to_string(buf->GetFormat()->SizeInBytes()));
621 }
622
623 cmd->SetOffset(token->AsUint32());
624
625 std::vector<Value> values;
626 Result r = ParseValues("ssbo", buf->GetFormat(), &values);
627 if (!r.IsSuccess())
628 return r;
629
630 buf->RecalculateMaxSizeInBytes(values, cmd->GetOffset());
631
632 cmd->SetValues(std::move(values));
633
634 } else {
635 if (token->IsEOL() || token->IsEOS())
636 return Result("Missing size value for ssbo command: " +
637 token->ToOriginalString());
638 if (!token->IsInteger())
639 return Result("Invalid size value for ssbo command: " +
640 token->ToOriginalString());
641
642 // Resize the buffer so we'll correctly create the descriptor sets.
643 auto* buf = cmd->GetBuffer();
644 buf->SetElementCount(token->AsUint32());
645
646 // Set a default format into the buffer if needed.
647 if (!buf->GetFormat()) {
648 TypeParser parser;
649 auto type = parser.Parse("R8_SINT");
650 auto fmt = MakeUnique<Format>(type.get());
651 buf->SetFormat(fmt.get());
652 script_->RegisterFormat(std::move(fmt));
653 script_->RegisterType(std::move(type));
654
655 // This has to come after the SetFormat() call because SetFormat() resets
656 // the value back to false.
657 buf->SetFormatIsDefault(true);
658 }
659
660 token = tokenizer_->NextToken();
661 if (!token->IsEOS() && !token->IsEOL())
662 return Result("Extra parameter for ssbo command: " +
663 token->ToOriginalString());
664 }
665
666 commands_.push_back(std::move(cmd));
667 return {};
668 }
669
ProcessUniform()670 Result CommandParser::ProcessUniform() {
671 auto token = tokenizer_->NextToken();
672 if (token->IsEOL() || token->IsEOS())
673 return Result("Missing binding and size values for uniform command: " +
674 token->ToOriginalString());
675 if (!token->IsIdentifier())
676 return Result("Invalid type value for uniform command: " +
677 token->ToOriginalString());
678
679 std::unique_ptr<BufferCommand> cmd;
680 bool is_ubo = false;
681 if (token->AsString() == "ubo") {
682 cmd = MakeUnique<BufferCommand>(BufferCommand::BufferType::kUniform,
683 pipeline_);
684 cmd->SetLine(tokenizer_->GetCurrentLine());
685
686 token = tokenizer_->NextToken();
687 if (!token->IsInteger()) {
688 return Result("Invalid binding value for uniform ubo command: " +
689 token->ToOriginalString());
690 }
691
692 uint32_t val = token->AsUint32();
693
694 token = tokenizer_->NextToken();
695 if (!token->IsIdentifier()) {
696 return Result("Invalid type value for uniform ubo command: " +
697 token->ToOriginalString());
698 }
699
700 auto& str = token->AsString();
701 if (str.size() >= 2 && str[0] == ':') {
702 cmd->SetDescriptorSet(val);
703
704 auto substr = str.substr(1, str.size());
705 uint64_t binding_val = strtoul(substr.c_str(), nullptr, 10);
706 if (binding_val > std::numeric_limits<uint32_t>::max())
707 return Result("binding value too large in uniform ubo command: " +
708 token->ToOriginalString());
709
710 cmd->SetBinding(static_cast<uint32_t>(binding_val));
711
712 token = tokenizer_->NextToken();
713 if (!token->IsIdentifier()) {
714 return Result("Invalid type value for uniform ubo command: " +
715 token->ToOriginalString());
716 }
717 } else {
718 cmd->SetBinding(val);
719 }
720 is_ubo = true;
721
722 auto set = cmd->GetDescriptorSet();
723 auto binding = cmd->GetBinding();
724
725 auto* buffer = pipeline_->GetBufferForBinding(set, binding);
726 if (!buffer) {
727 auto b = MakeUnique<Buffer>();
728 b->SetName("AutoBuf-" + std::to_string(script_->GetBuffers().size()));
729 buffer = b.get();
730 script_->AddBuffer(std::move(b));
731 pipeline_->ClearBuffers(set, binding);
732 pipeline_->AddBuffer(buffer, BufferType::kUniform, set, binding, 0, 0);
733 }
734 cmd->SetBuffer(buffer);
735
736 } else {
737 cmd = MakeUnique<BufferCommand>(BufferCommand::BufferType::kPushConstant,
738 pipeline_);
739 cmd->SetLine(tokenizer_->GetCurrentLine());
740
741 // Push constants don't have descriptor set and binding values. So, we do
742 // not want to try to lookup the buffer or we'll accidentally get whatever
743 // is bound at 0:0.
744 auto b = MakeUnique<Buffer>();
745 b->SetName("AutoBuf-" + std::to_string(script_->GetBuffers().size()));
746 cmd->SetBuffer(b.get());
747 script_->AddBuffer(std::move(b));
748 }
749
750 DatumTypeParser tp;
751 auto type = tp.Parse(token->AsString());
752 if (!type)
753 return Result("Invalid type provided: " + token->AsString());
754
755 auto fmt = MakeUnique<Format>(type.get());
756
757 // uniform is always std140.
758 if (is_ubo)
759 fmt->SetLayout(Format::Layout::kStd140);
760
761 auto* buf = cmd->GetBuffer();
762 if (buf->FormatIsDefault() || !buf->GetFormat()) {
763 buf->SetFormat(fmt.get());
764 script_->RegisterFormat(std::move(fmt));
765 script_->RegisterType(std::move(type));
766 } else if (!buf->GetFormat()->Equal(fmt.get())) {
767 return Result("probe ssbo format does not match buffer format");
768 }
769
770 token = tokenizer_->NextToken();
771 if (!token->IsInteger()) {
772 return Result("Invalid offset value for uniform command: " +
773 token->ToOriginalString());
774 }
775 if (token->AsInt32() < 0) {
776 return Result("offset for uniform must be positive, got: " +
777 std::to_string(token->AsInt32()));
778 }
779
780 auto buf_size = static_cast<int32_t>(buf->GetFormat()->SizeInBytes());
781 if (token->AsInt32() % buf_size != 0)
782 return Result("offset for uniform must be multiple of data size");
783
784 cmd->SetOffset(token->AsUint32());
785
786 std::vector<Value> values;
787 Result r = ParseValues("uniform", buf->GetFormat(), &values);
788 if (!r.IsSuccess())
789 return r;
790
791 buf->RecalculateMaxSizeInBytes(values, cmd->GetOffset());
792
793 if (cmd->IsPushConstant())
794 buf->SetData(values);
795 else
796 cmd->SetValues(std::move(values));
797
798 commands_.push_back(std::move(cmd));
799 return {};
800 }
801
ProcessTolerance()802 Result CommandParser::ProcessTolerance() {
803 current_tolerances_.clear();
804
805 auto token = tokenizer_->NextToken();
806 size_t found_tokens = 0;
807 while (!token->IsEOL() && !token->IsEOS() && found_tokens < 4) {
808 if (token->IsIdentifier() && token->AsString() == ",") {
809 token = tokenizer_->NextToken();
810 continue;
811 }
812
813 if (token->IsInteger() || token->IsDouble()) {
814 Result r = token->ConvertToDouble();
815 if (!r.IsSuccess())
816 return r;
817 double value = token->AsDouble();
818
819 token = tokenizer_->NextToken();
820 if (token->IsIdentifier() && token->AsString() != ",") {
821 if (token->AsString() != "%")
822 return Result("Invalid value for tolerance command: " +
823 token->ToOriginalString());
824
825 current_tolerances_.push_back(Probe::Tolerance{true, value});
826 token = tokenizer_->NextToken();
827 } else {
828 current_tolerances_.push_back(Probe::Tolerance{false, value});
829 }
830 } else {
831 return Result("Invalid value for tolerance command: " +
832 token->ToOriginalString());
833 }
834
835 ++found_tokens;
836 }
837 if (found_tokens == 0)
838 return Result("Missing value for tolerance command");
839 if (found_tokens != 1 && found_tokens != 4)
840 return Result("Invalid number of tolerance parameters provided");
841
842 if (!token->IsEOS() && !token->IsEOL())
843 return Result("Extra parameter for tolerance command: " +
844 token->ToOriginalString());
845
846 return {};
847 }
848
ProcessPatch()849 Result CommandParser::ProcessPatch() {
850 auto cmd = MakeUnique<PatchParameterVerticesCommand>(pipeline_);
851 cmd->SetLine(tokenizer_->GetCurrentLine());
852
853 auto token = tokenizer_->NextToken();
854 if (!token->IsIdentifier() || token->AsString() != "parameter")
855 return Result("Missing parameter flag to patch command: " +
856 token->ToOriginalString());
857
858 token = tokenizer_->NextToken();
859 if (!token->IsIdentifier() || token->AsString() != "vertices")
860 return Result("Missing vertices flag to patch command: " +
861 token->ToOriginalString());
862
863 token = tokenizer_->NextToken();
864 if (!token->IsInteger())
865 return Result("Invalid count parameter for patch parameter vertices: " +
866 token->ToOriginalString());
867 cmd->SetControlPointCount(token->AsUint32());
868
869 token = tokenizer_->NextToken();
870 if (!token->IsEOS() && !token->IsEOL())
871 return Result("Extra parameter for patch parameter vertices command: " +
872 token->ToOriginalString());
873
874 commands_.push_back(std::move(cmd));
875 return {};
876 }
877
ProcessEntryPoint(const std::string & name)878 Result CommandParser::ProcessEntryPoint(const std::string& name) {
879 auto cmd = MakeUnique<EntryPointCommand>(pipeline_);
880 cmd->SetLine(tokenizer_->GetCurrentLine());
881
882 auto token = tokenizer_->NextToken();
883 if (token->IsEOL() || token->IsEOS())
884 return Result("Missing entrypoint name");
885
886 if (!token->IsIdentifier())
887 return Result("Entrypoint name must be a string: " +
888 token->ToOriginalString());
889
890 cmd->SetShaderType(ShaderNameToType(name));
891 cmd->SetEntryPointName(token->AsString());
892
893 token = tokenizer_->NextToken();
894 if (!token->IsEOS() && !token->IsEOL())
895 return Result("Extra parameter for entrypoint command: " +
896 token->ToOriginalString());
897
898 commands_.push_back(std::move(cmd));
899
900 return {};
901 }
902
ProcessProbe(bool relative)903 Result CommandParser::ProcessProbe(bool relative) {
904 auto token = tokenizer_->NextToken();
905 if (!token->IsIdentifier())
906 return Result("Invalid token in probe command: " +
907 token->ToOriginalString());
908
909 // The SSBO syntax is different from probe or probe all so handle specially.
910 if (token->AsString() == "ssbo")
911 return ProcessProbeSSBO();
912
913 if (pipeline_->GetColorAttachments().empty())
914 return Result("Pipeline missing color buffers. Something went wrong.");
915
916 // VkScript has a single generated colour buffer which should always be
917 // available.
918 auto* buffer = pipeline_->GetColorAttachments()[0].buffer;
919 if (!buffer)
920 return Result("Pipeline missing color buffers, something went wrong.");
921
922 auto cmd = MakeUnique<ProbeCommand>(buffer);
923 cmd->SetLine(tokenizer_->GetCurrentLine());
924
925 cmd->SetTolerances(current_tolerances_);
926 if (relative)
927 cmd->SetRelative();
928
929 bool is_rect = false;
930 if (token->AsString() == "rect") {
931 is_rect = true;
932 cmd->SetProbeRect();
933
934 token = tokenizer_->NextToken();
935 if (!token->IsIdentifier())
936 return Result("Invalid token in probe command: " +
937 token->ToOriginalString());
938 } else if (token->AsString() == "all") {
939 cmd->SetWholeWindow();
940 cmd->SetProbeRect();
941
942 token = tokenizer_->NextToken();
943 if (!token->IsIdentifier())
944 return Result("Invalid token in probe command: " +
945 token->ToOriginalString());
946 }
947
948 std::string format = token->AsString();
949 if (format != "rgba" && format != "rgb")
950 return Result("Invalid format specified to probe command: " +
951 token->ToOriginalString());
952
953 if (format == "rgba")
954 cmd->SetIsRGBA();
955
956 token = tokenizer_->NextToken();
957 if (!cmd->IsWholeWindow()) {
958 bool got_rect_open_bracket = false;
959 if (token->IsOpenBracket()) {
960 got_rect_open_bracket = true;
961 token = tokenizer_->NextToken();
962 }
963
964 Result r = token->ConvertToDouble();
965 if (!r.IsSuccess())
966 return r;
967 cmd->SetX(token->AsFloat());
968
969 token = tokenizer_->NextToken();
970 if (token->IsComma())
971 token = tokenizer_->NextToken();
972
973 r = token->ConvertToDouble();
974 if (!r.IsSuccess())
975 return r;
976 cmd->SetY(token->AsFloat());
977
978 if (is_rect) {
979 token = tokenizer_->NextToken();
980 if (token->IsComma())
981 token = tokenizer_->NextToken();
982
983 r = token->ConvertToDouble();
984 if (!r.IsSuccess())
985 return r;
986 cmd->SetWidth(token->AsFloat());
987
988 token = tokenizer_->NextToken();
989 if (token->IsComma())
990 token = tokenizer_->NextToken();
991
992 r = token->ConvertToDouble();
993 if (!r.IsSuccess())
994 return r;
995 cmd->SetHeight(token->AsFloat());
996 }
997
998 token = tokenizer_->NextToken();
999 if (token->IsCloseBracket()) {
1000 // Close bracket without an open
1001 if (!got_rect_open_bracket)
1002 return Result("Missing open bracket for probe command");
1003
1004 token = tokenizer_->NextToken();
1005 } else if (got_rect_open_bracket) {
1006 // An open bracket without a close bracket.
1007 return Result("Missing close bracket for probe command");
1008 }
1009 }
1010
1011 bool got_color_open_bracket = false;
1012 if (token->IsOpenBracket()) {
1013 got_color_open_bracket = true;
1014 token = tokenizer_->NextToken();
1015 }
1016
1017 Result r = token->ConvertToDouble();
1018 if (!r.IsSuccess())
1019 return r;
1020 cmd->SetR(token->AsFloat());
1021
1022 token = tokenizer_->NextToken();
1023 if (token->IsComma())
1024 token = tokenizer_->NextToken();
1025
1026 r = token->ConvertToDouble();
1027 if (!r.IsSuccess())
1028 return r;
1029 cmd->SetG(token->AsFloat());
1030
1031 token = tokenizer_->NextToken();
1032 if (token->IsComma())
1033 token = tokenizer_->NextToken();
1034
1035 r = token->ConvertToDouble();
1036 if (!r.IsSuccess())
1037 return r;
1038 cmd->SetB(token->AsFloat());
1039
1040 if (format == "rgba") {
1041 token = tokenizer_->NextToken();
1042 if (token->IsComma())
1043 token = tokenizer_->NextToken();
1044
1045 r = token->ConvertToDouble();
1046 if (!r.IsSuccess())
1047 return r;
1048 cmd->SetA(token->AsFloat());
1049 }
1050
1051 token = tokenizer_->NextToken();
1052 if (token->IsCloseBracket()) {
1053 if (!got_color_open_bracket) {
1054 // Close without an open.
1055 return Result("Missing open bracket for probe command");
1056 }
1057 token = tokenizer_->NextToken();
1058 } else if (got_color_open_bracket) {
1059 // Open bracket without a close.
1060 return Result("Missing close bracket for probe command");
1061 }
1062
1063 if (!token->IsEOS() && !token->IsEOL())
1064 return Result("Extra parameter to probe command: " +
1065 token->ToOriginalString());
1066
1067 commands_.push_back(std::move(cmd));
1068 return {};
1069 }
1070
ProcessTopology()1071 Result CommandParser::ProcessTopology() {
1072 auto token = tokenizer_->NextToken();
1073 if (token->IsEOS() || token->IsEOL())
1074 return Result("Missing value for topology command");
1075 if (!token->IsIdentifier())
1076 return Result("Invalid value for topology command: " +
1077 token->ToOriginalString());
1078
1079 Topology topology = Topology::kPatchList;
1080 std::string topo = token->AsString();
1081
1082 if (topo == "VK_PRIMITIVE_TOPOLOGY_PATCH_LIST")
1083 topology = Topology::kPatchList;
1084 else if (topo == "VK_PRIMITIVE_TOPOLOGY_POINT_LIST")
1085 topology = Topology::kPointList;
1086 else if (topo == "VK_PRIMITIVE_TOPOLOGY_LINE_LIST")
1087 topology = Topology::kLineList;
1088 else if (topo == "VK_PRIMITIVE_TOPOLOGY_LINE_LIST_WITH_ADJACENCY")
1089 topology = Topology::kLineListWithAdjacency;
1090 else if (topo == "VK_PRIMITIVE_TOPOLOGY_LINE_STRIP")
1091 topology = Topology::kLineStrip;
1092 else if (topo == "VK_PRIMITIVE_TOPOLOGY_LINE_STRIP_WITH_ADJACENCY")
1093 topology = Topology::kLineStripWithAdjacency;
1094 else if (topo == "VK_PRIMITIVE_TOPOLOGY_TRIANGLE_FAN")
1095 topology = Topology::kTriangleFan;
1096 else if (topo == "VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST")
1097 topology = Topology::kTriangleList;
1098 else if (topo == "VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST_WITH_ADJACENCY")
1099 topology = Topology::kTriangleListWithAdjacency;
1100 else if (topo == "VK_PRIMITIVE_TOPOLOGY_TRIANGLE_STRIP")
1101 topology = Topology::kTriangleStrip;
1102 else if (topo == "VK_PRIMITIVE_TOPOLOGY_TRIANGLE_STRIP_WITH_ADJACENCY")
1103 topology = Topology::kTriangleStripWithAdjacency;
1104 else
1105 return Result("Unknown value for topology command: " +
1106 token->ToOriginalString());
1107
1108 token = tokenizer_->NextToken();
1109 if (!token->IsEOS() && !token->IsEOL())
1110 return Result("Extra parameter for topology command: " +
1111 token->ToOriginalString());
1112
1113 pipeline_data_.SetTopology(topology);
1114 return {};
1115 }
1116
ProcessPolygonMode()1117 Result CommandParser::ProcessPolygonMode() {
1118 auto token = tokenizer_->NextToken();
1119 if (token->IsEOS() || token->IsEOL())
1120 return Result("Missing value for polygonMode command");
1121 if (!token->IsIdentifier())
1122 return Result("Invalid value for polygonMode command: " +
1123 token->ToOriginalString());
1124
1125 PolygonMode mode = PolygonMode::kFill;
1126 std::string m = token->AsString();
1127 if (m == "VK_POLYGON_MODE_FILL")
1128 mode = PolygonMode::kFill;
1129 else if (m == "VK_POLYGON_MODE_LINE")
1130 mode = PolygonMode::kLine;
1131 else if (m == "VK_POLYGON_MODE_POINT")
1132 mode = PolygonMode::kPoint;
1133 else
1134 return Result("Unknown value for polygonMode command: " +
1135 token->ToOriginalString());
1136
1137 token = tokenizer_->NextToken();
1138 if (!token->IsEOS() && !token->IsEOL())
1139 return Result("Extra parameter for polygonMode command: " +
1140 token->ToOriginalString());
1141
1142 pipeline_data_.SetPolygonMode(mode);
1143 return {};
1144 }
1145
ProcessLogicOp()1146 Result CommandParser::ProcessLogicOp() {
1147 auto token = tokenizer_->NextToken();
1148 if (token->IsEOS() || token->IsEOL())
1149 return Result("Missing value for logicOp command");
1150 if (!token->IsIdentifier())
1151 return Result("Invalid value for logicOp command: " +
1152 token->ToOriginalString());
1153
1154 LogicOp op = LogicOp::kClear;
1155 std::string name = token->AsString();
1156 if (name == "VK_LOGIC_OP_CLEAR")
1157 op = LogicOp::kClear;
1158 else if (name == "VK_LOGIC_OP_AND")
1159 op = LogicOp::kAnd;
1160 else if (name == "VK_LOGIC_OP_AND_REVERSE")
1161 op = LogicOp::kAndReverse;
1162 else if (name == "VK_LOGIC_OP_COPY")
1163 op = LogicOp::kCopy;
1164 else if (name == "VK_LOGIC_OP_AND_INVERTED")
1165 op = LogicOp::kAndInverted;
1166 else if (name == "VK_LOGIC_OP_NO_OP")
1167 op = LogicOp::kNoOp;
1168 else if (name == "VK_LOGIC_OP_XOR")
1169 op = LogicOp::kXor;
1170 else if (name == "VK_LOGIC_OP_OR")
1171 op = LogicOp::kOr;
1172 else if (name == "VK_LOGIC_OP_NOR")
1173 op = LogicOp::kNor;
1174 else if (name == "VK_LOGIC_OP_EQUIVALENT")
1175 op = LogicOp::kEquivalent;
1176 else if (name == "VK_LOGIC_OP_INVERT")
1177 op = LogicOp::kInvert;
1178 else if (name == "VK_LOGIC_OP_OR_REVERSE")
1179 op = LogicOp::kOrReverse;
1180 else if (name == "VK_LOGIC_OP_COPY_INVERTED")
1181 op = LogicOp::kCopyInverted;
1182 else if (name == "VK_LOGIC_OP_OR_INVERTED")
1183 op = LogicOp::kOrInverted;
1184 else if (name == "VK_LOGIC_OP_NAND")
1185 op = LogicOp::kNand;
1186 else if (name == "VK_LOGIC_OP_SET")
1187 op = LogicOp::kSet;
1188 else
1189 return Result("Unknown value for logicOp command: " +
1190 token->ToOriginalString());
1191
1192 token = tokenizer_->NextToken();
1193 if (!token->IsEOS() && !token->IsEOL())
1194 return Result("Extra parameter for logicOp command: " +
1195 token->ToOriginalString());
1196
1197 pipeline_data_.SetLogicOp(op);
1198 return {};
1199 }
1200
ProcessCullMode()1201 Result CommandParser::ProcessCullMode() {
1202 auto token = tokenizer_->NextToken();
1203 if (token->IsEOS() || token->IsEOL())
1204 return Result("Missing value for cullMode command");
1205 if (!token->IsIdentifier())
1206 return Result("Invalid value for cullMode command: " +
1207 token->ToOriginalString());
1208
1209 CullMode mode = CullMode::kNone;
1210 while (!token->IsEOS() && !token->IsEOL()) {
1211 std::string name = token->AsString();
1212
1213 if (name == "|") {
1214 // We treat everything as an |.
1215 } else if (name == "VK_CULL_MODE_FRONT_BIT") {
1216 if (mode == CullMode::kNone)
1217 mode = CullMode::kFront;
1218 else if (mode == CullMode::kBack)
1219 mode = CullMode::kFrontAndBack;
1220 } else if (name == "VK_CULL_MODE_BACK_BIT") {
1221 if (mode == CullMode::kNone)
1222 mode = CullMode::kBack;
1223 else if (mode == CullMode::kFront)
1224 mode = CullMode::kFrontAndBack;
1225 } else if (name == "VK_CULL_MODE_FRONT_AND_BACK") {
1226 mode = CullMode::kFrontAndBack;
1227 } else if (name == "VK_CULL_MODE_NONE") {
1228 // Do nothing ...
1229 } else {
1230 return Result("Unknown value for cullMode command: " +
1231 token->ToOriginalString());
1232 }
1233
1234 token = tokenizer_->NextToken();
1235 }
1236
1237 pipeline_data_.SetCullMode(mode);
1238 return {};
1239 }
1240
ProcessFrontFace()1241 Result CommandParser::ProcessFrontFace() {
1242 auto token = tokenizer_->NextToken();
1243 if (token->IsEOS() || token->IsEOL())
1244 return Result("Missing value for frontFace command");
1245 if (!token->IsIdentifier())
1246 return Result("Invalid value for frontFace command: " +
1247 token->ToOriginalString());
1248
1249 FrontFace face = FrontFace::kCounterClockwise;
1250 std::string f = token->AsString();
1251 if (f == "VK_FRONT_FACE_COUNTER_CLOCKWISE")
1252 face = FrontFace::kCounterClockwise;
1253 else if (f == "VK_FRONT_FACE_CLOCKWISE")
1254 face = FrontFace::kClockwise;
1255 else
1256 return Result("Unknown value for frontFace command: " +
1257 token->ToOriginalString());
1258
1259 token = tokenizer_->NextToken();
1260 if (!token->IsEOS() && !token->IsEOL())
1261 return Result("Extra parameter for frontFace command: " +
1262 token->ToOriginalString());
1263
1264 pipeline_data_.SetFrontFace(face);
1265 return {};
1266 }
1267
ProcessBooleanPipelineData(const std::string & name,bool * value)1268 Result CommandParser::ProcessBooleanPipelineData(const std::string& name,
1269 bool* value) {
1270 auto token = tokenizer_->NextToken();
1271 if (token->IsEOS() || token->IsEOL())
1272 return Result("Missing value for " + name + " command");
1273 if (!token->IsIdentifier())
1274 return Result("Invalid value for " + name +
1275 " command: " + token->ToOriginalString());
1276
1277 Result r = ParseBoolean(token->AsString(), value);
1278 if (!r.IsSuccess())
1279 return r;
1280
1281 token = tokenizer_->NextToken();
1282 if (!token->IsEOS() && !token->IsEOL())
1283 return Result("Extra parameter for " + name +
1284 " command: " + token->ToOriginalString());
1285
1286 return {};
1287 }
1288
ProcessPrimitiveRestartEnable()1289 Result CommandParser::ProcessPrimitiveRestartEnable() {
1290 bool value = false;
1291 Result r = ProcessBooleanPipelineData("primitiveRestartEnable", &value);
1292 if (!r.IsSuccess())
1293 return r;
1294
1295 pipeline_data_.SetEnablePrimitiveRestart(value);
1296 return {};
1297 }
1298
ProcessDepthClampEnable()1299 Result CommandParser::ProcessDepthClampEnable() {
1300 bool value = false;
1301 Result r = ProcessBooleanPipelineData("depthClampEnable", &value);
1302 if (!r.IsSuccess())
1303 return r;
1304
1305 pipeline_data_.SetEnableDepthClamp(value);
1306 return {};
1307 }
1308
ProcessRasterizerDiscardEnable()1309 Result CommandParser::ProcessRasterizerDiscardEnable() {
1310 bool value = false;
1311 Result r = ProcessBooleanPipelineData("rasterizerDiscardEnable", &value);
1312 if (!r.IsSuccess())
1313 return r;
1314
1315 pipeline_data_.SetEnableRasterizerDiscard(value);
1316 return {};
1317 }
1318
ProcessDepthBiasEnable()1319 Result CommandParser::ProcessDepthBiasEnable() {
1320 bool value = false;
1321 Result r = ProcessBooleanPipelineData("depthBiasEnable", &value);
1322 if (!r.IsSuccess())
1323 return r;
1324
1325 pipeline_data_.SetEnableDepthBias(value);
1326 return {};
1327 }
1328
ProcessLogicOpEnable()1329 Result CommandParser::ProcessLogicOpEnable() {
1330 bool value = false;
1331 Result r = ProcessBooleanPipelineData("logicOpEnable", &value);
1332 if (!r.IsSuccess())
1333 return r;
1334
1335 pipeline_data_.SetEnableLogicOp(value);
1336 return {};
1337 }
1338
ProcessBlendEnable()1339 Result CommandParser::ProcessBlendEnable() {
1340 bool value = false;
1341 Result r = ProcessBooleanPipelineData("blendEnable", &value);
1342 if (!r.IsSuccess())
1343 return r;
1344
1345 pipeline_data_.SetEnableBlend(value);
1346 return {};
1347 }
1348
ProcessDepthTestEnable()1349 Result CommandParser::ProcessDepthTestEnable() {
1350 bool value = false;
1351 Result r = ProcessBooleanPipelineData("depthTestEnable", &value);
1352 if (!r.IsSuccess())
1353 return r;
1354
1355 pipeline_data_.SetEnableDepthTest(value);
1356 return {};
1357 }
1358
ProcessDepthWriteEnable()1359 Result CommandParser::ProcessDepthWriteEnable() {
1360 bool value = false;
1361 Result r = ProcessBooleanPipelineData("depthWriteEnable", &value);
1362 if (!r.IsSuccess())
1363 return r;
1364
1365 pipeline_data_.SetEnableDepthWrite(value);
1366 return {};
1367 }
1368
ProcessDepthBoundsTestEnable()1369 Result CommandParser::ProcessDepthBoundsTestEnable() {
1370 bool value = false;
1371 Result r = ProcessBooleanPipelineData("depthBoundsTestEnable", &value);
1372 if (!r.IsSuccess())
1373 return r;
1374
1375 pipeline_data_.SetEnableDepthBoundsTest(value);
1376 return {};
1377 }
1378
ProcessStencilTestEnable()1379 Result CommandParser::ProcessStencilTestEnable() {
1380 bool value = false;
1381 Result r = ProcessBooleanPipelineData("stencilTestEnable", &value);
1382 if (!r.IsSuccess())
1383 return r;
1384
1385 pipeline_data_.SetEnableStencilTest(value);
1386 return {};
1387 }
1388
ProcessFloatPipelineData(const std::string & name,float * value)1389 Result CommandParser::ProcessFloatPipelineData(const std::string& name,
1390 float* value) {
1391 assert(value);
1392
1393 auto token = tokenizer_->NextToken();
1394 if (token->IsEOS() || token->IsEOL())
1395 return Result("Missing value for " + name + " command");
1396
1397 Result r = token->ConvertToDouble();
1398 if (!r.IsSuccess())
1399 return r;
1400
1401 *value = token->AsFloat();
1402
1403 token = tokenizer_->NextToken();
1404 if (!token->IsEOS() && !token->IsEOL())
1405 return Result("Extra parameter for " + name +
1406 " command: " + token->ToOriginalString());
1407
1408 return {};
1409 }
1410
ProcessDepthBiasConstantFactor()1411 Result CommandParser::ProcessDepthBiasConstantFactor() {
1412 float value = 0.0;
1413 Result r = ProcessFloatPipelineData("depthBiasConstantFactor", &value);
1414 if (!r.IsSuccess())
1415 return r;
1416
1417 pipeline_data_.SetDepthBiasConstantFactor(value);
1418 return {};
1419 }
1420
ProcessDepthBiasClamp()1421 Result CommandParser::ProcessDepthBiasClamp() {
1422 float value = 0.0;
1423 Result r = ProcessFloatPipelineData("depthBiasClamp", &value);
1424 if (!r.IsSuccess())
1425 return r;
1426
1427 pipeline_data_.SetDepthBiasClamp(value);
1428 return {};
1429 }
1430
ProcessDepthBiasSlopeFactor()1431 Result CommandParser::ProcessDepthBiasSlopeFactor() {
1432 float value = 0.0;
1433 Result r = ProcessFloatPipelineData("depthBiasSlopeFactor", &value);
1434 if (!r.IsSuccess())
1435 return r;
1436
1437 pipeline_data_.SetDepthBiasSlopeFactor(value);
1438 return {};
1439 }
1440
ProcessLineWidth()1441 Result CommandParser::ProcessLineWidth() {
1442 float value = 0.0;
1443 Result r = ProcessFloatPipelineData("lineWidth", &value);
1444 if (!r.IsSuccess())
1445 return r;
1446
1447 pipeline_data_.SetLineWidth(value);
1448 return {};
1449 }
1450
ProcessMinDepthBounds()1451 Result CommandParser::ProcessMinDepthBounds() {
1452 float value = 0.0;
1453 Result r = ProcessFloatPipelineData("minDepthBounds", &value);
1454 if (!r.IsSuccess())
1455 return r;
1456
1457 pipeline_data_.SetMinDepthBounds(value);
1458 return {};
1459 }
1460
ProcessMaxDepthBounds()1461 Result CommandParser::ProcessMaxDepthBounds() {
1462 float value = 0.0;
1463 Result r = ProcessFloatPipelineData("maxDepthBounds", &value);
1464 if (!r.IsSuccess())
1465 return r;
1466
1467 pipeline_data_.SetMaxDepthBounds(value);
1468 return {};
1469 }
1470
ParseBlendFactorName(const std::string & name,BlendFactor * factor)1471 Result CommandParser::ParseBlendFactorName(const std::string& name,
1472 BlendFactor* factor) {
1473 assert(factor);
1474
1475 if (name == "VK_BLEND_FACTOR_ZERO")
1476 *factor = BlendFactor::kZero;
1477 else if (name == "VK_BLEND_FACTOR_ONE")
1478 *factor = BlendFactor::kOne;
1479 else if (name == "VK_BLEND_FACTOR_SRC_COLOR")
1480 *factor = BlendFactor::kSrcColor;
1481 else if (name == "VK_BLEND_FACTOR_ONE_MINUS_SRC_COLOR")
1482 *factor = BlendFactor::kOneMinusSrcColor;
1483 else if (name == "VK_BLEND_FACTOR_DST_COLOR")
1484 *factor = BlendFactor::kDstColor;
1485 else if (name == "VK_BLEND_FACTOR_ONE_MINUS_DST_COLOR")
1486 *factor = BlendFactor::kOneMinusDstColor;
1487 else if (name == "VK_BLEND_FACTOR_SRC_ALPHA")
1488 *factor = BlendFactor::kSrcAlpha;
1489 else if (name == "VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA")
1490 *factor = BlendFactor::kOneMinusSrcAlpha;
1491 else if (name == "VK_BLEND_FACTOR_DST_ALPHA")
1492 *factor = BlendFactor::kDstAlpha;
1493 else if (name == "VK_BLEND_FACTOR_ONE_MINUS_DST_ALPHA")
1494 *factor = BlendFactor::kOneMinusDstAlpha;
1495 else if (name == "VK_BLEND_FACTOR_CONSTANT_COLOR")
1496 *factor = BlendFactor::kConstantColor;
1497 else if (name == "VK_BLEND_FACTOR_ONE_MINUS_CONSTANT_COLOR")
1498 *factor = BlendFactor::kOneMinusConstantColor;
1499 else if (name == "VK_BLEND_FACTOR_CONSTANT_ALPHA")
1500 *factor = BlendFactor::kConstantAlpha;
1501 else if (name == "VK_BLEND_FACTOR_ONE_MINUS_CONSTANT_ALPHA")
1502 *factor = BlendFactor::kOneMinusConstantAlpha;
1503 else if (name == "VK_BLEND_FACTOR_SRC_ALPHA_SATURATE")
1504 *factor = BlendFactor::kSrcAlphaSaturate;
1505 else if (name == "VK_BLEND_FACTOR_SRC1_COLOR")
1506 *factor = BlendFactor::kSrc1Color;
1507 else if (name == "VK_BLEND_FACTOR_ONE_MINUS_SRC1_COLOR")
1508 *factor = BlendFactor::kOneMinusSrc1Color;
1509 else if (name == "VK_BLEND_FACTOR_SRC1_ALPHA")
1510 *factor = BlendFactor::kSrc1Alpha;
1511 else if (name == "VK_BLEND_FACTOR_ONE_MINUS_SRC1_ALPHA")
1512 *factor = BlendFactor::kOneMinusSrc1Alpha;
1513 else
1514 return Result("Unknown BlendFactor provided: " + name);
1515
1516 return {};
1517 }
1518
ParseBlendFactor(const std::string & name,BlendFactor * factor)1519 Result CommandParser::ParseBlendFactor(const std::string& name,
1520 BlendFactor* factor) {
1521 auto token = tokenizer_->NextToken();
1522 if (token->IsEOL() || token->IsEOS())
1523 return Result(std::string("Missing parameter for ") + name + " command");
1524 if (!token->IsIdentifier())
1525 return Result(std::string("Invalid parameter for ") + name +
1526 " command: " + token->ToOriginalString());
1527
1528 Result r = ParseBlendFactorName(token->AsString(), factor);
1529 if (!r.IsSuccess())
1530 return r;
1531
1532 token = tokenizer_->NextToken();
1533 if (!token->IsEOS() && !token->IsEOL())
1534 return Result(std::string("Extra parameter for ") + name +
1535 " command: " + token->ToOriginalString());
1536
1537 return {};
1538 }
1539
ProcessSrcAlphaBlendFactor()1540 Result CommandParser::ProcessSrcAlphaBlendFactor() {
1541 BlendFactor factor = BlendFactor::kZero;
1542 Result r = ParseBlendFactor("srcAlphaBlendFactor", &factor);
1543 if (!r.IsSuccess())
1544 return r;
1545
1546 pipeline_data_.SetSrcAlphaBlendFactor(factor);
1547 return {};
1548 }
1549
ProcessDstAlphaBlendFactor()1550 Result CommandParser::ProcessDstAlphaBlendFactor() {
1551 BlendFactor factor = BlendFactor::kZero;
1552 Result r = ParseBlendFactor("dstAlphaBlendFactor", &factor);
1553 if (!r.IsSuccess())
1554 return r;
1555
1556 pipeline_data_.SetDstAlphaBlendFactor(factor);
1557 return {};
1558 }
1559
ProcessSrcColorBlendFactor()1560 Result CommandParser::ProcessSrcColorBlendFactor() {
1561 BlendFactor factor = BlendFactor::kZero;
1562 Result r = ParseBlendFactor("srcColorBlendFactor", &factor);
1563 if (!r.IsSuccess())
1564 return r;
1565
1566 pipeline_data_.SetSrcColorBlendFactor(factor);
1567 return {};
1568 }
1569
ProcessDstColorBlendFactor()1570 Result CommandParser::ProcessDstColorBlendFactor() {
1571 BlendFactor factor = BlendFactor::kZero;
1572 Result r = ParseBlendFactor("dstColorBlendFactor", &factor);
1573 if (!r.IsSuccess())
1574 return r;
1575
1576 pipeline_data_.SetDstColorBlendFactor(factor);
1577 return {};
1578 }
1579
ParseBlendOpName(const std::string & name,BlendOp * op)1580 Result CommandParser::ParseBlendOpName(const std::string& name, BlendOp* op) {
1581 assert(op);
1582
1583 if (name == "VK_BLEND_OP_ADD")
1584 *op = BlendOp::kAdd;
1585 else if (name == "VK_BLEND_OP_ADD")
1586 *op = BlendOp::kAdd;
1587 else if (name == "VK_BLEND_OP_SUBTRACT")
1588 *op = BlendOp::kSubtract;
1589 else if (name == "VK_BLEND_OP_REVERSE_SUBTRACT")
1590 *op = BlendOp::kReverseSubtract;
1591 else if (name == "VK_BLEND_OP_MIN")
1592 *op = BlendOp::kMin;
1593 else if (name == "VK_BLEND_OP_MAX")
1594 *op = BlendOp::kMax;
1595 else if (name == "VK_BLEND_OP_ZERO_EXT")
1596 *op = BlendOp::kZero;
1597 else if (name == "VK_BLEND_OP_SRC_EXT")
1598 *op = BlendOp::kSrc;
1599 else if (name == "VK_BLEND_OP_DST_EXT")
1600 *op = BlendOp::kDst;
1601 else if (name == "VK_BLEND_OP_SRC_OVER_EXT")
1602 *op = BlendOp::kSrcOver;
1603 else if (name == "VK_BLEND_OP_DST_OVER_EXT")
1604 *op = BlendOp::kDstOver;
1605 else if (name == "VK_BLEND_OP_SRC_IN_EXT")
1606 *op = BlendOp::kSrcIn;
1607 else if (name == "VK_BLEND_OP_DST_IN_EXT")
1608 *op = BlendOp::kDstIn;
1609 else if (name == "VK_BLEND_OP_SRC_OUT_EXT")
1610 *op = BlendOp::kSrcOut;
1611 else if (name == "VK_BLEND_OP_DST_OUT_EXT")
1612 *op = BlendOp::kDstOut;
1613 else if (name == "VK_BLEND_OP_SRC_ATOP_EXT")
1614 *op = BlendOp::kSrcAtop;
1615 else if (name == "VK_BLEND_OP_DST_ATOP_EXT")
1616 *op = BlendOp::kDstAtop;
1617 else if (name == "VK_BLEND_OP_XOR_EXT")
1618 *op = BlendOp::kXor;
1619 else if (name == "VK_BLEND_OP_MULTIPLY_EXT")
1620 *op = BlendOp::kMultiply;
1621 else if (name == "VK_BLEND_OP_SCREEN_EXT")
1622 *op = BlendOp::kScreen;
1623 else if (name == "VK_BLEND_OP_OVERLAY_EXT")
1624 *op = BlendOp::kOverlay;
1625 else if (name == "VK_BLEND_OP_DARKEN_EXT")
1626 *op = BlendOp::kDarken;
1627 else if (name == "VK_BLEND_OP_LIGHTEN_EXT")
1628 *op = BlendOp::kLighten;
1629 else if (name == "VK_BLEND_OP_COLORDODGE_EXT")
1630 *op = BlendOp::kColorDodge;
1631 else if (name == "VK_BLEND_OP_COLORBURN_EXT")
1632 *op = BlendOp::kColorBurn;
1633 else if (name == "VK_BLEND_OP_HARDLIGHT_EXT")
1634 *op = BlendOp::kHardLight;
1635 else if (name == "VK_BLEND_OP_SOFTLIGHT_EXT")
1636 *op = BlendOp::kSoftLight;
1637 else if (name == "VK_BLEND_OP_DIFFERENCE_EXT")
1638 *op = BlendOp::kDifference;
1639 else if (name == "VK_BLEND_OP_EXCLUSION_EXT")
1640 *op = BlendOp::kExclusion;
1641 else if (name == "VK_BLEND_OP_INVERT_EXT")
1642 *op = BlendOp::kInvert;
1643 else if (name == "VK_BLEND_OP_INVERT_RGB_EXT")
1644 *op = BlendOp::kInvertRGB;
1645 else if (name == "VK_BLEND_OP_LINEARDODGE_EXT")
1646 *op = BlendOp::kLinearDodge;
1647 else if (name == "VK_BLEND_OP_LINEARBURN_EXT")
1648 *op = BlendOp::kLinearBurn;
1649 else if (name == "VK_BLEND_OP_VIVIDLIGHT_EXT")
1650 *op = BlendOp::kVividLight;
1651 else if (name == "VK_BLEND_OP_LINEARLIGHT_EXT")
1652 *op = BlendOp::kLinearLight;
1653 else if (name == "VK_BLEND_OP_PINLIGHT_EXT")
1654 *op = BlendOp::kPinLight;
1655 else if (name == "VK_BLEND_OP_HARDMIX_EXT")
1656 *op = BlendOp::kHardMix;
1657 else if (name == "VK_BLEND_OP_HSL_HUE_EXT")
1658 *op = BlendOp::kHslHue;
1659 else if (name == "VK_BLEND_OP_HSL_SATURATION_EXT")
1660 *op = BlendOp::kHslSaturation;
1661 else if (name == "VK_BLEND_OP_HSL_COLOR_EXT")
1662 *op = BlendOp::kHslColor;
1663 else if (name == "VK_BLEND_OP_HSL_LUMINOSITY_EXT")
1664 *op = BlendOp::kHslLuminosity;
1665 else if (name == "VK_BLEND_OP_PLUS_EXT")
1666 *op = BlendOp::kPlus;
1667 else if (name == "VK_BLEND_OP_PLUS_CLAMPED_EXT")
1668 *op = BlendOp::kPlusClamped;
1669 else if (name == "VK_BLEND_OP_PLUS_CLAMPED_ALPHA_EXT")
1670 *op = BlendOp::kPlusClampedAlpha;
1671 else if (name == "VK_BLEND_OP_PLUS_DARKER_EXT")
1672 *op = BlendOp::kPlusDarker;
1673 else if (name == "VK_BLEND_OP_MINUS_EXT")
1674 *op = BlendOp::kMinus;
1675 else if (name == "VK_BLEND_OP_MINUS_CLAMPED_EXT")
1676 *op = BlendOp::kMinusClamped;
1677 else if (name == "VK_BLEND_OP_CONTRAST_EXT")
1678 *op = BlendOp::kContrast;
1679 else if (name == "VK_BLEND_OP_INVERT_OVG_EXT")
1680 *op = BlendOp::kInvertOvg;
1681 else if (name == "VK_BLEND_OP_RED_EXT")
1682 *op = BlendOp::kRed;
1683 else if (name == "VK_BLEND_OP_GREEN_EXT")
1684 *op = BlendOp::kGreen;
1685 else if (name == "VK_BLEND_OP_BLUE_EXT")
1686 *op = BlendOp::kBlue;
1687 else
1688 return Result("Unknown BlendOp provided: " + name);
1689
1690 return {};
1691 }
1692
ParseBlendOp(const std::string & name,BlendOp * op)1693 Result CommandParser::ParseBlendOp(const std::string& name, BlendOp* op) {
1694 auto token = tokenizer_->NextToken();
1695 if (token->IsEOL() || token->IsEOS())
1696 return Result(std::string("Missing parameter for ") + name + " command");
1697 if (!token->IsIdentifier())
1698 return Result(std::string("Invalid parameter for ") + name +
1699 " command: " + token->ToOriginalString());
1700
1701 Result r = ParseBlendOpName(token->AsString(), op);
1702 if (!r.IsSuccess())
1703 return r;
1704
1705 token = tokenizer_->NextToken();
1706 if (!token->IsEOS() && !token->IsEOL())
1707 return Result(std::string("Extra parameter for ") + name +
1708 " command: " + token->ToOriginalString());
1709
1710 return {};
1711 }
1712
ProcessColorBlendOp()1713 Result CommandParser::ProcessColorBlendOp() {
1714 BlendOp op = BlendOp::kAdd;
1715 Result r = ParseBlendOp("colorBlendOp", &op);
1716 if (!r.IsSuccess())
1717 return r;
1718
1719 pipeline_data_.SetColorBlendOp(op);
1720 return {};
1721 }
1722
ProcessAlphaBlendOp()1723 Result CommandParser::ProcessAlphaBlendOp() {
1724 BlendOp op = BlendOp::kAdd;
1725 Result r = ParseBlendOp("alphaBlendOp", &op);
1726 if (!r.IsSuccess())
1727 return r;
1728
1729 pipeline_data_.SetAlphaBlendOp(op);
1730 return {};
1731 }
1732
ParseCompareOp(const std::string & name,CompareOp * op)1733 Result CommandParser::ParseCompareOp(const std::string& name, CompareOp* op) {
1734 auto token = tokenizer_->NextToken();
1735 if (token->IsEOL() || token->IsEOS())
1736 return Result(std::string("Missing parameter for ") + name + " command");
1737 if (!token->IsIdentifier())
1738 return Result(std::string("Invalid parameter for ") + name +
1739 " command: " + token->ToOriginalString());
1740
1741 Result r = ParseCompareOpName(token->AsString(), op);
1742 if (!r.IsSuccess())
1743 return r;
1744
1745 token = tokenizer_->NextToken();
1746 if (!token->IsEOS() && !token->IsEOL())
1747 return Result(std::string("Extra parameter for ") + name +
1748 " command: " + token->ToOriginalString());
1749
1750 return {};
1751 }
1752
ParseCompareOpName(const std::string & name,CompareOp * op)1753 Result CommandParser::ParseCompareOpName(const std::string& name,
1754 CompareOp* op) {
1755 assert(op);
1756
1757 if (name == "VK_COMPARE_OP_NEVER")
1758 *op = CompareOp::kNever;
1759 else if (name == "VK_COMPARE_OP_LESS")
1760 *op = CompareOp::kLess;
1761 else if (name == "VK_COMPARE_OP_EQUAL")
1762 *op = CompareOp::kEqual;
1763 else if (name == "VK_COMPARE_OP_LESS_OR_EQUAL")
1764 *op = CompareOp::kLessOrEqual;
1765 else if (name == "VK_COMPARE_OP_GREATER")
1766 *op = CompareOp::kGreater;
1767 else if (name == "VK_COMPARE_OP_NOT_EQUAL")
1768 *op = CompareOp::kNotEqual;
1769 else if (name == "VK_COMPARE_OP_GREATER_OR_EQUAL")
1770 *op = CompareOp::kGreaterOrEqual;
1771 else if (name == "VK_COMPARE_OP_ALWAYS")
1772 *op = CompareOp::kAlways;
1773 else
1774 return Result("Unknown CompareOp provided: " + name);
1775
1776 return {};
1777 }
1778
ProcessDepthCompareOp()1779 Result CommandParser::ProcessDepthCompareOp() {
1780 CompareOp op = CompareOp::kNever;
1781 Result r = ParseCompareOp("depthCompareOp", &op);
1782 if (!r.IsSuccess())
1783 return r;
1784
1785 pipeline_data_.SetDepthCompareOp(op);
1786 return {};
1787 }
1788
ProcessFrontCompareOp()1789 Result CommandParser::ProcessFrontCompareOp() {
1790 CompareOp op = CompareOp::kNever;
1791 Result r = ParseCompareOp("front.compareOp", &op);
1792 if (!r.IsSuccess())
1793 return r;
1794
1795 pipeline_data_.SetFrontCompareOp(op);
1796 return {};
1797 }
1798
ProcessBackCompareOp()1799 Result CommandParser::ProcessBackCompareOp() {
1800 CompareOp op = CompareOp::kNever;
1801 Result r = ParseCompareOp("back.compareOp", &op);
1802 if (!r.IsSuccess())
1803 return r;
1804
1805 pipeline_data_.SetBackCompareOp(op);
1806 return {};
1807 }
1808
ParseStencilOp(const std::string & name,StencilOp * op)1809 Result CommandParser::ParseStencilOp(const std::string& name, StencilOp* op) {
1810 auto token = tokenizer_->NextToken();
1811 if (token->IsEOL() || token->IsEOS())
1812 return Result(std::string("Missing parameter for ") + name + " command");
1813 if (!token->IsIdentifier())
1814 return Result(std::string("Invalid parameter for ") + name +
1815 " command: " + token->ToOriginalString());
1816
1817 Result r = ParseStencilOpName(token->AsString(), op);
1818 if (!r.IsSuccess())
1819 return r;
1820
1821 token = tokenizer_->NextToken();
1822 if (!token->IsEOS() && !token->IsEOL())
1823 return Result(std::string("Extra parameter for ") + name +
1824 " command: " + token->ToOriginalString());
1825
1826 return {};
1827 }
1828
ParseStencilOpName(const std::string & name,StencilOp * op)1829 Result CommandParser::ParseStencilOpName(const std::string& name,
1830 StencilOp* op) {
1831 assert(op);
1832
1833 if (name == "VK_STENCIL_OP_KEEP")
1834 *op = StencilOp::kKeep;
1835 else if (name == "VK_STENCIL_OP_ZERO")
1836 *op = StencilOp::kZero;
1837 else if (name == "VK_STENCIL_OP_REPLACE")
1838 *op = StencilOp::kReplace;
1839 else if (name == "VK_STENCIL_OP_INCREMENT_AND_CLAMP")
1840 *op = StencilOp::kIncrementAndClamp;
1841 else if (name == "VK_STENCIL_OP_DECREMENT_AND_CLAMP")
1842 *op = StencilOp::kDecrementAndClamp;
1843 else if (name == "VK_STENCIL_OP_INVERT")
1844 *op = StencilOp::kInvert;
1845 else if (name == "VK_STENCIL_OP_INCREMENT_AND_WRAP")
1846 *op = StencilOp::kIncrementAndWrap;
1847 else if (name == "VK_STENCIL_OP_DECREMENT_AND_WRAP")
1848 *op = StencilOp::kDecrementAndWrap;
1849 else
1850 return Result("Unknown StencilOp provided: " + name);
1851
1852 return {};
1853 }
1854
ProcessFrontFailOp()1855 Result CommandParser::ProcessFrontFailOp() {
1856 StencilOp op = StencilOp::kKeep;
1857 Result r = ParseStencilOp("front.failOp", &op);
1858 if (!r.IsSuccess())
1859 return r;
1860
1861 pipeline_data_.SetFrontFailOp(op);
1862 return {};
1863 }
1864
ProcessFrontPassOp()1865 Result CommandParser::ProcessFrontPassOp() {
1866 StencilOp op = StencilOp::kKeep;
1867 Result r = ParseStencilOp("front.passOp", &op);
1868 if (!r.IsSuccess())
1869 return r;
1870
1871 pipeline_data_.SetFrontPassOp(op);
1872 return {};
1873 }
1874
ProcessFrontDepthFailOp()1875 Result CommandParser::ProcessFrontDepthFailOp() {
1876 StencilOp op = StencilOp::kKeep;
1877 Result r = ParseStencilOp("front.depthFailOp", &op);
1878 if (!r.IsSuccess())
1879 return r;
1880
1881 pipeline_data_.SetFrontDepthFailOp(op);
1882 return {};
1883 }
1884
ProcessBackFailOp()1885 Result CommandParser::ProcessBackFailOp() {
1886 StencilOp op = StencilOp::kKeep;
1887 Result r = ParseStencilOp("back.failOp", &op);
1888 if (!r.IsSuccess())
1889 return r;
1890
1891 pipeline_data_.SetBackFailOp(op);
1892 return {};
1893 }
1894
ProcessBackPassOp()1895 Result CommandParser::ProcessBackPassOp() {
1896 StencilOp op = StencilOp::kKeep;
1897 Result r = ParseStencilOp("back.passOp", &op);
1898 if (!r.IsSuccess())
1899 return r;
1900
1901 pipeline_data_.SetBackPassOp(op);
1902 return {};
1903 }
1904
ProcessBackDepthFailOp()1905 Result CommandParser::ProcessBackDepthFailOp() {
1906 StencilOp op = StencilOp::kKeep;
1907 Result r = ParseStencilOp("back.depthFailOp", &op);
1908 if (!r.IsSuccess())
1909 return r;
1910
1911 pipeline_data_.SetBackDepthFailOp(op);
1912 return {};
1913 }
1914
ProcessFrontCompareMask()1915 Result CommandParser::ProcessFrontCompareMask() {
1916 return Result("front.compareMask not implemented");
1917 }
1918
ProcessFrontWriteMask()1919 Result CommandParser::ProcessFrontWriteMask() {
1920 return Result("front.writeMask not implemented");
1921 }
1922
ProcessBackCompareMask()1923 Result CommandParser::ProcessBackCompareMask() {
1924 return Result("back.compareMask not implemented");
1925 }
1926
ProcessBackWriteMask()1927 Result CommandParser::ProcessBackWriteMask() {
1928 return Result("back.writeMask not implemented");
1929 }
1930
ProcessFrontReference()1931 Result CommandParser::ProcessFrontReference() {
1932 auto token = tokenizer_->NextToken();
1933 if (token->IsEOL() || token->IsEOS())
1934 return Result("Missing parameter for front.reference command");
1935 if (!token->IsInteger())
1936 return Result("Invalid parameter for front.reference command: " +
1937 token->ToOriginalString());
1938
1939 pipeline_data_.SetFrontReference(token->AsUint32());
1940
1941 token = tokenizer_->NextToken();
1942 if (!token->IsEOS() && !token->IsEOL())
1943 return Result("Extra parameter for front.reference command: " +
1944 token->ToOriginalString());
1945
1946 return {};
1947 }
1948
ProcessBackReference()1949 Result CommandParser::ProcessBackReference() {
1950 auto token = tokenizer_->NextToken();
1951 if (token->IsEOL() || token->IsEOS())
1952 return Result("Missing parameter for back.reference command");
1953 if (!token->IsInteger())
1954 return Result("Invalid parameter for back.reference command: " +
1955 token->ToOriginalString());
1956
1957 pipeline_data_.SetBackReference(token->AsUint32());
1958
1959 token = tokenizer_->NextToken();
1960 if (!token->IsEOS() && !token->IsEOL())
1961 return Result("Extra parameter for back.reference command: " +
1962 token->ToOriginalString());
1963
1964 return {};
1965 }
1966
ProcessColorWriteMask()1967 Result CommandParser::ProcessColorWriteMask() {
1968 auto token = tokenizer_->NextToken();
1969 if (token->IsEOS() || token->IsEOL())
1970 return Result("Missing parameter for colorWriteMask command");
1971 if (!token->IsIdentifier())
1972 return Result("Invalid parameter for colorWriteMask command: " +
1973 token->ToOriginalString());
1974
1975 uint8_t mask = 0;
1976 while (!token->IsEOS() && !token->IsEOL()) {
1977 std::string name = token->AsString();
1978
1979 if (name == "|") {
1980 // We treat everything as an |.
1981 } else if (name == "VK_COLOR_COMPONENT_R_BIT") {
1982 mask |= kColorMaskR;
1983 } else if (name == "VK_COLOR_COMPONENT_G_BIT") {
1984 mask |= kColorMaskG;
1985 } else if (name == "VK_COLOR_COMPONENT_B_BIT") {
1986 mask |= kColorMaskB;
1987 } else if (name == "VK_COLOR_COMPONENT_A_BIT") {
1988 mask |= kColorMaskA;
1989 } else {
1990 return Result("Unknown parameter for colorWriteMask command: " + name);
1991 }
1992
1993 token = tokenizer_->NextToken();
1994 }
1995
1996 pipeline_data_.SetColorWriteMask(mask);
1997 return {};
1998 }
1999
ParseComparator(const std::string & name,ProbeSSBOCommand::Comparator * op)2000 Result CommandParser::ParseComparator(const std::string& name,
2001 ProbeSSBOCommand::Comparator* op) {
2002 if (name == "==")
2003 *op = ProbeSSBOCommand::Comparator::kEqual;
2004 else if (name == "!=")
2005 *op = ProbeSSBOCommand::Comparator::kNotEqual;
2006 else if (name == "~=")
2007 *op = ProbeSSBOCommand::Comparator::kFuzzyEqual;
2008 else if (name == "<")
2009 *op = ProbeSSBOCommand::Comparator::kLess;
2010 else if (name == "<=")
2011 *op = ProbeSSBOCommand::Comparator::kLessOrEqual;
2012 else if (name == ">")
2013 *op = ProbeSSBOCommand::Comparator::kGreater;
2014 else if (name == ">=")
2015 *op = ProbeSSBOCommand::Comparator::kGreaterOrEqual;
2016 else
2017 return Result("Invalid comparator: " + name);
2018 return {};
2019 }
2020
ProcessProbeSSBO()2021 Result CommandParser::ProcessProbeSSBO() {
2022 size_t cur_line = tokenizer_->GetCurrentLine();
2023
2024 auto token = tokenizer_->NextToken();
2025 if (token->IsEOL() || token->IsEOS())
2026 return Result("Missing values for probe ssbo command");
2027 if (!token->IsIdentifier())
2028 return Result("Invalid type for probe ssbo command: " +
2029 token->ToOriginalString());
2030
2031 DatumTypeParser tp;
2032 auto type = tp.Parse(token->AsString());
2033 if (!type)
2034 return Result("Invalid type provided: " + token->AsString());
2035
2036 token = tokenizer_->NextToken();
2037 if (!token->IsInteger())
2038 return Result("Invalid binding value for probe ssbo command: " +
2039 token->ToOriginalString());
2040
2041 uint32_t val = token->AsUint32();
2042
2043 uint32_t set = 0;
2044 uint32_t binding = 0;
2045 token = tokenizer_->NextToken();
2046 if (token->IsIdentifier()) {
2047 auto& str = token->AsString();
2048 if (str.size() >= 2 && str[0] == ':') {
2049 set = val;
2050
2051 auto substr = str.substr(1, str.size());
2052 uint64_t binding_val = strtoul(substr.c_str(), nullptr, 10);
2053 if (binding_val > std::numeric_limits<uint32_t>::max())
2054 return Result("binding value too large in probe ssbo command: " +
2055 token->ToOriginalString());
2056
2057 binding = static_cast<uint32_t>(binding_val);
2058 } else {
2059 return Result("Invalid value for probe ssbo command: " +
2060 token->ToOriginalString());
2061 }
2062
2063 token = tokenizer_->NextToken();
2064 } else {
2065 binding = val;
2066 }
2067
2068 auto* buffer = pipeline_->GetBufferForBinding(set, binding);
2069 if (!buffer) {
2070 return Result("unable to find buffer at descriptor set " +
2071 std::to_string(set) + " and binding " +
2072 std::to_string(binding));
2073 }
2074
2075 auto fmt = MakeUnique<Format>(type.get());
2076 if (buffer->FormatIsDefault() || !buffer->GetFormat()) {
2077 buffer->SetFormat(fmt.get());
2078 } else if (buffer->GetFormat() && !buffer->GetFormat()->Equal(fmt.get())) {
2079 return Result("probe format does not match buffer format");
2080 }
2081
2082 auto cmd = MakeUnique<ProbeSSBOCommand>(buffer);
2083 cmd->SetLine(cur_line);
2084 cmd->SetTolerances(current_tolerances_);
2085 cmd->SetFormat(fmt.get());
2086 cmd->SetDescriptorSet(set);
2087 cmd->SetBinding(binding);
2088
2089 script_->RegisterFormat(std::move(fmt));
2090 script_->RegisterType(std::move(type));
2091
2092 if (!token->IsInteger())
2093 return Result("Invalid offset for probe ssbo command: " +
2094 token->ToOriginalString());
2095
2096 cmd->SetOffset(token->AsUint32());
2097
2098 token = tokenizer_->NextToken();
2099 if (!token->IsIdentifier())
2100 return Result("Invalid comparator for probe ssbo command: " +
2101 token->ToOriginalString());
2102
2103 ProbeSSBOCommand::Comparator comp = ProbeSSBOCommand::Comparator::kEqual;
2104 Result r = ParseComparator(token->AsString(), &comp);
2105 if (!r.IsSuccess())
2106 return r;
2107
2108 cmd->SetComparator(comp);
2109
2110 std::vector<Value> values;
2111 r = ParseValues("probe ssbo", cmd->GetFormat(), &values);
2112 if (!r.IsSuccess())
2113 return r;
2114
2115 cmd->SetValues(std::move(values));
2116
2117 commands_.push_back(std::move(cmd));
2118 return {};
2119 }
2120
2121 } // namespace vkscript
2122 } // namespace amber
2123