• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2003-2009 Google Inc.  All rights reserved.
2 // Use of this source code is governed by a BSD-style
3 // license that can be found in the LICENSE file.
4 
5 // This is a variant of PCRE's pcrecpp.cc, originally written at Google.
6 // The main changes are the addition of the HitLimit method and
7 // compilation as PCRE in namespace re2.
8 
9 #include <assert.h>
10 #include <ctype.h>
11 #include <errno.h>
12 #include <stdlib.h>
13 #include <string.h>
14 #include <limits>
15 #include <string>
16 #include <utility>
17 
18 #include "util/util.h"
19 #include "util/flags.h"
20 #include "util/logging.h"
21 #include "util/pcre.h"
22 #include "util/strutil.h"
23 
24 // Silence warnings about the wacky formatting in the operator() functions.
25 // Note that we test for Clang first because it defines __GNUC__ as well.
26 #if defined(__clang__)
27 #elif defined(__GNUC__) && __GNUC__ >= 6
28 #pragma GCC diagnostic ignored "-Wmisleading-indentation"
29 #endif
30 
31 #define PCREPORT(level) LOG(level)
32 
33 // Default PCRE limits.
34 // Defaults chosen to allow a plausible amount of CPU and
35 // not exceed main thread stacks.  Note that other threads
36 // often have smaller stacks, and therefore tightening
37 // regexp_stack_limit may frequently be necessary.
38 DEFINE_FLAG(int, regexp_stack_limit, 256 << 10,
39             "default PCRE stack limit (bytes)");
40 DEFINE_FLAG(int, regexp_match_limit, 1000000,
41             "default PCRE match limit (function calls)");
42 
43 #ifndef USEPCRE
44 
45 // Fake just enough of the PCRE API to allow this file to build. :)
46 
47 struct pcre_extra {
48   int flags;
49   int match_limit;
50   int match_limit_recursion;
51 };
52 
53 #define PCRE_EXTRA_MATCH_LIMIT 0
54 #define PCRE_EXTRA_MATCH_LIMIT_RECURSION 0
55 #define PCRE_ANCHORED 0
56 #define PCRE_NOTEMPTY 0
57 #define PCRE_ERROR_NOMATCH 1
58 #define PCRE_ERROR_MATCHLIMIT 2
59 #define PCRE_ERROR_RECURSIONLIMIT 3
60 #define PCRE_INFO_CAPTURECOUNT 0
61 
pcre_free(void *)62 void pcre_free(void*) {
63 }
64 
pcre_compile(const char *,int,const char **,int *,const unsigned char *)65 pcre* pcre_compile(const char*, int, const char**, int*, const unsigned char*) {
66   return NULL;
67 }
68 
pcre_exec(const pcre *,const pcre_extra *,const char *,int,int,int,int *,int)69 int pcre_exec(const pcre*, const pcre_extra*, const char*, int, int, int, int*, int) {
70   return 0;
71 }
72 
pcre_fullinfo(const pcre *,const pcre_extra *,int,void *)73 int pcre_fullinfo(const pcre*, const pcre_extra*, int, void*) {
74   return 0;
75 }
76 
77 #endif
78 
79 namespace re2 {
80 
81 // Maximum number of args we can set
82 static const int kMaxArgs = 16;
83 static const int kVecSize = (1 + kMaxArgs) * 3;  // results + PCRE workspace
84 
85 // Approximate size of a recursive invocation of PCRE's
86 // internal "match()" frame.  This varies depending on the
87 // compiler and architecture, of course, so the constant is
88 // just a conservative estimate.  To find the exact number,
89 // run regexp_unittest with --regexp_stack_limit=0 under
90 // a debugger and look at the frames when it crashes.
91 // The exact frame size was 656 in production on 2008/02/03.
92 static const int kPCREFrameSize = 700;
93 
94 // Special name for missing C++ arguments.
95 PCRE::Arg PCRE::no_more_args((void*)NULL);
96 
97 const PCRE::PartialMatchFunctor PCRE::PartialMatch = { };
98 const PCRE::FullMatchFunctor PCRE::FullMatch = { } ;
99 const PCRE::ConsumeFunctor PCRE::Consume = { };
100 const PCRE::FindAndConsumeFunctor PCRE::FindAndConsume = { };
101 
102 // If a regular expression has no error, its error_ field points here
103 static const std::string empty_string;
104 
Init(const char * pattern,Option options,int match_limit,int stack_limit,bool report_errors)105 void PCRE::Init(const char* pattern, Option options, int match_limit,
106               int stack_limit, bool report_errors) {
107   pattern_ = pattern;
108   options_ = options;
109   match_limit_ = match_limit;
110   stack_limit_ = stack_limit;
111   hit_limit_ = false;
112   error_ = &empty_string;
113   report_errors_ = report_errors;
114   re_full_ = NULL;
115   re_partial_ = NULL;
116 
117   if (options & ~(EnabledCompileOptions | EnabledExecOptions)) {
118     error_ = new std::string("illegal regexp option");
119     PCREPORT(ERROR)
120         << "Error compiling '" << pattern << "': illegal regexp option";
121   } else {
122     re_partial_ = Compile(UNANCHORED);
123     if (re_partial_ != NULL) {
124       re_full_ = Compile(ANCHOR_BOTH);
125     }
126   }
127 }
128 
PCRE(const char * pattern)129 PCRE::PCRE(const char* pattern) {
130   Init(pattern, None, 0, 0, true);
131 }
PCRE(const char * pattern,Option option)132 PCRE::PCRE(const char* pattern, Option option) {
133   Init(pattern, option, 0, 0, true);
134 }
PCRE(const std::string & pattern)135 PCRE::PCRE(const std::string& pattern) {
136   Init(pattern.c_str(), None, 0, 0, true);
137 }
PCRE(const std::string & pattern,Option option)138 PCRE::PCRE(const std::string& pattern, Option option) {
139   Init(pattern.c_str(), option, 0, 0, true);
140 }
PCRE(const std::string & pattern,const PCRE_Options & re_option)141 PCRE::PCRE(const std::string& pattern, const PCRE_Options& re_option) {
142   Init(pattern.c_str(), re_option.option(), re_option.match_limit(),
143        re_option.stack_limit(), re_option.report_errors());
144 }
145 
PCRE(const char * pattern,const PCRE_Options & re_option)146 PCRE::PCRE(const char *pattern, const PCRE_Options& re_option) {
147   Init(pattern, re_option.option(), re_option.match_limit(),
148        re_option.stack_limit(), re_option.report_errors());
149 }
150 
~PCRE()151 PCRE::~PCRE() {
152   if (re_full_ != NULL)         pcre_free(re_full_);
153   if (re_partial_ != NULL)      pcre_free(re_partial_);
154   if (error_ != &empty_string)  delete error_;
155 }
156 
Compile(Anchor anchor)157 pcre* PCRE::Compile(Anchor anchor) {
158   // Special treatment for anchoring.  This is needed because at
159   // runtime pcre only provides an option for anchoring at the
160   // beginning of a string.
161   //
162   // There are three types of anchoring we want:
163   //    UNANCHORED      Compile the original pattern, and use
164   //                    a pcre unanchored match.
165   //    ANCHOR_START    Compile the original pattern, and use
166   //                    a pcre anchored match.
167   //    ANCHOR_BOTH     Tack a "\z" to the end of the original pattern
168   //                    and use a pcre anchored match.
169 
170   const char* error = "";
171   int eoffset;
172   pcre* re;
173   if (anchor != ANCHOR_BOTH) {
174     re = pcre_compile(pattern_.c_str(),
175                       (options_ & EnabledCompileOptions),
176                       &error, &eoffset, NULL);
177   } else {
178     // Tack a '\z' at the end of PCRE.  Parenthesize it first so that
179     // the '\z' applies to all top-level alternatives in the regexp.
180     std::string wrapped = "(?:";  // A non-counting grouping operator
181     wrapped += pattern_;
182     wrapped += ")\\z";
183     re = pcre_compile(wrapped.c_str(),
184                       (options_ & EnabledCompileOptions),
185                       &error, &eoffset, NULL);
186   }
187   if (re == NULL) {
188     if (error_ == &empty_string) error_ = new std::string(error);
189     PCREPORT(ERROR) << "Error compiling '" << pattern_ << "': " << error;
190   }
191   return re;
192 }
193 
194 /***** Convenience interfaces *****/
195 
operator ()(const StringPiece & text,const PCRE & re,const Arg & a0,const Arg & a1,const Arg & a2,const Arg & a3,const Arg & a4,const Arg & a5,const Arg & a6,const Arg & a7,const Arg & a8,const Arg & a9,const Arg & a10,const Arg & a11,const Arg & a12,const Arg & a13,const Arg & a14,const Arg & a15) const196 bool PCRE::FullMatchFunctor::operator ()(const StringPiece& text,
197                                        const PCRE& re,
198                                        const Arg& a0,
199                                        const Arg& a1,
200                                        const Arg& a2,
201                                        const Arg& a3,
202                                        const Arg& a4,
203                                        const Arg& a5,
204                                        const Arg& a6,
205                                        const Arg& a7,
206                                        const Arg& a8,
207                                        const Arg& a9,
208                                        const Arg& a10,
209                                        const Arg& a11,
210                                        const Arg& a12,
211                                        const Arg& a13,
212                                        const Arg& a14,
213                                        const Arg& a15) const {
214   const Arg* args[kMaxArgs];
215   int n = 0;
216   if (&a0 == &no_more_args)  goto done; args[n++] = &a0;
217   if (&a1 == &no_more_args)  goto done; args[n++] = &a1;
218   if (&a2 == &no_more_args)  goto done; args[n++] = &a2;
219   if (&a3 == &no_more_args)  goto done; args[n++] = &a3;
220   if (&a4 == &no_more_args)  goto done; args[n++] = &a4;
221   if (&a5 == &no_more_args)  goto done; args[n++] = &a5;
222   if (&a6 == &no_more_args)  goto done; args[n++] = &a6;
223   if (&a7 == &no_more_args)  goto done; args[n++] = &a7;
224   if (&a8 == &no_more_args)  goto done; args[n++] = &a8;
225   if (&a9 == &no_more_args)  goto done; args[n++] = &a9;
226   if (&a10 == &no_more_args) goto done; args[n++] = &a10;
227   if (&a11 == &no_more_args) goto done; args[n++] = &a11;
228   if (&a12 == &no_more_args) goto done; args[n++] = &a12;
229   if (&a13 == &no_more_args) goto done; args[n++] = &a13;
230   if (&a14 == &no_more_args) goto done; args[n++] = &a14;
231   if (&a15 == &no_more_args) goto done; args[n++] = &a15;
232 done:
233 
234   size_t consumed;
235   int vec[kVecSize] = {};
236   return re.DoMatchImpl(text, ANCHOR_BOTH, &consumed, args, n, vec, kVecSize);
237 }
238 
operator ()(const StringPiece & text,const PCRE & re,const Arg & a0,const Arg & a1,const Arg & a2,const Arg & a3,const Arg & a4,const Arg & a5,const Arg & a6,const Arg & a7,const Arg & a8,const Arg & a9,const Arg & a10,const Arg & a11,const Arg & a12,const Arg & a13,const Arg & a14,const Arg & a15) const239 bool PCRE::PartialMatchFunctor::operator ()(const StringPiece& text,
240                                           const PCRE& re,
241                                           const Arg& a0,
242                                           const Arg& a1,
243                                           const Arg& a2,
244                                           const Arg& a3,
245                                           const Arg& a4,
246                                           const Arg& a5,
247                                           const Arg& a6,
248                                           const Arg& a7,
249                                           const Arg& a8,
250                                           const Arg& a9,
251                                           const Arg& a10,
252                                           const Arg& a11,
253                                           const Arg& a12,
254                                           const Arg& a13,
255                                           const Arg& a14,
256                                           const Arg& a15) const {
257   const Arg* args[kMaxArgs];
258   int n = 0;
259   if (&a0 == &no_more_args)  goto done; args[n++] = &a0;
260   if (&a1 == &no_more_args)  goto done; args[n++] = &a1;
261   if (&a2 == &no_more_args)  goto done; args[n++] = &a2;
262   if (&a3 == &no_more_args)  goto done; args[n++] = &a3;
263   if (&a4 == &no_more_args)  goto done; args[n++] = &a4;
264   if (&a5 == &no_more_args)  goto done; args[n++] = &a5;
265   if (&a6 == &no_more_args)  goto done; args[n++] = &a6;
266   if (&a7 == &no_more_args)  goto done; args[n++] = &a7;
267   if (&a8 == &no_more_args)  goto done; args[n++] = &a8;
268   if (&a9 == &no_more_args)  goto done; args[n++] = &a9;
269   if (&a10 == &no_more_args) goto done; args[n++] = &a10;
270   if (&a11 == &no_more_args) goto done; args[n++] = &a11;
271   if (&a12 == &no_more_args) goto done; args[n++] = &a12;
272   if (&a13 == &no_more_args) goto done; args[n++] = &a13;
273   if (&a14 == &no_more_args) goto done; args[n++] = &a14;
274   if (&a15 == &no_more_args) goto done; args[n++] = &a15;
275 done:
276 
277   size_t consumed;
278   int vec[kVecSize] = {};
279   return re.DoMatchImpl(text, UNANCHORED, &consumed, args, n, vec, kVecSize);
280 }
281 
operator ()(StringPiece * input,const PCRE & pattern,const Arg & a0,const Arg & a1,const Arg & a2,const Arg & a3,const Arg & a4,const Arg & a5,const Arg & a6,const Arg & a7,const Arg & a8,const Arg & a9,const Arg & a10,const Arg & a11,const Arg & a12,const Arg & a13,const Arg & a14,const Arg & a15) const282 bool PCRE::ConsumeFunctor::operator ()(StringPiece* input,
283                                      const PCRE& pattern,
284                                      const Arg& a0,
285                                      const Arg& a1,
286                                      const Arg& a2,
287                                      const Arg& a3,
288                                      const Arg& a4,
289                                      const Arg& a5,
290                                      const Arg& a6,
291                                      const Arg& a7,
292                                      const Arg& a8,
293                                      const Arg& a9,
294                                      const Arg& a10,
295                                      const Arg& a11,
296                                      const Arg& a12,
297                                      const Arg& a13,
298                                      const Arg& a14,
299                                      const Arg& a15) const {
300   const Arg* args[kMaxArgs];
301   int n = 0;
302   if (&a0 == &no_more_args)  goto done; args[n++] = &a0;
303   if (&a1 == &no_more_args)  goto done; args[n++] = &a1;
304   if (&a2 == &no_more_args)  goto done; args[n++] = &a2;
305   if (&a3 == &no_more_args)  goto done; args[n++] = &a3;
306   if (&a4 == &no_more_args)  goto done; args[n++] = &a4;
307   if (&a5 == &no_more_args)  goto done; args[n++] = &a5;
308   if (&a6 == &no_more_args)  goto done; args[n++] = &a6;
309   if (&a7 == &no_more_args)  goto done; args[n++] = &a7;
310   if (&a8 == &no_more_args)  goto done; args[n++] = &a8;
311   if (&a9 == &no_more_args)  goto done; args[n++] = &a9;
312   if (&a10 == &no_more_args) goto done; args[n++] = &a10;
313   if (&a11 == &no_more_args) goto done; args[n++] = &a11;
314   if (&a12 == &no_more_args) goto done; args[n++] = &a12;
315   if (&a13 == &no_more_args) goto done; args[n++] = &a13;
316   if (&a14 == &no_more_args) goto done; args[n++] = &a14;
317   if (&a15 == &no_more_args) goto done; args[n++] = &a15;
318 done:
319 
320   size_t consumed;
321   int vec[kVecSize] = {};
322   if (pattern.DoMatchImpl(*input, ANCHOR_START, &consumed,
323                           args, n, vec, kVecSize)) {
324     input->remove_prefix(consumed);
325     return true;
326   } else {
327     return false;
328   }
329 }
330 
operator ()(StringPiece * input,const PCRE & pattern,const Arg & a0,const Arg & a1,const Arg & a2,const Arg & a3,const Arg & a4,const Arg & a5,const Arg & a6,const Arg & a7,const Arg & a8,const Arg & a9,const Arg & a10,const Arg & a11,const Arg & a12,const Arg & a13,const Arg & a14,const Arg & a15) const331 bool PCRE::FindAndConsumeFunctor::operator ()(StringPiece* input,
332                                             const PCRE& pattern,
333                                             const Arg& a0,
334                                             const Arg& a1,
335                                             const Arg& a2,
336                                             const Arg& a3,
337                                             const Arg& a4,
338                                             const Arg& a5,
339                                             const Arg& a6,
340                                             const Arg& a7,
341                                             const Arg& a8,
342                                             const Arg& a9,
343                                             const Arg& a10,
344                                             const Arg& a11,
345                                             const Arg& a12,
346                                             const Arg& a13,
347                                             const Arg& a14,
348                                             const Arg& a15) const {
349   const Arg* args[kMaxArgs];
350   int n = 0;
351   if (&a0 == &no_more_args)  goto done; args[n++] = &a0;
352   if (&a1 == &no_more_args)  goto done; args[n++] = &a1;
353   if (&a2 == &no_more_args)  goto done; args[n++] = &a2;
354   if (&a3 == &no_more_args)  goto done; args[n++] = &a3;
355   if (&a4 == &no_more_args)  goto done; args[n++] = &a4;
356   if (&a5 == &no_more_args)  goto done; args[n++] = &a5;
357   if (&a6 == &no_more_args)  goto done; args[n++] = &a6;
358   if (&a7 == &no_more_args)  goto done; args[n++] = &a7;
359   if (&a8 == &no_more_args)  goto done; args[n++] = &a8;
360   if (&a9 == &no_more_args)  goto done; args[n++] = &a9;
361   if (&a10 == &no_more_args) goto done; args[n++] = &a10;
362   if (&a11 == &no_more_args) goto done; args[n++] = &a11;
363   if (&a12 == &no_more_args) goto done; args[n++] = &a12;
364   if (&a13 == &no_more_args) goto done; args[n++] = &a13;
365   if (&a14 == &no_more_args) goto done; args[n++] = &a14;
366   if (&a15 == &no_more_args) goto done; args[n++] = &a15;
367 done:
368 
369   size_t consumed;
370   int vec[kVecSize] = {};
371   if (pattern.DoMatchImpl(*input, UNANCHORED, &consumed,
372                           args, n, vec, kVecSize)) {
373     input->remove_prefix(consumed);
374     return true;
375   } else {
376     return false;
377   }
378 }
379 
Replace(std::string * str,const PCRE & pattern,const StringPiece & rewrite)380 bool PCRE::Replace(std::string *str,
381                  const PCRE& pattern,
382                  const StringPiece& rewrite) {
383   int vec[kVecSize] = {};
384   int matches = pattern.TryMatch(*str, 0, UNANCHORED, true, vec, kVecSize);
385   if (matches == 0)
386     return false;
387 
388   std::string s;
389   if (!pattern.Rewrite(&s, rewrite, *str, vec, matches))
390     return false;
391 
392   assert(vec[0] >= 0);
393   assert(vec[1] >= 0);
394   str->replace(vec[0], vec[1] - vec[0], s);
395   return true;
396 }
397 
GlobalReplace(std::string * str,const PCRE & pattern,const StringPiece & rewrite)398 int PCRE::GlobalReplace(std::string *str,
399                       const PCRE& pattern,
400                       const StringPiece& rewrite) {
401   int count = 0;
402   int vec[kVecSize] = {};
403   std::string out;
404   size_t start = 0;
405   bool last_match_was_empty_string = false;
406 
407   while (start <= str->size()) {
408     // If the previous match was for the empty string, we shouldn't
409     // just match again: we'll match in the same way and get an
410     // infinite loop.  Instead, we do the match in a special way:
411     // anchored -- to force another try at the same position --
412     // and with a flag saying that this time, ignore empty matches.
413     // If this special match returns, that means there's a non-empty
414     // match at this position as well, and we can continue.  If not,
415     // we do what perl does, and just advance by one.
416     // Notice that perl prints '@@@' for this;
417     //    perl -le '$_ = "aa"; s/b*|aa/@/g; print'
418     int matches;
419     if (last_match_was_empty_string) {
420       matches = pattern.TryMatch(*str, start, ANCHOR_START, false,
421                                  vec, kVecSize);
422       if (matches <= 0) {
423         if (start < str->size())
424           out.push_back((*str)[start]);
425         start++;
426         last_match_was_empty_string = false;
427         continue;
428       }
429     } else {
430       matches = pattern.TryMatch(*str, start, UNANCHORED, true,
431                                  vec, kVecSize);
432       if (matches <= 0)
433         break;
434     }
435     size_t matchstart = vec[0], matchend = vec[1];
436     assert(matchstart >= start);
437     assert(matchend >= matchstart);
438 
439     out.append(*str, start, matchstart - start);
440     pattern.Rewrite(&out, rewrite, *str, vec, matches);
441     start = matchend;
442     count++;
443     last_match_was_empty_string = (matchstart == matchend);
444   }
445 
446   if (count == 0)
447     return 0;
448 
449   if (start < str->size())
450     out.append(*str, start, str->size() - start);
451   using std::swap;
452   swap(out, *str);
453   return count;
454 }
455 
Extract(const StringPiece & text,const PCRE & pattern,const StringPiece & rewrite,std::string * out)456 bool PCRE::Extract(const StringPiece &text,
457                  const PCRE& pattern,
458                  const StringPiece &rewrite,
459                  std::string *out) {
460   int vec[kVecSize] = {};
461   int matches = pattern.TryMatch(text, 0, UNANCHORED, true, vec, kVecSize);
462   if (matches == 0)
463     return false;
464   out->clear();
465   return pattern.Rewrite(out, rewrite, text, vec, matches);
466 }
467 
QuoteMeta(const StringPiece & unquoted)468 std::string PCRE::QuoteMeta(const StringPiece& unquoted) {
469   std::string result;
470   result.reserve(unquoted.size() << 1);
471 
472   // Escape any ascii character not in [A-Za-z_0-9].
473   //
474   // Note that it's legal to escape a character even if it has no
475   // special meaning in a regular expression -- so this function does
476   // that.  (This also makes it identical to the perl function of the
477   // same name except for the null-character special case;
478   // see `perldoc -f quotemeta`.)
479   for (size_t ii = 0; ii < unquoted.size(); ++ii) {
480     // Note that using 'isalnum' here raises the benchmark time from
481     // 32ns to 58ns:
482     if ((unquoted[ii] < 'a' || unquoted[ii] > 'z') &&
483         (unquoted[ii] < 'A' || unquoted[ii] > 'Z') &&
484         (unquoted[ii] < '0' || unquoted[ii] > '9') &&
485         unquoted[ii] != '_' &&
486         // If this is the part of a UTF8 or Latin1 character, we need
487         // to copy this byte without escaping.  Experimentally this is
488         // what works correctly with the regexp library.
489         !(unquoted[ii] & 128)) {
490       if (unquoted[ii] == '\0') {  // Special handling for null chars.
491         // Can't use "\\0" since the next character might be a digit.
492         result += "\\x00";
493         continue;
494       }
495       result += '\\';
496     }
497     result += unquoted[ii];
498   }
499 
500   return result;
501 }
502 
503 /***** Actual matching and rewriting code *****/
504 
HitLimit()505 bool PCRE::HitLimit() {
506   return hit_limit_ != 0;
507 }
508 
ClearHitLimit()509 void PCRE::ClearHitLimit() {
510   hit_limit_ = 0;
511 }
512 
TryMatch(const StringPiece & text,size_t startpos,Anchor anchor,bool empty_ok,int * vec,int vecsize) const513 int PCRE::TryMatch(const StringPiece& text,
514                    size_t startpos,
515                    Anchor anchor,
516                    bool empty_ok,
517                    int *vec,
518                    int vecsize) const {
519   pcre* re = (anchor == ANCHOR_BOTH) ? re_full_ : re_partial_;
520   if (re == NULL) {
521     PCREPORT(ERROR) << "Matching against invalid re: " << *error_;
522     return 0;
523   }
524 
525   int match_limit = match_limit_;
526   if (match_limit <= 0) {
527     match_limit = GetFlag(FLAGS_regexp_match_limit);
528   }
529 
530   int stack_limit = stack_limit_;
531   if (stack_limit <= 0) {
532     stack_limit = GetFlag(FLAGS_regexp_stack_limit);
533   }
534 
535   pcre_extra extra = { 0 };
536   if (match_limit > 0) {
537     extra.flags |= PCRE_EXTRA_MATCH_LIMIT;
538     extra.match_limit = match_limit;
539   }
540   if (stack_limit > 0) {
541     extra.flags |= PCRE_EXTRA_MATCH_LIMIT_RECURSION;
542     extra.match_limit_recursion = stack_limit / kPCREFrameSize;
543   }
544 
545   int options = 0;
546   if (anchor != UNANCHORED)
547     options |= PCRE_ANCHORED;
548   if (!empty_ok)
549     options |= PCRE_NOTEMPTY;
550 
551   int rc = pcre_exec(re,              // The regular expression object
552                      &extra,
553                      (text.data() == NULL) ? "" : text.data(),
554                      static_cast<int>(text.size()),
555                      static_cast<int>(startpos),
556                      options,
557                      vec,
558                      vecsize);
559 
560   // Handle errors
561   if (rc == 0) {
562     // pcre_exec() returns 0 as a special case when the number of
563     // capturing subpatterns exceeds the size of the vector.
564     // When this happens, there is a match and the output vector
565     // is filled, but we miss out on the positions of the extra subpatterns.
566     rc = vecsize / 2;
567   } else if (rc < 0) {
568     switch (rc) {
569       case PCRE_ERROR_NOMATCH:
570         return 0;
571       case PCRE_ERROR_MATCHLIMIT:
572         // Writing to hit_limit is not safe if multiple threads
573         // are using the PCRE, but the flag is only intended
574         // for use by unit tests anyway, so we let it go.
575         hit_limit_ = true;
576         PCREPORT(WARNING) << "Exceeded match limit of " << match_limit
577                         << " when matching '" << pattern_ << "'"
578                         << " against text that is " << text.size() << " bytes.";
579         return 0;
580       case PCRE_ERROR_RECURSIONLIMIT:
581         // See comment about hit_limit above.
582         hit_limit_ = true;
583         PCREPORT(WARNING) << "Exceeded stack limit of " << stack_limit
584                         << " when matching '" << pattern_ << "'"
585                         << " against text that is " << text.size() << " bytes.";
586         return 0;
587       default:
588         // There are other return codes from pcre.h :
589         // PCRE_ERROR_NULL           (-2)
590         // PCRE_ERROR_BADOPTION      (-3)
591         // PCRE_ERROR_BADMAGIC       (-4)
592         // PCRE_ERROR_UNKNOWN_NODE   (-5)
593         // PCRE_ERROR_NOMEMORY       (-6)
594         // PCRE_ERROR_NOSUBSTRING    (-7)
595         // ...
596         PCREPORT(ERROR) << "Unexpected return code: " << rc
597                       << " when matching '" << pattern_ << "'"
598                       << ", re=" << re
599                       << ", text=" << text
600                       << ", vec=" << vec
601                       << ", vecsize=" << vecsize;
602         return 0;
603     }
604   }
605 
606   return rc;
607 }
608 
DoMatchImpl(const StringPiece & text,Anchor anchor,size_t * consumed,const Arg * const * args,int n,int * vec,int vecsize) const609 bool PCRE::DoMatchImpl(const StringPiece& text,
610                        Anchor anchor,
611                        size_t* consumed,
612                        const Arg* const* args,
613                        int n,
614                        int* vec,
615                        int vecsize) const {
616   assert((1 + n) * 3 <= vecsize);  // results + PCRE workspace
617   if (NumberOfCapturingGroups() < n) {
618     // RE has fewer capturing groups than number of Arg pointers passed in.
619     return false;
620   }
621 
622   int matches = TryMatch(text, 0, anchor, true, vec, vecsize);
623   assert(matches >= 0);  // TryMatch never returns negatives
624   if (matches == 0)
625     return false;
626 
627   *consumed = vec[1];
628 
629   if (n == 0 || args == NULL) {
630     // We are not interested in results
631     return true;
632   }
633 
634   // If we got here, we must have matched the whole pattern.
635   // We do not need (can not do) any more checks on the value of 'matches' here
636   // -- see the comment for TryMatch.
637   for (int i = 0; i < n; i++) {
638     const int start = vec[2*(i+1)];
639     const int limit = vec[2*(i+1)+1];
640 
641     // Avoid invoking undefined behavior when text.data() happens
642     // to be null and start happens to be -1, the latter being the
643     // case for an unmatched subexpression. Even if text.data() is
644     // not null, pointing one byte before was a longstanding bug.
645     const char* addr = NULL;
646     if (start != -1) {
647       addr = text.data() + start;
648     }
649 
650     if (!args[i]->Parse(addr, limit-start)) {
651       // TODO: Should we indicate what the error was?
652       return false;
653     }
654   }
655 
656   return true;
657 }
658 
DoMatch(const StringPiece & text,Anchor anchor,size_t * consumed,const Arg * const args[],int n) const659 bool PCRE::DoMatch(const StringPiece& text,
660                    Anchor anchor,
661                    size_t* consumed,
662                    const Arg* const args[],
663                    int n) const {
664   assert(n >= 0);
665   const int vecsize = (1 + n) * 3;  // results + PCRE workspace
666                                     // (as for kVecSize)
667   int* vec = new int[vecsize];
668   bool b = DoMatchImpl(text, anchor, consumed, args, n, vec, vecsize);
669   delete[] vec;
670   return b;
671 }
672 
Rewrite(std::string * out,const StringPiece & rewrite,const StringPiece & text,int * vec,int veclen) const673 bool PCRE::Rewrite(std::string *out, const StringPiece &rewrite,
674                  const StringPiece &text, int *vec, int veclen) const {
675   int number_of_capturing_groups = NumberOfCapturingGroups();
676   for (const char *s = rewrite.data(), *end = s + rewrite.size();
677        s < end; s++) {
678     int c = *s;
679     if (c == '\\') {
680       c = *++s;
681       if (isdigit(c)) {
682         int n = (c - '0');
683         if (n >= veclen) {
684           if (n <= number_of_capturing_groups) {
685             // unmatched optional capturing group. treat
686             // its value as empty string; i.e., nothing to append.
687           } else {
688             PCREPORT(ERROR) << "requested group " << n
689                           << " in regexp " << rewrite.data();
690             return false;
691           }
692         }
693         int start = vec[2 * n];
694         if (start >= 0)
695           out->append(text.data() + start, vec[2 * n + 1] - start);
696       } else if (c == '\\') {
697         out->push_back('\\');
698       } else {
699         PCREPORT(ERROR) << "invalid rewrite pattern: " << rewrite.data();
700         return false;
701       }
702     } else {
703       out->push_back(c);
704     }
705   }
706   return true;
707 }
708 
CheckRewriteString(const StringPiece & rewrite,std::string * error) const709 bool PCRE::CheckRewriteString(const StringPiece& rewrite,
710                               std::string* error) const {
711   int max_token = -1;
712   for (const char *s = rewrite.data(), *end = s + rewrite.size();
713        s < end; s++) {
714     int c = *s;
715     if (c != '\\') {
716       continue;
717     }
718     if (++s == end) {
719       *error = "Rewrite schema error: '\\' not allowed at end.";
720       return false;
721     }
722     c = *s;
723     if (c == '\\') {
724       continue;
725     }
726     if (!isdigit(c)) {
727       *error = "Rewrite schema error: "
728                "'\\' must be followed by a digit or '\\'.";
729       return false;
730     }
731     int n = (c - '0');
732     if (max_token < n) {
733       max_token = n;
734     }
735   }
736 
737   if (max_token > NumberOfCapturingGroups()) {
738     *error = StringPrintf(
739         "Rewrite schema requests %d matches, but the regexp only has %d "
740         "parenthesized subexpressions.",
741         max_token, NumberOfCapturingGroups());
742     return false;
743   }
744   return true;
745 }
746 
747 
748 // Return the number of capturing subpatterns, or -1 if the
749 // regexp wasn't valid on construction.
NumberOfCapturingGroups() const750 int PCRE::NumberOfCapturingGroups() const {
751   if (re_partial_ == NULL) return -1;
752 
753   int result;
754   int rc = pcre_fullinfo(re_partial_,       // The regular expression object
755                          NULL,              // We did not study the pattern
756                          PCRE_INFO_CAPTURECOUNT,
757                          &result);
758   if (rc != 0) {
759     PCREPORT(ERROR) << "Unexpected return code: " << rc;
760     return -1;
761   }
762   return result;
763 }
764 
765 
766 /***** Parsers for various types *****/
767 
parse_null(const char * str,size_t n,void * dest)768 bool PCRE::Arg::parse_null(const char* str, size_t n, void* dest) {
769   // We fail if somebody asked us to store into a non-NULL void* pointer
770   return (dest == NULL);
771 }
772 
parse_string(const char * str,size_t n,void * dest)773 bool PCRE::Arg::parse_string(const char* str, size_t n, void* dest) {
774   if (dest == NULL) return true;
775   reinterpret_cast<std::string*>(dest)->assign(str, n);
776   return true;
777 }
778 
parse_stringpiece(const char * str,size_t n,void * dest)779 bool PCRE::Arg::parse_stringpiece(const char* str, size_t n, void* dest) {
780   if (dest == NULL) return true;
781   *(reinterpret_cast<StringPiece*>(dest)) = StringPiece(str, n);
782   return true;
783 }
784 
parse_char(const char * str,size_t n,void * dest)785 bool PCRE::Arg::parse_char(const char* str, size_t n, void* dest) {
786   if (n != 1) return false;
787   if (dest == NULL) return true;
788   *(reinterpret_cast<char*>(dest)) = str[0];
789   return true;
790 }
791 
parse_schar(const char * str,size_t n,void * dest)792 bool PCRE::Arg::parse_schar(const char* str, size_t n, void* dest) {
793   if (n != 1) return false;
794   if (dest == NULL) return true;
795   *(reinterpret_cast<signed char*>(dest)) = str[0];
796   return true;
797 }
798 
parse_uchar(const char * str,size_t n,void * dest)799 bool PCRE::Arg::parse_uchar(const char* str, size_t n, void* dest) {
800   if (n != 1) return false;
801   if (dest == NULL) return true;
802   *(reinterpret_cast<unsigned char*>(dest)) = str[0];
803   return true;
804 }
805 
806 // Largest number spec that we are willing to parse
807 static const int kMaxNumberLength = 32;
808 
809 // PCREQUIPCRES "buf" must have length at least kMaxNumberLength+1
810 // PCREQUIPCRES "n > 0"
811 // Copies "str" into "buf" and null-terminates if necessary.
812 // Returns one of:
813 //      a. "str" if no termination is needed
814 //      b. "buf" if the string was copied and null-terminated
815 //      c. "" if the input was invalid and has no hope of being parsed
TerminateNumber(char * buf,const char * str,size_t n)816 static const char* TerminateNumber(char* buf, const char* str, size_t n) {
817   if ((n > 0) && isspace(*str)) {
818     // We are less forgiving than the strtoxxx() routines and do not
819     // allow leading spaces.
820     return "";
821   }
822 
823   // See if the character right after the input text may potentially
824   // look like a digit.
825   if (isdigit(str[n]) ||
826       ((str[n] >= 'a') && (str[n] <= 'f')) ||
827       ((str[n] >= 'A') && (str[n] <= 'F'))) {
828     if (n > kMaxNumberLength) return ""; // Input too big to be a valid number
829     memcpy(buf, str, n);
830     buf[n] = '\0';
831     return buf;
832   } else {
833     // We can parse right out of the supplied string, so return it.
834     return str;
835   }
836 }
837 
parse_long_radix(const char * str,size_t n,void * dest,int radix)838 bool PCRE::Arg::parse_long_radix(const char* str,
839                                  size_t n,
840                                  void* dest,
841                                  int radix) {
842   if (n == 0) return false;
843   char buf[kMaxNumberLength+1];
844   str = TerminateNumber(buf, str, n);
845   char* end;
846   errno = 0;
847   long r = strtol(str, &end, radix);
848   if (end != str + n) return false;   // Leftover junk
849   if (errno) return false;
850   if (dest == NULL) return true;
851   *(reinterpret_cast<long*>(dest)) = r;
852   return true;
853 }
854 
parse_ulong_radix(const char * str,size_t n,void * dest,int radix)855 bool PCRE::Arg::parse_ulong_radix(const char* str,
856                                   size_t n,
857                                   void* dest,
858                                   int radix) {
859   if (n == 0) return false;
860   char buf[kMaxNumberLength+1];
861   str = TerminateNumber(buf, str, n);
862   if (str[0] == '-') {
863     // strtoul() will silently accept negative numbers and parse
864     // them.  This module is more strict and treats them as errors.
865     return false;
866   }
867 
868   char* end;
869   errno = 0;
870   unsigned long r = strtoul(str, &end, radix);
871   if (end != str + n) return false;   // Leftover junk
872   if (errno) return false;
873   if (dest == NULL) return true;
874   *(reinterpret_cast<unsigned long*>(dest)) = r;
875   return true;
876 }
877 
parse_short_radix(const char * str,size_t n,void * dest,int radix)878 bool PCRE::Arg::parse_short_radix(const char* str,
879                                   size_t n,
880                                   void* dest,
881                                   int radix) {
882   long r;
883   if (!parse_long_radix(str, n, &r, radix)) return false;  // Could not parse
884   if ((short)r != r) return false;                         // Out of range
885   if (dest == NULL) return true;
886   *(reinterpret_cast<short*>(dest)) = (short)r;
887   return true;
888 }
889 
parse_ushort_radix(const char * str,size_t n,void * dest,int radix)890 bool PCRE::Arg::parse_ushort_radix(const char* str,
891                                    size_t n,
892                                    void* dest,
893                                    int radix) {
894   unsigned long r;
895   if (!parse_ulong_radix(str, n, &r, radix)) return false;  // Could not parse
896   if ((unsigned short)r != r) return false;                 // Out of range
897   if (dest == NULL) return true;
898   *(reinterpret_cast<unsigned short*>(dest)) = (unsigned short)r;
899   return true;
900 }
901 
parse_int_radix(const char * str,size_t n,void * dest,int radix)902 bool PCRE::Arg::parse_int_radix(const char* str,
903                                 size_t n,
904                                 void* dest,
905                                 int radix) {
906   long r;
907   if (!parse_long_radix(str, n, &r, radix)) return false;  // Could not parse
908   if ((int)r != r) return false;                           // Out of range
909   if (dest == NULL) return true;
910   *(reinterpret_cast<int*>(dest)) = (int)r;
911   return true;
912 }
913 
parse_uint_radix(const char * str,size_t n,void * dest,int radix)914 bool PCRE::Arg::parse_uint_radix(const char* str,
915                                  size_t n,
916                                  void* dest,
917                                  int radix) {
918   unsigned long r;
919   if (!parse_ulong_radix(str, n, &r, radix)) return false;  // Could not parse
920   if ((unsigned int)r != r) return false;                   // Out of range
921   if (dest == NULL) return true;
922   *(reinterpret_cast<unsigned int*>(dest)) = (unsigned int)r;
923   return true;
924 }
925 
parse_longlong_radix(const char * str,size_t n,void * dest,int radix)926 bool PCRE::Arg::parse_longlong_radix(const char* str,
927                                      size_t n,
928                                      void* dest,
929                                      int radix) {
930   if (n == 0) return false;
931   char buf[kMaxNumberLength+1];
932   str = TerminateNumber(buf, str, n);
933   char* end;
934   errno = 0;
935   long long r = strtoll(str, &end, radix);
936   if (end != str + n) return false;   // Leftover junk
937   if (errno) return false;
938   if (dest == NULL) return true;
939   *(reinterpret_cast<long long*>(dest)) = r;
940   return true;
941 }
942 
parse_ulonglong_radix(const char * str,size_t n,void * dest,int radix)943 bool PCRE::Arg::parse_ulonglong_radix(const char* str,
944                                       size_t n,
945                                       void* dest,
946                                       int radix) {
947   if (n == 0) return false;
948   char buf[kMaxNumberLength+1];
949   str = TerminateNumber(buf, str, n);
950   if (str[0] == '-') {
951     // strtoull() will silently accept negative numbers and parse
952     // them.  This module is more strict and treats them as errors.
953     return false;
954   }
955   char* end;
956   errno = 0;
957   unsigned long long r = strtoull(str, &end, radix);
958   if (end != str + n) return false;   // Leftover junk
959   if (errno) return false;
960   if (dest == NULL) return true;
961   *(reinterpret_cast<unsigned long long*>(dest)) = r;
962   return true;
963 }
964 
parse_double_float(const char * str,size_t n,bool isfloat,void * dest)965 static bool parse_double_float(const char* str, size_t n, bool isfloat,
966                                void* dest) {
967   if (n == 0) return false;
968   static const int kMaxLength = 200;
969   char buf[kMaxLength];
970   if (n >= kMaxLength) return false;
971   memcpy(buf, str, n);
972   buf[n] = '\0';
973   char* end;
974   errno = 0;
975   double r;
976   if (isfloat) {
977     r = strtof(buf, &end);
978   } else {
979     r = strtod(buf, &end);
980   }
981   if (end != buf + n) return false;   // Leftover junk
982   if (errno) return false;
983   if (dest == NULL) return true;
984   if (isfloat) {
985     *(reinterpret_cast<float*>(dest)) = (float)r;
986   } else {
987     *(reinterpret_cast<double*>(dest)) = r;
988   }
989   return true;
990 }
991 
parse_double(const char * str,size_t n,void * dest)992 bool PCRE::Arg::parse_double(const char* str, size_t n, void* dest) {
993   return parse_double_float(str, n, false, dest);
994 }
995 
parse_float(const char * str,size_t n,void * dest)996 bool PCRE::Arg::parse_float(const char* str, size_t n, void* dest) {
997   return parse_double_float(str, n, true, dest);
998 }
999 
1000 #define DEFINE_INTEGER_PARSER(name)                                           \
1001   bool PCRE::Arg::parse_##name(const char* str, size_t n, void* dest) {       \
1002     return parse_##name##_radix(str, n, dest, 10);                            \
1003   }                                                                           \
1004   bool PCRE::Arg::parse_##name##_hex(const char* str, size_t n, void* dest) { \
1005     return parse_##name##_radix(str, n, dest, 16);                            \
1006   }                                                                           \
1007   bool PCRE::Arg::parse_##name##_octal(const char* str, size_t n,             \
1008                                        void* dest) {                          \
1009     return parse_##name##_radix(str, n, dest, 8);                             \
1010   }                                                                           \
1011   bool PCRE::Arg::parse_##name##_cradix(const char* str, size_t n,            \
1012                                         void* dest) {                         \
1013     return parse_##name##_radix(str, n, dest, 0);                             \
1014   }
1015 
1016 DEFINE_INTEGER_PARSER(short);
1017 DEFINE_INTEGER_PARSER(ushort);
1018 DEFINE_INTEGER_PARSER(int);
1019 DEFINE_INTEGER_PARSER(uint);
1020 DEFINE_INTEGER_PARSER(long);
1021 DEFINE_INTEGER_PARSER(ulong);
1022 DEFINE_INTEGER_PARSER(longlong);
1023 DEFINE_INTEGER_PARSER(ulonglong);
1024 
1025 #undef DEFINE_INTEGER_PARSER
1026 
1027 }  // namespace re2
1028