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