mirror of
https://github.com/Samsung/escargot.git
synced 2026-06-29 10:02:14 +00:00
* check index property set in DerivedObject::defineOwnProperty method only * index property check is removed for normal Object because PrototypeObject would check it instead Signed-off-by: HyukWoo Park <hyukwoo.park@samsung.com>
210 lines
8.6 KiB
C++
210 lines
8.6 KiB
C++
/*
|
|
* Copyright (c) 2016-present Samsung Electronics Co., Ltd
|
|
*
|
|
* 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.1 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
|
|
*/
|
|
|
|
#include "Escargot.h"
|
|
#include "StringObject.h"
|
|
#include "Context.h"
|
|
|
|
namespace Escargot {
|
|
|
|
StringObject::StringObject(ExecutionState& state, String* value)
|
|
: StringObject(state, state.context()->globalObject()->stringPrototype(), value)
|
|
{
|
|
}
|
|
|
|
StringObject::StringObject(ExecutionState& state, Object* proto, String* value)
|
|
: DerivedObject(state, proto, ESCARGOT_OBJECT_BUILTIN_PROPERTY_NUMBER + 1)
|
|
, m_primitiveValue(value)
|
|
{
|
|
m_structure = state.context()->defaultStructureForStringObject();
|
|
m_values[ESCARGOT_OBJECT_BUILTIN_PROPERTY_NUMBER] = Value();
|
|
}
|
|
|
|
void* StringObject::operator new(size_t size)
|
|
{
|
|
static MAY_THREAD_LOCAL bool typeInited = false;
|
|
static MAY_THREAD_LOCAL GC_descr descr;
|
|
if (!typeInited) {
|
|
GC_word obj_bitmap[GC_BITMAP_SIZE(StringObject)] = { 0 };
|
|
Object::fillGCDescriptor(obj_bitmap);
|
|
GC_set_bit(obj_bitmap, GC_WORD_OFFSET(StringObject, m_primitiveValue));
|
|
descr = GC_make_descriptor(obj_bitmap, GC_WORD_LEN(StringObject));
|
|
typeInited = true;
|
|
}
|
|
return GC_MALLOC_EXPLICITLY_TYPED(size, descr);
|
|
}
|
|
|
|
ObjectHasPropertyResult StringObject::hasProperty(ExecutionState& state, const ObjectPropertyName& P)
|
|
{
|
|
size_t idx = P.tryToUseAsIndexProperty();
|
|
if (idx != Value::InvalidIndexPropertyValue) {
|
|
size_t strLen = m_primitiveValue->length();
|
|
if (LIKELY(idx < strLen)) {
|
|
return ObjectHasPropertyResult(ObjectGetResult(state.context()->staticStrings().charCodeToString(m_primitiveValue->charAt(idx)), false, true, false));
|
|
}
|
|
}
|
|
return Object::hasProperty(state, P);
|
|
}
|
|
|
|
|
|
ObjectGetResult StringObject::getOwnProperty(ExecutionState& state, const ObjectPropertyName& P)
|
|
{
|
|
size_t idx = P.tryToUseAsIndexProperty();
|
|
if (idx != Value::InvalidIndexPropertyValue) {
|
|
size_t strLen = m_primitiveValue->length();
|
|
if (LIKELY(idx < strLen)) {
|
|
return ObjectGetResult(state.context()->staticStrings().charCodeToString(m_primitiveValue->charAt(idx)), false, true, false);
|
|
}
|
|
}
|
|
return Object::getOwnProperty(state, P);
|
|
}
|
|
|
|
bool StringObject::defineOwnProperty(ExecutionState& state, const ObjectPropertyName& P, const ObjectPropertyDescriptor& desc)
|
|
{
|
|
auto r = getOwnProperty(state, P);
|
|
if (r.hasValue() && !r.isConfigurable())
|
|
return false;
|
|
return DerivedObject::defineOwnProperty(state, P, desc);
|
|
}
|
|
|
|
bool StringObject::deleteOwnProperty(ExecutionState& state, const ObjectPropertyName& P)
|
|
{
|
|
auto r = getOwnProperty(state, P);
|
|
if (r.hasValue() && !r.isConfigurable())
|
|
return false;
|
|
return Object::deleteOwnProperty(state, P);
|
|
}
|
|
|
|
void StringObject::enumeration(ExecutionState& state, bool (*callback)(ExecutionState& state, Object* self, const ObjectPropertyName&, const ObjectStructurePropertyDescriptor& desc, void* data), void* data, bool shouldSkipSymbolKey)
|
|
{
|
|
size_t len = m_primitiveValue->length();
|
|
for (size_t i = 0; i < len; i++) {
|
|
if (!callback(state, this, ObjectPropertyName(state, i), ObjectStructurePropertyDescriptor::createDataDescriptor(ObjectStructurePropertyDescriptor::EnumerablePresent), data)) {
|
|
return;
|
|
}
|
|
}
|
|
Object::enumeration(state, callback, data, shouldSkipSymbolKey);
|
|
}
|
|
|
|
ObjectGetResult StringObject::getIndexedProperty(ExecutionState& state, const Value& property, const Value& receiver)
|
|
{
|
|
size_t idx = property.tryToUseAsIndexProperty(state);
|
|
if (idx != Value::InvalidIndexPropertyValue) {
|
|
size_t strLen = m_primitiveValue->length();
|
|
if (LIKELY(idx < strLen)) {
|
|
return ObjectGetResult(state.context()->staticStrings().charCodeToString(m_primitiveValue->charAt(idx)), false, true, false);
|
|
}
|
|
}
|
|
return get(state, ObjectPropertyName(state, property), receiver);
|
|
}
|
|
|
|
Value StringObject::getIndexedPropertyValue(ExecutionState& state, const Value& property, const Value& receiver)
|
|
{
|
|
size_t idx = property.tryToUseAsIndexProperty(state);
|
|
if (idx != Value::InvalidIndexPropertyValue) {
|
|
size_t strLen = m_primitiveValue->length();
|
|
if (LIKELY(idx < strLen)) {
|
|
return state.context()->staticStrings().charCodeToString(m_primitiveValue->charAt(idx));
|
|
}
|
|
}
|
|
return get(state, ObjectPropertyName(state, property), receiver).value(state, receiver);
|
|
}
|
|
|
|
ObjectHasPropertyResult StringObject::hasIndexedProperty(ExecutionState& state, const Value& propertyName)
|
|
{
|
|
size_t idx = propertyName.tryToUseAsIndexProperty(state);
|
|
if (idx != Value::InvalidIndexPropertyValue) {
|
|
size_t strLen = m_primitiveValue->length();
|
|
if (LIKELY(idx < strLen)) {
|
|
return ObjectHasPropertyResult(ObjectGetResult(state.context()->staticStrings().charCodeToString(m_primitiveValue->charAt(idx)), false, true, false));
|
|
}
|
|
}
|
|
return hasProperty(state, ObjectPropertyName(state, propertyName));
|
|
}
|
|
|
|
StringIteratorObject::StringIteratorObject(ExecutionState& state, String* s)
|
|
: IteratorObject(state, state.context()->globalObject()->stringIteratorPrototype())
|
|
, m_string(s)
|
|
, m_iteratorNextIndex(0)
|
|
{
|
|
}
|
|
|
|
void* StringIteratorObject::operator new(size_t size)
|
|
{
|
|
static MAY_THREAD_LOCAL bool typeInited = false;
|
|
static MAY_THREAD_LOCAL GC_descr descr;
|
|
if (!typeInited) {
|
|
GC_word obj_bitmap[GC_BITMAP_SIZE(StringIteratorObject)] = { 0 };
|
|
Object::fillGCDescriptor(obj_bitmap);
|
|
GC_set_bit(obj_bitmap, GC_WORD_OFFSET(StringIteratorObject, m_string));
|
|
descr = GC_make_descriptor(obj_bitmap, GC_WORD_LEN(StringIteratorObject));
|
|
typeInited = true;
|
|
}
|
|
return GC_MALLOC_EXPLICITLY_TYPED(size, descr);
|
|
}
|
|
|
|
std::pair<Value, bool> StringIteratorObject::advance(ExecutionState& state)
|
|
{
|
|
// Let s be the value of the [[IteratedString]] internal slot of O.
|
|
String* s = m_string;
|
|
// If a is undefined, return CreateIterResultObject(undefined, true).
|
|
if (s == nullptr) {
|
|
return std::make_pair(Value(), true);
|
|
}
|
|
// Let position be the value of the [[StringIteratorNextIndex]] internal slot of O.
|
|
size_t position = m_iteratorNextIndex;
|
|
// Let len be the number of elements in s.
|
|
auto len = s->length();
|
|
|
|
// If position ≥ len, then
|
|
if (position >= len) {
|
|
// Set the value of the [[IteratedString]] internal slot of O to undefined.
|
|
m_string = nullptr;
|
|
// Return CreateIterResultObject(undefined, true).
|
|
return std::make_pair(Value(), true);
|
|
}
|
|
|
|
// Let first be the code unit value at index position in s.
|
|
auto first = s->charAt(position);
|
|
// If first < 0xD800 or first > 0xDBFF or position+1 = len, let resultString be the string consisting of the single code unit first.
|
|
String* resultString;
|
|
if (first < 0xD800 || first > 0xDBFF || (position + 1 == len)) {
|
|
resultString = state.context()->staticStrings().charCodeToString(first);
|
|
} else {
|
|
// Let second be the code unit value at index position+1 in the String S.
|
|
auto second = s->charAt(position + 1);
|
|
// If second < 0xDC00 or second > 0xDFFF, let resultString be the string consisting of the single code unit first.
|
|
if (second < 0xDC00 || second > 0xDFFF) {
|
|
resultString = state.context()->staticStrings().charCodeToString(first);
|
|
} else {
|
|
// Else, let resultString be the string consisting of the code unit first followed by the code unit second.
|
|
char16_t s[2] = { first, second };
|
|
resultString = new UTF16String(s, 2);
|
|
}
|
|
}
|
|
// Let resultSize be the number of code units in resultString.
|
|
auto resultSize = resultString->length();
|
|
|
|
// Set the value of the [[StringIteratorNextIndex]] internal slot of O to position + resultSize.
|
|
m_iteratorNextIndex = position + resultSize;
|
|
|
|
// Return CreateIterResultObject(resultString, false).
|
|
return std::make_pair(Value(resultString), false);
|
|
}
|
|
} // namespace Escargot
|