mirror of
https://github.com/darlinghq/darling-JavaScriptCore.git
synced 2024-12-12 06:16:06 +00:00
144 lines
4.6 KiB
C++
144 lines
4.6 KiB
C++
/*
|
|
* Copyright (C) 1999-2000 Harri Porten (porten@kde.org)
|
|
* Copyright (C) 2003, 2007, 2008, 2012, 2016 Apple Inc. All Rights Reserved.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include "ButterflyInlines.h"
|
|
#include "Error.h"
|
|
#include "ExceptionHelpers.h"
|
|
#include "JSArray.h"
|
|
#include "JSGlobalObject.h"
|
|
#include "JSString.h"
|
|
#include "JSCInlines.h"
|
|
#include "RegExpConstructor.h"
|
|
#include "RegExpMatchesArray.h"
|
|
#include "RegExpObject.h"
|
|
|
|
namespace JSC {
|
|
|
|
ALWAYS_INLINE unsigned getRegExpObjectLastIndexAsUnsigned(
|
|
ExecState* exec, RegExpObject* regExpObject, const String& input)
|
|
{
|
|
VM& vm = exec->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
JSValue jsLastIndex = regExpObject->getLastIndex();
|
|
unsigned lastIndex;
|
|
if (LIKELY(jsLastIndex.isUInt32())) {
|
|
lastIndex = jsLastIndex.asUInt32();
|
|
if (lastIndex > input.length()) {
|
|
scope.release();
|
|
regExpObject->setLastIndex(exec, 0);
|
|
return UINT_MAX;
|
|
}
|
|
} else {
|
|
double doubleLastIndex = jsLastIndex.toInteger(exec);
|
|
RETURN_IF_EXCEPTION(scope, UINT_MAX);
|
|
if (doubleLastIndex < 0 || doubleLastIndex > input.length()) {
|
|
scope.release();
|
|
regExpObject->setLastIndex(exec, 0);
|
|
return UINT_MAX;
|
|
}
|
|
lastIndex = static_cast<unsigned>(doubleLastIndex);
|
|
}
|
|
return lastIndex;
|
|
}
|
|
|
|
JSValue RegExpObject::execInline(ExecState* exec, JSGlobalObject* globalObject, JSString* string)
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
|
|
RegExp* regExp = this->regExp();
|
|
RegExpConstructor* regExpConstructor = globalObject->regExpConstructor();
|
|
String input = string->value(exec);
|
|
RETURN_IF_EXCEPTION(scope, { });
|
|
|
|
bool globalOrSticky = regExp->globalOrSticky();
|
|
|
|
unsigned lastIndex;
|
|
if (globalOrSticky) {
|
|
lastIndex = getRegExpObjectLastIndexAsUnsigned(exec, this, input);
|
|
ASSERT(!scope.exception() || lastIndex == UINT_MAX);
|
|
if (lastIndex == UINT_MAX)
|
|
return jsNull();
|
|
} else
|
|
lastIndex = 0;
|
|
|
|
MatchResult result;
|
|
JSArray* array =
|
|
createRegExpMatchesArray(vm, globalObject, string, input, regExp, lastIndex, result);
|
|
if (!array) {
|
|
scope.release();
|
|
if (globalOrSticky)
|
|
setLastIndex(exec, 0);
|
|
return jsNull();
|
|
}
|
|
|
|
if (globalOrSticky)
|
|
setLastIndex(exec, result.end);
|
|
RETURN_IF_EXCEPTION(scope, { });
|
|
regExpConstructor->recordMatch(vm, regExp, string, result);
|
|
return array;
|
|
}
|
|
|
|
// Shared implementation used by test and exec.
|
|
MatchResult RegExpObject::matchInline(
|
|
ExecState* exec, JSGlobalObject* globalObject, JSString* string)
|
|
{
|
|
VM& vm = globalObject->vm();
|
|
auto scope = DECLARE_THROW_SCOPE(vm);
|
|
|
|
RegExp* regExp = this->regExp();
|
|
RegExpConstructor* regExpConstructor = globalObject->regExpConstructor();
|
|
String input = string->value(exec);
|
|
RETURN_IF_EXCEPTION(scope, { });
|
|
|
|
if (!regExp->global() && !regExp->sticky())
|
|
return regExpConstructor->performMatch(vm, regExp, string, input, 0);
|
|
|
|
unsigned lastIndex = getRegExpObjectLastIndexAsUnsigned(exec, this, input);
|
|
ASSERT(!scope.exception() || (lastIndex == UINT_MAX));
|
|
if (lastIndex == UINT_MAX)
|
|
return MatchResult::failed();
|
|
|
|
MatchResult result = regExpConstructor->performMatch(vm, regExp, string, input, lastIndex);
|
|
scope.release();
|
|
setLastIndex(exec, result.end);
|
|
return result;
|
|
}
|
|
|
|
unsigned RegExpObject::advanceStringUnicode(String s, unsigned length, unsigned currentIndex)
|
|
{
|
|
if (currentIndex + 1 >= length)
|
|
return currentIndex + 1;
|
|
|
|
UChar first = s[currentIndex];
|
|
if (first < 0xD800 || first > 0xDBFF)
|
|
return currentIndex + 1;
|
|
|
|
UChar second = s[currentIndex + 1];
|
|
if (second < 0xDC00 || second > 0xDFFF)
|
|
return currentIndex + 1;
|
|
|
|
return currentIndex + 2;
|
|
}
|
|
|
|
} // namespace JSC
|