1 //===-------------------------- regex.cpp ---------------------------------===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is dual licensed under the MIT and the University of Illinois Open
6 // Source Licenses. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 
10 #include "regex"
11 #include "algorithm"
12 #include "iterator"
13 
14 _LIBCPP_BEGIN_NAMESPACE_STD
15 
16 static
17 const char*
18 make_error_type_string(regex_constants::error_type ecode)
19 {
20     switch (ecode)
21     {
22     case regex_constants::error_collate:
23         return "The expression contained an invalid collating element name.";
24     case regex_constants::error_ctype:
25         return "The expression contained an invalid character class name.";
26     case regex_constants::error_escape:
27         return "The expression contained an invalid escaped character, or a "
28                "trailing escape.";
29     case regex_constants::error_backref:
30         return "The expression contained an invalid back reference.";
31     case regex_constants::error_brack:
32         return "The expression contained mismatched [ and ].";
33     case regex_constants::error_paren:
34         return "The expression contained mismatched ( and ).";
35     case regex_constants::error_brace:
36         return "The expression contained mismatched { and }.";
37     case regex_constants::error_badbrace:
38         return "The expression contained an invalid range in a {} expression.";
39     case regex_constants::error_range:
40         return "The expression contained an invalid character range, "
41                "such as [b-a] in most encodings.";
42     case regex_constants::error_space:
43         return "There was insufficient memory to convert the expression into "
44                "a finite state machine.";
45     case regex_constants::error_badrepeat:
46         return "One of *?+{ was not preceded by a valid regular expression.";
47     case regex_constants::error_complexity:
48         return "The complexity of an attempted match against a regular "
49                "expression exceeded a pre-set level.";
50     case regex_constants::error_stack:
51         return "There was insufficient memory to determine whether the regular "
52                "expression could match the specified character sequence.";
53     case regex_constants::__re_err_grammar:
54         return "An invalid regex grammar has been requested.";
55     case regex_constants::__re_err_empty:
56         return "An empty regex is not allowed in the POSIX grammar.";
57     default:
58         break;
59     }
60     return "Unknown error type";
61 }
62 
63 regex_error::regex_error(regex_constants::error_type ecode)
64     : runtime_error(make_error_type_string(ecode)),
65       __code_(ecode)
66 {}
67 
68 regex_error::~regex_error() throw() {}
69 
70 namespace {
71 
72 struct collationnames
73 {
74     const char* elem_;
75     char char_;
76 };
77 
78 const collationnames collatenames[] =
79 {
80     {"A", 0x41},
81     {"B", 0x42},
82     {"C", 0x43},
83     {"D", 0x44},
84     {"E", 0x45},
85     {"F", 0x46},
86     {"G", 0x47},
87     {"H", 0x48},
88     {"I", 0x49},
89     {"J", 0x4a},
90     {"K", 0x4b},
91     {"L", 0x4c},
92     {"M", 0x4d},
93     {"N", 0x4e},
94     {"NUL", 0x00},
95     {"O", 0x4f},
96     {"P", 0x50},
97     {"Q", 0x51},
98     {"R", 0x52},
99     {"S", 0x53},
100     {"T", 0x54},
101     {"U", 0x55},
102     {"V", 0x56},
103     {"W", 0x57},
104     {"X", 0x58},
105     {"Y", 0x59},
106     {"Z", 0x5a},
107     {"a", 0x61},
108     {"alert", 0x07},
109     {"ampersand", 0x26},
110     {"apostrophe", 0x27},
111     {"asterisk", 0x2a},
112     {"b", 0x62},
113     {"backslash", 0x5c},
114     {"backspace", 0x08},
115     {"c", 0x63},
116     {"carriage-return", 0x0d},
117     {"circumflex", 0x5e},
118     {"circumflex-accent", 0x5e},
119     {"colon", 0x3a},
120     {"comma", 0x2c},
121     {"commercial-at", 0x40},
122     {"d", 0x64},
123     {"dollar-sign", 0x24},
124     {"e", 0x65},
125     {"eight", 0x38},
126     {"equals-sign", 0x3d},
127     {"exclamation-mark", 0x21},
128     {"f", 0x66},
129     {"five", 0x35},
130     {"form-feed", 0x0c},
131     {"four", 0x34},
132     {"full-stop", 0x2e},
133     {"g", 0x67},
134     {"grave-accent", 0x60},
135     {"greater-than-sign", 0x3e},
136     {"h", 0x68},
137     {"hyphen", 0x2d},
138     {"hyphen-minus", 0x2d},
139     {"i", 0x69},
140     {"j", 0x6a},
141     {"k", 0x6b},
142     {"l", 0x6c},
143     {"left-brace", 0x7b},
144     {"left-curly-bracket", 0x7b},
145     {"left-parenthesis", 0x28},
146     {"left-square-bracket", 0x5b},
147     {"less-than-sign", 0x3c},
148     {"low-line", 0x5f},
149     {"m", 0x6d},
150     {"n", 0x6e},
151     {"newline", 0x0a},
152     {"nine", 0x39},
153     {"number-sign", 0x23},
154     {"o", 0x6f},
155     {"one", 0x31},
156     {"p", 0x70},
157     {"percent-sign", 0x25},
158     {"period", 0x2e},
159     {"plus-sign", 0x2b},
160     {"q", 0x71},
161     {"question-mark", 0x3f},
162     {"quotation-mark", 0x22},
163     {"r", 0x72},
164     {"reverse-solidus", 0x5c},
165     {"right-brace", 0x7d},
166     {"right-curly-bracket", 0x7d},
167     {"right-parenthesis", 0x29},
168     {"right-square-bracket", 0x5d},
169     {"s", 0x73},
170     {"semicolon", 0x3b},
171     {"seven", 0x37},
172     {"six", 0x36},
173     {"slash", 0x2f},
174     {"solidus", 0x2f},
175     {"space", 0x20},
176     {"t", 0x74},
177     {"tab", 0x09},
178     {"three", 0x33},
179     {"tilde", 0x7e},
180     {"two", 0x32},
181     {"u", 0x75},
182     {"underscore", 0x5f},
183     {"v", 0x76},
184     {"vertical-line", 0x7c},
185     {"vertical-tab", 0x0b},
186     {"w", 0x77},
187     {"x", 0x78},
188     {"y", 0x79},
189     {"z", 0x7a},
190     {"zero", 0x30}
191 };
192 
193 struct classnames
194 {
195     const char* elem_;
196     regex_traits<char>::char_class_type mask_;
197 };
198 
199 const classnames ClassNames[] =
200 {
201     {"alnum",  ctype_base::alnum},
202     {"alpha",  ctype_base::alpha},
203     {"blank",  ctype_base::blank},
204     {"cntrl",  ctype_base::cntrl},
205     {"d",      ctype_base::digit},
206     {"digit",  ctype_base::digit},
207     {"graph",  ctype_base::graph},
208     {"lower",  ctype_base::lower},
209     {"print",  ctype_base::print},
210     {"punct",  ctype_base::punct},
211     {"s",      ctype_base::space},
212     {"space",  ctype_base::space},
213     {"upper",  ctype_base::upper},
214     {"w",      regex_traits<char>::__regex_word},
215     {"xdigit", ctype_base::xdigit}
216 };
217 
218 struct use_strcmp
219 {
220     bool operator()(const collationnames& x, const char* y)
221         {return strcmp(x.elem_, y) < 0;}
222     bool operator()(const classnames& x, const char* y)
223         {return strcmp(x.elem_, y) < 0;}
224 };
225 
226 }
227 
228 string
229 __get_collation_name(const char* s)
230 {
231     const collationnames* i =
232             _VSTD::lower_bound(begin(collatenames), end(collatenames), s, use_strcmp());
233     string r;
234     if (i != end(collatenames) && strcmp(s, i->elem_) == 0)
235         r = char(i->char_);
236     return r;
237 }
238 
239 regex_traits<char>::char_class_type
240 __get_classname(const char* s, bool __icase)
241 {
242     const classnames* i =
243             _VSTD::lower_bound(begin(ClassNames), end(ClassNames), s, use_strcmp());
244     regex_traits<char>::char_class_type r = 0;
245     if (i != end(ClassNames) && strcmp(s, i->elem_) == 0)
246     {
247         r = i->mask_;
248         if (r == regex_traits<char>::__regex_word)
249             r |= ctype_base::alnum | ctype_base::upper | ctype_base::lower;
250         else if (__icase)
251         {
252             if (r & (ctype_base::lower | ctype_base::upper))
253                 r |= ctype_base::alpha;
254         }
255     }
256     return r;
257 }
258 
259 template <>
260 void
261 __match_any_but_newline<char>::__exec(__state& __s) const
262 {
263     if (__s.__current_ != __s.__last_)
264     {
265         switch (*__s.__current_)
266         {
267         case '\r':
268         case '\n':
269             __s.__do_ = __state::__reject;
270             __s.__node_ = nullptr;
271             break;
272         default:
273             __s.__do_ = __state::__accept_and_consume;
274             ++__s.__current_;
275             __s.__node_ = this->first();
276             break;
277         }
278     }
279     else
280     {
281         __s.__do_ = __state::__reject;
282         __s.__node_ = nullptr;
283     }
284 }
285 
286 template <>
287 void
288 __match_any_but_newline<wchar_t>::__exec(__state& __s) const
289 {
290     if (__s.__current_ != __s.__last_)
291     {
292         switch (*__s.__current_)
293         {
294         case '\r':
295         case '\n':
296         case 0x2028:
297         case 0x2029:
298             __s.__do_ = __state::__reject;
299             __s.__node_ = nullptr;
300             break;
301         default:
302             __s.__do_ = __state::__accept_and_consume;
303             ++__s.__current_;
304             __s.__node_ = this->first();
305             break;
306         }
307     }
308     else
309     {
310         __s.__do_ = __state::__reject;
311         __s.__node_ = nullptr;
312     }
313 }
314 
315 _LIBCPP_END_NAMESPACE_STD
316