• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 #include "src/sksl/lex/RegexNode.h"
9 
10 #include "src/sksl/lex/NFA.h"
11 
createStates(NFA * nfa,const std::vector<int> & accept) const12 std::vector<int> RegexNode::createStates(NFA* nfa, const std::vector<int>& accept) const {
13     std::vector<int> result;
14     switch (fKind) {
15         case kChar_Kind:
16             result.push_back(nfa->addState(NFAState(fPayload.fChar, accept)));
17             break;
18         case kCharset_Kind: {
19             std::vector<bool> chars;
20             for (const RegexNode& child : fChildren) {
21                 if (child.fKind == kChar_Kind) {
22                     while (chars.size() <= (size_t) child.fPayload.fChar) {
23                         chars.push_back(false);
24                     }
25                     chars[child.fPayload.fChar] = true;
26                 } else {
27                     SkASSERT(child.fKind == kRange_Kind);
28                     while (chars.size() <= (size_t) child.fChildren[1].fPayload.fChar) {
29                         chars.push_back(false);
30                     }
31                     for (char c = child.fChildren[0].fPayload.fChar;
32                          c <= child.fChildren[1].fPayload.fChar;
33                          ++c) {
34                         chars[c] = true;
35                     }
36                 }
37             }
38             result.push_back(nfa->addState(NFAState(fPayload.fBool, chars, accept)));
39             break;
40         }
41         case kConcat_Kind: {
42             std::vector<int> right = fChildren[1].createStates(nfa, accept);
43             result = fChildren[0].createStates(nfa, right);
44             break;
45         }
46         case kDot_Kind:
47             result.push_back(nfa->addState(NFAState(NFAState::kDot_Kind, accept)));
48             break;
49         case kOr_Kind: {
50             std::vector<int> states = fChildren[0].createStates(nfa, accept);
51             result.insert(result.end(), states.begin(), states.end());
52             states = fChildren[1].createStates(nfa, accept);
53             result.insert(result.end(), states.begin(), states.end());
54             break;
55         }
56         case kPlus_Kind: {
57             std::vector<int> next = accept;
58             std::vector<int> placeholder;
59             int id = nfa->addState(NFAState(placeholder));
60             next.push_back(id);
61             result = fChildren[0].createStates(nfa, next);
62             nfa->fStates[id] = NFAState(result);
63             break;
64         }
65         case kQuestion_Kind:
66             result = fChildren[0].createStates(nfa, accept);
67             result.insert(result.end(), accept.begin(), accept.end());
68             break;
69         case kRange_Kind:
70             SkUNREACHABLE;
71         case kStar_Kind: {
72             std::vector<int> next = accept;
73             std::vector<int> placeholder;
74             int id = nfa->addState(NFAState(placeholder));
75             next.push_back(id);
76             result = fChildren[0].createStates(nfa, next);
77             result.insert(result.end(), accept.begin(), accept.end());
78             nfa->fStates[id] = NFAState(result);
79             break;
80         }
81     }
82     return result;
83 }
84 
85 #ifdef SK_DEBUG
description() const86 std::string RegexNode::description() const {
87     switch (fKind) {
88         case kChar_Kind:
89             return std::string(1, fPayload.fChar);
90         case kCharset_Kind: {
91             std::string result("[");
92             if (fPayload.fBool) {
93                 result += "^";
94             }
95             for (const RegexNode& c : fChildren) {
96                 result += c.description();
97             }
98             result += "]";
99             return result;
100         }
101         case kConcat_Kind:
102             return fChildren[0].description() + fChildren[1].description();
103         case kDot_Kind:
104             return ".";
105         case kOr_Kind:
106             return "(" + fChildren[0].description() + "|" + fChildren[1].description() + ")";
107         case kPlus_Kind:
108             return fChildren[0].description() + "+";
109         case kQuestion_Kind:
110             return fChildren[0].description() + "?";
111         case kRange_Kind:
112             return fChildren[0].description() + "-" + fChildren[1].description();
113         case kStar_Kind:
114             return fChildren[0].description() + "*";
115         default:
116             return "<" + std::to_string(fKind) + ">";
117     }
118 }
119 #endif
120