1 /*
2  * Copyright 2017 Google Inc.
3  *
4  * Use of this source code is governed by a BSD-style license that can be
5  * found in the LICENSE file.
6  */
7 
8 #ifndef SKSL_SECTIONANDPARAMETERHELPER
9 #define SKSL_SECTIONANDPARAMETERHELPER
10 
11 #include "SkSLErrorReporter.h"
12 #include "ir/SkSLProgram.h"
13 #include "ir/SkSLSection.h"
14 #include "ir/SkSLVarDeclarations.h"
15 #include <unordered_map>
16 #include <vector>
17 
18 namespace SkSL {
19 
20 #define CLASS_SECTION              "class"
21 #define CLONE_SECTION              "clone"
22 #define CONSTRUCTOR_SECTION        "constructor"
23 #define CONSTRUCTOR_CODE_SECTION   "constructorCode"
24 #define CONSTRUCTOR_PARAMS_SECTION "constructorParams"
25 #define COORD_TRANSFORM_SECTION    "coordTransform"
26 #define CPP_SECTION                "cpp"
27 #define CPP_END_SECTION            "cppEnd"
28 #define HEADER_SECTION             "header"
29 #define HEADER_END_SECTION         "headerEnd"
30 #define EMIT_CODE_SECTION          "emitCode"
31 #define FIELDS_SECTION             "fields"
32 #define INITIALIZERS_SECTION       "initializers"
33 #define MAKE_SECTION               "make"
34 #define OPTIMIZATION_FLAGS_SECTION "optimizationFlags"
35 #define SAMPLER_PARAMS_SECTION     "samplerParams"
36 #define SET_DATA_SECTION           "setData"
37 #define TEST_CODE_SECTION          "test"
38 
39 class SectionAndParameterHelper {
40 public:
SectionAndParameterHelper(const Program & program,ErrorReporter & errors)41     SectionAndParameterHelper(const Program& program, ErrorReporter& errors) {
42         for (const auto& p : program) {
43             switch (p.fKind) {
44                 case ProgramElement::kVar_Kind: {
45                     const VarDeclarations& decls = (const VarDeclarations&) p;
46                     for (const auto& raw : decls.fVars) {
47                         const VarDeclaration& decl = (VarDeclaration&) *raw;
48                         if (IsParameter(*decl.fVar)) {
49                             fParameters.push_back(decl.fVar);
50                         }
51                     }
52                     break;
53                 }
54                 case ProgramElement::kSection_Kind: {
55                     const Section& s = (const Section&) p;
56                     if (IsSupportedSection(s.fName.c_str())) {
57                         if (SectionRequiresArgument(s.fName.c_str()) && !s.fArgument.size()) {
58                             errors.error(s.fOffset,
59                                          ("section '@" + s.fName +
60                                           "' requires one parameter").c_str());
61                         }
62                         if (!SectionAcceptsArgument(s.fName.c_str()) && s.fArgument.size()) {
63                             errors.error(s.fOffset,
64                                          ("section '@" + s.fName + "' has no parameters").c_str());
65                         }
66                     } else {
67                         errors.error(s.fOffset,
68                                      ("unsupported section '@" + s.fName + "'").c_str());
69                     }
70                     if (!SectionPermitsDuplicates(s.fName.c_str()) &&
71                             fSections.find(s.fName) != fSections.end()) {
72                         errors.error(s.fOffset,
73                                      ("duplicate section '@" + s.fName + "'").c_str());
74                     }
75                     fSections[s.fName].push_back(&s);
76                     break;
77                 }
78                 default:
79                     break;
80             }
81         }
82     }
83 
getSection(const char * name)84     const Section* getSection(const char* name) {
85         SkASSERT(!SectionPermitsDuplicates(name));
86         auto found = fSections.find(name);
87         if (found == fSections.end()) {
88             return nullptr;
89         }
90         SkASSERT(found->second.size() == 1);
91         return found->second[0];
92     }
93 
getSections(const char * name)94     std::vector<const Section*> getSections(const char* name) {
95         auto found = fSections.find(name);
96         if (found == fSections.end()) {
97             return std::vector<const Section*>();
98         }
99         return found->second;
100     }
101 
getParameters()102     const std::vector<const Variable*>& getParameters() {
103         return fParameters;
104     }
105 
IsParameter(const Variable & var)106     static bool IsParameter(const Variable& var) {
107         return (var.fModifiers.fFlags & Modifiers::kIn_Flag) &&
108                -1 == var.fModifiers.fLayout.fBuiltin;
109     }
110 
IsSupportedSection(const char * name)111     static bool IsSupportedSection(const char* name) {
112         return !strcmp(name, CLASS_SECTION) ||
113                !strcmp(name, CLONE_SECTION) ||
114                !strcmp(name, CONSTRUCTOR_SECTION) ||
115                !strcmp(name, CONSTRUCTOR_CODE_SECTION) ||
116                !strcmp(name, CONSTRUCTOR_PARAMS_SECTION) ||
117                !strcmp(name, COORD_TRANSFORM_SECTION) ||
118                !strcmp(name, CPP_SECTION) ||
119                !strcmp(name, CPP_END_SECTION) ||
120                !strcmp(name, EMIT_CODE_SECTION) ||
121                !strcmp(name, FIELDS_SECTION) ||
122                !strcmp(name, HEADER_SECTION) ||
123                !strcmp(name, HEADER_END_SECTION) ||
124                !strcmp(name, INITIALIZERS_SECTION) ||
125                !strcmp(name, MAKE_SECTION) ||
126                !strcmp(name, OPTIMIZATION_FLAGS_SECTION) ||
127                !strcmp(name, SAMPLER_PARAMS_SECTION) ||
128                !strcmp(name, SET_DATA_SECTION) ||
129                !strcmp(name, TEST_CODE_SECTION);
130     }
131 
SectionAcceptsArgument(const char * name)132     static bool SectionAcceptsArgument(const char* name) {
133         return !strcmp(name, COORD_TRANSFORM_SECTION) ||
134                !strcmp(name, SAMPLER_PARAMS_SECTION) ||
135                !strcmp(name, SET_DATA_SECTION) ||
136                !strcmp(name, TEST_CODE_SECTION);
137     }
138 
SectionRequiresArgument(const char * name)139     static bool SectionRequiresArgument(const char* name) {
140         return !strcmp(name, SAMPLER_PARAMS_SECTION) ||
141                !strcmp(name, SET_DATA_SECTION) ||
142                !strcmp(name, TEST_CODE_SECTION);
143     }
144 
SectionPermitsDuplicates(const char * name)145     static bool SectionPermitsDuplicates(const char* name) {
146         return !strcmp(name, COORD_TRANSFORM_SECTION) ||
147                !strcmp(name, SAMPLER_PARAMS_SECTION);
148     }
149 
150 private:
151     std::vector<const Variable*> fParameters;
152     std::unordered_map<String, std::vector<const Section*>> fSections;
153 };
154 
155 } // namespace SkSL
156 
157 #endif
158