2  *  Copyright (C) 1999-2000 Harri Porten (porten@kde.org) 
   3  *  Copyright (C) 2003, 2007, 2008 Apple Inc. All Rights Reserved. 
   5  *  This library is free software; you can redistribute it and/or 
   6  *  modify it under the terms of the GNU Lesser General Public 
   7  *  License as published by the Free Software Foundation; either 
   8  *  version 2 of the License, or (at your option) any later version. 
  10  *  This library is distributed in the hope that it will be useful, 
  11  *  but WITHOUT ANY WARRANTY; without even the implied warranty of 
  12  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU 
  13  *  Lesser General Public License for more details. 
  15  *  You should have received a copy of the GNU Lesser General Public 
  16  *  License along with this library; if not, write to the Free Software 
  17  *  Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA 
  22 #include "RegExpObject.h" 
  25 #include "ExceptionHelpers.h" 
  27 #include "JSGlobalObject.h" 
  30 #include "RegExpConstructor.h" 
  31 #include "RegExpPrototype.h" 
  32 #include "UStringConcatenate.h" 
  33 #include <wtf/PassOwnPtr.h> 
  35 #include <wtf/PassOwnPtr.h> 
  39 static JSValue 
regExpObjectGlobal(ExecState
*, JSValue
, const Identifier
&); 
  40 static JSValue 
regExpObjectIgnoreCase(ExecState
*, JSValue
, const Identifier
&); 
  41 static JSValue 
regExpObjectMultiline(ExecState
*, JSValue
, const Identifier
&); 
  42 static JSValue 
regExpObjectSource(ExecState
*, JSValue
, const Identifier
&); 
  43 static JSValue 
regExpObjectLastIndex(ExecState
*, JSValue
, const Identifier
&); 
  44 static void setRegExpObjectLastIndex(ExecState
*, JSObject
*, JSValue
); 
  48 #include "RegExpObject.lut.h" 
  52 ASSERT_CLASS_FITS_IN_CELL(RegExpObject
); 
  54 const ClassInfo 
RegExpObject::s_info 
= { "RegExp", &JSObjectWithGlobalObject::s_info
, 0, ExecState::regExpTable 
}; 
  56 /* Source for RegExpObject.lut.h 
  58     global        regExpObjectGlobal       DontDelete|ReadOnly|DontEnum 
  59     ignoreCase    regExpObjectIgnoreCase   DontDelete|ReadOnly|DontEnum 
  60     multiline     regExpObjectMultiline    DontDelete|ReadOnly|DontEnum 
  61     source        regExpObjectSource       DontDelete|ReadOnly|DontEnum 
  62     lastIndex     regExpObjectLastIndex    DontDelete|DontEnum 
  66 RegExpObject::RegExpObject(JSGlobalObject
* globalObject
, Structure
* structure
, RegExp
* regExp
) 
  67     : JSObjectWithGlobalObject(globalObject
, structure
) 
  68     , d(adoptPtr(new RegExpObjectData(globalObject
->globalData(), this, regExp
))) 
  70     ASSERT(inherits(&s_info
)); 
  73 RegExpObject::~RegExpObject() 
  77 void RegExpObject::visitChildren(SlotVisitor
& visitor
) 
  79     ASSERT_GC_OBJECT_INHERITS(this, &s_info
); 
  80     COMPILE_ASSERT(StructureFlags 
& OverridesVisitChildren
, OverridesVisitChildrenWithoutSettingFlag
); 
  81     ASSERT(structure()->typeInfo().overridesVisitChildren()); 
  82     Base::visitChildren(visitor
); 
  84         visitor
.append(&d
->regExp
); 
  85     if (UNLIKELY(!d
->lastIndex
.get().isInt32())) 
  86         visitor
.append(&d
->lastIndex
); 
  89 bool RegExpObject::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
) 
  91     return getStaticValueSlot
<RegExpObject
, JSObject
>(exec
, ExecState::regExpTable(exec
), this, propertyName
, slot
); 
  94 bool RegExpObject::getOwnPropertyDescriptor(ExecState
* exec
, const Identifier
& propertyName
, PropertyDescriptor
& descriptor
) 
  96     return getStaticValueDescriptor
<RegExpObject
, JSObject
>(exec
, ExecState::regExpTable(exec
), this, propertyName
, descriptor
); 
  99 JSValue 
regExpObjectGlobal(ExecState
*, JSValue slotBase
, const Identifier
&) 
 101     return jsBoolean(asRegExpObject(slotBase
)->regExp()->global()); 
 104 JSValue 
regExpObjectIgnoreCase(ExecState
*, JSValue slotBase
, const Identifier
&) 
 106     return jsBoolean(asRegExpObject(slotBase
)->regExp()->ignoreCase()); 
 109 JSValue 
regExpObjectMultiline(ExecState
*, JSValue slotBase
, const Identifier
&) 
 111     return jsBoolean(asRegExpObject(slotBase
)->regExp()->multiline()); 
 114 JSValue 
regExpObjectSource(ExecState
* exec
, JSValue slotBase
, const Identifier
&) 
 116     return jsString(exec
, asRegExpObject(slotBase
)->regExp()->pattern()); 
 119 JSValue 
regExpObjectLastIndex(ExecState
*, JSValue slotBase
, const Identifier
&) 
 121     return asRegExpObject(slotBase
)->getLastIndex(); 
 124 void RegExpObject::put(ExecState
* exec
, const Identifier
& propertyName
, JSValue value
, PutPropertySlot
& slot
) 
 126     lookupPut
<RegExpObject
, JSObject
>(exec
, propertyName
, value
, ExecState::regExpTable(exec
), this, slot
); 
 129 void setRegExpObjectLastIndex(ExecState
* exec
, JSObject
* baseObject
, JSValue value
) 
 131     asRegExpObject(baseObject
)->setLastIndex(exec
->globalData(), value
); 
 134 JSValue 
RegExpObject::test(ExecState
* exec
) 
 136     return jsBoolean(match(exec
)); 
 139 JSValue 
RegExpObject::exec(ExecState
* exec
) 
 142         return exec
->lexicalGlobalObject()->regExpConstructor()->arrayOfMatches(exec
); 
 146 // Shared implementation used by test and exec. 
 147 bool RegExpObject::match(ExecState
* exec
) 
 149     RegExpConstructor
* regExpConstructor 
= exec
->lexicalGlobalObject()->regExpConstructor(); 
 150     UString input 
= exec
->argument(0).toString(exec
); 
 151     JSGlobalData
* globalData 
= &exec
->globalData(); 
 152     if (!regExp()->global()) { 
 155         regExpConstructor
->performMatch(*globalData
, d
->regExp
.get(), input
, 0, position
, length
); 
 156         return position 
>= 0; 
 159     JSValue jsLastIndex 
= getLastIndex(); 
 161     if (LIKELY(jsLastIndex
.isUInt32())) { 
 162         lastIndex 
= jsLastIndex
.asUInt32(); 
 163         if (lastIndex 
> input
.length()) { 
 168         double doubleLastIndex 
= jsLastIndex
.toInteger(exec
); 
 169         if (doubleLastIndex 
< 0 || doubleLastIndex 
> input
.length()) { 
 173         lastIndex 
= static_cast<unsigned>(doubleLastIndex
); 
 178     regExpConstructor
->performMatch(*globalData
, d
->regExp
.get(), input
, lastIndex
, position
, length
); 
 184     setLastIndex(position 
+ length
);