2 * Copyright (C) 1999-2000 Harri Porten (porten@kde.org)
3 * Copyright (C) 2003, 2007, 2008 Apple Inc. All Rights Reserved.
4 * Copyright (C) 2009 Torch Mobile, Inc.
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
23 #include "RegExpConstructor.h"
25 #include "ArrayPrototype.h"
28 #include "JSFunction.h"
31 #include "ObjectPrototype.h"
32 #include "RegExpMatchesArray.h"
33 #include "RegExpObject.h"
34 #include "RegExpPrototype.h"
36 #include "RegExpCache.h"
40 static JSValue
regExpConstructorInput(ExecState
*, JSValue
, const Identifier
&);
41 static JSValue
regExpConstructorMultiline(ExecState
*, JSValue
, const Identifier
&);
42 static JSValue
regExpConstructorLastMatch(ExecState
*, JSValue
, const Identifier
&);
43 static JSValue
regExpConstructorLastParen(ExecState
*, JSValue
, const Identifier
&);
44 static JSValue
regExpConstructorLeftContext(ExecState
*, JSValue
, const Identifier
&);
45 static JSValue
regExpConstructorRightContext(ExecState
*, JSValue
, const Identifier
&);
46 static JSValue
regExpConstructorDollar1(ExecState
*, JSValue
, const Identifier
&);
47 static JSValue
regExpConstructorDollar2(ExecState
*, JSValue
, const Identifier
&);
48 static JSValue
regExpConstructorDollar3(ExecState
*, JSValue
, const Identifier
&);
49 static JSValue
regExpConstructorDollar4(ExecState
*, JSValue
, const Identifier
&);
50 static JSValue
regExpConstructorDollar5(ExecState
*, JSValue
, const Identifier
&);
51 static JSValue
regExpConstructorDollar6(ExecState
*, JSValue
, const Identifier
&);
52 static JSValue
regExpConstructorDollar7(ExecState
*, JSValue
, const Identifier
&);
53 static JSValue
regExpConstructorDollar8(ExecState
*, JSValue
, const Identifier
&);
54 static JSValue
regExpConstructorDollar9(ExecState
*, JSValue
, const Identifier
&);
56 static void setRegExpConstructorInput(ExecState
*, JSObject
*, JSValue
);
57 static void setRegExpConstructorMultiline(ExecState
*, JSObject
*, JSValue
);
61 #include "RegExpConstructor.lut.h"
65 ASSERT_CLASS_FITS_IN_CELL(RegExpConstructor
);
67 const ClassInfo
RegExpConstructor::info
= { "Function", &InternalFunction::info
, 0, ExecState::regExpConstructorTable
};
69 /* Source for RegExpConstructor.lut.h
70 @begin regExpConstructorTable
71 input regExpConstructorInput None
72 $_ regExpConstructorInput DontEnum
73 multiline regExpConstructorMultiline None
74 $* regExpConstructorMultiline DontEnum
75 lastMatch regExpConstructorLastMatch DontDelete|ReadOnly
76 $& regExpConstructorLastMatch DontDelete|ReadOnly|DontEnum
77 lastParen regExpConstructorLastParen DontDelete|ReadOnly
78 $+ regExpConstructorLastParen DontDelete|ReadOnly|DontEnum
79 leftContext regExpConstructorLeftContext DontDelete|ReadOnly
80 $` regExpConstructorLeftContext DontDelete|ReadOnly|DontEnum
81 rightContext regExpConstructorRightContext DontDelete|ReadOnly
82 $' regExpConstructorRightContext DontDelete|ReadOnly|DontEnum
83 $1 regExpConstructorDollar1 DontDelete|ReadOnly
84 $2 regExpConstructorDollar2 DontDelete|ReadOnly
85 $3 regExpConstructorDollar3 DontDelete|ReadOnly
86 $4 regExpConstructorDollar4 DontDelete|ReadOnly
87 $5 regExpConstructorDollar5 DontDelete|ReadOnly
88 $6 regExpConstructorDollar6 DontDelete|ReadOnly
89 $7 regExpConstructorDollar7 DontDelete|ReadOnly
90 $8 regExpConstructorDollar8 DontDelete|ReadOnly
91 $9 regExpConstructorDollar9 DontDelete|ReadOnly
95 RegExpConstructor::RegExpConstructor(ExecState
* exec
, NonNullPassRefPtr
<Structure
> structure
, RegExpPrototype
* regExpPrototype
)
96 : InternalFunction(&exec
->globalData(), structure
, Identifier(exec
, "RegExp"))
97 , d(new RegExpConstructorPrivate
)
99 // ECMA 15.10.5.1 RegExp.prototype
100 putDirectWithoutTransition(exec
->propertyNames().prototype
, regExpPrototype
, DontEnum
| DontDelete
| ReadOnly
);
102 // no. of arguments for constructor
103 putDirectWithoutTransition(exec
->propertyNames().length
, jsNumber(exec
, 2), ReadOnly
| DontDelete
| DontEnum
);
106 RegExpMatchesArray::RegExpMatchesArray(ExecState
* exec
, RegExpConstructorPrivate
* data
)
107 : JSArray(exec
->lexicalGlobalObject()->regExpMatchesArrayStructure(), data
->lastNumSubPatterns
+ 1)
109 RegExpConstructorPrivate
* d
= new RegExpConstructorPrivate
;
110 d
->input
= data
->lastInput
;
111 d
->lastInput
= data
->lastInput
;
112 d
->lastNumSubPatterns
= data
->lastNumSubPatterns
;
113 unsigned offsetVectorSize
= (data
->lastNumSubPatterns
+ 1) * 2; // only copying the result part of the vector
114 d
->lastOvector().resize(offsetVectorSize
);
115 memcpy(d
->lastOvector().data(), data
->lastOvector().data(), offsetVectorSize
* sizeof(int));
116 // d->multiline is not needed, and remains uninitialized
121 RegExpMatchesArray::~RegExpMatchesArray()
123 delete static_cast<RegExpConstructorPrivate
*>(subclassData());
126 void RegExpMatchesArray::fillArrayInstance(ExecState
* exec
)
128 RegExpConstructorPrivate
* d
= static_cast<RegExpConstructorPrivate
*>(subclassData());
131 unsigned lastNumSubpatterns
= d
->lastNumSubPatterns
;
133 for (unsigned i
= 0; i
<= lastNumSubpatterns
; ++i
) {
134 int start
= d
->lastOvector()[2 * i
];
136 JSArray::put(exec
, i
, jsSubstring(exec
, d
->lastInput
, start
, d
->lastOvector()[2 * i
+ 1] - start
));
138 JSArray::put(exec
, i
, jsUndefined());
141 PutPropertySlot slot
;
142 JSArray::put(exec
, exec
->propertyNames().index
, jsNumber(exec
, d
->lastOvector()[0]), slot
);
143 JSArray::put(exec
, exec
->propertyNames().input
, jsString(exec
, d
->input
), slot
);
149 JSObject
* RegExpConstructor::arrayOfMatches(ExecState
* exec
) const
151 return new (exec
) RegExpMatchesArray(exec
, d
.get());
154 JSValue
RegExpConstructor::getBackref(ExecState
* exec
, unsigned i
) const
156 if (!d
->lastOvector().isEmpty() && i
<= d
->lastNumSubPatterns
) {
157 int start
= d
->lastOvector()[2 * i
];
159 return jsSubstring(exec
, d
->lastInput
, start
, d
->lastOvector()[2 * i
+ 1] - start
);
161 return jsEmptyString(exec
);
164 JSValue
RegExpConstructor::getLastParen(ExecState
* exec
) const
166 unsigned i
= d
->lastNumSubPatterns
;
168 ASSERT(!d
->lastOvector().isEmpty());
169 int start
= d
->lastOvector()[2 * i
];
171 return jsSubstring(exec
, d
->lastInput
, start
, d
->lastOvector()[2 * i
+ 1] - start
);
173 return jsEmptyString(exec
);
176 JSValue
RegExpConstructor::getLeftContext(ExecState
* exec
) const
178 if (!d
->lastOvector().isEmpty())
179 return jsSubstring(exec
, d
->lastInput
, 0, d
->lastOvector()[0]);
180 return jsEmptyString(exec
);
183 JSValue
RegExpConstructor::getRightContext(ExecState
* exec
) const
185 if (!d
->lastOvector().isEmpty())
186 return jsSubstring(exec
, d
->lastInput
, d
->lastOvector()[1], d
->lastInput
.size() - d
->lastOvector()[1]);
187 return jsEmptyString(exec
);
190 bool RegExpConstructor::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
192 return getStaticValueSlot
<RegExpConstructor
, InternalFunction
>(exec
, ExecState::regExpConstructorTable(exec
), this, propertyName
, slot
);
195 bool RegExpConstructor::getOwnPropertyDescriptor(ExecState
* exec
, const Identifier
& propertyName
, PropertyDescriptor
& descriptor
)
197 return getStaticValueDescriptor
<RegExpConstructor
, InternalFunction
>(exec
, ExecState::regExpConstructorTable(exec
), this, propertyName
, descriptor
);
200 JSValue
regExpConstructorDollar1(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
202 return asRegExpConstructor(slotBase
)->getBackref(exec
, 1);
205 JSValue
regExpConstructorDollar2(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
207 return asRegExpConstructor(slotBase
)->getBackref(exec
, 2);
210 JSValue
regExpConstructorDollar3(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
212 return asRegExpConstructor(slotBase
)->getBackref(exec
, 3);
215 JSValue
regExpConstructorDollar4(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
217 return asRegExpConstructor(slotBase
)->getBackref(exec
, 4);
220 JSValue
regExpConstructorDollar5(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
222 return asRegExpConstructor(slotBase
)->getBackref(exec
, 5);
225 JSValue
regExpConstructorDollar6(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
227 return asRegExpConstructor(slotBase
)->getBackref(exec
, 6);
230 JSValue
regExpConstructorDollar7(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
232 return asRegExpConstructor(slotBase
)->getBackref(exec
, 7);
235 JSValue
regExpConstructorDollar8(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
237 return asRegExpConstructor(slotBase
)->getBackref(exec
, 8);
240 JSValue
regExpConstructorDollar9(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
242 return asRegExpConstructor(slotBase
)->getBackref(exec
, 9);
245 JSValue
regExpConstructorInput(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
247 return jsString(exec
, asRegExpConstructor(slotBase
)->input());
250 JSValue
regExpConstructorMultiline(ExecState
*, JSValue slotBase
, const Identifier
&)
252 return jsBoolean(asRegExpConstructor(slotBase
)->multiline());
255 JSValue
regExpConstructorLastMatch(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
257 return asRegExpConstructor(slotBase
)->getBackref(exec
, 0);
260 JSValue
regExpConstructorLastParen(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
262 return asRegExpConstructor(slotBase
)->getLastParen(exec
);
265 JSValue
regExpConstructorLeftContext(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
267 return asRegExpConstructor(slotBase
)->getLeftContext(exec
);
270 JSValue
regExpConstructorRightContext(ExecState
* exec
, JSValue slotBase
, const Identifier
&)
272 return asRegExpConstructor(slotBase
)->getRightContext(exec
);
275 void RegExpConstructor::put(ExecState
* exec
, const Identifier
& propertyName
, JSValue value
, PutPropertySlot
& slot
)
277 lookupPut
<RegExpConstructor
, InternalFunction
>(exec
, propertyName
, value
, ExecState::regExpConstructorTable(exec
), this, slot
);
280 void setRegExpConstructorInput(ExecState
* exec
, JSObject
* baseObject
, JSValue value
)
282 asRegExpConstructor(baseObject
)->setInput(value
.toString(exec
));
285 void setRegExpConstructorMultiline(ExecState
* exec
, JSObject
* baseObject
, JSValue value
)
287 asRegExpConstructor(baseObject
)->setMultiline(value
.toBoolean(exec
));
291 JSObject
* constructRegExp(ExecState
* exec
, const ArgList
& args
)
293 JSValue arg0
= args
.at(0);
294 JSValue arg1
= args
.at(1);
296 if (arg0
.inherits(&RegExpObject::info
)) {
297 if (!arg1
.isUndefined())
298 return throwError(exec
, TypeError
, "Cannot supply flags when constructing one RegExp from another.");
299 return asObject(arg0
);
302 UString pattern
= arg0
.isUndefined() ? UString("") : arg0
.toString(exec
);
303 UString flags
= arg1
.isUndefined() ? UString("") : arg1
.toString(exec
);
305 RefPtr
<RegExp
> regExp
= exec
->globalData().regExpCache()->lookupOrCreate(pattern
, flags
);
306 if (!regExp
->isValid())
307 return throwError(exec
, SyntaxError
, makeString("Invalid regular expression: ", regExp
->errorMessage()));
308 return new (exec
) RegExpObject(exec
->lexicalGlobalObject()->regExpStructure(), regExp
.release());
311 static JSObject
* constructWithRegExpConstructor(ExecState
* exec
, JSObject
*, const ArgList
& args
)
313 return constructRegExp(exec
, args
);
316 ConstructType
RegExpConstructor::getConstructData(ConstructData
& constructData
)
318 constructData
.native
.function
= constructWithRegExpConstructor
;
319 return ConstructTypeHost
;
323 static JSValue JSC_HOST_CALL
callRegExpConstructor(ExecState
* exec
, JSObject
*, JSValue
, const ArgList
& args
)
325 return constructRegExp(exec
, args
);
328 CallType
RegExpConstructor::getCallData(CallData
& callData
)
330 callData
.native
.function
= callRegExpConstructor
;
334 void RegExpConstructor::setInput(const UString
& input
)
339 const UString
& RegExpConstructor::input() const
341 // Can detect a distinct initial state that is invisible to JavaScript, by checking for null
342 // state (since jsString turns null strings to empty strings).
346 void RegExpConstructor::setMultiline(bool multiline
)
348 d
->multiline
= multiline
;
351 bool RegExpConstructor::multiline() const