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