2014-04-24 11:44:22 +00:00
|
|
|
// Copyright 2012 the V8 project authors. All rights reserved.
|
2014-04-29 06:42:26 +00:00
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
2014-04-24 11:44:22 +00:00
|
|
|
//
|
|
|
|
// Review notes:
|
|
|
|
//
|
|
|
|
// - The use of macros in these inline functions may seem superfluous
|
|
|
|
// but it is absolutely needed to make sure gcc generates optimal
|
|
|
|
// code. gcc is not happy when attempting to inline too deep.
|
|
|
|
//
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
#ifndef V8_OBJECTS_INL_H_
|
|
|
|
#define V8_OBJECTS_INL_H_
|
|
|
|
|
2014-04-24 11:44:22 +00:00
|
|
|
#include "elements.h"
|
2010-08-30 07:10:40 +00:00
|
|
|
#include "objects.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
#include "contexts.h"
|
|
|
|
#include "conversions-inl.h"
|
2010-08-30 07:10:40 +00:00
|
|
|
#include "heap.h"
|
2014-04-24 11:44:22 +00:00
|
|
|
#include "isolate.h"
|
2014-01-16 17:08:45 +00:00
|
|
|
#include "heap-inl.h"
|
2008-07-03 15:10:15 +00:00
|
|
|
#include "property.h"
|
2010-08-30 07:10:40 +00:00
|
|
|
#include "spaces.h"
|
2011-09-19 18:36:47 +00:00
|
|
|
#include "store-buffer.h"
|
2014-04-24 11:32:48 +00:00
|
|
|
#include "v8memory.h"
|
2014-04-24 11:44:22 +00:00
|
|
|
#include "factory.h"
|
|
|
|
#include "incremental-marking.h"
|
|
|
|
#include "transitions-inl.h"
|
|
|
|
#include "objects-visiting.h"
|
2011-09-19 18:36:47 +00:00
|
|
|
|
2009-05-25 10:05:56 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
PropertyDetails::PropertyDetails(Smi* smi) {
|
|
|
|
value_ = smi->value();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-18 11:30:51 +00:00
|
|
|
Smi* PropertyDetails::AsSmi() const {
|
2013-05-31 19:11:09 +00:00
|
|
|
// Ensure the upper 2 bits have the same value by sign extending it. This is
|
|
|
|
// necessary to be able to use the 31st bit of the property details.
|
|
|
|
int value = value_ << 1;
|
|
|
|
return Smi::FromInt(value >> 1);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-18 11:30:51 +00:00
|
|
|
PropertyDetails PropertyDetails::AsDeleted() const {
|
2011-01-27 08:35:39 +00:00
|
|
|
Smi* smi = Smi::FromInt(value_ | DeletedField::encode(1));
|
2009-06-30 10:05:36 +00:00
|
|
|
return PropertyDetails(smi);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
#define TYPE_CHECKER(type, instancetype) \
|
|
|
|
bool Object::Is##type() { \
|
|
|
|
return Object::IsHeapObject() && \
|
|
|
|
HeapObject::cast(this)->map()->instance_type() == instancetype; \
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
#define CAST_ACCESSOR(type) \
|
|
|
|
type* type::cast(Object* object) { \
|
2013-10-25 11:10:28 +00:00
|
|
|
SLOW_ASSERT(object->Is##type()); \
|
2008-07-03 15:10:15 +00:00
|
|
|
return reinterpret_cast<type*>(object); \
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define INT_ACCESSORS(holder, name, offset) \
|
|
|
|
int holder::name() { return READ_INT_FIELD(this, offset); } \
|
|
|
|
void holder::set_##name(int value) { WRITE_INT_FIELD(this, offset, value); }
|
|
|
|
|
|
|
|
|
|
|
|
#define ACCESSORS(holder, name, type, offset) \
|
|
|
|
type* holder::name() { return type::cast(READ_FIELD(this, offset)); } \
|
2008-10-23 08:46:32 +00:00
|
|
|
void holder::set_##name(type* value, WriteBarrierMode mode) { \
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_FIELD(this, offset, value); \
|
2011-09-19 18:36:47 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, offset, value, mode); \
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-14 16:16:46 +00:00
|
|
|
// Getter that returns a tagged Smi and setter that writes a tagged Smi.
|
|
|
|
#define ACCESSORS_TO_SMI(holder, name, offset) \
|
|
|
|
Smi* holder::name() { return Smi::cast(READ_FIELD(this, offset)); } \
|
|
|
|
void holder::set_##name(Smi* value, WriteBarrierMode mode) { \
|
|
|
|
WRITE_FIELD(this, offset, value); \
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Getter that returns a Smi as an int and writes an int as a Smi.
|
2008-07-03 15:10:15 +00:00
|
|
|
#define SMI_ACCESSORS(holder, name, offset) \
|
|
|
|
int holder::name() { \
|
|
|
|
Object* value = READ_FIELD(this, offset); \
|
|
|
|
return Smi::cast(value)->value(); \
|
|
|
|
} \
|
|
|
|
void holder::set_##name(int value) { \
|
|
|
|
WRITE_FIELD(this, offset, Smi::FromInt(value)); \
|
|
|
|
}
|
|
|
|
|
2014-03-31 14:29:01 +00:00
|
|
|
#define SYNCHRONIZED_SMI_ACCESSORS(holder, name, offset) \
|
|
|
|
int holder::synchronized_##name() { \
|
|
|
|
Object* value = ACQUIRE_READ_FIELD(this, offset); \
|
|
|
|
return Smi::cast(value)->value(); \
|
|
|
|
} \
|
|
|
|
void holder::synchronized_set_##name(int value) { \
|
|
|
|
RELEASE_WRITE_FIELD(this, offset, Smi::FromInt(value)); \
|
|
|
|
}
|
|
|
|
|
2014-04-08 16:31:57 +00:00
|
|
|
#define NOBARRIER_SMI_ACCESSORS(holder, name, offset) \
|
|
|
|
int holder::nobarrier_##name() { \
|
|
|
|
Object* value = NOBARRIER_READ_FIELD(this, offset); \
|
|
|
|
return Smi::cast(value)->value(); \
|
|
|
|
} \
|
|
|
|
void holder::nobarrier_set_##name(int value) { \
|
|
|
|
NOBARRIER_WRITE_FIELD(this, offset, Smi::FromInt(value)); \
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-08-19 07:30:20 +00:00
|
|
|
#define BOOL_GETTER(holder, field, name, offset) \
|
|
|
|
bool holder::name() { \
|
|
|
|
return BooleanBit::get(field(), offset); \
|
|
|
|
} \
|
|
|
|
|
|
|
|
|
|
|
|
#define BOOL_ACCESSORS(holder, field, name, offset) \
|
2008-07-03 15:10:15 +00:00
|
|
|
bool holder::name() { \
|
|
|
|
return BooleanBit::get(field(), offset); \
|
|
|
|
} \
|
|
|
|
void holder::set_##name(bool value) { \
|
|
|
|
set_##field(BooleanBit::set(field(), offset, value)); \
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-20 09:38:24 +00:00
|
|
|
bool Object::IsFixedArrayBase() {
|
2014-01-16 17:08:45 +00:00
|
|
|
return IsFixedArray() || IsFixedDoubleArray() || IsConstantPoolArray() ||
|
|
|
|
IsFixedTypedArrayBase() || IsExternalArray();
|
2011-10-20 09:38:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-13 12:27:03 +00:00
|
|
|
// External objects are not extensible, so the map check is enough.
|
|
|
|
bool Object::IsExternal() {
|
|
|
|
return Object::IsHeapObject() &&
|
|
|
|
HeapObject::cast(this)->map() ==
|
|
|
|
HeapObject::cast(this)->GetHeap()->external_map();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-12 14:33:08 +00:00
|
|
|
bool Object::IsAccessorInfo() {
|
|
|
|
return IsExecutableAccessorInfo() || IsDeclaredAccessorInfo();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Object::IsSmi() {
|
|
|
|
return HAS_SMI_TAG(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsHeapObject() {
|
2009-08-26 10:33:11 +00:00
|
|
|
return Internals::HasHeapObjectTag(this);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(HeapNumber, HEAP_NUMBER_TYPE)
|
2013-03-01 10:34:31 +00:00
|
|
|
TYPE_CHECKER(Symbol, SYMBOL_TYPE)
|
|
|
|
|
|
|
|
|
2013-03-01 13:28:55 +00:00
|
|
|
bool Object::IsString() {
|
2013-03-01 10:34:31 +00:00
|
|
|
return Object::IsHeapObject()
|
2013-03-01 13:28:55 +00:00
|
|
|
&& HeapObject::cast(this)->map()->instance_type() < FIRST_NONSTRING_TYPE;
|
2013-03-01 10:34:31 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2013-03-01 13:28:55 +00:00
|
|
|
bool Object::IsName() {
|
|
|
|
return IsString() || IsSymbol();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsUniqueName() {
|
|
|
|
return IsInternalizedString() || IsSymbol();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-21 13:51:04 +00:00
|
|
|
bool Object::IsSpecObject() {
|
|
|
|
return Object::IsHeapObject()
|
|
|
|
&& HeapObject::cast(this)->map()->instance_type() >= FIRST_SPEC_OBJECT_TYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-22 17:12:41 +00:00
|
|
|
bool Object::IsSpecFunction() {
|
|
|
|
if (!Object::IsHeapObject()) return false;
|
|
|
|
InstanceType type = HeapObject::cast(this)->map()->instance_type();
|
|
|
|
return type == JS_FUNCTION_TYPE || type == JS_FUNCTION_PROXY_TYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-28 13:42:03 +00:00
|
|
|
bool Object::IsTemplateInfo() {
|
|
|
|
return IsObjectTemplateInfo() || IsFunctionTemplateInfo();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-28 17:03:34 +00:00
|
|
|
bool Object::IsInternalizedString() {
|
2008-11-03 10:16:05 +00:00
|
|
|
if (!this->IsHeapObject()) return false;
|
|
|
|
uint32_t type = HeapObject::cast(this)->map()->instance_type();
|
2013-07-19 11:29:11 +00:00
|
|
|
STATIC_ASSERT(kNotInternalizedTag != 0);
|
|
|
|
return (type & (kIsNotStringMask | kIsNotInternalizedMask)) ==
|
|
|
|
(kStringTag | kInternalizedTag);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
bool Object::IsConsString() {
|
2011-08-26 13:03:30 +00:00
|
|
|
if (!IsString()) return false;
|
|
|
|
return StringShape(String::cast(this)).IsCons();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsSlicedString() {
|
|
|
|
if (!IsString()) return false;
|
|
|
|
return StringShape(String::cast(this)).IsSliced();
|
2008-11-03 10:16:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsSeqString() {
|
|
|
|
if (!IsString()) return false;
|
|
|
|
return StringShape(String::cast(this)).IsSequential();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-15 13:31:27 +00:00
|
|
|
bool Object::IsSeqOneByteString() {
|
2008-11-03 10:16:05 +00:00
|
|
|
if (!IsString()) return false;
|
2009-03-17 09:33:06 +00:00
|
|
|
return StringShape(String::cast(this)).IsSequential() &&
|
2012-11-21 10:01:05 +00:00
|
|
|
String::cast(this)->IsOneByteRepresentation();
|
2008-10-20 11:38:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-09 08:08:04 +00:00
|
|
|
bool Object::IsSeqTwoByteString() {
|
2008-11-03 10:16:05 +00:00
|
|
|
if (!IsString()) return false;
|
2009-03-17 09:33:06 +00:00
|
|
|
return StringShape(String::cast(this)).IsSequential() &&
|
2009-05-01 11:16:29 +00:00
|
|
|
String::cast(this)->IsTwoByteRepresentation();
|
2008-10-09 08:08:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
bool Object::IsExternalString() {
|
|
|
|
if (!IsString()) return false;
|
|
|
|
return StringShape(String::cast(this)).IsExternal();
|
2008-10-09 08:08:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
bool Object::IsExternalAsciiString() {
|
|
|
|
if (!IsString()) return false;
|
2009-03-17 09:33:06 +00:00
|
|
|
return StringShape(String::cast(this)).IsExternal() &&
|
2012-11-21 10:01:05 +00:00
|
|
|
String::cast(this)->IsOneByteRepresentation();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
bool Object::IsExternalTwoByteString() {
|
|
|
|
if (!IsString()) return false;
|
2009-03-17 09:33:06 +00:00
|
|
|
return StringShape(String::cast(this)).IsExternal() &&
|
2009-05-01 11:16:29 +00:00
|
|
|
String::cast(this)->IsTwoByteRepresentation();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2014-04-15 07:36:47 +00:00
|
|
|
|
2011-06-09 10:03:35 +00:00
|
|
|
bool Object::HasValidElements() {
|
|
|
|
// Dictionary is covered under FixedArray.
|
2014-01-16 17:08:45 +00:00
|
|
|
return IsFixedArray() || IsFixedDoubleArray() || IsExternalArray() ||
|
|
|
|
IsFixedTypedArrayBase();
|
2011-06-09 10:03:35 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-05-08 15:02:08 +00:00
|
|
|
|
2014-04-15 12:11:39 +00:00
|
|
|
Handle<Object> Object::NewStorageFor(Isolate* isolate,
|
|
|
|
Handle<Object> object,
|
|
|
|
Representation representation) {
|
|
|
|
if (representation.IsSmi() && object->IsUninitialized()) {
|
|
|
|
return handle(Smi::FromInt(0), isolate);
|
2013-12-18 17:53:50 +00:00
|
|
|
}
|
2014-04-15 12:11:39 +00:00
|
|
|
if (!representation.IsDouble()) return object;
|
|
|
|
if (object->IsUninitialized()) {
|
|
|
|
return isolate->factory()->NewHeapNumber(0);
|
2013-06-06 14:21:35 +00:00
|
|
|
}
|
2014-04-15 12:11:39 +00:00
|
|
|
return isolate->factory()->NewHeapNumber(object->Number());
|
2013-05-08 15:02:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
StringShape::StringShape(String* str)
|
2008-11-03 12:08:01 +00:00
|
|
|
: type_(str->map()->instance_type()) {
|
|
|
|
set_valid();
|
2008-11-03 10:16:05 +00:00
|
|
|
ASSERT((type_ & kIsNotStringMask) == kStringTag);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
StringShape::StringShape(Map* map)
|
2008-11-03 12:08:01 +00:00
|
|
|
: type_(map->instance_type()) {
|
|
|
|
set_valid();
|
2008-11-03 10:16:05 +00:00
|
|
|
ASSERT((type_ & kIsNotStringMask) == kStringTag);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
StringShape::StringShape(InstanceType t)
|
2008-11-03 12:08:01 +00:00
|
|
|
: type_(static_cast<uint32_t>(t)) {
|
|
|
|
set_valid();
|
2008-11-03 10:16:05 +00:00
|
|
|
ASSERT((type_ & kIsNotStringMask) == kStringTag);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-28 17:03:34 +00:00
|
|
|
bool StringShape::IsInternalized() {
|
2008-11-03 10:16:05 +00:00
|
|
|
ASSERT(valid());
|
2013-07-19 11:29:11 +00:00
|
|
|
STATIC_ASSERT(kNotInternalizedTag != 0);
|
|
|
|
return (type_ & (kIsNotStringMask | kIsNotInternalizedMask)) ==
|
|
|
|
(kStringTag | kInternalizedTag);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-21 10:01:05 +00:00
|
|
|
bool String::IsOneByteRepresentation() {
|
2009-05-01 11:16:29 +00:00
|
|
|
uint32_t type = map()->instance_type();
|
2012-11-08 12:14:29 +00:00
|
|
|
return (type & kStringEncodingMask) == kOneByteStringTag;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-01 11:16:29 +00:00
|
|
|
bool String::IsTwoByteRepresentation() {
|
|
|
|
uint32_t type = map()->instance_type();
|
|
|
|
return (type & kStringEncodingMask) == kTwoByteStringTag;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-21 10:01:05 +00:00
|
|
|
bool String::IsOneByteRepresentationUnderneath() {
|
2011-08-26 13:03:30 +00:00
|
|
|
uint32_t type = map()->instance_type();
|
|
|
|
STATIC_ASSERT(kIsIndirectStringTag != 0);
|
|
|
|
STATIC_ASSERT((kIsIndirectStringMask & kStringEncodingMask) == 0);
|
|
|
|
ASSERT(IsFlat());
|
|
|
|
switch (type & (kIsIndirectStringMask | kStringEncodingMask)) {
|
2012-11-08 12:14:29 +00:00
|
|
|
case kOneByteStringTag:
|
2011-08-26 13:03:30 +00:00
|
|
|
return true;
|
|
|
|
case kTwoByteStringTag:
|
|
|
|
return false;
|
|
|
|
default: // Cons or sliced string. Need to go deeper.
|
2012-11-21 10:01:05 +00:00
|
|
|
return GetUnderlying()->IsOneByteRepresentation();
|
2011-08-26 13:03:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool String::IsTwoByteRepresentationUnderneath() {
|
|
|
|
uint32_t type = map()->instance_type();
|
|
|
|
STATIC_ASSERT(kIsIndirectStringTag != 0);
|
|
|
|
STATIC_ASSERT((kIsIndirectStringMask & kStringEncodingMask) == 0);
|
|
|
|
ASSERT(IsFlat());
|
|
|
|
switch (type & (kIsIndirectStringMask | kStringEncodingMask)) {
|
2012-11-08 12:14:29 +00:00
|
|
|
case kOneByteStringTag:
|
2011-08-26 13:03:30 +00:00
|
|
|
return false;
|
|
|
|
case kTwoByteStringTag:
|
|
|
|
return true;
|
|
|
|
default: // Cons or sliced string. Need to go deeper.
|
|
|
|
return GetUnderlying()->IsTwoByteRepresentation();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-26 11:34:44 +00:00
|
|
|
bool String::HasOnlyOneByteChars() {
|
2010-06-17 16:19:28 +00:00
|
|
|
uint32_t type = map()->instance_type();
|
2013-04-26 14:08:18 +00:00
|
|
|
return (type & kOneByteDataHintMask) == kOneByteDataHintTag ||
|
|
|
|
IsOneByteRepresentation();
|
2013-01-09 10:30:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
bool StringShape::IsCons() {
|
|
|
|
return (type_ & kStringRepresentationMask) == kConsStringTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-26 13:03:30 +00:00
|
|
|
bool StringShape::IsSliced() {
|
|
|
|
return (type_ & kStringRepresentationMask) == kSlicedStringTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool StringShape::IsIndirect() {
|
|
|
|
return (type_ & kIsIndirectStringMask) == kIsIndirectStringTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
bool StringShape::IsExternal() {
|
|
|
|
return (type_ & kStringRepresentationMask) == kExternalStringTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool StringShape::IsSequential() {
|
|
|
|
return (type_ & kStringRepresentationMask) == kSeqStringTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
StringRepresentationTag StringShape::representation_tag() {
|
|
|
|
uint32_t tag = (type_ & kStringRepresentationMask);
|
|
|
|
return static_cast<StringRepresentationTag>(tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-23 12:22:12 +00:00
|
|
|
uint32_t StringShape::encoding_tag() {
|
|
|
|
return type_ & kStringEncodingMask;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
uint32_t StringShape::full_representation_tag() {
|
|
|
|
return (type_ & (kStringRepresentationMask | kStringEncodingMask));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-08-26 10:33:11 +00:00
|
|
|
STATIC_CHECK((kStringRepresentationMask | kStringEncodingMask) ==
|
|
|
|
Internals::kFullStringRepresentationMask);
|
|
|
|
|
2012-09-12 11:29:50 +00:00
|
|
|
STATIC_CHECK(static_cast<uint32_t>(kStringEncodingMask) ==
|
|
|
|
Internals::kStringEncodingMask);
|
|
|
|
|
2009-08-26 10:33:11 +00:00
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
bool StringShape::IsSequentialAscii() {
|
2012-11-08 12:14:29 +00:00
|
|
|
return full_representation_tag() == (kSeqStringTag | kOneByteStringTag);
|
2008-11-03 10:16:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool StringShape::IsSequentialTwoByte() {
|
2009-04-17 09:17:04 +00:00
|
|
|
return full_representation_tag() == (kSeqStringTag | kTwoByteStringTag);
|
2008-11-03 10:16:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool StringShape::IsExternalAscii() {
|
2012-11-08 12:14:29 +00:00
|
|
|
return full_representation_tag() == (kExternalStringTag | kOneByteStringTag);
|
2008-11-03 10:16:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-08 12:14:29 +00:00
|
|
|
STATIC_CHECK((kExternalStringTag | kOneByteStringTag) ==
|
2012-09-12 11:29:50 +00:00
|
|
|
Internals::kExternalAsciiRepresentationTag);
|
|
|
|
|
2012-11-08 12:14:29 +00:00
|
|
|
STATIC_CHECK(v8::String::ASCII_ENCODING == kOneByteStringTag);
|
2012-09-12 11:29:50 +00:00
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
bool StringShape::IsExternalTwoByte() {
|
2009-04-17 09:17:04 +00:00
|
|
|
return full_representation_tag() == (kExternalStringTag | kTwoByteStringTag);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-08-26 10:33:11 +00:00
|
|
|
STATIC_CHECK((kExternalStringTag | kTwoByteStringTag) ==
|
|
|
|
Internals::kExternalTwoByteRepresentationTag);
|
|
|
|
|
2012-09-12 11:29:50 +00:00
|
|
|
STATIC_CHECK(v8::String::TWO_BYTE_ENCODING == kTwoByteStringTag);
|
2009-08-26 10:33:11 +00:00
|
|
|
|
2008-11-25 11:07:48 +00:00
|
|
|
uc32 FlatStringReader::Get(int index) {
|
|
|
|
ASSERT(0 <= index && index <= length_);
|
|
|
|
if (is_ascii_) {
|
|
|
|
return static_cast<const byte*>(start_)[index];
|
|
|
|
} else {
|
|
|
|
return static_cast<const uc16*>(start_)[index];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-25 13:50:19 +00:00
|
|
|
Handle<Object> StringTableShape::AsHandle(Isolate* isolate, HashTableKey* key) {
|
|
|
|
return key->AsHandle(isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> MapCacheShape::AsHandle(Isolate* isolate, HashTableKey* key) {
|
|
|
|
return key->AsHandle(isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> CompilationCacheShape::AsHandle(Isolate* isolate,
|
|
|
|
HashTableKey* key) {
|
|
|
|
return key->AsHandle(isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> CodeCacheHashTableShape::AsHandle(Isolate* isolate,
|
|
|
|
HashTableKey* key) {
|
|
|
|
return key->AsHandle(isolate);
|
|
|
|
}
|
|
|
|
|
2014-01-17 10:27:57 +00:00
|
|
|
template <typename Char>
|
|
|
|
class SequentialStringKey : public HashTableKey {
|
|
|
|
public:
|
|
|
|
explicit SequentialStringKey(Vector<const Char> string, uint32_t seed)
|
|
|
|
: string_(string), hash_field_(0), seed_(seed) { }
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual uint32_t Hash() V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
hash_field_ = StringHasher::HashSequentialString<Char>(string_.start(),
|
|
|
|
string_.length(),
|
|
|
|
seed_);
|
|
|
|
|
|
|
|
uint32_t result = hash_field_ >> String::kHashShift;
|
|
|
|
ASSERT(result != 0); // Ensure that the hash value of 0 is never computed.
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual uint32_t HashForObject(Object* other) V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
return String::cast(other)->Hash();
|
|
|
|
}
|
|
|
|
|
|
|
|
Vector<const Char> string_;
|
|
|
|
uint32_t hash_field_;
|
|
|
|
uint32_t seed_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class OneByteStringKey : public SequentialStringKey<uint8_t> {
|
|
|
|
public:
|
|
|
|
OneByteStringKey(Vector<const uint8_t> str, uint32_t seed)
|
|
|
|
: SequentialStringKey<uint8_t>(str, seed) { }
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual bool IsMatch(Object* string) V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
return String::cast(string)->IsOneByteEqualTo(string_);
|
|
|
|
}
|
|
|
|
|
2014-04-25 13:59:33 +00:00
|
|
|
virtual Handle<Object> AsHandle(Isolate* isolate) V8_OVERRIDE;
|
2014-01-17 10:27:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-01-29 14:31:34 +00:00
|
|
|
template<class Char>
|
|
|
|
class SubStringKey : public HashTableKey {
|
2014-01-17 10:27:57 +00:00
|
|
|
public:
|
2014-01-29 14:31:34 +00:00
|
|
|
SubStringKey(Handle<String> string, int from, int length)
|
|
|
|
: string_(string), from_(from), length_(length) {
|
|
|
|
if (string_->IsSlicedString()) {
|
|
|
|
string_ = Handle<String>(Unslice(*string_, &from_));
|
|
|
|
}
|
|
|
|
ASSERT(string_->IsSeqString() || string->IsExternalString());
|
|
|
|
}
|
2014-01-17 10:27:57 +00:00
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual uint32_t Hash() V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
ASSERT(length_ >= 0);
|
|
|
|
ASSERT(from_ + length_ <= string_->length());
|
2014-01-29 14:31:34 +00:00
|
|
|
const Char* chars = GetChars() + from_;
|
2014-01-17 10:27:57 +00:00
|
|
|
hash_field_ = StringHasher::HashSequentialString(
|
|
|
|
chars, length_, string_->GetHeap()->HashSeed());
|
|
|
|
uint32_t result = hash_field_ >> String::kHashShift;
|
|
|
|
ASSERT(result != 0); // Ensure that the hash value of 0 is never computed.
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual uint32_t HashForObject(Object* other) V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
return String::cast(other)->Hash();
|
|
|
|
}
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual bool IsMatch(Object* string) V8_OVERRIDE;
|
2014-04-25 13:59:33 +00:00
|
|
|
virtual Handle<Object> AsHandle(Isolate* isolate) V8_OVERRIDE;
|
2014-01-17 10:27:57 +00:00
|
|
|
|
|
|
|
private:
|
2014-01-29 14:31:34 +00:00
|
|
|
const Char* GetChars();
|
|
|
|
String* Unslice(String* string, int* offset) {
|
|
|
|
while (string->IsSlicedString()) {
|
|
|
|
SlicedString* sliced = SlicedString::cast(string);
|
|
|
|
*offset += sliced->offset();
|
|
|
|
string = sliced->parent();
|
|
|
|
}
|
|
|
|
return string;
|
|
|
|
}
|
|
|
|
|
|
|
|
Handle<String> string_;
|
2014-01-17 10:27:57 +00:00
|
|
|
int from_;
|
|
|
|
int length_;
|
|
|
|
uint32_t hash_field_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class TwoByteStringKey : public SequentialStringKey<uc16> {
|
|
|
|
public:
|
|
|
|
explicit TwoByteStringKey(Vector<const uc16> str, uint32_t seed)
|
|
|
|
: SequentialStringKey<uc16>(str, seed) { }
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual bool IsMatch(Object* string) V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
return String::cast(string)->IsTwoByteEqualTo(string_);
|
|
|
|
}
|
|
|
|
|
2014-04-25 13:59:33 +00:00
|
|
|
virtual Handle<Object> AsHandle(Isolate* isolate) V8_OVERRIDE;
|
2014-01-17 10:27:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// Utf8StringKey carries a vector of chars as key.
|
|
|
|
class Utf8StringKey : public HashTableKey {
|
|
|
|
public:
|
|
|
|
explicit Utf8StringKey(Vector<const char> string, uint32_t seed)
|
|
|
|
: string_(string), hash_field_(0), seed_(seed) { }
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual bool IsMatch(Object* string) V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
return String::cast(string)->IsUtf8EqualTo(string_);
|
|
|
|
}
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual uint32_t Hash() V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
if (hash_field_ != 0) return hash_field_ >> String::kHashShift;
|
|
|
|
hash_field_ = StringHasher::ComputeUtf8Hash(string_, seed_, &chars_);
|
|
|
|
uint32_t result = hash_field_ >> String::kHashShift;
|
|
|
|
ASSERT(result != 0); // Ensure that the hash value of 0 is never computed.
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-04-23 15:43:39 +00:00
|
|
|
virtual uint32_t HashForObject(Object* other) V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
return String::cast(other)->Hash();
|
|
|
|
}
|
|
|
|
|
2014-04-25 13:59:33 +00:00
|
|
|
virtual Handle<Object> AsHandle(Isolate* isolate) V8_OVERRIDE {
|
2014-01-17 10:27:57 +00:00
|
|
|
if (hash_field_ == 0) Hash();
|
2014-04-25 13:59:33 +00:00
|
|
|
return isolate->factory()->NewInternalizedStringFromUtf8(
|
|
|
|
string_, chars_, hash_field_);
|
2014-01-17 10:27:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Vector<const char> string_;
|
|
|
|
uint32_t hash_field_;
|
|
|
|
int chars_; // Caches the number of characters when computing the hash code.
|
|
|
|
uint32_t seed_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Object::IsNumber() {
|
|
|
|
return IsSmi() || IsHeapNumber();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(ByteArray, BYTE_ARRAY_TYPE)
|
|
|
|
TYPE_CHECKER(FreeSpace, FREE_SPACE_TYPE)
|
2011-09-19 18:36:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsFiller() {
|
|
|
|
if (!Object::IsHeapObject()) return false;
|
|
|
|
InstanceType instance_type = HeapObject::cast(this)->map()->instance_type();
|
|
|
|
return instance_type == FREE_SPACE_TYPE || instance_type == FILLER_TYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-10-20 15:26:17 +00:00
|
|
|
bool Object::IsExternalArray() {
|
|
|
|
if (!Object::IsHeapObject())
|
|
|
|
return false;
|
|
|
|
InstanceType instance_type =
|
|
|
|
HeapObject::cast(this)->map()->instance_type();
|
2009-12-22 13:34:02 +00:00
|
|
|
return (instance_type >= FIRST_EXTERNAL_ARRAY_TYPE &&
|
|
|
|
instance_type <= LAST_EXTERNAL_ARRAY_TYPE);
|
2009-10-20 15:26:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
#define TYPED_ARRAY_TYPE_CHECKER(Type, type, TYPE, ctype, size) \
|
|
|
|
TYPE_CHECKER(External##Type##Array, EXTERNAL_##TYPE##_ARRAY_TYPE) \
|
|
|
|
TYPE_CHECKER(Fixed##Type##Array, FIXED_##TYPE##_ARRAY_TYPE)
|
|
|
|
|
|
|
|
TYPED_ARRAYS(TYPED_ARRAY_TYPE_CHECKER)
|
|
|
|
#undef TYPED_ARRAY_TYPE_CHECKER
|
2011-04-21 07:15:43 +00:00
|
|
|
|
|
|
|
|
2014-01-16 17:08:45 +00:00
|
|
|
bool Object::IsFixedTypedArrayBase() {
|
|
|
|
if (!Object::IsHeapObject()) return false;
|
|
|
|
|
|
|
|
InstanceType instance_type =
|
|
|
|
HeapObject::cast(this)->map()->instance_type();
|
|
|
|
return (instance_type >= FIRST_FIXED_TYPED_ARRAY_TYPE &&
|
|
|
|
instance_type <= LAST_FIXED_TYPED_ARRAY_TYPE);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Implement set trap for proxies, and revamp class hierarchy in preparation:
- Introduce a class JSReceiver, that is a common superclass of JSObject and
JSProxy. Use JSReceiver where appropriate (probably lots of places that we
still have to migrate, but we will find those later with proxy test suite).
- Move appropriate methods to JSReceiver class (SetProperty,
GetPropertyAttribute, Get/SetPrototype, Lookup, and so on).
- Introduce new JSFunctionProxy subclass of JSProxy. Currently only a stub.
- Overhaul enum InstanceType:
* Introduce FIRST/LAST_SPEC_OBJECT_TYPE that ranges over all types that
represent JS objects, and use that consistently to check language types.
* Rename FIRST/LAST_JS_OBJECT_TYPE and FIRST/LAST_FUNCTION_CLASS_TYPE
to FIRST/LAST_[NON]CALLABLE_SPEC_OBJECT_TYPE for clarity.
* Eliminate the overlap over JS_REGEXP_TYPE.
* Also replace FIRST_JS_OBJECT with FIRST_JS_RECEIVER, but only use it where
we exclusively talk about the internal representation type.
* Insert JS_PROXY and JS_FUNCTION_PROXY in the appropriate places.
- Fix all checks concerning classification, especially for functions, to
use the CALLABLE_SPEC_OBJECT range (that includes funciton proxies).
- Handle proxies in SetProperty (that was the easiest part :) ).
- A few simple test cases.
R=kmillikin@chromium.org
Review URL: http://codereview.chromium.org/6992072
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8126 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-05-31 16:38:40 +00:00
|
|
|
bool Object::IsJSReceiver() {
|
2011-09-21 14:46:54 +00:00
|
|
|
STATIC_ASSERT(LAST_JS_RECEIVER_TYPE == LAST_TYPE);
|
Implement set trap for proxies, and revamp class hierarchy in preparation:
- Introduce a class JSReceiver, that is a common superclass of JSObject and
JSProxy. Use JSReceiver where appropriate (probably lots of places that we
still have to migrate, but we will find those later with proxy test suite).
- Move appropriate methods to JSReceiver class (SetProperty,
GetPropertyAttribute, Get/SetPrototype, Lookup, and so on).
- Introduce new JSFunctionProxy subclass of JSProxy. Currently only a stub.
- Overhaul enum InstanceType:
* Introduce FIRST/LAST_SPEC_OBJECT_TYPE that ranges over all types that
represent JS objects, and use that consistently to check language types.
* Rename FIRST/LAST_JS_OBJECT_TYPE and FIRST/LAST_FUNCTION_CLASS_TYPE
to FIRST/LAST_[NON]CALLABLE_SPEC_OBJECT_TYPE for clarity.
* Eliminate the overlap over JS_REGEXP_TYPE.
* Also replace FIRST_JS_OBJECT with FIRST_JS_RECEIVER, but only use it where
we exclusively talk about the internal representation type.
* Insert JS_PROXY and JS_FUNCTION_PROXY in the appropriate places.
- Fix all checks concerning classification, especially for functions, to
use the CALLABLE_SPEC_OBJECT range (that includes funciton proxies).
- Handle proxies in SetProperty (that was the easiest part :) ).
- A few simple test cases.
R=kmillikin@chromium.org
Review URL: http://codereview.chromium.org/6992072
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8126 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-05-31 16:38:40 +00:00
|
|
|
return IsHeapObject() &&
|
|
|
|
HeapObject::cast(this)->map()->instance_type() >= FIRST_JS_RECEIVER_TYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Object::IsJSObject() {
|
2011-09-21 14:46:54 +00:00
|
|
|
STATIC_ASSERT(LAST_JS_OBJECT_TYPE == LAST_TYPE);
|
|
|
|
return IsHeapObject() &&
|
|
|
|
HeapObject::cast(this)->map()->instance_type() >= FIRST_JS_OBJECT_TYPE;
|
Implement set trap for proxies, and revamp class hierarchy in preparation:
- Introduce a class JSReceiver, that is a common superclass of JSObject and
JSProxy. Use JSReceiver where appropriate (probably lots of places that we
still have to migrate, but we will find those later with proxy test suite).
- Move appropriate methods to JSReceiver class (SetProperty,
GetPropertyAttribute, Get/SetPrototype, Lookup, and so on).
- Introduce new JSFunctionProxy subclass of JSProxy. Currently only a stub.
- Overhaul enum InstanceType:
* Introduce FIRST/LAST_SPEC_OBJECT_TYPE that ranges over all types that
represent JS objects, and use that consistently to check language types.
* Rename FIRST/LAST_JS_OBJECT_TYPE and FIRST/LAST_FUNCTION_CLASS_TYPE
to FIRST/LAST_[NON]CALLABLE_SPEC_OBJECT_TYPE for clarity.
* Eliminate the overlap over JS_REGEXP_TYPE.
* Also replace FIRST_JS_OBJECT with FIRST_JS_RECEIVER, but only use it where
we exclusively talk about the internal representation type.
* Insert JS_PROXY and JS_FUNCTION_PROXY in the appropriate places.
- Fix all checks concerning classification, especially for functions, to
use the CALLABLE_SPEC_OBJECT range (that includes funciton proxies).
- Handle proxies in SetProperty (that was the easiest part :) ).
- A few simple test cases.
R=kmillikin@chromium.org
Review URL: http://codereview.chromium.org/6992072
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8126 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-05-31 16:38:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsJSProxy() {
|
2011-09-21 14:46:54 +00:00
|
|
|
if (!Object::IsHeapObject()) return false;
|
|
|
|
InstanceType type = HeapObject::cast(this)->map()->instance_type();
|
|
|
|
return FIRST_JS_PROXY_TYPE <= type && type <= LAST_JS_PROXY_TYPE;
|
Implement set trap for proxies, and revamp class hierarchy in preparation:
- Introduce a class JSReceiver, that is a common superclass of JSObject and
JSProxy. Use JSReceiver where appropriate (probably lots of places that we
still have to migrate, but we will find those later with proxy test suite).
- Move appropriate methods to JSReceiver class (SetProperty,
GetPropertyAttribute, Get/SetPrototype, Lookup, and so on).
- Introduce new JSFunctionProxy subclass of JSProxy. Currently only a stub.
- Overhaul enum InstanceType:
* Introduce FIRST/LAST_SPEC_OBJECT_TYPE that ranges over all types that
represent JS objects, and use that consistently to check language types.
* Rename FIRST/LAST_JS_OBJECT_TYPE and FIRST/LAST_FUNCTION_CLASS_TYPE
to FIRST/LAST_[NON]CALLABLE_SPEC_OBJECT_TYPE for clarity.
* Eliminate the overlap over JS_REGEXP_TYPE.
* Also replace FIRST_JS_OBJECT with FIRST_JS_RECEIVER, but only use it where
we exclusively talk about the internal representation type.
* Insert JS_PROXY and JS_FUNCTION_PROXY in the appropriate places.
- Fix all checks concerning classification, especially for functions, to
use the CALLABLE_SPEC_OBJECT range (that includes funciton proxies).
- Handle proxies in SetProperty (that was the easiest part :) ).
- A few simple test cases.
R=kmillikin@chromium.org
Review URL: http://codereview.chromium.org/6992072
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8126 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-05-31 16:38:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSFunctionProxy, JS_FUNCTION_PROXY_TYPE)
|
|
|
|
TYPE_CHECKER(JSSet, JS_SET_TYPE)
|
|
|
|
TYPE_CHECKER(JSMap, JS_MAP_TYPE)
|
2014-04-17 17:45:32 +00:00
|
|
|
TYPE_CHECKER(JSSetIterator, JS_SET_ITERATOR_TYPE)
|
|
|
|
TYPE_CHECKER(JSMapIterator, JS_MAP_ITERATOR_TYPE)
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSWeakMap, JS_WEAK_MAP_TYPE)
|
2013-07-22 08:32:24 +00:00
|
|
|
TYPE_CHECKER(JSWeakSet, JS_WEAK_SET_TYPE)
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSContextExtensionObject, JS_CONTEXT_EXTENSION_OBJECT_TYPE)
|
|
|
|
TYPE_CHECKER(Map, MAP_TYPE)
|
|
|
|
TYPE_CHECKER(FixedArray, FIXED_ARRAY_TYPE)
|
|
|
|
TYPE_CHECKER(FixedDoubleArray, FIXED_DOUBLE_ARRAY_TYPE)
|
2013-10-14 13:35:06 +00:00
|
|
|
TYPE_CHECKER(ConstantPoolArray, CONSTANT_POOL_ARRAY_TYPE)
|
2011-06-09 10:03:35 +00:00
|
|
|
|
|
|
|
|
2013-07-22 08:32:24 +00:00
|
|
|
bool Object::IsJSWeakCollection() {
|
|
|
|
return IsJSWeakMap() || IsJSWeakSet();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Object::IsDescriptorArray() {
|
|
|
|
return IsFixedArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-05 13:54:20 +00:00
|
|
|
bool Object::IsTransitionArray() {
|
|
|
|
return IsFixedArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
bool Object::IsDeoptimizationInputData() {
|
|
|
|
// Must be a fixed array.
|
|
|
|
if (!IsFixedArray()) return false;
|
|
|
|
|
|
|
|
// There's no sure way to detect the difference between a fixed array and
|
|
|
|
// a deoptimization data array. Since this is used for asserts we can
|
|
|
|
// check that the length is zero or else the fixed size plus a multiple of
|
|
|
|
// the entry size.
|
|
|
|
int length = FixedArray::cast(this)->length();
|
|
|
|
if (length == 0) return true;
|
|
|
|
|
|
|
|
length -= DeoptimizationInputData::kFirstDeoptEntryIndex;
|
|
|
|
return length >= 0 &&
|
|
|
|
length % DeoptimizationInputData::kDeoptEntrySize == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsDeoptimizationOutputData() {
|
|
|
|
if (!IsFixedArray()) return false;
|
|
|
|
// There's actually no way to see the difference between a fixed array and
|
|
|
|
// a deoptimization data array. Since this is used for asserts we can check
|
|
|
|
// that the length is plausible though.
|
|
|
|
if (FixedArray::cast(this)->length() % 2 != 0) return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-20 11:49:54 +00:00
|
|
|
bool Object::IsDependentCode() {
|
2013-01-24 11:55:05 +00:00
|
|
|
if (!IsFixedArray()) return false;
|
|
|
|
// There's actually no way to see the difference between a fixed array and
|
|
|
|
// a dependent codes array.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Object::IsContext() {
|
2012-08-27 09:40:26 +00:00
|
|
|
if (!Object::IsHeapObject()) return false;
|
|
|
|
Map* map = HeapObject::cast(this)->map();
|
|
|
|
Heap* heap = map->GetHeap();
|
|
|
|
return (map == heap->function_context_map() ||
|
|
|
|
map == heap->catch_context_map() ||
|
|
|
|
map == heap->with_context_map() ||
|
|
|
|
map == heap->native_context_map() ||
|
|
|
|
map == heap->block_context_map() ||
|
|
|
|
map == heap->module_context_map() ||
|
|
|
|
map == heap->global_context_map());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-17 09:03:08 +00:00
|
|
|
bool Object::IsNativeContext() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return Object::IsHeapObject() &&
|
|
|
|
HeapObject::cast(this)->map() ==
|
2012-08-17 09:03:08 +00:00
|
|
|
HeapObject::cast(this)->GetHeap()->native_context_map();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-03 10:36:55 +00:00
|
|
|
bool Object::IsScopeInfo() {
|
2011-08-11 16:29:28 +00:00
|
|
|
return Object::IsHeapObject() &&
|
|
|
|
HeapObject::cast(this)->map() ==
|
2011-11-03 10:36:55 +00:00
|
|
|
HeapObject::cast(this)->GetHeap()->scope_info_map();
|
2011-08-11 16:29:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSFunction, JS_FUNCTION_TYPE)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-09-03 08:19:44 +00:00
|
|
|
template <> inline bool Is<JSFunction>(Object* obj) {
|
2008-07-03 15:10:15 +00:00
|
|
|
return obj->IsJSFunction();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(Code, CODE_TYPE)
|
|
|
|
TYPE_CHECKER(Oddball, ODDBALL_TYPE)
|
2013-06-12 15:03:44 +00:00
|
|
|
TYPE_CHECKER(Cell, CELL_TYPE)
|
2013-06-14 16:06:12 +00:00
|
|
|
TYPE_CHECKER(PropertyCell, PROPERTY_CELL_TYPE)
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(SharedFunctionInfo, SHARED_FUNCTION_INFO_TYPE)
|
2013-04-15 12:29:44 +00:00
|
|
|
TYPE_CHECKER(JSGeneratorObject, JS_GENERATOR_OBJECT_TYPE)
|
2012-04-16 14:43:27 +00:00
|
|
|
TYPE_CHECKER(JSModule, JS_MODULE_TYPE)
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSValue, JS_VALUE_TYPE)
|
2012-03-09 11:11:55 +00:00
|
|
|
TYPE_CHECKER(JSDate, JS_DATE_TYPE)
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSMessageObject, JS_MESSAGE_OBJECT_TYPE)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2008-10-27 14:36:08 +00:00
|
|
|
bool Object::IsStringWrapper() {
|
|
|
|
return IsJSValue() && JSValue::cast(this)->value()->IsString();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(Foreign, FOREIGN_TYPE)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsBoolean() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return IsOddball() &&
|
|
|
|
((Oddball::cast(this)->kind() & Oddball::kNotBooleanMask) == 0);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSArray, JS_ARRAY_TYPE)
|
2013-03-28 12:50:18 +00:00
|
|
|
TYPE_CHECKER(JSArrayBuffer, JS_ARRAY_BUFFER_TYPE)
|
2013-04-16 14:16:30 +00:00
|
|
|
TYPE_CHECKER(JSTypedArray, JS_TYPED_ARRAY_TYPE)
|
2013-06-21 13:02:38 +00:00
|
|
|
TYPE_CHECKER(JSDataView, JS_DATA_VIEW_TYPE)
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsJSArrayBufferView() {
|
|
|
|
return IsJSDataView() || IsJSTypedArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSRegExp, JS_REGEXP_TYPE)
|
2008-09-23 11:45:43 +00:00
|
|
|
|
|
|
|
|
2008-09-03 08:19:44 +00:00
|
|
|
template <> inline bool Is<JSArray>(Object* obj) {
|
2008-07-03 15:10:15 +00:00
|
|
|
return obj->IsJSArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsHashTable() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return Object::IsHeapObject() &&
|
|
|
|
HeapObject::cast(this)->map() ==
|
|
|
|
HeapObject::cast(this)->GetHeap()->hash_table_map();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsDictionary() {
|
2011-06-16 14:12:58 +00:00
|
|
|
return IsHashTable() &&
|
2013-02-28 17:03:34 +00:00
|
|
|
this != HeapObject::cast(this)->GetHeap()->string_table();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-28 17:03:34 +00:00
|
|
|
bool Object::IsStringTable() {
|
2014-04-14 15:56:57 +00:00
|
|
|
return IsHashTable();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-04 16:42:11 +00:00
|
|
|
bool Object::IsJSFunctionResultCache() {
|
|
|
|
if (!IsFixedArray()) return false;
|
|
|
|
FixedArray* self = FixedArray::cast(this);
|
|
|
|
int length = self->length();
|
|
|
|
if (length < JSFunctionResultCache::kEntriesIndex) return false;
|
|
|
|
if ((length - JSFunctionResultCache::kEntriesIndex)
|
|
|
|
% JSFunctionResultCache::kEntrySize != 0) {
|
|
|
|
return false;
|
|
|
|
}
|
2012-10-12 11:41:14 +00:00
|
|
|
#ifdef VERIFY_HEAP
|
2011-10-25 13:27:46 +00:00
|
|
|
if (FLAG_verify_heap) {
|
|
|
|
reinterpret_cast<JSFunctionResultCache*>(this)->
|
|
|
|
JSFunctionResultCacheVerify();
|
|
|
|
}
|
2010-05-04 16:42:11 +00:00
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-25 13:25:54 +00:00
|
|
|
bool Object::IsNormalizedMapCache() {
|
2014-05-02 10:27:12 +00:00
|
|
|
return NormalizedMapCache::IsNormalizedMapCache(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int NormalizedMapCache::GetIndex(Handle<Map> map) {
|
|
|
|
return map->Hash() % NormalizedMapCache::kEntries;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool NormalizedMapCache::IsNormalizedMapCache(Object* obj) {
|
|
|
|
if (!obj->IsFixedArray()) return false;
|
|
|
|
if (FixedArray::cast(obj)->length() != NormalizedMapCache::kEntries) {
|
2010-08-25 13:25:54 +00:00
|
|
|
return false;
|
|
|
|
}
|
2012-10-12 11:41:14 +00:00
|
|
|
#ifdef VERIFY_HEAP
|
2011-10-25 13:27:46 +00:00
|
|
|
if (FLAG_verify_heap) {
|
2014-05-02 10:27:12 +00:00
|
|
|
reinterpret_cast<NormalizedMapCache*>(obj)->NormalizedMapCacheVerify();
|
2011-10-25 13:27:46 +00:00
|
|
|
}
|
2010-08-25 13:25:54 +00:00
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-09-11 10:51:52 +00:00
|
|
|
bool Object::IsCompilationCacheTable() {
|
|
|
|
return IsHashTable();
|
2008-09-05 16:27:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-09 10:49:41 +00:00
|
|
|
bool Object::IsCodeCacheHashTable() {
|
|
|
|
return IsHashTable();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-06 13:15:11 +00:00
|
|
|
bool Object::IsPolymorphicCodeCacheHashTable() {
|
|
|
|
return IsHashTable();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-09-25 07:46:07 +00:00
|
|
|
bool Object::IsMapCache() {
|
|
|
|
return IsHashTable();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-06 16:47:15 +00:00
|
|
|
bool Object::IsObjectHashTable() {
|
|
|
|
return IsHashTable();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 20:06:35 +00:00
|
|
|
bool Object::IsOrderedHashTable() {
|
|
|
|
return IsHeapObject() &&
|
|
|
|
HeapObject::cast(this)->map() ==
|
|
|
|
HeapObject::cast(this)->GetHeap()->ordered_hash_table_map();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Object::IsPrimitive() {
|
|
|
|
return IsOddball() || IsNumber() || IsString();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Split window support from V8.
Here is a description of the background and design of split window in Chrome and V8:
https://docs.google.com/a/google.com/Doc?id=chhjkpg_47fwddxbfr
This change list splits the window object into two parts: 1) an inner window object used as the global object of contexts; 2) an outer window object exposed to JavaScript and accessible by the name 'window'. Firefox did it awhile ago, here are some discussions: https://wiki.mozilla.org/Gecko:SplitWindow. One additional benefit of splitting window in Chrome is that accessing global variables don't need security checks anymore, it can improve applications that use many global variables.
V8 support of split window:
There are a small number of changes on V8 api to support split window:
Security context is removed from V8, so does related API functions;
A global object can be detached from its context and reused by a new context;
Access checks on an object template can be turned on/off by default;
An object can turn on its access checks later;
V8 has a new object type, ApiGlobalObject, which is the outer window object type. The existing JSGlobalObject becomes the inner window object type. Security checks are moved from JSGlobalObject to ApiGlobalObject. ApiGlobalObject is the one exposed to JavaScript, it is accessible through Context::Global(). ApiGlobalObject's prototype is set to JSGlobalObject so that property lookups are forwarded to JSGlobalObject. ApiGlobalObject forwards all other property access requests to JSGlobalObject, such as SetProperty, DeleteProperty, etc.
Security token is moved to a global context, and ApiGlobalObject has a reference to its global context. JSGlobalObject has a reference to its global context as well. When accessing properties on a global object in JavaScript, the domain security check is performed by comparing the security token of the lexical context (Top::global_context()) to the token of global object's context. The check is only needed when the receiver is a window object, such as 'window.document'. Accessing global variables, such as 'var foo = 3; foo' does not need checks because the receiver is the inner window object.
When an outer window is detached from its global context (when a frame navigates away from a page), it is completely detached from the inner window. A new context is created for the new page, and the outer global object is reused. At this point, the access check on the DOMWindow wrapper of the old context is turned on. The code in old context is still able to access DOMWindow properties, but it has to go through domain security checks.
It is debatable on how to implement the outer window object. Currently each property access function has to check if the receiver is ApiGlobalObject type. This approach might be error-prone that one may forget to check the receiver when adding new functions. It is unlikely a performance issue because accessing global variables are more common than 'window.foo' style coding.
I am still working on the ARM port, and I'd like to hear comments and suggestions on the best way to support it in V8.
Review URL: http://codereview.chromium.org/7366
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@540 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2008-10-21 19:07:58 +00:00
|
|
|
bool Object::IsJSGlobalProxy() {
|
|
|
|
bool result = IsHeapObject() &&
|
|
|
|
(HeapObject::cast(this)->map()->instance_type() ==
|
|
|
|
JS_GLOBAL_PROXY_TYPE);
|
2014-03-10 12:23:05 +00:00
|
|
|
ASSERT(!result ||
|
|
|
|
HeapObject::cast(this)->map()->is_access_check_needed());
|
Split window support from V8.
Here is a description of the background and design of split window in Chrome and V8:
https://docs.google.com/a/google.com/Doc?id=chhjkpg_47fwddxbfr
This change list splits the window object into two parts: 1) an inner window object used as the global object of contexts; 2) an outer window object exposed to JavaScript and accessible by the name 'window'. Firefox did it awhile ago, here are some discussions: https://wiki.mozilla.org/Gecko:SplitWindow. One additional benefit of splitting window in Chrome is that accessing global variables don't need security checks anymore, it can improve applications that use many global variables.
V8 support of split window:
There are a small number of changes on V8 api to support split window:
Security context is removed from V8, so does related API functions;
A global object can be detached from its context and reused by a new context;
Access checks on an object template can be turned on/off by default;
An object can turn on its access checks later;
V8 has a new object type, ApiGlobalObject, which is the outer window object type. The existing JSGlobalObject becomes the inner window object type. Security checks are moved from JSGlobalObject to ApiGlobalObject. ApiGlobalObject is the one exposed to JavaScript, it is accessible through Context::Global(). ApiGlobalObject's prototype is set to JSGlobalObject so that property lookups are forwarded to JSGlobalObject. ApiGlobalObject forwards all other property access requests to JSGlobalObject, such as SetProperty, DeleteProperty, etc.
Security token is moved to a global context, and ApiGlobalObject has a reference to its global context. JSGlobalObject has a reference to its global context as well. When accessing properties on a global object in JavaScript, the domain security check is performed by comparing the security token of the lexical context (Top::global_context()) to the token of global object's context. The check is only needed when the receiver is a window object, such as 'window.document'. Accessing global variables, such as 'var foo = 3; foo' does not need checks because the receiver is the inner window object.
When an outer window is detached from its global context (when a frame navigates away from a page), it is completely detached from the inner window. A new context is created for the new page, and the outer global object is reused. At this point, the access check on the DOMWindow wrapper of the old context is turned on. The code in old context is still able to access DOMWindow properties, but it has to go through domain security checks.
It is debatable on how to implement the outer window object. Currently each property access function has to check if the receiver is ApiGlobalObject type. This approach might be error-prone that one may forget to check the receiver when adding new functions. It is unlikely a performance issue because accessing global variables are more common than 'window.foo' style coding.
I am still working on the ARM port, and I'd like to hear comments and suggestions on the best way to support it in V8.
Review URL: http://codereview.chromium.org/7366
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@540 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2008-10-21 19:07:58 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Object::IsGlobalObject() {
|
Split window support from V8.
Here is a description of the background and design of split window in Chrome and V8:
https://docs.google.com/a/google.com/Doc?id=chhjkpg_47fwddxbfr
This change list splits the window object into two parts: 1) an inner window object used as the global object of contexts; 2) an outer window object exposed to JavaScript and accessible by the name 'window'. Firefox did it awhile ago, here are some discussions: https://wiki.mozilla.org/Gecko:SplitWindow. One additional benefit of splitting window in Chrome is that accessing global variables don't need security checks anymore, it can improve applications that use many global variables.
V8 support of split window:
There are a small number of changes on V8 api to support split window:
Security context is removed from V8, so does related API functions;
A global object can be detached from its context and reused by a new context;
Access checks on an object template can be turned on/off by default;
An object can turn on its access checks later;
V8 has a new object type, ApiGlobalObject, which is the outer window object type. The existing JSGlobalObject becomes the inner window object type. Security checks are moved from JSGlobalObject to ApiGlobalObject. ApiGlobalObject is the one exposed to JavaScript, it is accessible through Context::Global(). ApiGlobalObject's prototype is set to JSGlobalObject so that property lookups are forwarded to JSGlobalObject. ApiGlobalObject forwards all other property access requests to JSGlobalObject, such as SetProperty, DeleteProperty, etc.
Security token is moved to a global context, and ApiGlobalObject has a reference to its global context. JSGlobalObject has a reference to its global context as well. When accessing properties on a global object in JavaScript, the domain security check is performed by comparing the security token of the lexical context (Top::global_context()) to the token of global object's context. The check is only needed when the receiver is a window object, such as 'window.document'. Accessing global variables, such as 'var foo = 3; foo' does not need checks because the receiver is the inner window object.
When an outer window is detached from its global context (when a frame navigates away from a page), it is completely detached from the inner window. A new context is created for the new page, and the outer global object is reused. At this point, the access check on the DOMWindow wrapper of the old context is turned on. The code in old context is still able to access DOMWindow properties, but it has to go through domain security checks.
It is debatable on how to implement the outer window object. Currently each property access function has to check if the receiver is ApiGlobalObject type. This approach might be error-prone that one may forget to check the receiver when adding new functions. It is unlikely a performance issue because accessing global variables are more common than 'window.foo' style coding.
I am still working on the ARM port, and I'd like to hear comments and suggestions on the best way to support it in V8.
Review URL: http://codereview.chromium.org/7366
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@540 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2008-10-21 19:07:58 +00:00
|
|
|
if (!IsHeapObject()) return false;
|
|
|
|
|
2008-10-28 13:46:04 +00:00
|
|
|
InstanceType type = HeapObject::cast(this)->map()->instance_type();
|
Split window support from V8.
Here is a description of the background and design of split window in Chrome and V8:
https://docs.google.com/a/google.com/Doc?id=chhjkpg_47fwddxbfr
This change list splits the window object into two parts: 1) an inner window object used as the global object of contexts; 2) an outer window object exposed to JavaScript and accessible by the name 'window'. Firefox did it awhile ago, here are some discussions: https://wiki.mozilla.org/Gecko:SplitWindow. One additional benefit of splitting window in Chrome is that accessing global variables don't need security checks anymore, it can improve applications that use many global variables.
V8 support of split window:
There are a small number of changes on V8 api to support split window:
Security context is removed from V8, so does related API functions;
A global object can be detached from its context and reused by a new context;
Access checks on an object template can be turned on/off by default;
An object can turn on its access checks later;
V8 has a new object type, ApiGlobalObject, which is the outer window object type. The existing JSGlobalObject becomes the inner window object type. Security checks are moved from JSGlobalObject to ApiGlobalObject. ApiGlobalObject is the one exposed to JavaScript, it is accessible through Context::Global(). ApiGlobalObject's prototype is set to JSGlobalObject so that property lookups are forwarded to JSGlobalObject. ApiGlobalObject forwards all other property access requests to JSGlobalObject, such as SetProperty, DeleteProperty, etc.
Security token is moved to a global context, and ApiGlobalObject has a reference to its global context. JSGlobalObject has a reference to its global context as well. When accessing properties on a global object in JavaScript, the domain security check is performed by comparing the security token of the lexical context (Top::global_context()) to the token of global object's context. The check is only needed when the receiver is a window object, such as 'window.document'. Accessing global variables, such as 'var foo = 3; foo' does not need checks because the receiver is the inner window object.
When an outer window is detached from its global context (when a frame navigates away from a page), it is completely detached from the inner window. A new context is created for the new page, and the outer global object is reused. At this point, the access check on the DOMWindow wrapper of the old context is turned on. The code in old context is still able to access DOMWindow properties, but it has to go through domain security checks.
It is debatable on how to implement the outer window object. Currently each property access function has to check if the receiver is ApiGlobalObject type. This approach might be error-prone that one may forget to check the receiver when adding new functions. It is unlikely a performance issue because accessing global variables are more common than 'window.foo' style coding.
I am still working on the ARM port, and I'd like to hear comments and suggestions on the best way to support it in V8.
Review URL: http://codereview.chromium.org/7366
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@540 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2008-10-21 19:07:58 +00:00
|
|
|
return type == JS_GLOBAL_OBJECT_TYPE ||
|
|
|
|
type == JS_BUILTINS_OBJECT_TYPE;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
TYPE_CHECKER(JSGlobalObject, JS_GLOBAL_OBJECT_TYPE)
|
|
|
|
TYPE_CHECKER(JSBuiltinsObject, JS_BUILTINS_OBJECT_TYPE)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsUndetectableObject() {
|
|
|
|
return IsHeapObject()
|
|
|
|
&& HeapObject::cast(this)->map()->is_undetectable();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsAccessCheckNeeded() {
|
2014-03-10 12:23:05 +00:00
|
|
|
if (!IsHeapObject()) return false;
|
|
|
|
if (IsJSGlobalProxy()) {
|
|
|
|
JSGlobalProxy* proxy = JSGlobalProxy::cast(this);
|
|
|
|
GlobalObject* global =
|
|
|
|
proxy->GetIsolate()->context()->global_object();
|
|
|
|
return proxy->IsDetachedFrom(global);
|
|
|
|
}
|
|
|
|
return HeapObject::cast(this)->map()->is_access_check_needed();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsStruct() {
|
|
|
|
if (!IsHeapObject()) return false;
|
|
|
|
switch (HeapObject::cast(this)->map()->instance_type()) {
|
|
|
|
#define MAKE_STRUCT_CASE(NAME, Name, name) case NAME##_TYPE: return true;
|
|
|
|
STRUCT_LIST(MAKE_STRUCT_CASE)
|
|
|
|
#undef MAKE_STRUCT_CASE
|
|
|
|
default: return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define MAKE_STRUCT_PREDICATE(NAME, Name, name) \
|
|
|
|
bool Object::Is##Name() { \
|
|
|
|
return Object::IsHeapObject() \
|
|
|
|
&& HeapObject::cast(this)->map()->instance_type() == NAME##_TYPE; \
|
|
|
|
}
|
|
|
|
STRUCT_LIST(MAKE_STRUCT_PREDICATE)
|
|
|
|
#undef MAKE_STRUCT_PREDICATE
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsUndefined() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return IsOddball() && Oddball::cast(this)->kind() == Oddball::kUndefined;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsNull() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return IsOddball() && Oddball::cast(this)->kind() == Oddball::kNull;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsTheHole() {
|
|
|
|
return IsOddball() && Oddball::cast(this)->kind() == Oddball::kTheHole;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-22 07:33:20 +00:00
|
|
|
bool Object::IsException() {
|
|
|
|
return IsOddball() && Oddball::cast(this)->kind() == Oddball::kException;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-06 14:21:35 +00:00
|
|
|
bool Object::IsUninitialized() {
|
|
|
|
return IsOddball() && Oddball::cast(this)->kind() == Oddball::kUninitialized;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Object::IsTrue() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return IsOddball() && Oddball::cast(this)->kind() == Oddball::kTrue;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsFalse() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return IsOddball() && Oddball::cast(this)->kind() == Oddball::kFalse;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-07 10:06:28 +00:00
|
|
|
bool Object::IsArgumentsMarker() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return IsOddball() && Oddball::cast(this)->kind() == Oddball::kArgumentMarker;
|
2011-01-07 10:06:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
double Object::Number() {
|
|
|
|
ASSERT(IsNumber());
|
|
|
|
return IsSmi()
|
|
|
|
? static_cast<double>(reinterpret_cast<Smi*>(this)->value())
|
|
|
|
: reinterpret_cast<HeapNumber*>(this)->value();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-09 11:19:55 +00:00
|
|
|
bool Object::IsNaN() {
|
2013-04-19 13:26:47 +00:00
|
|
|
return this->IsHeapNumber() && std::isnan(HeapNumber::cast(this)->value());
|
2012-03-09 11:19:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-29 13:18:27 +00:00
|
|
|
MaybeHandle<Smi> Object::ToSmi(Isolate* isolate, Handle<Object> object) {
|
|
|
|
if (object->IsSmi()) return Handle<Smi>::cast(object);
|
2014-03-19 16:29:19 +00:00
|
|
|
if (object->IsHeapNumber()) {
|
|
|
|
double value = Handle<HeapNumber>::cast(object)->value();
|
|
|
|
int int_value = FastD2I(value);
|
|
|
|
if (value == FastI2D(int_value) && Smi::IsValid(int_value)) {
|
|
|
|
return handle(Smi::FromInt(int_value), isolate);
|
|
|
|
}
|
|
|
|
}
|
2014-04-29 13:18:27 +00:00
|
|
|
return Handle<Smi>();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-14 09:19:09 +00:00
|
|
|
MaybeHandle<JSReceiver> Object::ToObject(Isolate* isolate,
|
|
|
|
Handle<Object> object) {
|
|
|
|
return ToObject(
|
|
|
|
isolate, object, handle(isolate->context()->native_context(), isolate));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-23 08:25:23 +00:00
|
|
|
bool Object::HasSpecificClassOf(String* name) {
|
|
|
|
return this->IsJSObject() && (JSObject::cast(this)->class_name() == name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 12:47:34 +00:00
|
|
|
MaybeHandle<Object> Object::GetProperty(Handle<Object> object,
|
|
|
|
Handle<Name> name) {
|
|
|
|
PropertyAttributes attributes;
|
|
|
|
return GetPropertyWithReceiver(object, object, name, &attributes);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-10 09:20:11 +00:00
|
|
|
MaybeHandle<Object> Object::GetElement(Isolate* isolate,
|
|
|
|
Handle<Object> object,
|
|
|
|
uint32_t index) {
|
2011-02-24 14:00:52 +00:00
|
|
|
// GetElement can trigger a getter which can cause allocation.
|
|
|
|
// This was not always the case. This ASSERT is here to catch
|
|
|
|
// leftover incorrect uses.
|
2013-06-03 15:32:22 +00:00
|
|
|
ASSERT(AllowHeapAllocation::IsAllowed());
|
2014-03-25 10:15:12 +00:00
|
|
|
return Object::GetElementWithReceiver(isolate, object, object, index);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 12:47:34 +00:00
|
|
|
MaybeHandle<Object> Object::GetPropertyOrElement(Handle<Object> object,
|
|
|
|
Handle<Name> name) {
|
|
|
|
uint32_t index;
|
|
|
|
Isolate* isolate = name->GetIsolate();
|
|
|
|
if (name->AsArrayIndex(&index)) return GetElement(isolate, object, index);
|
|
|
|
return GetProperty(object, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-16 13:28:11 +00:00
|
|
|
MaybeHandle<Object> Object::GetProperty(Isolate* isolate,
|
|
|
|
Handle<Object> object,
|
|
|
|
const char* name) {
|
|
|
|
Handle<String> str = isolate->factory()->InternalizeUtf8String(name);
|
|
|
|
ASSERT(!str.is_null());
|
|
|
|
#ifdef DEBUG
|
|
|
|
uint32_t index; // Assert that the name is not an array index.
|
|
|
|
ASSERT(!str->AsArrayIndex(&index));
|
|
|
|
#endif // DEBUG
|
|
|
|
return GetProperty(object, str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 12:47:34 +00:00
|
|
|
MaybeHandle<Object> JSProxy::GetElementWithHandler(Handle<JSProxy> proxy,
|
|
|
|
Handle<Object> receiver,
|
2014-03-20 12:22:13 +00:00
|
|
|
uint32_t index) {
|
2014-04-11 12:47:34 +00:00
|
|
|
return GetPropertyWithHandler(
|
|
|
|
proxy, receiver, proxy->GetIsolate()->factory()->Uint32ToString(index));
|
2014-04-11 11:26:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 12:47:34 +00:00
|
|
|
MaybeHandle<Object> JSProxy::SetElementWithHandler(Handle<JSProxy> proxy,
|
|
|
|
Handle<JSReceiver> receiver,
|
|
|
|
uint32_t index,
|
|
|
|
Handle<Object> value,
|
|
|
|
StrictMode strict_mode) {
|
|
|
|
Isolate* isolate = proxy->GetIsolate();
|
|
|
|
Handle<String> name = isolate->factory()->Uint32ToString(index);
|
|
|
|
return SetPropertyWithHandler(
|
|
|
|
proxy, receiver, name, value, NONE, strict_mode);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool JSProxy::HasElementWithHandler(Handle<JSProxy> proxy, uint32_t index) {
|
|
|
|
Isolate* isolate = proxy->GetIsolate();
|
|
|
|
Handle<String> name = isolate->factory()->Uint32ToString(index);
|
|
|
|
return HasPropertyWithHandler(proxy, name);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define FIELD_ADDR(p, offset) \
|
|
|
|
(reinterpret_cast<byte*>(p) + offset - kHeapObjectTag)
|
|
|
|
|
|
|
|
#define READ_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<Object**>(FIELD_ADDR(p, offset)))
|
|
|
|
|
2014-03-31 14:29:01 +00:00
|
|
|
#define ACQUIRE_READ_FIELD(p, offset) \
|
|
|
|
reinterpret_cast<Object*>( \
|
|
|
|
Acquire_Load(reinterpret_cast<AtomicWord*>(FIELD_ADDR(p, offset))))
|
|
|
|
|
2014-04-08 16:31:57 +00:00
|
|
|
#define NOBARRIER_READ_FIELD(p, offset) \
|
2014-03-31 14:29:01 +00:00
|
|
|
reinterpret_cast<Object*>( \
|
|
|
|
NoBarrier_Load(reinterpret_cast<AtomicWord*>(FIELD_ADDR(p, offset))))
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
#define WRITE_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<Object**>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
|
2014-03-31 14:29:01 +00:00
|
|
|
#define RELEASE_WRITE_FIELD(p, offset, value) \
|
|
|
|
Release_Store(reinterpret_cast<AtomicWord*>(FIELD_ADDR(p, offset)), \
|
|
|
|
reinterpret_cast<AtomicWord>(value));
|
|
|
|
|
2014-04-08 16:31:57 +00:00
|
|
|
#define NOBARRIER_WRITE_FIELD(p, offset, value) \
|
2014-03-31 14:29:01 +00:00
|
|
|
NoBarrier_Store(reinterpret_cast<AtomicWord*>(FIELD_ADDR(p, offset)), \
|
|
|
|
reinterpret_cast<AtomicWord>(value));
|
|
|
|
|
2011-09-19 18:36:47 +00:00
|
|
|
#define WRITE_BARRIER(heap, object, offset, value) \
|
|
|
|
heap->incremental_marking()->RecordWrite( \
|
|
|
|
object, HeapObject::RawField(object, offset), value); \
|
|
|
|
if (heap->InNewSpace(value)) { \
|
|
|
|
heap->RecordWrite(object->address(), offset); \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define CONDITIONAL_WRITE_BARRIER(heap, object, offset, value, mode) \
|
|
|
|
if (mode == UPDATE_WRITE_BARRIER) { \
|
|
|
|
heap->incremental_marking()->RecordWrite( \
|
|
|
|
object, HeapObject::RawField(object, offset), value); \
|
|
|
|
if (heap->InNewSpace(value)) { \
|
|
|
|
heap->RecordWrite(object->address(), offset); \
|
|
|
|
} \
|
2008-10-23 08:46:32 +00:00
|
|
|
}
|
|
|
|
|
2011-03-28 13:05:36 +00:00
|
|
|
#ifndef V8_TARGET_ARCH_MIPS
|
|
|
|
#define READ_DOUBLE_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<double*>(FIELD_ADDR(p, offset)))
|
|
|
|
#else // V8_TARGET_ARCH_MIPS
|
|
|
|
// Prevent gcc from using load-double (mips ldc1) on (possibly)
|
|
|
|
// non-64-bit aligned HeapNumber::value.
|
2011-06-10 07:09:14 +00:00
|
|
|
static inline double read_double_field(void* p, int offset) {
|
2011-03-28 13:05:36 +00:00
|
|
|
union conversion {
|
|
|
|
double d;
|
|
|
|
uint32_t u[2];
|
|
|
|
} c;
|
|
|
|
c.u[0] = (*reinterpret_cast<uint32_t*>(FIELD_ADDR(p, offset)));
|
|
|
|
c.u[1] = (*reinterpret_cast<uint32_t*>(FIELD_ADDR(p, offset + 4)));
|
|
|
|
return c.d;
|
|
|
|
}
|
|
|
|
#define READ_DOUBLE_FIELD(p, offset) read_double_field(p, offset)
|
|
|
|
#endif // V8_TARGET_ARCH_MIPS
|
|
|
|
|
|
|
|
#ifndef V8_TARGET_ARCH_MIPS
|
|
|
|
#define WRITE_DOUBLE_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<double*>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
#else // V8_TARGET_ARCH_MIPS
|
|
|
|
// Prevent gcc from using store-double (mips sdc1) on (possibly)
|
|
|
|
// non-64-bit aligned HeapNumber::value.
|
2011-06-10 07:09:14 +00:00
|
|
|
static inline void write_double_field(void* p, int offset,
|
2011-03-28 13:05:36 +00:00
|
|
|
double value) {
|
|
|
|
union conversion {
|
|
|
|
double d;
|
|
|
|
uint32_t u[2];
|
|
|
|
} c;
|
|
|
|
c.d = value;
|
|
|
|
(*reinterpret_cast<uint32_t*>(FIELD_ADDR(p, offset))) = c.u[0];
|
|
|
|
(*reinterpret_cast<uint32_t*>(FIELD_ADDR(p, offset + 4))) = c.u[1];
|
|
|
|
}
|
|
|
|
#define WRITE_DOUBLE_FIELD(p, offset, value) \
|
|
|
|
write_double_field(p, offset, value)
|
|
|
|
#endif // V8_TARGET_ARCH_MIPS
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
#define READ_INT_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<int*>(FIELD_ADDR(p, offset)))
|
|
|
|
|
|
|
|
#define WRITE_INT_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<int*>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
|
2009-06-29 08:04:39 +00:00
|
|
|
#define READ_INTPTR_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<intptr_t*>(FIELD_ADDR(p, offset)))
|
|
|
|
|
|
|
|
#define WRITE_INTPTR_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<intptr_t*>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
|
2008-10-06 09:35:52 +00:00
|
|
|
#define READ_UINT32_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<uint32_t*>(FIELD_ADDR(p, offset)))
|
|
|
|
|
|
|
|
#define WRITE_UINT32_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<uint32_t*>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
|
2013-10-14 13:35:06 +00:00
|
|
|
#define READ_INT32_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<int32_t*>(FIELD_ADDR(p, offset)))
|
|
|
|
|
|
|
|
#define WRITE_INT32_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<int32_t*>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
|
2012-03-23 10:14:51 +00:00
|
|
|
#define READ_INT64_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<int64_t*>(FIELD_ADDR(p, offset)))
|
|
|
|
|
|
|
|
#define WRITE_INT64_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<int64_t*>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
#define READ_SHORT_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<uint16_t*>(FIELD_ADDR(p, offset)))
|
|
|
|
|
|
|
|
#define WRITE_SHORT_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<uint16_t*>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
|
|
|
|
#define READ_BYTE_FIELD(p, offset) \
|
|
|
|
(*reinterpret_cast<byte*>(FIELD_ADDR(p, offset)))
|
|
|
|
|
2014-04-09 09:50:25 +00:00
|
|
|
#define NOBARRIER_READ_BYTE_FIELD(p, offset) \
|
|
|
|
static_cast<byte>(NoBarrier_Load( \
|
|
|
|
reinterpret_cast<Atomic8*>(FIELD_ADDR(p, offset))) )
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
#define WRITE_BYTE_FIELD(p, offset, value) \
|
|
|
|
(*reinterpret_cast<byte*>(FIELD_ADDR(p, offset)) = value)
|
|
|
|
|
2014-04-09 09:50:25 +00:00
|
|
|
#define NOBARRIER_WRITE_BYTE_FIELD(p, offset, value) \
|
|
|
|
NoBarrier_Store(reinterpret_cast<Atomic8*>(FIELD_ADDR(p, offset)), \
|
|
|
|
static_cast<Atomic8>(value));
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2008-10-29 10:37:14 +00:00
|
|
|
Object** HeapObject::RawField(HeapObject* obj, int byte_offset) {
|
|
|
|
return &READ_FIELD(obj, byte_offset);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Smi::value() {
|
2009-08-26 10:33:11 +00:00
|
|
|
return Internals::SmiValue(this);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Smi* Smi::FromInt(int value) {
|
2009-05-06 07:53:08 +00:00
|
|
|
ASSERT(Smi::IsValid(value));
|
2013-05-22 06:35:38 +00:00
|
|
|
return reinterpret_cast<Smi*>(Internals::IntToSmi(value));
|
2009-05-06 07:53:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Smi* Smi::FromIntptr(intptr_t value) {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(Smi::IsValid(value));
|
2009-10-08 12:36:12 +00:00
|
|
|
int smi_shift_bits = kSmiTagSize + kSmiShiftSize;
|
|
|
|
return reinterpret_cast<Smi*>((value << smi_shift_bits) | kSmiTag);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-08-24 11:56:29 +00:00
|
|
|
bool Smi::IsValid(intptr_t value) {
|
2013-05-22 06:35:38 +00:00
|
|
|
bool result = Internals::IsValidSmi(value);
|
|
|
|
ASSERT_EQ(result, value >= kMinValue && value <= kMaxValue);
|
2009-05-06 07:53:08 +00:00
|
|
|
return result;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-30 08:49:36 +00:00
|
|
|
MapWord MapWord::FromMap(Map* map) {
|
|
|
|
return MapWord(reinterpret_cast<uintptr_t>(map));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Map* MapWord::ToMap() {
|
|
|
|
return reinterpret_cast<Map*>(value_);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool MapWord::IsForwardingAddress() {
|
2008-10-16 07:17:26 +00:00
|
|
|
return HAS_SMI_TAG(reinterpret_cast<Object*>(value_));
|
2008-07-30 08:49:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
MapWord MapWord::FromForwardingAddress(HeapObject* object) {
|
2008-10-16 07:17:26 +00:00
|
|
|
Address raw = reinterpret_cast<Address>(object) - kHeapObjectTag;
|
|
|
|
return MapWord(reinterpret_cast<uintptr_t>(raw));
|
2008-07-30 08:49:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
HeapObject* MapWord::ToForwardingAddress() {
|
|
|
|
ASSERT(IsForwardingAddress());
|
2008-10-16 07:17:26 +00:00
|
|
|
return HeapObject::FromAddress(reinterpret_cast<Address>(value_));
|
2008-07-30 08:49:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-12 11:41:14 +00:00
|
|
|
#ifdef VERIFY_HEAP
|
2008-07-03 15:10:15 +00:00
|
|
|
void HeapObject::VerifyObjectField(int offset) {
|
|
|
|
VerifyPointer(READ_FIELD(this, offset));
|
|
|
|
}
|
2010-05-27 12:30:45 +00:00
|
|
|
|
|
|
|
void HeapObject::VerifySmiField(int offset) {
|
2012-10-12 11:41:14 +00:00
|
|
|
CHECK(READ_FIELD(this, offset)->IsSmi());
|
2010-05-27 12:30:45 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2011-03-18 20:35:07 +00:00
|
|
|
Heap* HeapObject::GetHeap() {
|
2011-09-19 18:36:47 +00:00
|
|
|
Heap* heap =
|
|
|
|
MemoryChunk::FromAddress(reinterpret_cast<Address>(this))->heap();
|
2013-10-25 11:10:28 +00:00
|
|
|
SLOW_ASSERT(heap != NULL);
|
2011-09-19 18:36:47 +00:00
|
|
|
return heap;
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Isolate* HeapObject::GetIsolate() {
|
2011-03-22 19:15:02 +00:00
|
|
|
return GetHeap()->isolate();
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Map* HeapObject::map() {
|
2008-07-30 08:49:36 +00:00
|
|
|
return map_word().ToMap();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HeapObject::set_map(Map* value) {
|
2008-07-30 08:49:36 +00:00
|
|
|
set_map_word(MapWord::FromMap(value));
|
2011-09-19 18:36:47 +00:00
|
|
|
if (value != NULL) {
|
|
|
|
// TODO(1600) We are passing NULL as a slot because maps can never be on
|
|
|
|
// evacuation candidate.
|
|
|
|
value->GetHeap()->incremental_marking()->RecordWrite(this, NULL, value);
|
|
|
|
}
|
2008-07-30 08:49:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-31 14:29:01 +00:00
|
|
|
Map* HeapObject::synchronized_map() {
|
|
|
|
return synchronized_map_word().ToMap();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HeapObject::synchronized_set_map(Map* value) {
|
|
|
|
synchronized_set_map_word(MapWord::FromMap(value));
|
|
|
|
if (value != NULL) {
|
|
|
|
// TODO(1600) We are passing NULL as a slot because maps can never be on
|
|
|
|
// evacuation candidate.
|
|
|
|
value->GetHeap()->incremental_marking()->RecordWrite(this, NULL, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 16:31:57 +00:00
|
|
|
void HeapObject::synchronized_set_map_no_write_barrier(Map* value) {
|
|
|
|
synchronized_set_map_word(MapWord::FromMap(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-30 13:03:48 +00:00
|
|
|
// Unsafe accessor omitting write barrier.
|
2011-12-07 08:43:18 +00:00
|
|
|
void HeapObject::set_map_no_write_barrier(Map* value) {
|
2011-09-30 13:03:48 +00:00
|
|
|
set_map_word(MapWord::FromMap(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-30 08:49:36 +00:00
|
|
|
MapWord HeapObject::map_word() {
|
2014-03-31 14:29:01 +00:00
|
|
|
return MapWord(
|
2014-04-08 16:31:57 +00:00
|
|
|
reinterpret_cast<uintptr_t>(NOBARRIER_READ_FIELD(this, kMapOffset)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-30 08:49:36 +00:00
|
|
|
void HeapObject::set_map_word(MapWord map_word) {
|
2014-04-08 16:31:57 +00:00
|
|
|
NOBARRIER_WRITE_FIELD(
|
2014-03-31 14:29:01 +00:00
|
|
|
this, kMapOffset, reinterpret_cast<Object*>(map_word.value_));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
MapWord HeapObject::synchronized_map_word() {
|
|
|
|
return MapWord(
|
|
|
|
reinterpret_cast<uintptr_t>(ACQUIRE_READ_FIELD(this, kMapOffset)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HeapObject::synchronized_set_map_word(MapWord map_word) {
|
|
|
|
RELEASE_WRITE_FIELD(
|
|
|
|
this, kMapOffset, reinterpret_cast<Object*>(map_word.value_));
|
2008-07-30 08:49:36 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
HeapObject* HeapObject::FromAddress(Address address) {
|
|
|
|
ASSERT_TAG_ALIGNED(address);
|
|
|
|
return reinterpret_cast<HeapObject*>(address + kHeapObjectTag);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Address HeapObject::address() {
|
|
|
|
return reinterpret_cast<Address>(this) - kHeapObjectTag;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int HeapObject::Size() {
|
|
|
|
return SizeFromMap(map());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HeapObject::IteratePointers(ObjectVisitor* v, int start, int end) {
|
|
|
|
v->VisitPointers(reinterpret_cast<Object**>(FIELD_ADDR(this, start)),
|
|
|
|
reinterpret_cast<Object**>(FIELD_ADDR(this, end)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HeapObject::IteratePointer(ObjectVisitor* v, int offset) {
|
|
|
|
v->VisitPointer(reinterpret_cast<Object**>(FIELD_ADDR(this, offset)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-26 15:14:51 +00:00
|
|
|
void HeapObject::IterateNextCodeLink(ObjectVisitor* v, int offset) {
|
|
|
|
v->VisitNextCodeLink(reinterpret_cast<Object**>(FIELD_ADDR(this, offset)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
double HeapNumber::value() {
|
|
|
|
return READ_DOUBLE_FIELD(this, kValueOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void HeapNumber::set_value(double value) {
|
|
|
|
WRITE_DOUBLE_FIELD(this, kValueOffset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-18 13:00:57 +00:00
|
|
|
int HeapNumber::get_exponent() {
|
|
|
|
return ((READ_INT_FIELD(this, kExponentOffset) & kExponentMask) >>
|
|
|
|
kExponentShift) - kExponentBias;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int HeapNumber::get_sign() {
|
|
|
|
return READ_INT_FIELD(this, kExponentOffset) & kSignMask;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(JSObject, properties, FixedArray, kPropertiesOffset)
|
2009-07-28 08:43:51 +00:00
|
|
|
|
|
|
|
|
2011-12-09 08:50:19 +00:00
|
|
|
Object** FixedArray::GetFirstElementAddress() {
|
|
|
|
return reinterpret_cast<Object**>(FIELD_ADDR(this, OffsetOfElementAt(0)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool FixedArray::ContainsOnlySmisOrHoles() {
|
|
|
|
Object* the_hole = GetHeap()->the_hole_value();
|
|
|
|
Object** current = GetFirstElementAddress();
|
|
|
|
for (int i = 0; i < length(); ++i) {
|
|
|
|
Object* candidate = *current++;
|
|
|
|
if (!candidate->IsSmi() && candidate != the_hole) return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-12 13:54:27 +00:00
|
|
|
FixedArrayBase* JSObject::elements() {
|
2009-07-28 08:43:51 +00:00
|
|
|
Object* array = READ_FIELD(this, kElementsOffset);
|
2011-08-12 13:54:27 +00:00
|
|
|
return static_cast<FixedArrayBase*>(array);
|
2009-07-28 08:43:51 +00:00
|
|
|
}
|
|
|
|
|
2012-05-23 14:24:29 +00:00
|
|
|
|
2014-04-07 10:00:14 +00:00
|
|
|
void JSObject::ValidateElements(Handle<JSObject> object) {
|
2013-10-25 11:10:28 +00:00
|
|
|
#ifdef ENABLE_SLOW_ASSERTS
|
2012-05-23 14:24:29 +00:00
|
|
|
if (FLAG_enable_slow_asserts) {
|
2014-04-07 10:00:14 +00:00
|
|
|
ElementsAccessor* accessor = object->GetElementsAccessor();
|
|
|
|
accessor->Validate(object);
|
2011-09-22 11:30:04 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-19 14:13:34 +00:00
|
|
|
void AllocationSite::Initialize() {
|
2013-11-14 12:05:09 +00:00
|
|
|
set_transition_info(Smi::FromInt(0));
|
2013-09-19 14:13:34 +00:00
|
|
|
SetElementsKind(GetInitialFastElementsKind());
|
2013-09-24 10:30:41 +00:00
|
|
|
set_nested_site(Smi::FromInt(0));
|
2014-01-13 10:28:01 +00:00
|
|
|
set_pretenure_data(Smi::FromInt(0));
|
|
|
|
set_pretenure_create_count(Smi::FromInt(0));
|
2013-09-19 14:13:34 +00:00
|
|
|
set_dependent_code(DependentCode::cast(GetHeap()->empty_fixed_array()),
|
|
|
|
SKIP_WRITE_BARRIER);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-27 14:03:40 +00:00
|
|
|
void AllocationSite::MarkZombie() {
|
|
|
|
ASSERT(!IsZombie());
|
2014-01-07 14:27:31 +00:00
|
|
|
Initialize();
|
2014-01-13 10:28:01 +00:00
|
|
|
set_pretenure_decision(kZombie);
|
2013-11-27 14:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-25 16:00:32 +00:00
|
|
|
// Heuristic: We only need to create allocation site info if the boilerplate
|
|
|
|
// elements kind is the initial elements kind.
|
2013-07-08 10:02:16 +00:00
|
|
|
AllocationSiteMode AllocationSite::GetMode(
|
2013-04-25 16:00:32 +00:00
|
|
|
ElementsKind boilerplate_elements_kind) {
|
2014-03-19 13:39:09 +00:00
|
|
|
if (FLAG_pretenuring_call_new ||
|
|
|
|
IsFastSmiElementsKind(boilerplate_elements_kind)) {
|
2013-04-25 16:00:32 +00:00
|
|
|
return TRACK_ALLOCATION_SITE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DONT_TRACK_ALLOCATION_SITE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-08 10:02:16 +00:00
|
|
|
AllocationSiteMode AllocationSite::GetMode(ElementsKind from,
|
|
|
|
ElementsKind to) {
|
2014-03-19 13:39:09 +00:00
|
|
|
if (FLAG_pretenuring_call_new ||
|
|
|
|
(IsFastSmiElementsKind(from) &&
|
|
|
|
IsMoreGeneralElementsKindTransition(from, to))) {
|
2013-04-25 16:00:32 +00:00
|
|
|
return TRACK_ALLOCATION_SITE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DONT_TRACK_ALLOCATION_SITE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-28 08:39:43 +00:00
|
|
|
inline bool AllocationSite::CanTrack(InstanceType type) {
|
2013-11-27 14:03:40 +00:00
|
|
|
if (FLAG_allocation_site_pretenuring) {
|
2014-01-02 15:31:27 +00:00
|
|
|
return type == JS_ARRAY_TYPE ||
|
|
|
|
type == JS_OBJECT_TYPE ||
|
|
|
|
type < FIRST_NONSTRING_TYPE;
|
2013-11-27 14:03:40 +00:00
|
|
|
}
|
2013-08-28 08:39:43 +00:00
|
|
|
return type == JS_ARRAY_TYPE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-14 12:05:09 +00:00
|
|
|
inline DependentCode::DependencyGroup AllocationSite::ToDependencyGroup(
|
|
|
|
Reason reason) {
|
|
|
|
switch (reason) {
|
|
|
|
case TENURING:
|
|
|
|
return DependentCode::kAllocationSiteTenuringChangedGroup;
|
|
|
|
break;
|
|
|
|
case TRANSITIONS:
|
|
|
|
return DependentCode::kAllocationSiteTransitionChangedGroup;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
UNREACHABLE();
|
|
|
|
return DependentCode::kAllocationSiteTransitionChangedGroup;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-13 10:28:01 +00:00
|
|
|
inline void AllocationSite::set_memento_found_count(int count) {
|
|
|
|
int value = pretenure_data()->value();
|
|
|
|
// Verify that we can count more mementos than we can possibly find in one
|
|
|
|
// new space collection.
|
|
|
|
ASSERT((GetHeap()->MaxSemiSpaceSize() /
|
|
|
|
(StaticVisitorBase::kMinObjectSizeInWords * kPointerSize +
|
|
|
|
AllocationMemento::kSize)) < MementoFoundCountBits::kMax);
|
|
|
|
ASSERT(count < MementoFoundCountBits::kMax);
|
|
|
|
set_pretenure_data(
|
|
|
|
Smi::FromInt(MementoFoundCountBits::update(value, count)),
|
|
|
|
SKIP_WRITE_BARRIER);
|
|
|
|
}
|
|
|
|
|
2013-12-18 21:23:56 +00:00
|
|
|
inline bool AllocationSite::IncrementMementoFoundCount() {
|
|
|
|
if (IsZombie()) return false;
|
|
|
|
|
2014-01-13 10:28:01 +00:00
|
|
|
int value = memento_found_count();
|
|
|
|
set_memento_found_count(value + 1);
|
2013-12-18 21:23:56 +00:00
|
|
|
return value == 0;
|
2013-11-27 14:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline void AllocationSite::IncrementMementoCreateCount() {
|
|
|
|
ASSERT(FLAG_allocation_site_pretenuring);
|
2014-01-13 10:28:01 +00:00
|
|
|
int value = memento_create_count();
|
|
|
|
set_memento_create_count(value + 1);
|
2013-11-27 14:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline bool AllocationSite::DigestPretenuringFeedback() {
|
2014-01-16 11:54:12 +00:00
|
|
|
bool decision_changed = false;
|
2014-01-13 10:28:01 +00:00
|
|
|
int create_count = memento_create_count();
|
2014-02-06 10:30:13 +00:00
|
|
|
int found_count = memento_found_count();
|
2014-02-06 11:27:08 +00:00
|
|
|
bool minimum_mementos_created = create_count >= kPretenureMinimumCreated;
|
|
|
|
double ratio =
|
|
|
|
minimum_mementos_created || FLAG_trace_pretenuring_statistics ?
|
|
|
|
static_cast<double>(found_count) / create_count : 0.0;
|
2014-02-06 10:30:13 +00:00
|
|
|
PretenureFlag current_mode = GetPretenureMode();
|
|
|
|
|
2014-02-06 11:27:08 +00:00
|
|
|
if (minimum_mementos_created) {
|
2014-01-13 10:28:01 +00:00
|
|
|
PretenureDecision result = ratio >= kPretenureRatio
|
|
|
|
? kTenure
|
|
|
|
: kDontTenure;
|
|
|
|
set_pretenure_decision(result);
|
2014-01-10 12:11:54 +00:00
|
|
|
if (current_mode != GetPretenureMode()) {
|
2014-01-16 11:54:12 +00:00
|
|
|
decision_changed = true;
|
2014-02-17 12:15:16 +00:00
|
|
|
set_deopt_dependent_code(true);
|
2013-11-27 14:03:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-06 10:30:13 +00:00
|
|
|
if (FLAG_trace_pretenuring_statistics) {
|
|
|
|
PrintF(
|
|
|
|
"AllocationSite(%p): (created, found, ratio) (%d, %d, %f) %s => %s\n",
|
|
|
|
static_cast<void*>(this), create_count, found_count, ratio,
|
|
|
|
current_mode == TENURED ? "tenured" : "not tenured",
|
|
|
|
GetPretenureMode() == TENURED ? "tenured" : "not tenured");
|
|
|
|
}
|
|
|
|
|
2013-11-27 14:03:40 +00:00
|
|
|
// Clear feedback calculation fields until the next gc.
|
2014-01-13 10:28:01 +00:00
|
|
|
set_memento_found_count(0);
|
|
|
|
set_memento_create_count(0);
|
2014-01-16 11:54:12 +00:00
|
|
|
return decision_changed;
|
2013-11-27 14:03:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-09 14:29:47 +00:00
|
|
|
void JSObject::EnsureCanContainHeapObjectElements(Handle<JSObject> object) {
|
2014-04-07 10:00:14 +00:00
|
|
|
JSObject::ValidateElements(object);
|
2013-09-09 14:29:47 +00:00
|
|
|
ElementsKind elements_kind = object->map()->elements_kind();
|
2012-05-23 14:24:29 +00:00
|
|
|
if (!IsFastObjectElementsKind(elements_kind)) {
|
|
|
|
if (IsFastHoleyElementsKind(elements_kind)) {
|
2013-09-09 14:29:47 +00:00
|
|
|
TransitionElementsKind(object, FAST_HOLEY_ELEMENTS);
|
2012-05-23 14:24:29 +00:00
|
|
|
} else {
|
2013-09-09 14:29:47 +00:00
|
|
|
TransitionElementsKind(object, FAST_ELEMENTS);
|
2012-05-23 14:24:29 +00:00
|
|
|
}
|
2011-09-22 11:30:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-21 14:29:27 +00:00
|
|
|
void JSObject::EnsureCanContainElements(Handle<JSObject> object,
|
|
|
|
Object** objects,
|
|
|
|
uint32_t count,
|
|
|
|
EnsureElementsMode mode) {
|
|
|
|
ElementsKind current_kind = object->map()->elements_kind();
|
2011-12-09 08:50:19 +00:00
|
|
|
ElementsKind target_kind = current_kind;
|
2014-03-21 14:29:27 +00:00
|
|
|
{
|
|
|
|
DisallowHeapAllocation no_allocation;
|
|
|
|
ASSERT(mode != ALLOW_COPIED_DOUBLE_ELEMENTS);
|
|
|
|
bool is_holey = IsFastHoleyElementsKind(current_kind);
|
|
|
|
if (current_kind == FAST_HOLEY_ELEMENTS) return;
|
|
|
|
Heap* heap = object->GetHeap();
|
|
|
|
Object* the_hole = heap->the_hole_value();
|
|
|
|
for (uint32_t i = 0; i < count; ++i) {
|
|
|
|
Object* current = *objects++;
|
|
|
|
if (current == the_hole) {
|
|
|
|
is_holey = true;
|
|
|
|
target_kind = GetHoleyElementsKind(target_kind);
|
|
|
|
} else if (!current->IsSmi()) {
|
|
|
|
if (mode == ALLOW_CONVERTED_DOUBLE_ELEMENTS && current->IsNumber()) {
|
|
|
|
if (IsFastSmiElementsKind(target_kind)) {
|
|
|
|
if (is_holey) {
|
|
|
|
target_kind = FAST_HOLEY_DOUBLE_ELEMENTS;
|
|
|
|
} else {
|
|
|
|
target_kind = FAST_DOUBLE_ELEMENTS;
|
|
|
|
}
|
2012-05-23 14:24:29 +00:00
|
|
|
}
|
2014-03-21 14:29:27 +00:00
|
|
|
} else if (is_holey) {
|
|
|
|
target_kind = FAST_HOLEY_ELEMENTS;
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
target_kind = FAST_ELEMENTS;
|
2012-05-23 14:24:29 +00:00
|
|
|
}
|
2011-09-22 11:30:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-12-09 08:50:19 +00:00
|
|
|
if (target_kind != current_kind) {
|
2014-03-21 14:29:27 +00:00
|
|
|
TransitionElementsKind(object, target_kind);
|
2011-12-09 08:50:19 +00:00
|
|
|
}
|
2011-09-22 11:30:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-21 08:30:42 +00:00
|
|
|
void JSObject::EnsureCanContainElements(Handle<JSObject> object,
|
|
|
|
Handle<FixedArrayBase> elements,
|
|
|
|
uint32_t length,
|
|
|
|
EnsureElementsMode mode) {
|
2014-03-21 14:29:27 +00:00
|
|
|
Heap* heap = object->GetHeap();
|
|
|
|
if (elements->map() != heap->fixed_double_array_map()) {
|
|
|
|
ASSERT(elements->map() == heap->fixed_array_map() ||
|
|
|
|
elements->map() == heap->fixed_cow_array_map());
|
2011-12-09 08:50:19 +00:00
|
|
|
if (mode == ALLOW_COPIED_DOUBLE_ELEMENTS) {
|
|
|
|
mode = DONT_ALLOW_DOUBLE_ELEMENTS;
|
|
|
|
}
|
2014-03-21 14:29:27 +00:00
|
|
|
Object** objects =
|
|
|
|
Handle<FixedArray>::cast(elements)->GetFirstElementAddress();
|
|
|
|
EnsureCanContainElements(object, objects, length, mode);
|
|
|
|
return;
|
2011-12-09 08:50:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(mode == ALLOW_COPIED_DOUBLE_ELEMENTS);
|
2014-03-21 14:29:27 +00:00
|
|
|
if (object->GetElementsKind() == FAST_HOLEY_SMI_ELEMENTS) {
|
|
|
|
TransitionElementsKind(object, FAST_HOLEY_DOUBLE_ELEMENTS);
|
|
|
|
} else if (object->GetElementsKind() == FAST_SMI_ELEMENTS) {
|
|
|
|
Handle<FixedDoubleArray> double_array =
|
|
|
|
Handle<FixedDoubleArray>::cast(elements);
|
2012-05-23 14:24:29 +00:00
|
|
|
for (uint32_t i = 0; i < length; ++i) {
|
|
|
|
if (double_array->is_the_hole(i)) {
|
2014-03-21 14:29:27 +00:00
|
|
|
TransitionElementsKind(object, FAST_HOLEY_DOUBLE_ELEMENTS);
|
|
|
|
return;
|
2012-05-23 14:24:29 +00:00
|
|
|
}
|
|
|
|
}
|
2014-03-21 14:29:27 +00:00
|
|
|
TransitionElementsKind(object, FAST_DOUBLE_ELEMENTS);
|
2011-12-09 08:50:19 +00:00
|
|
|
}
|
2011-09-22 11:30:04 +00:00
|
|
|
}
|
|
|
|
|
2009-07-28 08:43:51 +00:00
|
|
|
|
2014-03-31 15:30:13 +00:00
|
|
|
void JSObject::SetMapAndElements(Handle<JSObject> object,
|
|
|
|
Handle<Map> new_map,
|
|
|
|
Handle<FixedArrayBase> value) {
|
|
|
|
JSObject::MigrateToMap(object, new_map);
|
|
|
|
ASSERT((object->map()->has_fast_smi_or_object_elements() ||
|
|
|
|
(*value == object->GetHeap()->empty_fixed_array())) ==
|
|
|
|
(value->map() == object->GetHeap()->fixed_array_map() ||
|
|
|
|
value->map() == object->GetHeap()->fixed_cow_array_map()));
|
|
|
|
ASSERT((*value == object->GetHeap()->empty_fixed_array()) ||
|
|
|
|
(object->map()->has_fast_double_elements() ==
|
|
|
|
value->IsFixedDoubleArray()));
|
|
|
|
object->set_elements(*value);
|
2009-07-28 08:43:51 +00:00
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2011-12-30 14:28:14 +00:00
|
|
|
void JSObject::set_elements(FixedArrayBase* value, WriteBarrierMode mode) {
|
2014-03-31 15:30:13 +00:00
|
|
|
WRITE_FIELD(this, kElementsOffset, value);
|
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kElementsOffset, value, mode);
|
2011-12-30 14:28:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void JSObject::initialize_properties() {
|
2011-03-18 20:35:07 +00:00
|
|
|
ASSERT(!GetHeap()->InNewSpace(GetHeap()->empty_fixed_array()));
|
|
|
|
WRITE_FIELD(this, kPropertiesOffset, GetHeap()->empty_fixed_array());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSObject::initialize_elements() {
|
2014-04-09 14:26:32 +00:00
|
|
|
FixedArrayBase* elements = map()->GetInitialElements();
|
|
|
|
WRITE_FIELD(this, kElementsOffset, elements);
|
2010-06-24 13:56:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-09 16:49:28 +00:00
|
|
|
Handle<String> JSObject::ExpectedTransitionKey(Handle<Map> map) {
|
2013-06-03 15:32:22 +00:00
|
|
|
DisallowHeapAllocation no_gc;
|
2013-04-09 16:49:28 +00:00
|
|
|
if (!map->HasTransitionArray()) return Handle<String>::null();
|
|
|
|
TransitionArray* transitions = map->transitions();
|
|
|
|
if (!transitions->IsSimpleTransition()) return Handle<String>::null();
|
|
|
|
int transition = TransitionArray::kSimpleTransitionIndex;
|
|
|
|
PropertyDetails details = transitions->GetTargetDetails(transition);
|
|
|
|
Name* name = transitions->GetKey(transition);
|
|
|
|
if (details.type() != FIELD) return Handle<String>::null();
|
|
|
|
if (details.attributes() != NONE) return Handle<String>::null();
|
|
|
|
if (!name->IsString()) return Handle<String>::null();
|
|
|
|
return Handle<String>(String::cast(name));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Map> JSObject::ExpectedTransitionTarget(Handle<Map> map) {
|
|
|
|
ASSERT(!ExpectedTransitionKey(map).is_null());
|
|
|
|
return Handle<Map>(map->transitions()->GetTarget(
|
|
|
|
TransitionArray::kSimpleTransitionIndex));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Map> JSObject::FindTransitionToField(Handle<Map> map, Handle<Name> key) {
|
2013-06-03 15:32:22 +00:00
|
|
|
DisallowHeapAllocation no_allocation;
|
2013-04-09 16:49:28 +00:00
|
|
|
if (!map->HasTransitionArray()) return Handle<Map>::null();
|
|
|
|
TransitionArray* transitions = map->transitions();
|
|
|
|
int transition = transitions->Search(*key);
|
|
|
|
if (transition == TransitionArray::kNotFound) return Handle<Map>::null();
|
|
|
|
PropertyDetails target_details = transitions->GetTargetDetails(transition);
|
|
|
|
if (target_details.type() != FIELD) return Handle<Map>::null();
|
|
|
|
if (target_details.attributes() != NONE) return Handle<Map>::null();
|
|
|
|
return Handle<Map>(transitions->GetTarget(transition));
|
2012-10-17 14:09:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(Oddball, to_string, String, kToStringOffset)
|
|
|
|
ACCESSORS(Oddball, to_number, Object, kToNumberOffset)
|
|
|
|
|
|
|
|
|
2011-03-18 20:35:07 +00:00
|
|
|
byte Oddball::kind() {
|
2011-09-19 18:36:47 +00:00
|
|
|
return Smi::cast(READ_FIELD(this, kKindOffset))->value();
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Oddball::set_kind(byte value) {
|
2011-09-19 18:36:47 +00:00
|
|
|
WRITE_FIELD(this, kKindOffset, Smi::FromInt(value));
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-12 15:03:44 +00:00
|
|
|
Object* Cell::value() {
|
2009-07-09 14:34:08 +00:00
|
|
|
return READ_FIELD(this, kValueOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-12 15:03:44 +00:00
|
|
|
void Cell::set_value(Object* val, WriteBarrierMode ignored) {
|
2009-07-09 14:34:08 +00:00
|
|
|
// The write barrier is not used for global property cells.
|
2013-06-14 16:06:12 +00:00
|
|
|
ASSERT(!val->IsPropertyCell() && !val->IsCell());
|
2009-07-09 14:34:08 +00:00
|
|
|
WRITE_FIELD(this, kValueOffset, val);
|
|
|
|
}
|
|
|
|
|
2013-06-26 16:17:12 +00:00
|
|
|
ACCESSORS(PropertyCell, dependent_code, DependentCode, kDependentCodeOffset)
|
2009-06-30 10:05:36 +00:00
|
|
|
|
2013-06-14 16:06:12 +00:00
|
|
|
Object* PropertyCell::type_raw() {
|
2013-06-12 15:03:44 +00:00
|
|
|
return READ_FIELD(this, kTypeOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-14 16:06:12 +00:00
|
|
|
void PropertyCell::set_type_raw(Object* val, WriteBarrierMode ignored) {
|
2013-06-12 15:03:44 +00:00
|
|
|
WRITE_FIELD(this, kTypeOffset, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
int JSObject::GetHeaderSize() {
|
2009-07-17 09:16:23 +00:00
|
|
|
InstanceType type = map()->instance_type();
|
|
|
|
// Check for the most common kind of JavaScript object before
|
|
|
|
// falling into the generic switch. This speeds up the internal
|
|
|
|
// field operations considerably on average.
|
|
|
|
if (type == JS_OBJECT_TYPE) return JSObject::kHeaderSize;
|
|
|
|
switch (type) {
|
2013-04-15 12:29:44 +00:00
|
|
|
case JS_GENERATOR_OBJECT_TYPE:
|
|
|
|
return JSGeneratorObject::kSize;
|
2012-04-16 14:43:27 +00:00
|
|
|
case JS_MODULE_TYPE:
|
|
|
|
return JSModule::kSize;
|
Split window support from V8.
Here is a description of the background and design of split window in Chrome and V8:
https://docs.google.com/a/google.com/Doc?id=chhjkpg_47fwddxbfr
This change list splits the window object into two parts: 1) an inner window object used as the global object of contexts; 2) an outer window object exposed to JavaScript and accessible by the name 'window'. Firefox did it awhile ago, here are some discussions: https://wiki.mozilla.org/Gecko:SplitWindow. One additional benefit of splitting window in Chrome is that accessing global variables don't need security checks anymore, it can improve applications that use many global variables.
V8 support of split window:
There are a small number of changes on V8 api to support split window:
Security context is removed from V8, so does related API functions;
A global object can be detached from its context and reused by a new context;
Access checks on an object template can be turned on/off by default;
An object can turn on its access checks later;
V8 has a new object type, ApiGlobalObject, which is the outer window object type. The existing JSGlobalObject becomes the inner window object type. Security checks are moved from JSGlobalObject to ApiGlobalObject. ApiGlobalObject is the one exposed to JavaScript, it is accessible through Context::Global(). ApiGlobalObject's prototype is set to JSGlobalObject so that property lookups are forwarded to JSGlobalObject. ApiGlobalObject forwards all other property access requests to JSGlobalObject, such as SetProperty, DeleteProperty, etc.
Security token is moved to a global context, and ApiGlobalObject has a reference to its global context. JSGlobalObject has a reference to its global context as well. When accessing properties on a global object in JavaScript, the domain security check is performed by comparing the security token of the lexical context (Top::global_context()) to the token of global object's context. The check is only needed when the receiver is a window object, such as 'window.document'. Accessing global variables, such as 'var foo = 3; foo' does not need checks because the receiver is the inner window object.
When an outer window is detached from its global context (when a frame navigates away from a page), it is completely detached from the inner window. A new context is created for the new page, and the outer global object is reused. At this point, the access check on the DOMWindow wrapper of the old context is turned on. The code in old context is still able to access DOMWindow properties, but it has to go through domain security checks.
It is debatable on how to implement the outer window object. Currently each property access function has to check if the receiver is ApiGlobalObject type. This approach might be error-prone that one may forget to check the receiver when adding new functions. It is unlikely a performance issue because accessing global variables are more common than 'window.foo' style coding.
I am still working on the ARM port, and I'd like to hear comments and suggestions on the best way to support it in V8.
Review URL: http://codereview.chromium.org/7366
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@540 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2008-10-21 19:07:58 +00:00
|
|
|
case JS_GLOBAL_PROXY_TYPE:
|
|
|
|
return JSGlobalProxy::kSize;
|
2008-07-03 15:10:15 +00:00
|
|
|
case JS_GLOBAL_OBJECT_TYPE:
|
|
|
|
return JSGlobalObject::kSize;
|
|
|
|
case JS_BUILTINS_OBJECT_TYPE:
|
|
|
|
return JSBuiltinsObject::kSize;
|
|
|
|
case JS_FUNCTION_TYPE:
|
|
|
|
return JSFunction::kSize;
|
|
|
|
case JS_VALUE_TYPE:
|
|
|
|
return JSValue::kSize;
|
2012-03-09 11:11:55 +00:00
|
|
|
case JS_DATE_TYPE:
|
|
|
|
return JSDate::kSize;
|
2008-07-03 15:10:15 +00:00
|
|
|
case JS_ARRAY_TYPE:
|
2012-01-18 17:01:57 +00:00
|
|
|
return JSArray::kSize;
|
2013-03-28 12:50:18 +00:00
|
|
|
case JS_ARRAY_BUFFER_TYPE:
|
|
|
|
return JSArrayBuffer::kSize;
|
2013-04-16 14:16:30 +00:00
|
|
|
case JS_TYPED_ARRAY_TYPE:
|
|
|
|
return JSTypedArray::kSize;
|
2013-06-21 13:02:38 +00:00
|
|
|
case JS_DATA_VIEW_TYPE:
|
|
|
|
return JSDataView::kSize;
|
2013-03-11 11:31:53 +00:00
|
|
|
case JS_SET_TYPE:
|
|
|
|
return JSSet::kSize;
|
|
|
|
case JS_MAP_TYPE:
|
|
|
|
return JSMap::kSize;
|
2014-04-17 17:45:32 +00:00
|
|
|
case JS_SET_ITERATOR_TYPE:
|
|
|
|
return JSSetIterator::kSize;
|
|
|
|
case JS_MAP_ITERATOR_TYPE:
|
|
|
|
return JSMapIterator::kSize;
|
2011-08-03 11:55:13 +00:00
|
|
|
case JS_WEAK_MAP_TYPE:
|
|
|
|
return JSWeakMap::kSize;
|
2013-07-22 08:32:24 +00:00
|
|
|
case JS_WEAK_SET_TYPE:
|
|
|
|
return JSWeakSet::kSize;
|
2008-09-23 11:45:43 +00:00
|
|
|
case JS_REGEXP_TYPE:
|
2012-01-18 17:01:57 +00:00
|
|
|
return JSRegExp::kSize;
|
2009-01-14 12:13:26 +00:00
|
|
|
case JS_CONTEXT_EXTENSION_OBJECT_TYPE:
|
2008-07-03 15:10:15 +00:00
|
|
|
return JSObject::kHeaderSize;
|
2011-02-02 13:31:52 +00:00
|
|
|
case JS_MESSAGE_OBJECT_TYPE:
|
|
|
|
return JSMessageObject::kSize;
|
2008-07-03 15:10:15 +00:00
|
|
|
default:
|
2013-10-28 17:54:43 +00:00
|
|
|
// TODO(jkummerow): Re-enable this. Blink currently hits this
|
|
|
|
// from its CustomElementConstructorBuilder.
|
|
|
|
// UNREACHABLE();
|
2008-07-03 15:10:15 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int JSObject::GetInternalFieldCount() {
|
|
|
|
ASSERT(1 << kPointerSizeLog2 == kPointerSize);
|
2008-10-15 06:03:26 +00:00
|
|
|
// Make sure to adjust for the number of in-object properties. These
|
|
|
|
// properties do contribute to the size, but are not internal fields.
|
|
|
|
return ((Size() - GetHeaderSize()) >> kPointerSizeLog2) -
|
|
|
|
map()->inobject_properties();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-18 12:49:27 +00:00
|
|
|
int JSObject::GetInternalFieldOffset(int index) {
|
|
|
|
ASSERT(index < GetInternalFieldCount() && index >= 0);
|
|
|
|
return GetHeaderSize() + (kPointerSize * index);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* JSObject::GetInternalField(int index) {
|
|
|
|
ASSERT(index < GetInternalFieldCount() && index >= 0);
|
2008-10-15 06:03:26 +00:00
|
|
|
// Internal objects do follow immediately after the header, whereas in-object
|
|
|
|
// properties are at the end of the object. Therefore there is no need
|
|
|
|
// to adjust the index here.
|
2008-07-03 15:10:15 +00:00
|
|
|
return READ_FIELD(this, GetHeaderSize() + (kPointerSize * index));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSObject::SetInternalField(int index, Object* value) {
|
|
|
|
ASSERT(index < GetInternalFieldCount() && index >= 0);
|
2008-10-15 06:03:26 +00:00
|
|
|
// Internal objects do follow immediately after the header, whereas in-object
|
|
|
|
// properties are at the end of the object. Therefore there is no need
|
|
|
|
// to adjust the index here.
|
2008-07-03 15:10:15 +00:00
|
|
|
int offset = GetHeaderSize() + (kPointerSize * index);
|
|
|
|
WRITE_FIELD(this, offset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
WRITE_BARRIER(GetHeap(), this, offset, value);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-15 12:18:24 +00:00
|
|
|
void JSObject::SetInternalField(int index, Smi* value) {
|
|
|
|
ASSERT(index < GetInternalFieldCount() && index >= 0);
|
|
|
|
// Internal objects do follow immediately after the header, whereas in-object
|
|
|
|
// properties are at the end of the object. Therefore there is no need
|
|
|
|
// to adjust the index here.
|
|
|
|
int offset = GetHeaderSize() + (kPointerSize * index);
|
|
|
|
WRITE_FIELD(this, offset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-15 06:03:26 +00:00
|
|
|
// Access fast-case object properties at index. The use of these routines
|
|
|
|
// is needed to correctly distinguish between properties stored in-object and
|
|
|
|
// properties stored in the properties array.
|
2013-05-08 15:02:08 +00:00
|
|
|
Object* JSObject::RawFastPropertyAt(int index) {
|
2008-10-15 06:03:26 +00:00
|
|
|
// Adjust for the number of properties stored in the object.
|
|
|
|
index -= map()->inobject_properties();
|
|
|
|
if (index < 0) {
|
|
|
|
int offset = map()->instance_size() + (index * kPointerSize);
|
|
|
|
return READ_FIELD(this, offset);
|
|
|
|
} else {
|
|
|
|
ASSERT(index < properties()->length());
|
|
|
|
return properties()->get(index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-08 15:02:08 +00:00
|
|
|
void JSObject::FastPropertyAtPut(int index, Object* value) {
|
2008-10-15 06:03:26 +00:00
|
|
|
// Adjust for the number of properties stored in the object.
|
|
|
|
index -= map()->inobject_properties();
|
|
|
|
if (index < 0) {
|
|
|
|
int offset = map()->instance_size() + (index * kPointerSize);
|
|
|
|
WRITE_FIELD(this, offset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
WRITE_BARRIER(GetHeap(), this, offset, value);
|
2008-10-15 06:03:26 +00:00
|
|
|
} else {
|
|
|
|
ASSERT(index < properties()->length());
|
|
|
|
properties()->set(index, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-18 12:49:27 +00:00
|
|
|
int JSObject::GetInObjectPropertyOffset(int index) {
|
2014-02-04 10:48:49 +00:00
|
|
|
return map()->GetInObjectPropertyOffset(index);
|
2011-03-18 12:49:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-23 07:27:47 +00:00
|
|
|
Object* JSObject::InObjectPropertyAt(int index) {
|
2014-02-04 10:48:49 +00:00
|
|
|
int offset = GetInObjectPropertyOffset(index);
|
2009-03-23 07:27:47 +00:00
|
|
|
return READ_FIELD(this, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-23 08:46:32 +00:00
|
|
|
Object* JSObject::InObjectPropertyAtPut(int index,
|
|
|
|
Object* value,
|
|
|
|
WriteBarrierMode mode) {
|
|
|
|
// Adjust for the number of properties stored in the object.
|
2014-02-04 10:48:49 +00:00
|
|
|
int offset = GetInObjectPropertyOffset(index);
|
2008-10-23 08:46:32 +00:00
|
|
|
WRITE_FIELD(this, offset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, offset, value, mode);
|
2008-10-23 08:46:32 +00:00
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-09-20 10:06:23 +00:00
|
|
|
void JSObject::InitializeBody(Map* map,
|
|
|
|
Object* pre_allocated_value,
|
|
|
|
Object* filler_value) {
|
|
|
|
ASSERT(!filler_value->IsHeapObject() ||
|
|
|
|
!GetHeap()->InNewSpace(filler_value));
|
|
|
|
ASSERT(!pre_allocated_value->IsHeapObject() ||
|
|
|
|
!GetHeap()->InNewSpace(pre_allocated_value));
|
|
|
|
int size = map->instance_size();
|
|
|
|
int offset = kHeaderSize;
|
|
|
|
if (filler_value != pre_allocated_value) {
|
|
|
|
int pre_allocated = map->pre_allocated_property_fields();
|
|
|
|
ASSERT(pre_allocated * kPointerSize + kHeaderSize <= size);
|
|
|
|
for (int i = 0; i < pre_allocated; i++) {
|
|
|
|
WRITE_FIELD(this, offset, pre_allocated_value);
|
|
|
|
offset += kPointerSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (offset < size) {
|
|
|
|
WRITE_FIELD(this, offset, filler_value);
|
|
|
|
offset += kPointerSize;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-02 11:27:57 +00:00
|
|
|
bool JSObject::HasFastProperties() {
|
2012-08-06 14:25:19 +00:00
|
|
|
ASSERT(properties()->IsDictionary() == map()->is_dictionary_map());
|
2010-07-02 11:27:57 +00:00
|
|
|
return !properties()->IsDictionary();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-26 15:30:30 +00:00
|
|
|
bool JSObject::TooManyFastProperties(StoreFromKeyed store_mode) {
|
2010-07-02 11:27:57 +00:00
|
|
|
// Allow extra fast properties if the object has more than
|
2013-08-26 15:30:30 +00:00
|
|
|
// kFastPropertiesSoftLimit in-object properties. When this is the case, it is
|
|
|
|
// very unlikely that the object is being used as a dictionary and there is a
|
|
|
|
// good chance that allowing more map transitions will be worth it.
|
|
|
|
Map* map = this->map();
|
|
|
|
if (map->unused_property_fields() != 0) return false;
|
|
|
|
|
|
|
|
int inobject = map->inobject_properties();
|
2012-06-06 10:17:26 +00:00
|
|
|
|
|
|
|
int limit;
|
2012-06-15 11:53:09 +00:00
|
|
|
if (store_mode == CERTAINLY_NOT_STORE_FROM_KEYED) {
|
2012-06-06 10:17:26 +00:00
|
|
|
limit = Max(inobject, kMaxFastProperties);
|
|
|
|
} else {
|
|
|
|
limit = Max(inobject, kFastPropertiesSoftLimit);
|
|
|
|
}
|
2013-08-26 15:30:30 +00:00
|
|
|
return properties()->length() > limit;
|
2010-07-02 11:27:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void Struct::InitializeBody(int object_size) {
|
2011-03-18 20:35:07 +00:00
|
|
|
Object* value = GetHeap()->undefined_value();
|
2008-09-23 11:45:43 +00:00
|
|
|
for (int offset = kHeaderSize; offset < object_size; offset += kPointerSize) {
|
2008-10-22 08:21:18 +00:00
|
|
|
WRITE_FIELD(this, offset, value);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-27 12:30:45 +00:00
|
|
|
bool Object::ToArrayIndex(uint32_t* index) {
|
|
|
|
if (IsSmi()) {
|
|
|
|
int value = Smi::cast(this)->value();
|
2008-07-03 15:10:15 +00:00
|
|
|
if (value < 0) return false;
|
|
|
|
*index = value;
|
|
|
|
return true;
|
|
|
|
}
|
2010-05-27 12:30:45 +00:00
|
|
|
if (IsHeapNumber()) {
|
|
|
|
double value = HeapNumber::cast(this)->value();
|
2008-07-03 15:10:15 +00:00
|
|
|
uint32_t uint_value = static_cast<uint32_t>(value);
|
|
|
|
if (value == static_cast<double>(uint_value)) {
|
|
|
|
*index = uint_value;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Object::IsStringObjectWithCharacterAt(uint32_t index) {
|
|
|
|
if (!this->IsJSValue()) return false;
|
|
|
|
|
|
|
|
JSValue* js_value = JSValue::cast(this);
|
|
|
|
if (!js_value->value()->IsString()) return false;
|
|
|
|
|
|
|
|
String* str = String::cast(js_value->value());
|
2013-01-07 15:02:56 +00:00
|
|
|
if (index >= static_cast<uint32_t>(str->length())) return false;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-09-10 13:38:21 +00:00
|
|
|
void Object::VerifyApiCallResultType() {
|
|
|
|
#if ENABLE_EXTRA_CHECKS
|
|
|
|
if (!(IsSmi() ||
|
|
|
|
IsString() ||
|
2014-03-20 16:13:09 +00:00
|
|
|
IsSymbol() ||
|
2012-09-10 13:38:21 +00:00
|
|
|
IsSpecObject() ||
|
|
|
|
IsHeapNumber() ||
|
|
|
|
IsUndefined() ||
|
|
|
|
IsTrue() ||
|
|
|
|
IsFalse() ||
|
|
|
|
IsNull())) {
|
|
|
|
FATAL("API call returned invalid object");
|
|
|
|
}
|
|
|
|
#endif // ENABLE_EXTRA_CHECKS
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-09 10:03:35 +00:00
|
|
|
FixedArrayBase* FixedArrayBase::cast(Object* object) {
|
2014-01-16 17:08:45 +00:00
|
|
|
ASSERT(object->IsFixedArrayBase());
|
2011-06-09 10:03:35 +00:00
|
|
|
return reinterpret_cast<FixedArrayBase*>(object);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* FixedArray::get(int index) {
|
2013-10-25 11:10:28 +00:00
|
|
|
SLOW_ASSERT(index >= 0 && index < this->length());
|
2008-07-03 15:10:15 +00:00
|
|
|
return READ_FIELD(this, kHeaderSize + index * kPointerSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> FixedArray::get(Handle<FixedArray> array, int index) {
|
|
|
|
return handle(array->get(index), array->GetIsolate());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-23 14:24:29 +00:00
|
|
|
bool FixedArray::is_the_hole(int index) {
|
|
|
|
return get(index) == GetHeap()->the_hole_value();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-25 11:07:48 +00:00
|
|
|
void FixedArray::set(int index, Smi* value) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map());
|
2011-08-03 13:56:55 +00:00
|
|
|
ASSERT(index >= 0 && index < this->length());
|
2008-11-25 11:07:48 +00:00
|
|
|
ASSERT(reinterpret_cast<Object*>(value)->IsSmi());
|
|
|
|
int offset = kHeaderSize + index * kPointerSize;
|
|
|
|
WRITE_FIELD(this, offset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void FixedArray::set(int index, Object* value) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map());
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(index >= 0 && index < this->length());
|
|
|
|
int offset = kHeaderSize + index * kPointerSize;
|
|
|
|
WRITE_FIELD(this, offset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
WRITE_BARRIER(GetHeap(), this, offset, value);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-13 13:50:27 +00:00
|
|
|
inline bool FixedDoubleArray::is_the_hole_nan(double value) {
|
|
|
|
return BitCast<uint64_t, double>(value) == kHoleNanInt64;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline double FixedDoubleArray::hole_nan_as_double() {
|
|
|
|
return BitCast<double, uint64_t>(kHoleNanInt64);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline double FixedDoubleArray::canonical_not_the_hole_nan_as_double() {
|
2011-07-13 16:36:26 +00:00
|
|
|
ASSERT(BitCast<uint64_t>(OS::nan_value()) != kHoleNanInt64);
|
|
|
|
ASSERT((BitCast<uint64_t>(OS::nan_value()) >> 32) != kHoleNanUpper32);
|
|
|
|
return OS::nan_value();
|
2011-07-13 13:50:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-03 11:12:46 +00:00
|
|
|
double FixedDoubleArray::get_scalar(int index) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map() &&
|
|
|
|
map() != GetHeap()->fixed_array_map());
|
2011-06-09 10:03:35 +00:00
|
|
|
ASSERT(index >= 0 && index < this->length());
|
|
|
|
double result = READ_DOUBLE_FIELD(this, kHeaderSize + index * kDoubleSize);
|
|
|
|
ASSERT(!is_the_hole_nan(result));
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2012-03-23 10:14:51 +00:00
|
|
|
int64_t FixedDoubleArray::get_representation(int index) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map() &&
|
|
|
|
map() != GetHeap()->fixed_array_map());
|
2012-03-23 10:14:51 +00:00
|
|
|
ASSERT(index >= 0 && index < this->length());
|
|
|
|
return READ_INT64_FIELD(this, kHeaderSize + index * kDoubleSize);
|
|
|
|
}
|
2011-06-09 10:03:35 +00:00
|
|
|
|
2011-08-03 11:12:46 +00:00
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> FixedDoubleArray::get(Handle<FixedDoubleArray> array,
|
|
|
|
int index) {
|
|
|
|
if (array->is_the_hole(index)) {
|
|
|
|
return array->GetIsolate()->factory()->the_hole_value();
|
2014-03-27 16:41:09 +00:00
|
|
|
} else {
|
2014-04-08 14:20:29 +00:00
|
|
|
return array->GetIsolate()->factory()->NewNumber(array->get_scalar(index));
|
2014-03-27 16:41:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-09 10:03:35 +00:00
|
|
|
void FixedDoubleArray::set(int index, double value) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map() &&
|
|
|
|
map() != GetHeap()->fixed_array_map());
|
2011-06-09 10:03:35 +00:00
|
|
|
int offset = kHeaderSize + index * kDoubleSize;
|
2013-04-19 13:26:47 +00:00
|
|
|
if (std::isnan(value)) value = canonical_not_the_hole_nan_as_double();
|
2011-06-09 10:03:35 +00:00
|
|
|
WRITE_DOUBLE_FIELD(this, offset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FixedDoubleArray::set_the_hole(int index) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map() &&
|
|
|
|
map() != GetHeap()->fixed_array_map());
|
2011-06-09 10:03:35 +00:00
|
|
|
int offset = kHeaderSize + index * kDoubleSize;
|
|
|
|
WRITE_DOUBLE_FIELD(this, offset, hole_nan_as_double());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool FixedDoubleArray::is_the_hole(int index) {
|
|
|
|
int offset = kHeaderSize + index * kDoubleSize;
|
|
|
|
return is_the_hole_nan(READ_DOUBLE_FIELD(this, offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-28 13:05:23 +00:00
|
|
|
double* FixedDoubleArray::data_start() {
|
|
|
|
return reinterpret_cast<double*>(FIELD_ADDR(this, kHeaderSize));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FixedDoubleArray::FillWithHoles(int from, int to) {
|
|
|
|
for (int i = from; i < to; i++) {
|
|
|
|
set_the_hole(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 10:00:57 +00:00
|
|
|
void ConstantPoolArray::set_weak_object_state(
|
|
|
|
ConstantPoolArray::WeakObjectState state) {
|
|
|
|
int old_layout_field = READ_INT_FIELD(this, kArrayLayoutOffset);
|
|
|
|
int new_layout_field = WeakObjectStateField::update(old_layout_field, state);
|
|
|
|
WRITE_INT_FIELD(this, kArrayLayoutOffset, new_layout_field);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ConstantPoolArray::WeakObjectState ConstantPoolArray::get_weak_object_state() {
|
|
|
|
int layout_field = READ_INT_FIELD(this, kArrayLayoutOffset);
|
|
|
|
return WeakObjectStateField::decode(layout_field);
|
|
|
|
}
|
2013-10-14 13:35:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
int ConstantPoolArray::first_int64_index() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 10:00:57 +00:00
|
|
|
int ConstantPoolArray::first_code_ptr_index() {
|
|
|
|
int layout_field = READ_INT_FIELD(this, kArrayLayoutOffset);
|
|
|
|
return first_int64_index() +
|
|
|
|
NumberOfInt64EntriesField::decode(layout_field);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ConstantPoolArray::first_heap_ptr_index() {
|
|
|
|
int layout_field = READ_INT_FIELD(this, kArrayLayoutOffset);
|
|
|
|
return first_code_ptr_index() +
|
|
|
|
NumberOfCodePtrEntriesField::decode(layout_field);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ConstantPoolArray::first_int32_index() {
|
|
|
|
int layout_field = READ_INT_FIELD(this, kArrayLayoutOffset);
|
|
|
|
return first_heap_ptr_index() +
|
|
|
|
NumberOfHeapPtrEntriesField::decode(layout_field);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-14 13:35:06 +00:00
|
|
|
int ConstantPoolArray::count_of_int64_entries() {
|
2014-03-10 19:05:43 +00:00
|
|
|
return first_code_ptr_index();
|
2013-10-14 13:35:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-10 19:05:43 +00:00
|
|
|
int ConstantPoolArray::count_of_code_ptr_entries() {
|
|
|
|
return first_heap_ptr_index() - first_code_ptr_index();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ConstantPoolArray::count_of_heap_ptr_entries() {
|
|
|
|
return first_int32_index() - first_heap_ptr_index();
|
2013-10-14 13:35:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ConstantPoolArray::count_of_int32_entries() {
|
|
|
|
return length() - first_int32_index();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 10:00:57 +00:00
|
|
|
void ConstantPoolArray::Init(int number_of_int64_entries,
|
|
|
|
int number_of_code_ptr_entries,
|
|
|
|
int number_of_heap_ptr_entries,
|
|
|
|
int number_of_int32_entries) {
|
|
|
|
set_length(number_of_int64_entries +
|
|
|
|
number_of_code_ptr_entries +
|
|
|
|
number_of_heap_ptr_entries +
|
|
|
|
number_of_int32_entries);
|
|
|
|
int layout_field =
|
|
|
|
NumberOfInt64EntriesField::encode(number_of_int64_entries) |
|
|
|
|
NumberOfCodePtrEntriesField::encode(number_of_code_ptr_entries) |
|
|
|
|
NumberOfHeapPtrEntriesField::encode(number_of_heap_ptr_entries) |
|
|
|
|
WeakObjectStateField::encode(NO_WEAK_OBJECTS);
|
|
|
|
WRITE_INT_FIELD(this, kArrayLayoutOffset, layout_field);
|
2013-10-14 13:35:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int64_t ConstantPoolArray::get_int64_entry(int index) {
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
2014-03-10 19:05:43 +00:00
|
|
|
ASSERT(index >= 0 && index < first_code_ptr_index());
|
2013-10-14 13:35:06 +00:00
|
|
|
return READ_INT64_FIELD(this, OffsetOfElementAt(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
double ConstantPoolArray::get_int64_entry_as_double(int index) {
|
|
|
|
STATIC_ASSERT(kDoubleSize == kInt64Size);
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
2014-03-10 19:05:43 +00:00
|
|
|
ASSERT(index >= 0 && index < first_code_ptr_index());
|
2013-10-14 13:35:06 +00:00
|
|
|
return READ_DOUBLE_FIELD(this, OffsetOfElementAt(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-10 19:05:43 +00:00
|
|
|
Address ConstantPoolArray::get_code_ptr_entry(int index) {
|
2013-10-14 13:35:06 +00:00
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
2014-03-10 19:05:43 +00:00
|
|
|
ASSERT(index >= first_code_ptr_index() && index < first_heap_ptr_index());
|
|
|
|
return reinterpret_cast<Address>(READ_FIELD(this, OffsetOfElementAt(index)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object* ConstantPoolArray::get_heap_ptr_entry(int index) {
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
|
|
|
ASSERT(index >= first_heap_ptr_index() && index < first_int32_index());
|
2013-10-14 13:35:06 +00:00
|
|
|
return READ_FIELD(this, OffsetOfElementAt(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int32_t ConstantPoolArray::get_int32_entry(int index) {
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
|
|
|
ASSERT(index >= first_int32_index() && index < length());
|
|
|
|
return READ_INT32_FIELD(this, OffsetOfElementAt(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-10 19:05:43 +00:00
|
|
|
void ConstantPoolArray::set(int index, Address value) {
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
|
|
|
ASSERT(index >= first_code_ptr_index() && index < first_heap_ptr_index());
|
|
|
|
WRITE_FIELD(this, OffsetOfElementAt(index), reinterpret_cast<Object*>(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-14 13:35:06 +00:00
|
|
|
void ConstantPoolArray::set(int index, Object* value) {
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
2014-03-10 19:05:43 +00:00
|
|
|
ASSERT(index >= first_code_ptr_index() && index < first_int32_index());
|
2013-10-14 13:35:06 +00:00
|
|
|
WRITE_FIELD(this, OffsetOfElementAt(index), value);
|
|
|
|
WRITE_BARRIER(GetHeap(), this, OffsetOfElementAt(index), value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ConstantPoolArray::set(int index, int64_t value) {
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
2014-03-10 19:05:43 +00:00
|
|
|
ASSERT(index >= first_int64_index() && index < first_code_ptr_index());
|
2013-10-14 13:35:06 +00:00
|
|
|
WRITE_INT64_FIELD(this, OffsetOfElementAt(index), value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ConstantPoolArray::set(int index, double value) {
|
|
|
|
STATIC_ASSERT(kDoubleSize == kInt64Size);
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
2014-03-10 19:05:43 +00:00
|
|
|
ASSERT(index >= first_int64_index() && index < first_code_ptr_index());
|
2013-10-14 13:35:06 +00:00
|
|
|
WRITE_DOUBLE_FIELD(this, OffsetOfElementAt(index), value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ConstantPoolArray::set(int index, int32_t value) {
|
|
|
|
ASSERT(map() == GetHeap()->constant_pool_array_map());
|
|
|
|
ASSERT(index >= this->first_int32_index() && index < length());
|
|
|
|
WRITE_INT32_FIELD(this, OffsetOfElementAt(index), value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-03 15:32:22 +00:00
|
|
|
WriteBarrierMode HeapObject::GetWriteBarrierMode(
|
|
|
|
const DisallowHeapAllocation& promise) {
|
2011-09-19 18:36:47 +00:00
|
|
|
Heap* heap = GetHeap();
|
|
|
|
if (heap->incremental_marking()->IsMarking()) return UPDATE_WRITE_BARRIER;
|
|
|
|
if (heap->InNewSpace(this)) return SKIP_WRITE_BARRIER;
|
2008-07-03 15:10:15 +00:00
|
|
|
return UPDATE_WRITE_BARRIER;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FixedArray::set(int index,
|
|
|
|
Object* value,
|
2008-10-23 08:46:32 +00:00
|
|
|
WriteBarrierMode mode) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map());
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(index >= 0 && index < this->length());
|
|
|
|
int offset = kHeaderSize + index * kPointerSize;
|
|
|
|
WRITE_FIELD(this, offset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, offset, value, mode);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-06 12:11:08 +00:00
|
|
|
void FixedArray::NoIncrementalWriteBarrierSet(FixedArray* array,
|
|
|
|
int index,
|
|
|
|
Object* value) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(array->map() != array->GetHeap()->fixed_cow_array_map());
|
2011-12-06 12:11:08 +00:00
|
|
|
ASSERT(index >= 0 && index < array->length());
|
|
|
|
int offset = kHeaderSize + index * kPointerSize;
|
|
|
|
WRITE_FIELD(array, offset, value);
|
|
|
|
Heap* heap = array->GetHeap();
|
|
|
|
if (heap->InNewSpace(value)) {
|
|
|
|
heap->RecordWrite(array->address(), offset);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-21 10:32:38 +00:00
|
|
|
void FixedArray::NoWriteBarrierSet(FixedArray* array,
|
|
|
|
int index,
|
|
|
|
Object* value) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(array->map() != array->GetHeap()->fixed_cow_array_map());
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(index >= 0 && index < array->length());
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(!array->GetHeap()->InNewSpace(value));
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_FIELD(array, kHeaderSize + index * kPointerSize, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void FixedArray::set_undefined(int index) {
|
2013-08-28 14:26:22 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map());
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(index >= 0 && index < this->length());
|
2013-08-28 14:26:22 +00:00
|
|
|
ASSERT(!GetHeap()->InNewSpace(GetHeap()->undefined_value()));
|
|
|
|
WRITE_FIELD(this,
|
|
|
|
kHeaderSize + index * kPointerSize,
|
|
|
|
GetHeap()->undefined_value());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-09-18 11:18:27 +00:00
|
|
|
void FixedArray::set_null(int index) {
|
|
|
|
ASSERT(index >= 0 && index < this->length());
|
2013-08-28 14:26:22 +00:00
|
|
|
ASSERT(!GetHeap()->InNewSpace(GetHeap()->null_value()));
|
|
|
|
WRITE_FIELD(this,
|
|
|
|
kHeaderSize + index * kPointerSize,
|
|
|
|
GetHeap()->null_value());
|
2008-09-18 11:18:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void FixedArray::set_the_hole(int index) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(map() != GetHeap()->fixed_cow_array_map());
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(index >= 0 && index < this->length());
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(!GetHeap()->InNewSpace(GetHeap()->the_hole_value()));
|
2011-03-18 20:35:07 +00:00
|
|
|
WRITE_FIELD(this,
|
|
|
|
kHeaderSize + index * kPointerSize,
|
|
|
|
GetHeap()->the_hole_value());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-28 13:05:23 +00:00
|
|
|
void FixedArray::FillWithHoles(int from, int to) {
|
|
|
|
for (int i = from; i < to; i++) {
|
|
|
|
set_the_hole(i);
|
|
|
|
}
|
2012-11-15 12:19:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-01 15:33:30 +00:00
|
|
|
Object** FixedArray::data_start() {
|
|
|
|
return HeapObject::RawField(this, kHeaderSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-27 10:11:39 +00:00
|
|
|
bool DescriptorArray::IsEmpty() {
|
2012-06-18 12:42:46 +00:00
|
|
|
ASSERT(length() >= kFirstIndex ||
|
2013-09-11 07:14:41 +00:00
|
|
|
this == GetHeap()->empty_descriptor_array());
|
2012-06-18 12:42:46 +00:00
|
|
|
return length() < kFirstIndex;
|
2012-06-11 06:59:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-09-18 13:25:12 +00:00
|
|
|
void DescriptorArray::SetNumberOfDescriptors(int number_of_descriptors) {
|
|
|
|
WRITE_FIELD(
|
|
|
|
this, kDescriptorLengthOffset, Smi::FromInt(number_of_descriptors));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-05 13:54:20 +00:00
|
|
|
// Perform a binary search in a fixed array. Low and high are entry indices. If
|
|
|
|
// there are three entries in this array it should be called with low=0 and
|
|
|
|
// high=2.
|
2012-10-10 14:48:07 +00:00
|
|
|
template<SearchMode search_mode, typename T>
|
2013-03-04 15:00:57 +00:00
|
|
|
int BinarySearch(T* array, Name* name, int low, int high, int valid_entries) {
|
2012-07-05 13:54:20 +00:00
|
|
|
uint32_t hash = name->Hash();
|
|
|
|
int limit = high;
|
|
|
|
|
|
|
|
ASSERT(low <= high);
|
|
|
|
|
|
|
|
while (low != high) {
|
|
|
|
int mid = (low + high) / 2;
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* mid_name = array->GetSortedKey(mid);
|
2012-07-05 13:54:20 +00:00
|
|
|
uint32_t mid_hash = mid_name->Hash();
|
|
|
|
|
|
|
|
if (mid_hash >= hash) {
|
|
|
|
high = mid;
|
|
|
|
} else {
|
|
|
|
low = mid + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-27 13:47:34 +00:00
|
|
|
for (; low <= limit; ++low) {
|
|
|
|
int sort_index = array->GetSortedKeyIndex(low);
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* entry = array->GetKey(sort_index);
|
2012-08-27 13:47:34 +00:00
|
|
|
if (entry->Hash() != hash) break;
|
2012-10-10 14:48:07 +00:00
|
|
|
if (entry->Equals(name)) {
|
|
|
|
if (search_mode == ALL_ENTRIES || sort_index < valid_entries) {
|
|
|
|
return sort_index;
|
|
|
|
}
|
|
|
|
return T::kNotFound;
|
|
|
|
}
|
2012-07-05 13:54:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return T::kNotFound;
|
|
|
|
}
|
|
|
|
|
2012-10-10 14:48:07 +00:00
|
|
|
|
2012-07-05 13:54:20 +00:00
|
|
|
// Perform a linear search in this fixed array. len is the number of entry
|
|
|
|
// indices that are valid.
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
template<SearchMode search_mode, typename T>
|
2013-03-04 15:00:57 +00:00
|
|
|
int LinearSearch(T* array, Name* name, int len, int valid_entries) {
|
2012-07-05 13:54:20 +00:00
|
|
|
uint32_t hash = name->Hash();
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
if (search_mode == ALL_ENTRIES) {
|
|
|
|
for (int number = 0; number < len; number++) {
|
|
|
|
int sorted_index = array->GetSortedKeyIndex(number);
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* entry = array->GetKey(sorted_index);
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
uint32_t current_hash = entry->Hash();
|
|
|
|
if (current_hash > hash) break;
|
|
|
|
if (current_hash == hash && entry->Equals(name)) return sorted_index;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ASSERT(len >= valid_entries);
|
|
|
|
for (int number = 0; number < valid_entries; number++) {
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* entry = array->GetKey(number);
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
uint32_t current_hash = entry->Hash();
|
|
|
|
if (current_hash == hash && entry->Equals(name)) return number;
|
|
|
|
}
|
2012-07-05 13:54:20 +00:00
|
|
|
}
|
|
|
|
return T::kNotFound;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
template<SearchMode search_mode, typename T>
|
2013-03-04 15:00:57 +00:00
|
|
|
int Search(T* array, Name* name, int valid_entries) {
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
if (search_mode == VALID_ENTRIES) {
|
|
|
|
SLOW_ASSERT(array->IsSortedNoDuplicates(valid_entries));
|
|
|
|
} else {
|
|
|
|
SLOW_ASSERT(array->IsSortedNoDuplicates());
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-07-05 13:54:20 +00:00
|
|
|
int nof = array->number_of_entries();
|
|
|
|
if (nof == 0) return T::kNotFound;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Fast case: do linear search for small arrays.
|
|
|
|
const int kMaxElementsForLinearSearch = 8;
|
2012-10-10 14:48:07 +00:00
|
|
|
if ((search_mode == ALL_ENTRIES &&
|
|
|
|
nof <= kMaxElementsForLinearSearch) ||
|
|
|
|
(search_mode == VALID_ENTRIES &&
|
|
|
|
valid_entries <= (kMaxElementsForLinearSearch * 3))) {
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
return LinearSearch<search_mode>(array, name, nof, valid_entries);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Slow case: perform binary search.
|
2012-10-10 14:48:07 +00:00
|
|
|
return BinarySearch<search_mode>(array, name, 0, nof - 1, valid_entries);
|
2012-07-05 13:54:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-04 15:00:57 +00:00
|
|
|
int DescriptorArray::Search(Name* name, int valid_descriptors) {
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
return internal::Search<VALID_ENTRIES>(this, name, valid_descriptors);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-04 15:00:57 +00:00
|
|
|
int DescriptorArray::SearchWithCache(Name* name, Map* map) {
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
int number_of_own_descriptors = map->NumberOfOwnDescriptors();
|
|
|
|
if (number_of_own_descriptors == 0) return kNotFound;
|
2012-08-27 13:47:34 +00:00
|
|
|
|
2012-07-05 13:54:20 +00:00
|
|
|
DescriptorLookupCache* cache = GetIsolate()->descriptor_lookup_cache();
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
int number = cache->Lookup(map, name);
|
2012-08-27 13:47:34 +00:00
|
|
|
|
2010-08-12 14:51:59 +00:00
|
|
|
if (number == DescriptorLookupCache::kAbsent) {
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
number = Search(name, number_of_own_descriptors);
|
|
|
|
cache->Update(map, name, number);
|
2010-08-12 14:51:59 +00:00
|
|
|
}
|
2012-08-27 13:47:34 +00:00
|
|
|
|
2010-08-12 14:51:59 +00:00
|
|
|
return number;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-18 12:19:32 +00:00
|
|
|
PropertyDetails Map::GetLastDescriptorDetails() {
|
|
|
|
return instance_descriptors()->GetDetails(LastAdded());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-27 13:47:34 +00:00
|
|
|
void Map::LookupDescriptor(JSObject* holder,
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* name,
|
2012-08-27 13:47:34 +00:00
|
|
|
LookupResult* result) {
|
|
|
|
DescriptorArray* descriptors = this->instance_descriptors();
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
int number = descriptors->SearchWithCache(name, this);
|
2012-08-27 13:47:34 +00:00
|
|
|
if (number == DescriptorArray::kNotFound) return result->NotFound();
|
|
|
|
result->DescriptorResult(holder, descriptors->GetDetails(number), number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Map::LookupTransition(JSObject* holder,
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* name,
|
2012-08-27 13:47:34 +00:00
|
|
|
LookupResult* result) {
|
2014-04-04 04:49:07 +00:00
|
|
|
int transition_index = this->SearchTransition(name);
|
|
|
|
if (transition_index == TransitionArray::kNotFound) return result->NotFound();
|
|
|
|
result->TransitionResult(holder, this->GetTransition(transition_index));
|
2012-08-27 13:47:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-09 14:26:32 +00:00
|
|
|
FixedArrayBase* Map::GetInitialElements() {
|
|
|
|
if (has_fast_smi_or_object_elements() ||
|
|
|
|
has_fast_double_elements()) {
|
|
|
|
ASSERT(!GetHeap()->InNewSpace(GetHeap()->empty_fixed_array()));
|
|
|
|
return GetHeap()->empty_fixed_array();
|
|
|
|
} else if (has_external_array_elements()) {
|
|
|
|
ExternalArray* empty_array = GetHeap()->EmptyExternalArrayForMap(this);
|
|
|
|
ASSERT(!GetHeap()->InNewSpace(empty_array));
|
|
|
|
return empty_array;
|
|
|
|
} else if (has_fixed_typed_array_elements()) {
|
|
|
|
FixedTypedArrayBase* empty_array =
|
|
|
|
GetHeap()->EmptyFixedTypedArrayForMap(this);
|
|
|
|
ASSERT(!GetHeap()->InNewSpace(empty_array));
|
|
|
|
return empty_array;
|
|
|
|
} else {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-04 07:20:11 +00:00
|
|
|
Object** DescriptorArray::GetKeySlot(int descriptor_number) {
|
|
|
|
ASSERT(descriptor_number < number_of_descriptors());
|
2013-12-23 14:42:42 +00:00
|
|
|
return RawFieldOfElementAt(ToKeyIndex(descriptor_number));
|
2012-06-04 07:20:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-31 10:50:42 +00:00
|
|
|
Object** DescriptorArray::GetDescriptorStartSlot(int descriptor_number) {
|
|
|
|
return GetKeySlot(descriptor_number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object** DescriptorArray::GetDescriptorEndSlot(int descriptor_number) {
|
|
|
|
return GetValueSlot(descriptor_number - 1) + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* DescriptorArray::GetKey(int descriptor_number) {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(descriptor_number < number_of_descriptors());
|
2013-03-04 15:00:57 +00:00
|
|
|
return Name::cast(get(ToKeyIndex(descriptor_number)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-27 13:47:34 +00:00
|
|
|
int DescriptorArray::GetSortedKeyIndex(int descriptor_number) {
|
|
|
|
return GetDetails(descriptor_number).pointer();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* DescriptorArray::GetSortedKey(int descriptor_number) {
|
2012-08-27 13:47:34 +00:00
|
|
|
return GetKey(GetSortedKeyIndex(descriptor_number));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
void DescriptorArray::SetSortedKey(int descriptor_index, int pointer) {
|
|
|
|
PropertyDetails details = GetDetails(descriptor_index);
|
|
|
|
set(ToDetailsIndex(descriptor_index), details.set_pointer(pointer).AsSmi());
|
2012-08-27 13:47:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-26 15:30:41 +00:00
|
|
|
void DescriptorArray::SetRepresentation(int descriptor_index,
|
|
|
|
Representation representation) {
|
|
|
|
ASSERT(!representation.IsNone());
|
|
|
|
PropertyDetails details = GetDetails(descriptor_index);
|
|
|
|
set(ToDetailsIndex(descriptor_index),
|
|
|
|
details.CopyWithRepresentation(representation).AsSmi());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-30 15:15:17 +00:00
|
|
|
Object** DescriptorArray::GetValueSlot(int descriptor_number) {
|
|
|
|
ASSERT(descriptor_number < number_of_descriptors());
|
2013-12-23 14:42:42 +00:00
|
|
|
return RawFieldOfElementAt(ToValueIndex(descriptor_number));
|
2012-05-30 15:15:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* DescriptorArray::GetValue(int descriptor_number) {
|
|
|
|
ASSERT(descriptor_number < number_of_descriptors());
|
2012-06-04 07:20:11 +00:00
|
|
|
return get(ToValueIndex(descriptor_number));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-15 07:36:47 +00:00
|
|
|
void DescriptorArray::SetValue(int descriptor_index, Object* value) {
|
|
|
|
set(ToValueIndex(descriptor_index), value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-17 07:16:19 +00:00
|
|
|
PropertyDetails DescriptorArray::GetDetails(int descriptor_number) {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(descriptor_number < number_of_descriptors());
|
2012-06-04 07:20:11 +00:00
|
|
|
Object* details = get(ToDetailsIndex(descriptor_number));
|
2012-04-17 07:16:19 +00:00
|
|
|
return PropertyDetails(Smi::cast(details));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-10 19:25:18 +00:00
|
|
|
PropertyType DescriptorArray::GetType(int descriptor_number) {
|
2012-04-17 07:16:19 +00:00
|
|
|
return GetDetails(descriptor_number).type();
|
2009-07-10 19:25:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int DescriptorArray::GetFieldIndex(int descriptor_number) {
|
2013-08-22 13:43:06 +00:00
|
|
|
ASSERT(GetDetails(descriptor_number).type() == FIELD);
|
2013-05-31 19:11:09 +00:00
|
|
|
return GetDetails(descriptor_number).field_index();
|
2009-07-10 19:25:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-15 07:36:47 +00:00
|
|
|
HeapType* DescriptorArray::GetFieldType(int descriptor_number) {
|
|
|
|
ASSERT(GetDetails(descriptor_number).type() == FIELD);
|
|
|
|
return HeapType::cast(GetValue(descriptor_number));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-24 12:34:50 +00:00
|
|
|
Object* DescriptorArray::GetConstant(int descriptor_number) {
|
|
|
|
return GetValue(descriptor_number);
|
2009-07-10 19:25:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object* DescriptorArray::GetCallbacksObject(int descriptor_number) {
|
|
|
|
ASSERT(GetType(descriptor_number) == CALLBACKS);
|
|
|
|
return GetValue(descriptor_number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
AccessorDescriptor* DescriptorArray::GetCallbacks(int descriptor_number) {
|
|
|
|
ASSERT(GetType(descriptor_number) == CALLBACKS);
|
2011-05-19 11:47:34 +00:00
|
|
|
Foreign* p = Foreign::cast(GetCallbacksObject(descriptor_number));
|
2011-10-28 12:37:29 +00:00
|
|
|
return reinterpret_cast<AccessorDescriptor*>(p->foreign_address());
|
2009-07-10 19:25:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void DescriptorArray::Get(int descriptor_number, Descriptor* desc) {
|
2014-04-09 14:26:32 +00:00
|
|
|
desc->Init(handle(GetKey(descriptor_number), GetIsolate()),
|
|
|
|
handle(GetValue(descriptor_number), GetIsolate()),
|
2012-04-17 07:16:19 +00:00
|
|
|
GetDetails(descriptor_number));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-21 10:32:38 +00:00
|
|
|
void DescriptorArray::Set(int descriptor_number,
|
|
|
|
Descriptor* desc,
|
|
|
|
const WhitenessWitness&) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// Range check.
|
|
|
|
ASSERT(descriptor_number < number_of_descriptors());
|
|
|
|
|
2011-12-06 12:11:08 +00:00
|
|
|
NoIncrementalWriteBarrierSet(this,
|
|
|
|
ToKeyIndex(descriptor_number),
|
2014-04-09 14:26:32 +00:00
|
|
|
*desc->GetKey());
|
2012-06-04 07:20:11 +00:00
|
|
|
NoIncrementalWriteBarrierSet(this,
|
2011-12-06 12:11:08 +00:00
|
|
|
ToValueIndex(descriptor_number),
|
2014-04-09 14:26:32 +00:00
|
|
|
*desc->GetValue());
|
2012-06-04 07:20:11 +00:00
|
|
|
NoIncrementalWriteBarrierSet(this,
|
2011-12-06 12:11:08 +00:00
|
|
|
ToDetailsIndex(descriptor_number),
|
|
|
|
desc->GetDetails().AsSmi());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-09-18 13:25:12 +00:00
|
|
|
void DescriptorArray::Set(int descriptor_number, Descriptor* desc) {
|
|
|
|
// Range check.
|
|
|
|
ASSERT(descriptor_number < number_of_descriptors());
|
|
|
|
|
2014-04-09 14:26:32 +00:00
|
|
|
set(ToKeyIndex(descriptor_number), *desc->GetKey());
|
|
|
|
set(ToValueIndex(descriptor_number), *desc->GetValue());
|
2012-09-18 13:25:12 +00:00
|
|
|
set(ToDetailsIndex(descriptor_number), desc->GetDetails().AsSmi());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-27 13:47:34 +00:00
|
|
|
void DescriptorArray::Append(Descriptor* desc,
|
2012-09-18 13:25:12 +00:00
|
|
|
const WhitenessWitness& witness) {
|
2014-04-09 14:26:32 +00:00
|
|
|
DisallowHeapAllocation no_gc;
|
2012-09-18 13:25:12 +00:00
|
|
|
int descriptor_number = number_of_descriptors();
|
|
|
|
SetNumberOfDescriptors(descriptor_number + 1);
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
Set(descriptor_number, desc, witness);
|
2012-07-18 15:38:58 +00:00
|
|
|
|
|
|
|
uint32_t hash = desc->GetKey()->Hash();
|
|
|
|
|
2012-08-27 13:47:34 +00:00
|
|
|
int insertion;
|
|
|
|
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
for (insertion = descriptor_number; insertion > 0; --insertion) {
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* key = GetSortedKey(insertion - 1);
|
2012-07-18 15:38:58 +00:00
|
|
|
if (key->Hash() <= hash) break;
|
2012-08-27 13:47:34 +00:00
|
|
|
SetSortedKey(insertion, GetSortedKeyIndex(insertion - 1));
|
2012-07-18 15:38:58 +00:00
|
|
|
}
|
|
|
|
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
SetSortedKey(insertion, descriptor_number);
|
2012-07-12 15:14:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-09-18 13:25:12 +00:00
|
|
|
void DescriptorArray::Append(Descriptor* desc) {
|
2014-04-09 14:26:32 +00:00
|
|
|
DisallowHeapAllocation no_gc;
|
2012-09-18 13:25:12 +00:00
|
|
|
int descriptor_number = number_of_descriptors();
|
|
|
|
SetNumberOfDescriptors(descriptor_number + 1);
|
|
|
|
Set(descriptor_number, desc);
|
|
|
|
|
|
|
|
uint32_t hash = desc->GetKey()->Hash();
|
|
|
|
|
|
|
|
int insertion;
|
|
|
|
|
|
|
|
for (insertion = descriptor_number; insertion > 0; --insertion) {
|
2013-03-04 15:00:57 +00:00
|
|
|
Name* key = GetSortedKey(insertion - 1);
|
2012-09-18 13:25:12 +00:00
|
|
|
if (key->Hash() <= hash) break;
|
|
|
|
SetSortedKey(insertion, GetSortedKeyIndex(insertion - 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
SetSortedKey(insertion, descriptor_number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-27 13:47:34 +00:00
|
|
|
void DescriptorArray::SwapSortedKeys(int first, int second) {
|
|
|
|
int first_key = GetSortedKeyIndex(first);
|
|
|
|
SetSortedKey(first, GetSortedKeyIndex(second));
|
|
|
|
SetSortedKey(second, first_key);
|
2011-10-21 10:32:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 12:13:53 +00:00
|
|
|
DescriptorArray::WhitenessWitness::WhitenessWitness(DescriptorArray* array)
|
2011-10-21 10:32:38 +00:00
|
|
|
: marking_(array->GetHeap()->incremental_marking()) {
|
|
|
|
marking_->EnterNoMarkingScope();
|
2014-03-19 10:48:54 +00:00
|
|
|
ASSERT(!marking_->IsMarking() ||
|
|
|
|
Marking::Color(array) == Marking::WHITE_OBJECT);
|
2011-10-21 10:32:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-09-14 15:10:31 +00:00
|
|
|
DescriptorArray::WhitenessWitness::~WhitenessWitness() {
|
2011-10-21 10:32:38 +00:00
|
|
|
marking_->LeaveNoMarkingScope();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-14 15:56:57 +00:00
|
|
|
template<typename Derived, typename Shape, typename Key>
|
|
|
|
int HashTable<Derived, Shape, Key>::ComputeCapacity(int at_least_space_for) {
|
2011-07-26 13:56:21 +00:00
|
|
|
const int kMinCapacity = 32;
|
|
|
|
int capacity = RoundUpToPowerOf2(at_least_space_for * 2);
|
|
|
|
if (capacity < kMinCapacity) {
|
|
|
|
capacity = kMinCapacity; // Guarantee min capacity.
|
|
|
|
}
|
|
|
|
return capacity;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-14 15:56:57 +00:00
|
|
|
template<typename Derived, typename Shape, typename Key>
|
|
|
|
int HashTable<Derived, Shape, Key>::FindEntry(Key key) {
|
2011-03-18 20:35:07 +00:00
|
|
|
return FindEntry(GetIsolate(), key);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Find entry for key otherwise return kNotFound.
|
2014-04-14 15:56:57 +00:00
|
|
|
template<typename Derived, typename Shape, typename Key>
|
|
|
|
int HashTable<Derived, Shape, Key>::FindEntry(Isolate* isolate, Key key) {
|
2011-03-18 20:35:07 +00:00
|
|
|
uint32_t capacity = Capacity();
|
2014-04-14 15:56:57 +00:00
|
|
|
uint32_t entry = FirstProbe(HashTable::Hash(key), capacity);
|
2011-03-18 20:35:07 +00:00
|
|
|
uint32_t count = 1;
|
|
|
|
// EnsureCapacity will guarantee the hash table is never full.
|
|
|
|
while (true) {
|
|
|
|
Object* element = KeyAt(entry);
|
2012-10-25 13:54:10 +00:00
|
|
|
// Empty entry. Uses raw unchecked accessors because it is called by the
|
2013-02-28 17:03:34 +00:00
|
|
|
// string table during bootstrapping.
|
2012-10-25 13:54:10 +00:00
|
|
|
if (element == isolate->heap()->raw_unchecked_undefined_value()) break;
|
|
|
|
if (element != isolate->heap()->raw_unchecked_the_hole_value() &&
|
2011-03-18 20:35:07 +00:00
|
|
|
Shape::IsMatch(key, element)) return entry;
|
|
|
|
entry = NextProbe(entry, count++, capacity);
|
|
|
|
}
|
|
|
|
return kNotFound;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-16 09:44:35 +00:00
|
|
|
bool SeededNumberDictionary::requires_slow_elements() {
|
2008-10-22 09:46:09 +00:00
|
|
|
Object* max_index_object = get(kMaxNumberKeyIndex);
|
2008-07-03 15:10:15 +00:00
|
|
|
if (!max_index_object->IsSmi()) return false;
|
|
|
|
return 0 !=
|
|
|
|
(Smi::cast(max_index_object)->value() & kRequiresSlowElementsMask);
|
|
|
|
}
|
|
|
|
|
2012-01-16 09:44:35 +00:00
|
|
|
uint32_t SeededNumberDictionary::max_number_key() {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(!requires_slow_elements());
|
2008-10-22 09:46:09 +00:00
|
|
|
Object* max_index_object = get(kMaxNumberKeyIndex);
|
2008-07-03 15:10:15 +00:00
|
|
|
if (!max_index_object->IsSmi()) return 0;
|
|
|
|
uint32_t value = static_cast<uint32_t>(Smi::cast(max_index_object)->value());
|
|
|
|
return value >> kRequiresSlowElementsTagSize;
|
|
|
|
}
|
|
|
|
|
2012-01-16 09:44:35 +00:00
|
|
|
void SeededNumberDictionary::set_requires_slow_elements() {
|
2010-01-29 11:46:55 +00:00
|
|
|
set(kMaxNumberKeyIndex, Smi::FromInt(kRequiresSlowElementsMask));
|
2009-03-13 11:40:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// ------------------------------------
|
|
|
|
// Cast operations
|
|
|
|
|
|
|
|
|
|
|
|
CAST_ACCESSOR(FixedArray)
|
2011-06-09 10:03:35 +00:00
|
|
|
CAST_ACCESSOR(FixedDoubleArray)
|
2014-01-16 17:08:45 +00:00
|
|
|
CAST_ACCESSOR(FixedTypedArrayBase)
|
2013-10-14 13:35:06 +00:00
|
|
|
CAST_ACCESSOR(ConstantPoolArray)
|
2008-07-03 15:10:15 +00:00
|
|
|
CAST_ACCESSOR(DescriptorArray)
|
2010-12-07 11:31:57 +00:00
|
|
|
CAST_ACCESSOR(DeoptimizationInputData)
|
|
|
|
CAST_ACCESSOR(DeoptimizationOutputData)
|
2013-02-20 11:49:54 +00:00
|
|
|
CAST_ACCESSOR(DependentCode)
|
2013-02-28 17:03:34 +00:00
|
|
|
CAST_ACCESSOR(StringTable)
|
2010-05-04 16:42:11 +00:00
|
|
|
CAST_ACCESSOR(JSFunctionResultCache)
|
2010-08-25 13:25:54 +00:00
|
|
|
CAST_ACCESSOR(NormalizedMapCache)
|
2011-11-03 10:36:55 +00:00
|
|
|
CAST_ACCESSOR(ScopeInfo)
|
2008-09-11 10:51:52 +00:00
|
|
|
CAST_ACCESSOR(CompilationCacheTable)
|
2010-03-09 10:49:41 +00:00
|
|
|
CAST_ACCESSOR(CodeCacheHashTable)
|
2011-06-06 13:15:11 +00:00
|
|
|
CAST_ACCESSOR(PolymorphicCodeCacheHashTable)
|
2008-09-25 07:46:07 +00:00
|
|
|
CAST_ACCESSOR(MapCache)
|
2008-07-03 15:10:15 +00:00
|
|
|
CAST_ACCESSOR(String)
|
|
|
|
CAST_ACCESSOR(SeqString)
|
2012-11-15 13:31:27 +00:00
|
|
|
CAST_ACCESSOR(SeqOneByteString)
|
2008-10-09 08:08:04 +00:00
|
|
|
CAST_ACCESSOR(SeqTwoByteString)
|
2011-08-26 13:03:30 +00:00
|
|
|
CAST_ACCESSOR(SlicedString)
|
2008-07-03 15:10:15 +00:00
|
|
|
CAST_ACCESSOR(ConsString)
|
|
|
|
CAST_ACCESSOR(ExternalString)
|
|
|
|
CAST_ACCESSOR(ExternalAsciiString)
|
|
|
|
CAST_ACCESSOR(ExternalTwoByteString)
|
2013-03-01 10:34:31 +00:00
|
|
|
CAST_ACCESSOR(Symbol)
|
2013-03-04 15:00:57 +00:00
|
|
|
CAST_ACCESSOR(Name)
|
Implement set trap for proxies, and revamp class hierarchy in preparation:
- Introduce a class JSReceiver, that is a common superclass of JSObject and
JSProxy. Use JSReceiver where appropriate (probably lots of places that we
still have to migrate, but we will find those later with proxy test suite).
- Move appropriate methods to JSReceiver class (SetProperty,
GetPropertyAttribute, Get/SetPrototype, Lookup, and so on).
- Introduce new JSFunctionProxy subclass of JSProxy. Currently only a stub.
- Overhaul enum InstanceType:
* Introduce FIRST/LAST_SPEC_OBJECT_TYPE that ranges over all types that
represent JS objects, and use that consistently to check language types.
* Rename FIRST/LAST_JS_OBJECT_TYPE and FIRST/LAST_FUNCTION_CLASS_TYPE
to FIRST/LAST_[NON]CALLABLE_SPEC_OBJECT_TYPE for clarity.
* Eliminate the overlap over JS_REGEXP_TYPE.
* Also replace FIRST_JS_OBJECT with FIRST_JS_RECEIVER, but only use it where
we exclusively talk about the internal representation type.
* Insert JS_PROXY and JS_FUNCTION_PROXY in the appropriate places.
- Fix all checks concerning classification, especially for functions, to
use the CALLABLE_SPEC_OBJECT range (that includes funciton proxies).
- Handle proxies in SetProperty (that was the easiest part :) ).
- A few simple test cases.
R=kmillikin@chromium.org
Review URL: http://codereview.chromium.org/6992072
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8126 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-05-31 16:38:40 +00:00
|
|
|
CAST_ACCESSOR(JSReceiver)
|
2008-07-03 15:10:15 +00:00
|
|
|
CAST_ACCESSOR(JSObject)
|
|
|
|
CAST_ACCESSOR(Smi)
|
|
|
|
CAST_ACCESSOR(HeapObject)
|
|
|
|
CAST_ACCESSOR(HeapNumber)
|
|
|
|
CAST_ACCESSOR(Oddball)
|
2013-06-12 15:03:44 +00:00
|
|
|
CAST_ACCESSOR(Cell)
|
2013-06-14 16:06:12 +00:00
|
|
|
CAST_ACCESSOR(PropertyCell)
|
2008-07-03 15:10:15 +00:00
|
|
|
CAST_ACCESSOR(SharedFunctionInfo)
|
|
|
|
CAST_ACCESSOR(Map)
|
|
|
|
CAST_ACCESSOR(JSFunction)
|
2008-10-23 08:42:22 +00:00
|
|
|
CAST_ACCESSOR(GlobalObject)
|
Split window support from V8.
Here is a description of the background and design of split window in Chrome and V8:
https://docs.google.com/a/google.com/Doc?id=chhjkpg_47fwddxbfr
This change list splits the window object into two parts: 1) an inner window object used as the global object of contexts; 2) an outer window object exposed to JavaScript and accessible by the name 'window'. Firefox did it awhile ago, here are some discussions: https://wiki.mozilla.org/Gecko:SplitWindow. One additional benefit of splitting window in Chrome is that accessing global variables don't need security checks anymore, it can improve applications that use many global variables.
V8 support of split window:
There are a small number of changes on V8 api to support split window:
Security context is removed from V8, so does related API functions;
A global object can be detached from its context and reused by a new context;
Access checks on an object template can be turned on/off by default;
An object can turn on its access checks later;
V8 has a new object type, ApiGlobalObject, which is the outer window object type. The existing JSGlobalObject becomes the inner window object type. Security checks are moved from JSGlobalObject to ApiGlobalObject. ApiGlobalObject is the one exposed to JavaScript, it is accessible through Context::Global(). ApiGlobalObject's prototype is set to JSGlobalObject so that property lookups are forwarded to JSGlobalObject. ApiGlobalObject forwards all other property access requests to JSGlobalObject, such as SetProperty, DeleteProperty, etc.
Security token is moved to a global context, and ApiGlobalObject has a reference to its global context. JSGlobalObject has a reference to its global context as well. When accessing properties on a global object in JavaScript, the domain security check is performed by comparing the security token of the lexical context (Top::global_context()) to the token of global object's context. The check is only needed when the receiver is a window object, such as 'window.document'. Accessing global variables, such as 'var foo = 3; foo' does not need checks because the receiver is the inner window object.
When an outer window is detached from its global context (when a frame navigates away from a page), it is completely detached from the inner window. A new context is created for the new page, and the outer global object is reused. At this point, the access check on the DOMWindow wrapper of the old context is turned on. The code in old context is still able to access DOMWindow properties, but it has to go through domain security checks.
It is debatable on how to implement the outer window object. Currently each property access function has to check if the receiver is ApiGlobalObject type. This approach might be error-prone that one may forget to check the receiver when adding new functions. It is unlikely a performance issue because accessing global variables are more common than 'window.foo' style coding.
I am still working on the ARM port, and I'd like to hear comments and suggestions on the best way to support it in V8.
Review URL: http://codereview.chromium.org/7366
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@540 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2008-10-21 19:07:58 +00:00
|
|
|
CAST_ACCESSOR(JSGlobalProxy)
|
2008-07-03 15:10:15 +00:00
|
|
|
CAST_ACCESSOR(JSGlobalObject)
|
|
|
|
CAST_ACCESSOR(JSBuiltinsObject)
|
|
|
|
CAST_ACCESSOR(Code)
|
|
|
|
CAST_ACCESSOR(JSArray)
|
2013-03-28 12:50:18 +00:00
|
|
|
CAST_ACCESSOR(JSArrayBuffer)
|
2013-06-21 13:02:38 +00:00
|
|
|
CAST_ACCESSOR(JSArrayBufferView)
|
2013-04-16 14:16:30 +00:00
|
|
|
CAST_ACCESSOR(JSTypedArray)
|
2013-06-21 13:02:38 +00:00
|
|
|
CAST_ACCESSOR(JSDataView)
|
2008-09-23 11:45:43 +00:00
|
|
|
CAST_ACCESSOR(JSRegExp)
|
2011-05-13 10:58:25 +00:00
|
|
|
CAST_ACCESSOR(JSProxy)
|
Implement set trap for proxies, and revamp class hierarchy in preparation:
- Introduce a class JSReceiver, that is a common superclass of JSObject and
JSProxy. Use JSReceiver where appropriate (probably lots of places that we
still have to migrate, but we will find those later with proxy test suite).
- Move appropriate methods to JSReceiver class (SetProperty,
GetPropertyAttribute, Get/SetPrototype, Lookup, and so on).
- Introduce new JSFunctionProxy subclass of JSProxy. Currently only a stub.
- Overhaul enum InstanceType:
* Introduce FIRST/LAST_SPEC_OBJECT_TYPE that ranges over all types that
represent JS objects, and use that consistently to check language types.
* Rename FIRST/LAST_JS_OBJECT_TYPE and FIRST/LAST_FUNCTION_CLASS_TYPE
to FIRST/LAST_[NON]CALLABLE_SPEC_OBJECT_TYPE for clarity.
* Eliminate the overlap over JS_REGEXP_TYPE.
* Also replace FIRST_JS_OBJECT with FIRST_JS_RECEIVER, but only use it where
we exclusively talk about the internal representation type.
* Insert JS_PROXY and JS_FUNCTION_PROXY in the appropriate places.
- Fix all checks concerning classification, especially for functions, to
use the CALLABLE_SPEC_OBJECT range (that includes funciton proxies).
- Handle proxies in SetProperty (that was the easiest part :) ).
- A few simple test cases.
R=kmillikin@chromium.org
Review URL: http://codereview.chromium.org/6992072
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8126 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-05-31 16:38:40 +00:00
|
|
|
CAST_ACCESSOR(JSFunctionProxy)
|
2011-10-25 14:14:56 +00:00
|
|
|
CAST_ACCESSOR(JSSet)
|
|
|
|
CAST_ACCESSOR(JSMap)
|
2014-04-17 17:45:32 +00:00
|
|
|
CAST_ACCESSOR(JSSetIterator)
|
|
|
|
CAST_ACCESSOR(JSMapIterator)
|
2011-08-03 11:55:13 +00:00
|
|
|
CAST_ACCESSOR(JSWeakMap)
|
2013-07-22 08:32:24 +00:00
|
|
|
CAST_ACCESSOR(JSWeakSet)
|
2011-05-19 11:47:34 +00:00
|
|
|
CAST_ACCESSOR(Foreign)
|
2008-07-03 15:10:15 +00:00
|
|
|
CAST_ACCESSOR(ByteArray)
|
2011-09-19 18:36:47 +00:00
|
|
|
CAST_ACCESSOR(FreeSpace)
|
2009-10-20 15:26:17 +00:00
|
|
|
CAST_ACCESSOR(ExternalArray)
|
2014-01-24 16:01:15 +00:00
|
|
|
CAST_ACCESSOR(ExternalInt8Array)
|
|
|
|
CAST_ACCESSOR(ExternalUint8Array)
|
|
|
|
CAST_ACCESSOR(ExternalInt16Array)
|
|
|
|
CAST_ACCESSOR(ExternalUint16Array)
|
|
|
|
CAST_ACCESSOR(ExternalInt32Array)
|
|
|
|
CAST_ACCESSOR(ExternalUint32Array)
|
|
|
|
CAST_ACCESSOR(ExternalFloat32Array)
|
|
|
|
CAST_ACCESSOR(ExternalFloat64Array)
|
|
|
|
CAST_ACCESSOR(ExternalUint8ClampedArray)
|
2008-07-03 15:10:15 +00:00
|
|
|
CAST_ACCESSOR(Struct)
|
2013-02-12 14:33:08 +00:00
|
|
|
CAST_ACCESSOR(AccessorInfo)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2014-01-16 17:08:45 +00:00
|
|
|
template <class Traits>
|
|
|
|
FixedTypedArray<Traits>* FixedTypedArray<Traits>::cast(Object* object) {
|
|
|
|
SLOW_ASSERT(object->IsHeapObject() &&
|
|
|
|
HeapObject::cast(object)->map()->instance_type() ==
|
|
|
|
Traits::kInstanceType);
|
|
|
|
return reinterpret_cast<FixedTypedArray<Traits>*>(object);
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
#define MAKE_STRUCT_CAST(NAME, Name, name) CAST_ACCESSOR(Name)
|
|
|
|
STRUCT_LIST(MAKE_STRUCT_CAST)
|
|
|
|
#undef MAKE_STRUCT_CAST
|
|
|
|
|
2009-07-02 06:50:43 +00:00
|
|
|
|
2014-04-14 15:56:57 +00:00
|
|
|
template <typename Derived, typename Shape, typename Key>
|
|
|
|
HashTable<Derived, Shape, Key>*
|
|
|
|
HashTable<Derived, Shape, Key>::cast(Object* obj) {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(obj->IsHashTable());
|
|
|
|
return reinterpret_cast<HashTable*>(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-09 10:03:35 +00:00
|
|
|
SMI_ACCESSORS(FixedArrayBase, length, kLengthOffset)
|
2014-03-31 14:29:01 +00:00
|
|
|
SYNCHRONIZED_SMI_ACCESSORS(FixedArrayBase, length, kLengthOffset)
|
|
|
|
|
2011-09-19 18:36:47 +00:00
|
|
|
SMI_ACCESSORS(FreeSpace, size, kSizeOffset)
|
2014-04-08 16:31:57 +00:00
|
|
|
NOBARRIER_SMI_ACCESSORS(FreeSpace, size, kSizeOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-05-04 14:49:50 +00:00
|
|
|
SMI_ACCESSORS(String, length, kLengthOffset)
|
2014-03-31 14:29:01 +00:00
|
|
|
SYNCHRONIZED_SMI_ACCESSORS(String, length, kLengthOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2008-10-27 12:39:34 +00:00
|
|
|
|
2013-03-01 10:34:31 +00:00
|
|
|
uint32_t Name::hash_field() {
|
2009-11-24 14:10:06 +00:00
|
|
|
return READ_UINT32_FIELD(this, kHashFieldOffset);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-01 10:34:31 +00:00
|
|
|
void Name::set_hash_field(uint32_t value) {
|
2009-11-24 14:10:06 +00:00
|
|
|
WRITE_UINT32_FIELD(this, kHashFieldOffset, value);
|
2010-05-27 12:30:45 +00:00
|
|
|
#if V8_HOST_ARCH_64_BIT
|
|
|
|
WRITE_UINT32_FIELD(this, kHashFieldOffset + kIntSize, 0);
|
|
|
|
#endif
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-04 15:00:57 +00:00
|
|
|
bool Name::Equals(Name* other) {
|
|
|
|
if (other == this) return true;
|
2013-04-10 09:53:43 +00:00
|
|
|
if ((this->IsInternalizedString() && other->IsInternalizedString()) ||
|
|
|
|
this->IsSymbol() || other->IsSymbol()) {
|
2013-04-03 17:06:22 +00:00
|
|
|
return false;
|
|
|
|
}
|
2013-03-04 15:00:57 +00:00
|
|
|
return String::cast(this)->SlowEquals(String::cast(other));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 07:27:25 +00:00
|
|
|
bool Name::Equals(Handle<Name> one, Handle<Name> two) {
|
|
|
|
if (one.is_identical_to(two)) return true;
|
|
|
|
if ((one->IsInternalizedString() && two->IsInternalizedString()) ||
|
|
|
|
one->IsSymbol() || two->IsSymbol()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return String::SlowEquals(Handle<String>::cast(one),
|
|
|
|
Handle<String>::cast(two));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-22 16:51:28 +00:00
|
|
|
ACCESSORS(Symbol, name, Object, kNameOffset)
|
Provide private symbols through internal APIs
Adds a notion of private symbols, mainly intended for internal use, especially, self-hosting of built-in types that would otherwise require new C++ classes.
On the JS side (i.e., in built-ins), private properties can be created and accessed through a set of macros:
NEW_PRIVATE(print_name)
HAS_PRIVATE(obj, sym)
GET_PRIVATE(obj, sym)
SET_PRIVATE(obj, sym, val)
DELETE_PRIVATE(obj, sym)
In the V8 API, they are accessible via a new class Private, and respective HasPrivate/Get/Private/SetPrivate/DeletePrivate methods on calss Object.
These APIs are designed and restricted such that their implementation can later be replaced by whatever ES7+ will officially provide.
R=yangguo@chromium.org
BUG=
Review URL: https://codereview.chromium.org/48923002
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@17683 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2013-11-13 10:34:06 +00:00
|
|
|
ACCESSORS(Symbol, flags, Smi, kFlagsOffset)
|
|
|
|
BOOL_ACCESSORS(Symbol, flags, is_private, kPrivateBit)
|
2013-03-22 16:51:28 +00:00
|
|
|
|
|
|
|
|
2009-11-24 14:10:06 +00:00
|
|
|
bool String::Equals(String* other) {
|
|
|
|
if (other == this) return true;
|
2013-03-04 15:00:57 +00:00
|
|
|
if (this->IsInternalizedString() && other->IsInternalizedString()) {
|
2009-11-24 14:10:06 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SlowEquals(other);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 07:27:25 +00:00
|
|
|
bool String::Equals(Handle<String> one, Handle<String> two) {
|
|
|
|
if (one.is_identical_to(two)) return true;
|
|
|
|
if (one->IsInternalizedString() && two->IsInternalizedString()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return SlowEquals(one, two);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 09:49:49 +00:00
|
|
|
Handle<String> String::Flatten(Handle<String> string, PretenureFlag pretenure) {
|
|
|
|
if (!string->IsConsString()) return string;
|
|
|
|
Handle<ConsString> cons = Handle<ConsString>::cast(string);
|
|
|
|
if (cons->IsFlat()) return handle(cons->first());
|
|
|
|
return SlowFlatten(cons, pretenure);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-17 09:33:06 +00:00
|
|
|
uint16_t String::Get(int index) {
|
|
|
|
ASSERT(index >= 0 && index < length());
|
|
|
|
switch (StringShape(this).full_representation_tag()) {
|
2012-11-08 12:14:29 +00:00
|
|
|
case kSeqStringTag | kOneByteStringTag:
|
2012-11-15 13:31:27 +00:00
|
|
|
return SeqOneByteString::cast(this)->SeqOneByteStringGet(index);
|
2008-11-03 10:16:05 +00:00
|
|
|
case kSeqStringTag | kTwoByteStringTag:
|
|
|
|
return SeqTwoByteString::cast(this)->SeqTwoByteStringGet(index);
|
2012-11-08 12:14:29 +00:00
|
|
|
case kConsStringTag | kOneByteStringTag:
|
2008-11-03 10:16:05 +00:00
|
|
|
case kConsStringTag | kTwoByteStringTag:
|
2008-07-03 15:10:15 +00:00
|
|
|
return ConsString::cast(this)->ConsStringGet(index);
|
2012-11-08 12:14:29 +00:00
|
|
|
case kExternalStringTag | kOneByteStringTag:
|
2008-11-03 10:16:05 +00:00
|
|
|
return ExternalAsciiString::cast(this)->ExternalAsciiStringGet(index);
|
|
|
|
case kExternalStringTag | kTwoByteStringTag:
|
|
|
|
return ExternalTwoByteString::cast(this)->ExternalTwoByteStringGet(index);
|
2012-11-08 12:14:29 +00:00
|
|
|
case kSlicedStringTag | kOneByteStringTag:
|
2011-08-26 13:03:30 +00:00
|
|
|
case kSlicedStringTag | kTwoByteStringTag:
|
|
|
|
return SlicedString::cast(this)->SlicedStringGet(index);
|
2008-07-03 15:10:15 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
UNREACHABLE();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-17 09:33:06 +00:00
|
|
|
void String::Set(int index, uint16_t value) {
|
|
|
|
ASSERT(index >= 0 && index < length());
|
|
|
|
ASSERT(StringShape(this).IsSequential());
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-11-21 10:01:05 +00:00
|
|
|
return this->IsOneByteRepresentation()
|
2012-11-15 13:31:27 +00:00
|
|
|
? SeqOneByteString::cast(this)->SeqOneByteStringSet(index, value)
|
2008-10-09 08:08:04 +00:00
|
|
|
: SeqTwoByteString::cast(this)->SeqTwoByteStringSet(index, value);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-17 09:33:06 +00:00
|
|
|
bool String::IsFlat() {
|
2011-08-26 13:03:30 +00:00
|
|
|
if (!StringShape(this).IsCons()) return true;
|
|
|
|
return ConsString::cast(this)->second()->length() == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String* String::GetUnderlying() {
|
|
|
|
// Giving direct access to underlying string only makes sense if the
|
|
|
|
// wrapping string is already flattened.
|
|
|
|
ASSERT(this->IsFlat());
|
|
|
|
ASSERT(StringShape(this).IsIndirect());
|
|
|
|
STATIC_ASSERT(ConsString::kFirstOffset == SlicedString::kParentOffset);
|
|
|
|
const int kUnderlyingOffset = SlicedString::kParentOffset;
|
|
|
|
return String::cast(READ_FIELD(this, kUnderlyingOffset));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-29 13:09:31 +00:00
|
|
|
template<class Visitor>
|
|
|
|
ConsString* String::VisitFlat(Visitor* visitor,
|
|
|
|
String* string,
|
|
|
|
const int offset) {
|
|
|
|
int slice_offset = offset;
|
|
|
|
const int length = string->length();
|
2012-12-06 11:49:15 +00:00
|
|
|
ASSERT(offset <= length);
|
|
|
|
while (true) {
|
2014-04-29 13:09:31 +00:00
|
|
|
int32_t type = string->map()->instance_type();
|
2012-12-06 11:49:15 +00:00
|
|
|
switch (type & (kStringRepresentationMask | kStringEncodingMask)) {
|
|
|
|
case kSeqStringTag | kOneByteStringTag:
|
2014-04-29 13:09:31 +00:00
|
|
|
visitor->VisitOneByteString(
|
2013-01-09 15:47:53 +00:00
|
|
|
SeqOneByteString::cast(string)->GetChars() + slice_offset,
|
|
|
|
length - offset);
|
2014-04-29 13:09:31 +00:00
|
|
|
return NULL;
|
2012-12-06 11:49:15 +00:00
|
|
|
|
|
|
|
case kSeqStringTag | kTwoByteStringTag:
|
2014-04-29 13:09:31 +00:00
|
|
|
visitor->VisitTwoByteString(
|
2013-01-09 15:47:53 +00:00
|
|
|
SeqTwoByteString::cast(string)->GetChars() + slice_offset,
|
|
|
|
length - offset);
|
2014-04-29 13:09:31 +00:00
|
|
|
return NULL;
|
2012-12-06 11:49:15 +00:00
|
|
|
|
|
|
|
case kExternalStringTag | kOneByteStringTag:
|
2014-04-29 13:09:31 +00:00
|
|
|
visitor->VisitOneByteString(
|
2013-01-09 15:47:53 +00:00
|
|
|
ExternalAsciiString::cast(string)->GetChars() + slice_offset,
|
|
|
|
length - offset);
|
2014-04-29 13:09:31 +00:00
|
|
|
return NULL;
|
2012-12-06 11:49:15 +00:00
|
|
|
|
|
|
|
case kExternalStringTag | kTwoByteStringTag:
|
2014-04-29 13:09:31 +00:00
|
|
|
visitor->VisitTwoByteString(
|
2013-01-09 15:47:53 +00:00
|
|
|
ExternalTwoByteString::cast(string)->GetChars() + slice_offset,
|
|
|
|
length - offset);
|
2014-04-29 13:09:31 +00:00
|
|
|
return NULL;
|
2012-12-06 11:49:15 +00:00
|
|
|
|
|
|
|
case kSlicedStringTag | kOneByteStringTag:
|
|
|
|
case kSlicedStringTag | kTwoByteStringTag: {
|
|
|
|
SlicedString* slicedString = SlicedString::cast(string);
|
2012-12-11 10:22:15 +00:00
|
|
|
slice_offset += slicedString->offset();
|
2012-12-06 11:49:15 +00:00
|
|
|
string = slicedString->parent();
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
case kConsStringTag | kOneByteStringTag:
|
|
|
|
case kConsStringTag | kTwoByteStringTag:
|
2014-04-29 13:09:31 +00:00
|
|
|
return ConsString::cast(string);
|
2012-12-06 11:49:15 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
2014-04-29 13:09:31 +00:00
|
|
|
return NULL;
|
2012-12-06 11:49:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-15 13:31:27 +00:00
|
|
|
uint16_t SeqOneByteString::SeqOneByteStringGet(int index) {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(index >= 0 && index < length());
|
|
|
|
return READ_BYTE_FIELD(this, kHeaderSize + index * kCharSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-15 13:31:27 +00:00
|
|
|
void SeqOneByteString::SeqOneByteStringSet(int index, uint16_t value) {
|
2013-01-09 10:30:54 +00:00
|
|
|
ASSERT(index >= 0 && index < length() && value <= kMaxOneByteCharCode);
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_BYTE_FIELD(this, kHeaderSize + index * kCharSize,
|
|
|
|
static_cast<byte>(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-15 13:31:27 +00:00
|
|
|
Address SeqOneByteString::GetCharsAddress() {
|
2008-07-03 15:10:15 +00:00
|
|
|
return FIELD_ADDR(this, kHeaderSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-09 15:47:53 +00:00
|
|
|
uint8_t* SeqOneByteString::GetChars() {
|
2013-01-09 10:30:54 +00:00
|
|
|
return reinterpret_cast<uint8_t*>(GetCharsAddress());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-09 08:08:04 +00:00
|
|
|
Address SeqTwoByteString::GetCharsAddress() {
|
2008-10-07 08:11:44 +00:00
|
|
|
return FIELD_ADDR(this, kHeaderSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-22 09:09:07 +00:00
|
|
|
uc16* SeqTwoByteString::GetChars() {
|
|
|
|
return reinterpret_cast<uc16*>(FIELD_ADDR(this, kHeaderSize));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-09 08:08:04 +00:00
|
|
|
uint16_t SeqTwoByteString::SeqTwoByteStringGet(int index) {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(index >= 0 && index < length());
|
|
|
|
return READ_SHORT_FIELD(this, kHeaderSize + index * kShortSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-09 08:08:04 +00:00
|
|
|
void SeqTwoByteString::SeqTwoByteStringSet(int index, uint16_t value) {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(index >= 0 && index < length());
|
|
|
|
WRITE_SHORT_FIELD(this, kHeaderSize + index * kShortSize, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-17 09:33:06 +00:00
|
|
|
int SeqTwoByteString::SeqTwoByteStringSize(InstanceType instance_type) {
|
2010-05-04 14:49:50 +00:00
|
|
|
return SizeFor(length());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-15 13:31:27 +00:00
|
|
|
int SeqOneByteString::SeqOneByteStringSize(InstanceType instance_type) {
|
2010-05-04 14:49:50 +00:00
|
|
|
return SizeFor(length());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-26 13:03:30 +00:00
|
|
|
String* SlicedString::parent() {
|
|
|
|
return String::cast(READ_FIELD(this, kParentOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-18 13:39:53 +00:00
|
|
|
void SlicedString::set_parent(String* parent, WriteBarrierMode mode) {
|
2011-09-15 12:47:06 +00:00
|
|
|
ASSERT(parent->IsSeqString() || parent->IsExternalString());
|
2011-08-26 13:03:30 +00:00
|
|
|
WRITE_FIELD(this, kParentOffset, parent);
|
2012-07-18 13:39:53 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kParentOffset, parent, mode);
|
2011-08-26 13:03:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SMI_ACCESSORS(SlicedString, offset, kOffsetOffset)
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
String* ConsString::first() {
|
|
|
|
return String::cast(READ_FIELD(this, kFirstOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object* ConsString::unchecked_first() {
|
2008-07-03 15:10:15 +00:00
|
|
|
return READ_FIELD(this, kFirstOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
void ConsString::set_first(String* value, WriteBarrierMode mode) {
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_FIELD(this, kFirstOffset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kFirstOffset, value, mode);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
String* ConsString::second() {
|
|
|
|
return String::cast(READ_FIELD(this, kSecondOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object* ConsString::unchecked_second() {
|
2008-07-03 15:10:15 +00:00
|
|
|
return READ_FIELD(this, kSecondOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-03 10:16:05 +00:00
|
|
|
void ConsString::set_second(String* value, WriteBarrierMode mode) {
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_FIELD(this, kSecondOffset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kSecondOffset, value, mode);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-23 13:08:28 +00:00
|
|
|
bool ExternalString::is_short() {
|
|
|
|
InstanceType type = map()->instance_type();
|
|
|
|
return (type & kShortExternalStringMask) == kShortExternalStringTag;
|
2011-11-17 17:05:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-21 13:28:09 +00:00
|
|
|
const ExternalAsciiString::Resource* ExternalAsciiString::resource() {
|
2008-07-03 15:10:15 +00:00
|
|
|
return *reinterpret_cast<Resource**>(FIELD_ADDR(this, kResourceOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-23 13:08:28 +00:00
|
|
|
void ExternalAsciiString::update_data_cache() {
|
|
|
|
if (is_short()) return;
|
|
|
|
const char** data_field =
|
|
|
|
reinterpret_cast<const char**>(FIELD_ADDR(this, kResourceDataOffset));
|
|
|
|
*data_field = resource()->data();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void ExternalAsciiString::set_resource(
|
2011-09-21 13:28:09 +00:00
|
|
|
const ExternalAsciiString::Resource* resource) {
|
2014-02-05 09:29:04 +00:00
|
|
|
ASSERT(IsAligned(reinterpret_cast<intptr_t>(resource), kPointerSize));
|
2011-09-21 13:28:09 +00:00
|
|
|
*reinterpret_cast<const Resource**>(
|
|
|
|
FIELD_ADDR(this, kResourceOffset)) = resource;
|
2011-11-23 13:08:28 +00:00
|
|
|
if (resource != NULL) update_data_cache();
|
2011-11-17 17:05:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-09 15:47:53 +00:00
|
|
|
const uint8_t* ExternalAsciiString::GetChars() {
|
|
|
|
return reinterpret_cast<const uint8_t*>(resource()->data());
|
2011-11-17 17:05:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint16_t ExternalAsciiString::ExternalAsciiStringGet(int index) {
|
|
|
|
ASSERT(index >= 0 && index < length());
|
|
|
|
return GetChars()[index];
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-21 13:28:09 +00:00
|
|
|
const ExternalTwoByteString::Resource* ExternalTwoByteString::resource() {
|
2008-07-03 15:10:15 +00:00
|
|
|
return *reinterpret_cast<Resource**>(FIELD_ADDR(this, kResourceOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-23 13:08:28 +00:00
|
|
|
void ExternalTwoByteString::update_data_cache() {
|
|
|
|
if (is_short()) return;
|
|
|
|
const uint16_t** data_field =
|
|
|
|
reinterpret_cast<const uint16_t**>(FIELD_ADDR(this, kResourceDataOffset));
|
|
|
|
*data_field = resource()->data();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void ExternalTwoByteString::set_resource(
|
2011-09-21 13:28:09 +00:00
|
|
|
const ExternalTwoByteString::Resource* resource) {
|
|
|
|
*reinterpret_cast<const Resource**>(
|
|
|
|
FIELD_ADDR(this, kResourceOffset)) = resource;
|
2011-11-23 13:08:28 +00:00
|
|
|
if (resource != NULL) update_data_cache();
|
2011-11-17 17:05:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const uint16_t* ExternalTwoByteString::GetChars() {
|
2011-11-23 13:08:28 +00:00
|
|
|
return resource()->data();
|
2011-11-17 17:05:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint16_t ExternalTwoByteString::ExternalTwoByteStringGet(int index) {
|
|
|
|
ASSERT(index >= 0 && index < length());
|
|
|
|
return GetChars()[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const uint16_t* ExternalTwoByteString::ExternalTwoByteStringGetData(
|
|
|
|
unsigned start) {
|
|
|
|
return GetChars() + start;
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-29 13:09:31 +00:00
|
|
|
int ConsStringIteratorOp::OffsetForDepth(int depth) {
|
2012-12-06 11:49:15 +00:00
|
|
|
return depth & kDepthMask;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ConsStringIteratorOp::PushLeft(ConsString* string) {
|
|
|
|
frames_[depth_++ & kDepthMask] = string;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-11 10:22:15 +00:00
|
|
|
void ConsStringIteratorOp::PushRight(ConsString* string) {
|
|
|
|
// Inplace update.
|
2012-12-06 11:49:15 +00:00
|
|
|
frames_[(depth_-1) & kDepthMask] = string;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ConsStringIteratorOp::AdjustMaximumDepth() {
|
|
|
|
if (depth_ > maximum_depth_) maximum_depth_ = depth_;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ConsStringIteratorOp::Pop() {
|
|
|
|
ASSERT(depth_ > 0);
|
|
|
|
ASSERT(depth_ <= maximum_depth_);
|
|
|
|
depth_--;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint16_t StringCharacterStream::GetNext() {
|
2012-12-31 11:13:50 +00:00
|
|
|
ASSERT(buffer8_ != NULL && end_ != NULL);
|
|
|
|
// Advance cursor if needed.
|
|
|
|
if (buffer8_ == end_) HasMore();
|
|
|
|
ASSERT(buffer8_ < end_);
|
2012-12-06 11:49:15 +00:00
|
|
|
return is_one_byte_ ? *buffer8_++ : *buffer16_++;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-31 11:13:50 +00:00
|
|
|
StringCharacterStream::StringCharacterStream(String* string,
|
|
|
|
ConsStringIteratorOp* op,
|
2014-04-29 13:09:31 +00:00
|
|
|
int offset)
|
2012-12-11 10:22:15 +00:00
|
|
|
: is_one_byte_(false),
|
2012-12-06 11:49:15 +00:00
|
|
|
op_(op) {
|
2012-12-31 11:13:50 +00:00
|
|
|
Reset(string, offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-29 13:09:31 +00:00
|
|
|
void StringCharacterStream::Reset(String* string, int offset) {
|
2012-12-31 11:13:50 +00:00
|
|
|
buffer8_ = NULL;
|
|
|
|
end_ = NULL;
|
2014-04-29 13:09:31 +00:00
|
|
|
ConsString* cons_string = String::VisitFlat(this, string, offset);
|
|
|
|
op_->Reset(cons_string, offset);
|
|
|
|
if (cons_string != NULL) {
|
|
|
|
string = op_->Next(&offset);
|
|
|
|
if (string != NULL) String::VisitFlat(this, string, offset);
|
|
|
|
}
|
2012-12-06 11:49:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool StringCharacterStream::HasMore() {
|
|
|
|
if (buffer8_ != end_) return true;
|
2014-04-29 13:09:31 +00:00
|
|
|
int offset;
|
|
|
|
String* string = op_->Next(&offset);
|
|
|
|
ASSERT_EQ(offset, 0);
|
2012-12-19 13:27:20 +00:00
|
|
|
if (string == NULL) return false;
|
2014-04-29 13:09:31 +00:00
|
|
|
String::VisitFlat(this, string);
|
2012-12-19 13:27:20 +00:00
|
|
|
ASSERT(buffer8_ != end_);
|
2012-12-06 11:49:15 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void StringCharacterStream::VisitOneByteString(
|
2014-04-29 13:09:31 +00:00
|
|
|
const uint8_t* chars, int length) {
|
2012-12-06 11:49:15 +00:00
|
|
|
is_one_byte_ = true;
|
|
|
|
buffer8_ = chars;
|
|
|
|
end_ = chars + length;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void StringCharacterStream::VisitTwoByteString(
|
2014-04-29 13:09:31 +00:00
|
|
|
const uint16_t* chars, int length) {
|
2012-12-06 11:49:15 +00:00
|
|
|
is_one_byte_ = false;
|
|
|
|
buffer16_ = chars;
|
|
|
|
end_ = reinterpret_cast<const uint8_t*>(chars + length);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-04 16:42:11 +00:00
|
|
|
void JSFunctionResultCache::MakeZeroSize() {
|
2011-01-17 16:54:56 +00:00
|
|
|
set_finger_index(kEntriesIndex);
|
|
|
|
set_size(kEntriesIndex);
|
2010-05-04 16:42:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSFunctionResultCache::Clear() {
|
2011-01-17 16:54:56 +00:00
|
|
|
int cache_size = size();
|
2013-12-23 14:42:42 +00:00
|
|
|
Object** entries_start = RawFieldOfElementAt(kEntriesIndex);
|
2010-10-29 08:13:19 +00:00
|
|
|
MemsetPointer(entries_start,
|
2011-03-18 20:35:07 +00:00
|
|
|
GetHeap()->the_hole_value(),
|
2010-10-29 08:13:19 +00:00
|
|
|
cache_size - kEntriesIndex);
|
2010-05-04 16:42:11 +00:00
|
|
|
MakeZeroSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-17 16:54:56 +00:00
|
|
|
int JSFunctionResultCache::size() {
|
|
|
|
return Smi::cast(get(kCacheSizeIndex))->value();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSFunctionResultCache::set_size(int size) {
|
|
|
|
set(kCacheSizeIndex, Smi::FromInt(size));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int JSFunctionResultCache::finger_index() {
|
|
|
|
return Smi::cast(get(kFingerIndex))->value();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSFunctionResultCache::set_finger_index(int finger_index) {
|
|
|
|
set(kFingerIndex, Smi::FromInt(finger_index));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
byte ByteArray::get(int index) {
|
|
|
|
ASSERT(index >= 0 && index < this->length());
|
|
|
|
return READ_BYTE_FIELD(this, kHeaderSize + index * kCharSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ByteArray::set(int index, byte value) {
|
|
|
|
ASSERT(index >= 0 && index < this->length());
|
|
|
|
WRITE_BYTE_FIELD(this, kHeaderSize + index * kCharSize, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int ByteArray::get_int(int index) {
|
|
|
|
ASSERT(index >= 0 && (index * kIntSize) < this->length());
|
|
|
|
return READ_INT_FIELD(this, kHeaderSize + index * kIntSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ByteArray* ByteArray::FromDataStartAddress(Address address) {
|
|
|
|
ASSERT_TAG_ALIGNED(address);
|
|
|
|
return reinterpret_cast<ByteArray*>(address - kHeaderSize + kHeapObjectTag);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Address ByteArray::GetDataStartAddress() {
|
|
|
|
return reinterpret_cast<Address>(this) - kHeapObjectTag + kHeaderSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
uint8_t* ExternalUint8ClampedArray::external_uint8_clamped_pointer() {
|
2011-03-09 15:01:16 +00:00
|
|
|
return reinterpret_cast<uint8_t*>(external_pointer());
|
2009-07-28 08:43:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
uint8_t ExternalUint8ClampedArray::get_scalar(int index) {
|
2009-07-28 08:43:51 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
2014-01-24 16:01:15 +00:00
|
|
|
uint8_t* ptr = external_uint8_clamped_pointer();
|
2009-07-28 08:43:51 +00:00
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalUint8ClampedArray::get(
|
|
|
|
Handle<ExternalUint8ClampedArray> array,
|
|
|
|
int index) {
|
2014-04-16 06:18:37 +00:00
|
|
|
return Handle<Smi>(Smi::FromInt(array->get_scalar(index)),
|
|
|
|
array->GetIsolate());
|
2014-04-08 14:20:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalUint8ClampedArray::set(int index, uint8_t value) {
|
2009-07-28 08:43:51 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
2014-01-24 16:01:15 +00:00
|
|
|
uint8_t* ptr = external_uint8_clamped_pointer();
|
2009-07-28 08:43:51 +00:00
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-10-20 15:26:17 +00:00
|
|
|
void* ExternalArray::external_pointer() {
|
|
|
|
intptr_t ptr = READ_INTPTR_FIELD(this, kExternalPointerOffset);
|
|
|
|
return reinterpret_cast<void*>(ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExternalArray::set_external_pointer(void* value, WriteBarrierMode mode) {
|
|
|
|
intptr_t ptr = reinterpret_cast<intptr_t>(value);
|
|
|
|
WRITE_INTPTR_FIELD(this, kExternalPointerOffset, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
int8_t ExternalInt8Array::get_scalar(int index) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
int8_t* ptr = static_cast<int8_t*>(external_pointer());
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalInt8Array::get(Handle<ExternalInt8Array> array,
|
|
|
|
int index) {
|
2014-04-16 06:18:37 +00:00
|
|
|
return Handle<Smi>(Smi::FromInt(array->get_scalar(index)),
|
|
|
|
array->GetIsolate());
|
2014-04-08 14:20:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalInt8Array::set(int index, int8_t value) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
int8_t* ptr = static_cast<int8_t*>(external_pointer());
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
uint8_t ExternalUint8Array::get_scalar(int index) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
uint8_t* ptr = static_cast<uint8_t*>(external_pointer());
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalUint8Array::get(Handle<ExternalUint8Array> array,
|
|
|
|
int index) {
|
2014-04-16 06:18:37 +00:00
|
|
|
return Handle<Smi>(Smi::FromInt(array->get_scalar(index)),
|
|
|
|
array->GetIsolate());
|
2014-04-08 14:20:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalUint8Array::set(int index, uint8_t value) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
uint8_t* ptr = static_cast<uint8_t*>(external_pointer());
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
int16_t ExternalInt16Array::get_scalar(int index) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
int16_t* ptr = static_cast<int16_t*>(external_pointer());
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalInt16Array::get(Handle<ExternalInt16Array> array,
|
|
|
|
int index) {
|
2014-04-16 06:18:37 +00:00
|
|
|
return Handle<Smi>(Smi::FromInt(array->get_scalar(index)),
|
|
|
|
array->GetIsolate());
|
2014-04-08 14:20:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalInt16Array::set(int index, int16_t value) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
int16_t* ptr = static_cast<int16_t*>(external_pointer());
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
uint16_t ExternalUint16Array::get_scalar(int index) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
uint16_t* ptr = static_cast<uint16_t*>(external_pointer());
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalUint16Array::get(Handle<ExternalUint16Array> array,
|
|
|
|
int index) {
|
2014-04-16 06:18:37 +00:00
|
|
|
return Handle<Smi>(Smi::FromInt(array->get_scalar(index)),
|
|
|
|
array->GetIsolate());
|
2014-04-08 14:20:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalUint16Array::set(int index, uint16_t value) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
uint16_t* ptr = static_cast<uint16_t*>(external_pointer());
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
int32_t ExternalInt32Array::get_scalar(int index) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
int32_t* ptr = static_cast<int32_t*>(external_pointer());
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalInt32Array::get(Handle<ExternalInt32Array> array,
|
|
|
|
int index) {
|
|
|
|
return array->GetIsolate()->factory()->
|
|
|
|
NewNumberFromInt(array->get_scalar(index));
|
2011-08-03 11:12:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalInt32Array::set(int index, int32_t value) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
int32_t* ptr = static_cast<int32_t*>(external_pointer());
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
uint32_t ExternalUint32Array::get_scalar(int index) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
uint32_t* ptr = static_cast<uint32_t*>(external_pointer());
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalUint32Array::get(Handle<ExternalUint32Array> array,
|
|
|
|
int index) {
|
|
|
|
return array->GetIsolate()->factory()->
|
|
|
|
NewNumberFromUint(array->get_scalar(index));
|
2011-08-03 11:12:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalUint32Array::set(int index, uint32_t value) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
uint32_t* ptr = static_cast<uint32_t*>(external_pointer());
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
float ExternalFloat32Array::get_scalar(int index) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
float* ptr = static_cast<float*>(external_pointer());
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalFloat32Array::get(Handle<ExternalFloat32Array> array,
|
|
|
|
int index) {
|
|
|
|
return array->GetIsolate()->factory()->NewNumber(array->get_scalar(index));
|
2011-08-03 11:12:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalFloat32Array::set(int index, float value) {
|
2009-10-20 15:26:17 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
float* ptr = static_cast<float*>(external_pointer());
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
2010-07-13 08:05:10 +00:00
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
double ExternalFloat64Array::get_scalar(int index) {
|
2011-04-21 07:15:43 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
double* ptr = static_cast<double*>(external_pointer());
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> ExternalFloat64Array::get(Handle<ExternalFloat64Array> array,
|
|
|
|
int index) {
|
|
|
|
return array->GetIsolate()->factory()->NewNumber(array->get_scalar(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
void ExternalFloat64Array::set(int index, double value) {
|
2011-04-21 07:15:43 +00:00
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
double* ptr = static_cast<double*>(external_pointer());
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-26 12:50:13 +00:00
|
|
|
void* FixedTypedArrayBase::DataPtr() {
|
|
|
|
return FIELD_ADDR(this, kDataOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int FixedTypedArrayBase::DataSize() {
|
2014-01-16 17:08:45 +00:00
|
|
|
InstanceType instance_type = map()->instance_type();
|
|
|
|
int element_size;
|
|
|
|
switch (instance_type) {
|
2014-03-26 12:50:13 +00:00
|
|
|
#define TYPED_ARRAY_CASE(Type, type, TYPE, ctype, size) \
|
|
|
|
case FIXED_##TYPE##_ARRAY_TYPE: \
|
|
|
|
element_size = size; \
|
2014-01-16 17:08:45 +00:00
|
|
|
break;
|
2014-03-26 12:50:13 +00:00
|
|
|
|
|
|
|
TYPED_ARRAYS(TYPED_ARRAY_CASE)
|
|
|
|
#undef TYPED_ARRAY_CASE
|
2014-01-16 17:08:45 +00:00
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
return 0;
|
|
|
|
}
|
2014-03-26 12:50:13 +00:00
|
|
|
return length() * element_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int FixedTypedArrayBase::size() {
|
|
|
|
return OBJECT_POINTER_ALIGN(kDataOffset + DataSize());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t Uint8ArrayTraits::defaultValue() { return 0; }
|
|
|
|
|
|
|
|
|
|
|
|
uint8_t Uint8ClampedArrayTraits::defaultValue() { return 0; }
|
|
|
|
|
|
|
|
|
|
|
|
int8_t Int8ArrayTraits::defaultValue() { return 0; }
|
|
|
|
|
|
|
|
|
|
|
|
uint16_t Uint16ArrayTraits::defaultValue() { return 0; }
|
|
|
|
|
|
|
|
|
|
|
|
int16_t Int16ArrayTraits::defaultValue() { return 0; }
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t Uint32ArrayTraits::defaultValue() { return 0; }
|
|
|
|
|
|
|
|
|
|
|
|
int32_t Int32ArrayTraits::defaultValue() { return 0; }
|
|
|
|
|
|
|
|
|
|
|
|
float Float32ArrayTraits::defaultValue() {
|
|
|
|
return static_cast<float>(OS::nan_value());
|
2014-01-16 17:08:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-26 12:50:13 +00:00
|
|
|
double Float64ArrayTraits::defaultValue() { return OS::nan_value(); }
|
|
|
|
|
|
|
|
|
2014-01-16 17:08:45 +00:00
|
|
|
template <class Traits>
|
|
|
|
typename Traits::ElementType FixedTypedArray<Traits>::get_scalar(int index) {
|
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
ElementType* ptr = reinterpret_cast<ElementType*>(
|
|
|
|
FIELD_ADDR(this, kDataOffset));
|
|
|
|
return ptr[index];
|
|
|
|
}
|
|
|
|
|
2014-01-30 20:05:11 +00:00
|
|
|
|
|
|
|
template<> inline
|
|
|
|
FixedTypedArray<Float64ArrayTraits>::ElementType
|
|
|
|
FixedTypedArray<Float64ArrayTraits>::get_scalar(int index) {
|
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
return READ_DOUBLE_FIELD(this, ElementOffset(index));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-16 17:08:45 +00:00
|
|
|
template <class Traits>
|
|
|
|
void FixedTypedArray<Traits>::set(int index, ElementType value) {
|
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
ElementType* ptr = reinterpret_cast<ElementType*>(
|
|
|
|
FIELD_ADDR(this, kDataOffset));
|
|
|
|
ptr[index] = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-30 20:05:11 +00:00
|
|
|
template<> inline
|
|
|
|
void FixedTypedArray<Float64ArrayTraits>::set(
|
|
|
|
int index, Float64ArrayTraits::ElementType value) {
|
|
|
|
ASSERT((index >= 0) && (index < this->length()));
|
|
|
|
WRITE_DOUBLE_FIELD(this, ElementOffset(index), value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-26 12:50:13 +00:00
|
|
|
template <class Traits>
|
|
|
|
typename Traits::ElementType FixedTypedArray<Traits>::from_int(int value) {
|
|
|
|
return static_cast<ElementType>(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template <> inline
|
|
|
|
uint8_t FixedTypedArray<Uint8ClampedArrayTraits>::from_int(int value) {
|
|
|
|
if (value < 0) return 0;
|
|
|
|
if (value > 0xFF) return 0xFF;
|
|
|
|
return static_cast<uint8_t>(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template <class Traits>
|
|
|
|
typename Traits::ElementType FixedTypedArray<Traits>::from_double(
|
|
|
|
double value) {
|
|
|
|
return static_cast<ElementType>(DoubleToInt32(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<> inline
|
|
|
|
uint8_t FixedTypedArray<Uint8ClampedArrayTraits>::from_double(double value) {
|
|
|
|
if (value < 0) return 0;
|
|
|
|
if (value > 0xFF) return 0xFF;
|
|
|
|
return static_cast<uint8_t>(lrint(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<> inline
|
|
|
|
float FixedTypedArray<Float32ArrayTraits>::from_double(double value) {
|
|
|
|
return static_cast<float>(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<> inline
|
|
|
|
double FixedTypedArray<Float64ArrayTraits>::from_double(double value) {
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
template <class Traits>
|
|
|
|
Handle<Object> FixedTypedArray<Traits>::get(
|
|
|
|
Handle<FixedTypedArray<Traits> > array,
|
|
|
|
int index) {
|
|
|
|
return Traits::ToHandle(array->GetIsolate(), array->get_scalar(index));
|
|
|
|
}
|
|
|
|
|
2014-04-16 06:18:37 +00:00
|
|
|
|
2014-01-16 17:08:45 +00:00
|
|
|
template <class Traits>
|
2014-04-16 06:18:37 +00:00
|
|
|
Handle<Object> FixedTypedArray<Traits>::SetValue(
|
|
|
|
Handle<FixedTypedArray<Traits> > array,
|
|
|
|
uint32_t index,
|
|
|
|
Handle<Object> value) {
|
2014-01-16 17:08:45 +00:00
|
|
|
ElementType cast_value = Traits::defaultValue();
|
2014-04-16 06:18:37 +00:00
|
|
|
if (index < static_cast<uint32_t>(array->length())) {
|
2014-01-16 17:08:45 +00:00
|
|
|
if (value->IsSmi()) {
|
2014-04-16 06:18:37 +00:00
|
|
|
int int_value = Handle<Smi>::cast(value)->value();
|
2014-03-26 12:50:13 +00:00
|
|
|
cast_value = from_int(int_value);
|
2014-01-16 17:08:45 +00:00
|
|
|
} else if (value->IsHeapNumber()) {
|
2014-04-16 06:18:37 +00:00
|
|
|
double double_value = Handle<HeapNumber>::cast(value)->value();
|
2014-03-26 12:50:13 +00:00
|
|
|
cast_value = from_double(double_value);
|
2014-01-16 17:08:45 +00:00
|
|
|
} else {
|
|
|
|
// Clamp undefined to the default value. All other types have been
|
|
|
|
// converted to a number type further up in the call chain.
|
|
|
|
ASSERT(value->IsUndefined());
|
|
|
|
}
|
2014-04-16 06:18:37 +00:00
|
|
|
array->set(index, cast_value);
|
2014-01-16 17:08:45 +00:00
|
|
|
}
|
2014-04-16 06:18:37 +00:00
|
|
|
return Traits::ToHandle(array->GetIsolate(), cast_value);
|
2014-01-16 17:08:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 14:20:29 +00:00
|
|
|
Handle<Object> Uint8ArrayTraits::ToHandle(Isolate* isolate, uint8_t scalar) {
|
|
|
|
return handle(Smi::FromInt(scalar), isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Uint8ClampedArrayTraits::ToHandle(Isolate* isolate,
|
|
|
|
uint8_t scalar) {
|
|
|
|
return handle(Smi::FromInt(scalar), isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Int8ArrayTraits::ToHandle(Isolate* isolate, int8_t scalar) {
|
|
|
|
return handle(Smi::FromInt(scalar), isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Uint16ArrayTraits::ToHandle(Isolate* isolate, uint16_t scalar) {
|
|
|
|
return handle(Smi::FromInt(scalar), isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Int16ArrayTraits::ToHandle(Isolate* isolate, int16_t scalar) {
|
|
|
|
return handle(Smi::FromInt(scalar), isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Uint32ArrayTraits::ToHandle(Isolate* isolate, uint32_t scalar) {
|
|
|
|
return isolate->factory()->NewNumberFromUint(scalar);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Int32ArrayTraits::ToHandle(Isolate* isolate, int32_t scalar) {
|
|
|
|
return isolate->factory()->NewNumberFromInt(scalar);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Float32ArrayTraits::ToHandle(Isolate* isolate, float scalar) {
|
|
|
|
return isolate->factory()->NewNumber(scalar);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<Object> Float64ArrayTraits::ToHandle(Isolate* isolate, double scalar) {
|
|
|
|
return isolate->factory()->NewNumber(scalar);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-06 12:50:11 +00:00
|
|
|
int Map::visitor_id() {
|
|
|
|
return READ_BYTE_FIELD(this, kVisitorIdOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Map::set_visitor_id(int id) {
|
|
|
|
ASSERT(0 <= id && id < 256);
|
|
|
|
WRITE_BYTE_FIELD(this, kVisitorIdOffset, static_cast<byte>(id));
|
|
|
|
}
|
|
|
|
|
2009-10-20 15:26:17 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
int Map::instance_size() {
|
2014-04-09 09:50:25 +00:00
|
|
|
return NOBARRIER_READ_BYTE_FIELD(
|
|
|
|
this, kInstanceSizeOffset) << kPointerSizeLog2;
|
2008-10-15 06:03:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Map::inobject_properties() {
|
|
|
|
return READ_BYTE_FIELD(this, kInObjectPropertiesOffset);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-08-19 07:30:20 +00:00
|
|
|
int Map::pre_allocated_property_fields() {
|
|
|
|
return READ_BYTE_FIELD(this, kPreAllocatedPropertyFieldsOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-04 10:48:49 +00:00
|
|
|
int Map::GetInObjectPropertyOffset(int index) {
|
|
|
|
// Adjust for the number of properties stored in the object.
|
|
|
|
index -= inobject_properties();
|
|
|
|
ASSERT(index < 0);
|
|
|
|
return instance_size() + (index * kPointerSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
int HeapObject::SizeFromMap(Map* map) {
|
2010-08-18 13:00:38 +00:00
|
|
|
int instance_size = map->instance_size();
|
|
|
|
if (instance_size != kVariableSizeSentinel) return instance_size;
|
2009-06-16 13:31:31 +00:00
|
|
|
// Only inline the most frequent cases.
|
2013-06-27 13:39:44 +00:00
|
|
|
int instance_type = static_cast<int>(map->instance_type());
|
2008-07-03 15:10:15 +00:00
|
|
|
if (instance_type == FIXED_ARRAY_TYPE) {
|
2010-08-11 14:30:14 +00:00
|
|
|
return FixedArray::BodyDescriptor::SizeOf(map, this);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
2013-06-27 13:39:44 +00:00
|
|
|
if (instance_type == ASCII_STRING_TYPE ||
|
|
|
|
instance_type == ASCII_INTERNALIZED_STRING_TYPE) {
|
2012-11-15 13:31:27 +00:00
|
|
|
return SeqOneByteString::SizeFor(
|
|
|
|
reinterpret_cast<SeqOneByteString*>(this)->length());
|
2010-08-18 13:00:38 +00:00
|
|
|
}
|
2009-06-16 13:31:31 +00:00
|
|
|
if (instance_type == BYTE_ARRAY_TYPE) {
|
|
|
|
return reinterpret_cast<ByteArray*>(this)->ByteArraySize();
|
|
|
|
}
|
2011-09-19 18:36:47 +00:00
|
|
|
if (instance_type == FREE_SPACE_TYPE) {
|
2014-04-08 16:31:57 +00:00
|
|
|
return reinterpret_cast<FreeSpace*>(this)->nobarrier_size();
|
2011-09-19 18:36:47 +00:00
|
|
|
}
|
2013-06-27 13:39:44 +00:00
|
|
|
if (instance_type == STRING_TYPE ||
|
|
|
|
instance_type == INTERNALIZED_STRING_TYPE) {
|
2010-08-18 13:00:38 +00:00
|
|
|
return SeqTwoByteString::SizeFor(
|
|
|
|
reinterpret_cast<SeqTwoByteString*>(this)->length());
|
|
|
|
}
|
2011-06-09 10:03:35 +00:00
|
|
|
if (instance_type == FIXED_DOUBLE_ARRAY_TYPE) {
|
|
|
|
return FixedDoubleArray::SizeFor(
|
|
|
|
reinterpret_cast<FixedDoubleArray*>(this)->length());
|
|
|
|
}
|
2013-10-14 13:35:06 +00:00
|
|
|
if (instance_type == CONSTANT_POOL_ARRAY_TYPE) {
|
|
|
|
return ConstantPoolArray::SizeFor(
|
|
|
|
reinterpret_cast<ConstantPoolArray*>(this)->count_of_int64_entries(),
|
2014-03-10 19:05:43 +00:00
|
|
|
reinterpret_cast<ConstantPoolArray*>(this)->count_of_code_ptr_entries(),
|
|
|
|
reinterpret_cast<ConstantPoolArray*>(this)->count_of_heap_ptr_entries(),
|
2013-10-14 13:35:06 +00:00
|
|
|
reinterpret_cast<ConstantPoolArray*>(this)->count_of_int32_entries());
|
|
|
|
}
|
2014-01-16 17:08:45 +00:00
|
|
|
if (instance_type >= FIRST_FIXED_TYPED_ARRAY_TYPE &&
|
|
|
|
instance_type <= LAST_FIXED_TYPED_ARRAY_TYPE) {
|
|
|
|
return reinterpret_cast<FixedTypedArrayBase*>(this)->size();
|
|
|
|
}
|
2010-08-18 13:00:38 +00:00
|
|
|
ASSERT(instance_type == CODE_TYPE);
|
|
|
|
return reinterpret_cast<Code*>(this)->CodeSize();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Map::set_instance_size(int value) {
|
2009-05-18 12:13:20 +00:00
|
|
|
ASSERT_EQ(0, value & (kPointerSize - 1));
|
2008-10-15 06:03:26 +00:00
|
|
|
value >>= kPointerSizeLog2;
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(0 <= value && value < 256);
|
2014-04-09 09:50:25 +00:00
|
|
|
NOBARRIER_WRITE_BYTE_FIELD(
|
|
|
|
this, kInstanceSizeOffset, static_cast<byte>(value));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-15 06:03:26 +00:00
|
|
|
void Map::set_inobject_properties(int value) {
|
|
|
|
ASSERT(0 <= value && value < 256);
|
|
|
|
WRITE_BYTE_FIELD(this, kInObjectPropertiesOffset, static_cast<byte>(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-08-19 07:30:20 +00:00
|
|
|
void Map::set_pre_allocated_property_fields(int value) {
|
|
|
|
ASSERT(0 <= value && value < 256);
|
|
|
|
WRITE_BYTE_FIELD(this,
|
|
|
|
kPreAllocatedPropertyFieldsOffset,
|
|
|
|
static_cast<byte>(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
InstanceType Map::instance_type() {
|
|
|
|
return static_cast<InstanceType>(READ_BYTE_FIELD(this, kInstanceTypeOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Map::set_instance_type(InstanceType value) {
|
|
|
|
WRITE_BYTE_FIELD(this, kInstanceTypeOffset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Map::unused_property_fields() {
|
|
|
|
return READ_BYTE_FIELD(this, kUnusedPropertyFieldsOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Map::set_unused_property_fields(int value) {
|
|
|
|
WRITE_BYTE_FIELD(this, kUnusedPropertyFieldsOffset, Min(value, 255));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
byte Map::bit_field() {
|
|
|
|
return READ_BYTE_FIELD(this, kBitFieldOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Map::set_bit_field(byte value) {
|
|
|
|
WRITE_BYTE_FIELD(this, kBitFieldOffset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-04-24 08:13:09 +00:00
|
|
|
byte Map::bit_field2() {
|
|
|
|
return READ_BYTE_FIELD(this, kBitField2Offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Map::set_bit_field2(byte value) {
|
|
|
|
WRITE_BYTE_FIELD(this, kBitField2Offset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void Map::set_non_instance_prototype(bool value) {
|
|
|
|
if (value) {
|
|
|
|
set_bit_field(bit_field() | (1 << kHasNonInstancePrototype));
|
|
|
|
} else {
|
|
|
|
set_bit_field(bit_field() & ~(1 << kHasNonInstancePrototype));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::has_non_instance_prototype() {
|
|
|
|
return ((1 << kHasNonInstancePrototype) & bit_field()) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-28 12:05:40 +00:00
|
|
|
void Map::set_function_with_prototype(bool value) {
|
2012-07-19 14:45:19 +00:00
|
|
|
set_bit_field3(FunctionWithPrototype::update(bit_field3(), value));
|
2010-04-28 12:05:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::function_with_prototype() {
|
2012-07-19 14:45:19 +00:00
|
|
|
return FunctionWithPrototype::decode(bit_field3());
|
2010-04-28 12:05:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-30 12:51:06 +00:00
|
|
|
void Map::set_is_access_check_needed(bool access_check_needed) {
|
|
|
|
if (access_check_needed) {
|
|
|
|
set_bit_field(bit_field() | (1 << kIsAccessCheckNeeded));
|
|
|
|
} else {
|
|
|
|
set_bit_field(bit_field() & ~(1 << kIsAccessCheckNeeded));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::is_access_check_needed() {
|
|
|
|
return ((1 << kIsAccessCheckNeeded) & bit_field()) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-02 14:36:34 +00:00
|
|
|
void Map::set_is_extensible(bool value) {
|
|
|
|
if (value) {
|
|
|
|
set_bit_field2(bit_field2() | (1 << kIsExtensible));
|
|
|
|
} else {
|
|
|
|
set_bit_field2(bit_field2() & ~(1 << kIsExtensible));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Map::is_extensible() {
|
|
|
|
return ((1 << kIsExtensible) & bit_field2()) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-23 09:15:26 +00:00
|
|
|
void Map::set_attached_to_shared_function_info(bool value) {
|
|
|
|
if (value) {
|
|
|
|
set_bit_field2(bit_field2() | (1 << kAttachedToSharedFunctionInfo));
|
|
|
|
} else {
|
|
|
|
set_bit_field2(bit_field2() & ~(1 << kAttachedToSharedFunctionInfo));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Map::attached_to_shared_function_info() {
|
|
|
|
return ((1 << kAttachedToSharedFunctionInfo) & bit_field2()) != 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-24 08:18:33 +00:00
|
|
|
void Map::set_is_shared(bool value) {
|
2012-07-19 14:45:19 +00:00
|
|
|
set_bit_field3(IsShared::update(bit_field3(), value));
|
2010-09-24 08:18:33 +00:00
|
|
|
}
|
|
|
|
|
2012-08-06 14:25:19 +00:00
|
|
|
|
2010-09-24 08:18:33 +00:00
|
|
|
bool Map::is_shared() {
|
2014-03-04 12:48:17 +00:00
|
|
|
return IsShared::decode(bit_field3()); }
|
2010-09-24 08:18:33 +00:00
|
|
|
|
|
|
|
|
2012-08-06 14:25:19 +00:00
|
|
|
void Map::set_dictionary_map(bool value) {
|
2014-01-31 11:38:43 +00:00
|
|
|
uint32_t new_bit_field3 = DictionaryMap::update(bit_field3(), value);
|
|
|
|
new_bit_field3 = IsUnstable::update(new_bit_field3, value);
|
|
|
|
set_bit_field3(new_bit_field3);
|
2012-08-06 14:25:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::is_dictionary_map() {
|
|
|
|
return DictionaryMap::decode(bit_field3());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Code::Flags Code::flags() {
|
|
|
|
return static_cast<Flags>(READ_INT_FIELD(this, kFlagsOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
void Map::set_owns_descriptors(bool is_shared) {
|
|
|
|
set_bit_field3(OwnsDescriptors::update(bit_field3(), is_shared));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::owns_descriptors() {
|
|
|
|
return OwnsDescriptors::decode(bit_field3());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-14 21:47:39 +00:00
|
|
|
void Map::set_has_instance_call_handler() {
|
|
|
|
set_bit_field3(HasInstanceCallHandler::update(bit_field3(), true));
|
2012-11-06 12:30:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-14 21:47:39 +00:00
|
|
|
bool Map::has_instance_call_handler() {
|
|
|
|
return HasInstanceCallHandler::decode(bit_field3());
|
2012-11-06 12:30:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-26 15:30:41 +00:00
|
|
|
void Map::deprecate() {
|
|
|
|
set_bit_field3(Deprecated::update(bit_field3(), true));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::is_deprecated() {
|
|
|
|
return Deprecated::decode(bit_field3());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-05 16:42:39 +00:00
|
|
|
void Map::set_migration_target(bool value) {
|
|
|
|
set_bit_field3(IsMigrationTarget::update(bit_field3(), value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::is_migration_target() {
|
|
|
|
return IsMigrationTarget::decode(bit_field3());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-23 07:05:58 +00:00
|
|
|
void Map::freeze() {
|
|
|
|
set_bit_field3(IsFrozen::update(bit_field3(), true));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::is_frozen() {
|
|
|
|
return IsFrozen::decode(bit_field3());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-30 16:33:58 +00:00
|
|
|
void Map::mark_unstable() {
|
|
|
|
set_bit_field3(IsUnstable::update(bit_field3(), true));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::is_stable() {
|
|
|
|
return !IsUnstable::decode(bit_field3());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-18 13:00:40 +00:00
|
|
|
bool Map::has_code_cache() {
|
|
|
|
return code_cache() != GetIsolate()->heap()->empty_fixed_array();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-26 15:30:41 +00:00
|
|
|
bool Map::CanBeDeprecated() {
|
|
|
|
int descriptor = LastAdded();
|
|
|
|
for (int i = 0; i <= descriptor; i++) {
|
|
|
|
PropertyDetails details = instance_descriptors()->GetDetails(i);
|
2014-03-04 12:48:17 +00:00
|
|
|
if (details.representation().IsNone()) return true;
|
|
|
|
if (details.representation().IsSmi()) return true;
|
|
|
|
if (details.representation().IsDouble()) return true;
|
|
|
|
if (details.representation().IsHeapObject()) return true;
|
|
|
|
if (details.type() == CONSTANT) return true;
|
2013-04-26 15:30:41 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-20 11:49:54 +00:00
|
|
|
void Map::NotifyLeafMapLayoutChange() {
|
2013-07-30 16:33:58 +00:00
|
|
|
if (is_stable()) {
|
|
|
|
mark_unstable();
|
|
|
|
dependent_code()->DeoptimizeDependentCodeGroup(
|
|
|
|
GetIsolate(),
|
|
|
|
DependentCode::kPrototypeCheckGroup);
|
|
|
|
}
|
2013-02-20 11:49:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-23 09:18:42 +00:00
|
|
|
bool Map::CanOmitMapChecks() {
|
2013-07-30 16:33:58 +00:00
|
|
|
return is_stable() && FLAG_omit_map_checks_for_leaf_maps;
|
2013-07-23 09:18:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-20 11:49:54 +00:00
|
|
|
int DependentCode::number_of_entries(DependencyGroup group) {
|
2013-01-24 11:55:05 +00:00
|
|
|
if (length() == 0) return 0;
|
2013-02-20 11:49:54 +00:00
|
|
|
return Smi::cast(get(group))->value();
|
2013-01-24 11:55:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-20 11:49:54 +00:00
|
|
|
void DependentCode::set_number_of_entries(DependencyGroup group, int value) {
|
|
|
|
set(group, Smi::FromInt(value));
|
2013-01-24 11:55:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-12 09:43:22 +00:00
|
|
|
bool DependentCode::is_code_at(int i) {
|
|
|
|
return get(kCodesStartIndex + i)->IsCode();
|
|
|
|
}
|
|
|
|
|
2013-02-20 11:49:54 +00:00
|
|
|
Code* DependentCode::code_at(int i) {
|
|
|
|
return Code::cast(get(kCodesStartIndex + i));
|
2013-01-24 11:55:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-12 09:43:22 +00:00
|
|
|
CompilationInfo* DependentCode::compilation_info_at(int i) {
|
|
|
|
return reinterpret_cast<CompilationInfo*>(
|
|
|
|
Foreign::cast(get(kCodesStartIndex + i))->foreign_address());
|
2013-01-24 11:55:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-12 09:43:22 +00:00
|
|
|
void DependentCode::set_object_at(int i, Object* object) {
|
|
|
|
set(kCodesStartIndex + i, object);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object* DependentCode::object_at(int i) {
|
|
|
|
return get(kCodesStartIndex + i);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object** DependentCode::slot_at(int i) {
|
2013-12-23 14:42:42 +00:00
|
|
|
return RawFieldOfElementAt(kCodesStartIndex + i);
|
2013-01-24 11:55:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-12 09:43:22 +00:00
|
|
|
void DependentCode::clear_at(int i) {
|
2013-02-20 11:49:54 +00:00
|
|
|
set_undefined(kCodesStartIndex + i);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-12 09:43:22 +00:00
|
|
|
void DependentCode::copy(int from, int to) {
|
|
|
|
set(kCodesStartIndex + to, get(kCodesStartIndex + from));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-20 11:49:54 +00:00
|
|
|
void DependentCode::ExtendGroup(DependencyGroup group) {
|
|
|
|
GroupStartIndexes starts(this);
|
|
|
|
for (int g = kGroupCount - 1; g > group; g--) {
|
|
|
|
if (starts.at(g) < starts.at(g + 1)) {
|
2013-06-12 09:43:22 +00:00
|
|
|
copy(starts.at(g), starts.at(g + 1));
|
2013-02-20 11:49:54 +00:00
|
|
|
}
|
|
|
|
}
|
2013-01-24 11:55:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void Code::set_flags(Code::Flags flags) {
|
2011-09-12 10:50:50 +00:00
|
|
|
STATIC_ASSERT(Code::NUMBER_OF_KINDS <= KindField::kMax + 1);
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_INT_FIELD(this, kFlagsOffset, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Code::Kind Code::kind() {
|
|
|
|
return ExtractKindFromFlags(flags());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-30 08:49:36 +00:00
|
|
|
InlineCacheState Code::ic_state() {
|
|
|
|
InlineCacheState result = ExtractICStateFromFlags(flags());
|
2008-07-03 15:10:15 +00:00
|
|
|
// Only allow uninitialized or debugger states for non-IC code
|
|
|
|
// objects. This is used in the debugger to determine whether or not
|
|
|
|
// a call to code object has been replaced with a debug break call.
|
|
|
|
ASSERT(is_inline_cache_stub() ||
|
|
|
|
result == UNINITIALIZED ||
|
2013-01-10 14:15:12 +00:00
|
|
|
result == DEBUG_STUB);
|
2008-07-03 15:10:15 +00:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-28 15:32:55 +00:00
|
|
|
ExtraICState Code::extra_ic_state() {
|
2013-04-24 11:32:17 +00:00
|
|
|
ASSERT(is_inline_cache_stub() || ic_state() == DEBUG_STUB);
|
2014-02-11 15:01:44 +00:00
|
|
|
return ExtractExtraICStateFromFlags(flags());
|
2013-04-24 11:32:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-25 11:35:23 +00:00
|
|
|
Code::StubType Code::type() {
|
2008-07-03 15:10:15 +00:00
|
|
|
return ExtractTypeFromFlags(flags());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-13 10:27:19 +00:00
|
|
|
// For initialization.
|
|
|
|
void Code::set_raw_kind_specific_flags1(int value) {
|
|
|
|
WRITE_INT_FIELD(this, kKindSpecificFlags1Offset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_raw_kind_specific_flags2(int value) {
|
|
|
|
WRITE_INT_FIELD(this, kKindSpecificFlags2Offset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-18 09:50:46 +00:00
|
|
|
inline bool Code::is_crankshafted() {
|
|
|
|
return IsCrankshaftedField::decode(
|
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags2Offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
inline void Code::set_is_crankshafted(bool value) {
|
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
|
|
|
|
int updated = IsCrankshaftedField::update(previous, value);
|
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-30 11:48:07 +00:00
|
|
|
int Code::major_key() {
|
2013-11-21 16:55:16 +00:00
|
|
|
ASSERT(has_major_key());
|
2012-07-06 22:08:27 +00:00
|
|
|
return StubMajorKeyField::decode(
|
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags2Offset));
|
2008-07-30 08:49:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-30 11:48:07 +00:00
|
|
|
void Code::set_major_key(int major) {
|
2013-11-21 16:55:16 +00:00
|
|
|
ASSERT(has_major_key());
|
2008-07-30 08:49:36 +00:00
|
|
|
ASSERT(0 <= major && major < 256);
|
2012-07-06 22:08:27 +00:00
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
|
|
|
|
int updated = StubMajorKeyField::update(previous, major);
|
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-21 16:55:16 +00:00
|
|
|
bool Code::has_major_key() {
|
|
|
|
return kind() == STUB ||
|
|
|
|
kind() == HANDLER ||
|
|
|
|
kind() == BINARY_OP_IC ||
|
|
|
|
kind() == COMPARE_IC ||
|
|
|
|
kind() == COMPARE_NIL_IC ||
|
|
|
|
kind() == LOAD_IC ||
|
|
|
|
kind() == KEYED_LOAD_IC ||
|
|
|
|
kind() == STORE_IC ||
|
2014-04-30 14:33:35 +00:00
|
|
|
kind() == CALL_IC ||
|
2013-11-21 16:55:16 +00:00
|
|
|
kind() == KEYED_STORE_IC ||
|
|
|
|
kind() == TO_BOOLEAN_IC;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
bool Code::optimizable() {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2010-12-07 11:31:57 +00:00
|
|
|
return READ_BYTE_FIELD(this, kOptimizableOffset) == 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_optimizable(bool value) {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2010-12-07 11:31:57 +00:00
|
|
|
WRITE_BYTE_FIELD(this, kOptimizableOffset, value ? 1 : 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Code::has_deoptimization_support() {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2011-09-13 08:31:21 +00:00
|
|
|
byte flags = READ_BYTE_FIELD(this, kFullCodeFlags);
|
|
|
|
return FullCodeFlagsHasDeoptimizationSupportField::decode(flags);
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_has_deoptimization_support(bool value) {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2011-09-13 08:31:21 +00:00
|
|
|
byte flags = READ_BYTE_FIELD(this, kFullCodeFlags);
|
|
|
|
flags = FullCodeFlagsHasDeoptimizationSupportField::update(flags, value);
|
|
|
|
WRITE_BYTE_FIELD(this, kFullCodeFlags, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Code::has_debug_break_slots() {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2011-09-13 08:31:21 +00:00
|
|
|
byte flags = READ_BYTE_FIELD(this, kFullCodeFlags);
|
|
|
|
return FullCodeFlagsHasDebugBreakSlotsField::decode(flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_has_debug_break_slots(bool value) {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2011-09-13 08:31:21 +00:00
|
|
|
byte flags = READ_BYTE_FIELD(this, kFullCodeFlags);
|
|
|
|
flags = FullCodeFlagsHasDebugBreakSlotsField::update(flags, value);
|
|
|
|
WRITE_BYTE_FIELD(this, kFullCodeFlags, flags);
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-18 13:40:33 +00:00
|
|
|
bool Code::is_compiled_optimizable() {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2011-10-18 13:40:33 +00:00
|
|
|
byte flags = READ_BYTE_FIELD(this, kFullCodeFlags);
|
|
|
|
return FullCodeFlagsIsCompiledOptimizable::decode(flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_compiled_optimizable(bool value) {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2011-10-18 13:40:33 +00:00
|
|
|
byte flags = READ_BYTE_FIELD(this, kFullCodeFlags);
|
|
|
|
flags = FullCodeFlagsIsCompiledOptimizable::update(flags, value);
|
|
|
|
WRITE_BYTE_FIELD(this, kFullCodeFlags, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
int Code::allow_osr_at_loop_nesting_level() {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2010-12-07 11:31:57 +00:00
|
|
|
return READ_BYTE_FIELD(this, kAllowOSRAtLoopNestingLevelOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_allow_osr_at_loop_nesting_level(int level) {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2010-12-07 11:31:57 +00:00
|
|
|
ASSERT(level >= 0 && level <= kMaxLoopNestingMarker);
|
|
|
|
WRITE_BYTE_FIELD(this, kAllowOSRAtLoopNestingLevelOffset, level);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-27 12:19:50 +00:00
|
|
|
int Code::profiler_ticks() {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2012-03-27 12:19:50 +00:00
|
|
|
return READ_BYTE_FIELD(this, kProfilerTicksOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_profiler_ticks(int ticks) {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2012-03-27 12:19:50 +00:00
|
|
|
ASSERT(ticks < 256);
|
|
|
|
WRITE_BYTE_FIELD(this, kProfilerTicksOffset, ticks);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
unsigned Code::stack_slots() {
|
2013-04-18 09:50:46 +00:00
|
|
|
ASSERT(is_crankshafted());
|
2012-07-06 22:08:27 +00:00
|
|
|
return StackSlotsField::decode(
|
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_stack_slots(unsigned slots) {
|
2012-07-06 22:08:27 +00:00
|
|
|
CHECK(slots <= (1 << kStackSlotsBitCount));
|
2013-04-18 09:50:46 +00:00
|
|
|
ASSERT(is_crankshafted());
|
2012-07-06 22:08:27 +00:00
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
|
|
|
|
int updated = StackSlotsField::update(previous, slots);
|
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-03 10:07:22 +00:00
|
|
|
unsigned Code::safepoint_table_offset() {
|
2013-04-18 09:50:46 +00:00
|
|
|
ASSERT(is_crankshafted());
|
2012-07-06 22:08:27 +00:00
|
|
|
return SafepointTableOffsetField::decode(
|
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags2Offset));
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-03 10:07:22 +00:00
|
|
|
void Code::set_safepoint_table_offset(unsigned offset) {
|
2012-07-06 22:08:27 +00:00
|
|
|
CHECK(offset <= (1 << kSafepointTableOffsetBitCount));
|
2013-04-18 09:50:46 +00:00
|
|
|
ASSERT(is_crankshafted());
|
2010-12-07 11:31:57 +00:00
|
|
|
ASSERT(IsAligned(offset, static_cast<unsigned>(kIntSize)));
|
2012-07-06 22:08:27 +00:00
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
|
|
|
|
int updated = SafepointTableOffsetField::update(previous, offset);
|
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-10 09:24:31 +00:00
|
|
|
unsigned Code::back_edge_table_offset() {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2013-04-10 09:24:31 +00:00
|
|
|
return BackEdgeTableOffsetField::decode(
|
2012-07-06 22:08:27 +00:00
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags2Offset));
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-10 09:24:31 +00:00
|
|
|
void Code::set_back_edge_table_offset(unsigned offset) {
|
2012-04-02 10:57:17 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2010-12-07 11:31:57 +00:00
|
|
|
ASSERT(IsAligned(offset, static_cast<unsigned>(kIntSize)));
|
2012-07-06 22:08:27 +00:00
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
|
2013-04-10 09:24:31 +00:00
|
|
|
int updated = BackEdgeTableOffsetField::update(previous, offset);
|
2012-07-06 22:08:27 +00:00
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-10 09:24:31 +00:00
|
|
|
bool Code::back_edges_patched_for_osr() {
|
2013-03-12 18:03:18 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
2013-04-10 09:24:31 +00:00
|
|
|
return BackEdgesPatchedForOSRField::decode(
|
2013-03-12 18:03:18 +00:00
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags2Offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-10 09:24:31 +00:00
|
|
|
void Code::set_back_edges_patched_for_osr(bool value) {
|
2013-03-12 18:03:18 +00:00
|
|
|
ASSERT_EQ(FUNCTION, kind());
|
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
|
2013-04-10 09:24:31 +00:00
|
|
|
int updated = BackEdgesPatchedForOSRField::update(previous, value);
|
2013-03-12 18:03:18 +00:00
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-07-21 13:51:04 +00:00
|
|
|
byte Code::to_boolean_state() {
|
2014-02-11 15:01:44 +00:00
|
|
|
return extra_ic_state();
|
2011-07-21 13:51:04 +00:00
|
|
|
}
|
|
|
|
|
2011-09-27 11:42:02 +00:00
|
|
|
|
2012-01-27 16:09:20 +00:00
|
|
|
bool Code::has_function_cache() {
|
|
|
|
ASSERT(kind() == STUB);
|
2012-07-06 22:08:27 +00:00
|
|
|
return HasFunctionCacheField::decode(
|
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
|
2012-01-27 16:09:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_has_function_cache(bool flag) {
|
|
|
|
ASSERT(kind() == STUB);
|
2012-07-06 22:08:27 +00:00
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
|
|
|
|
int updated = HasFunctionCacheField::update(previous, flag);
|
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
|
2012-01-27 16:09:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-24 11:55:05 +00:00
|
|
|
bool Code::marked_for_deoptimization() {
|
|
|
|
ASSERT(kind() == OPTIMIZED_FUNCTION);
|
|
|
|
return MarkedForDeoptimizationField::decode(
|
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_marked_for_deoptimization(bool flag) {
|
|
|
|
ASSERT(kind() == OPTIMIZED_FUNCTION);
|
2014-04-28 06:47:05 +00:00
|
|
|
ASSERT(!flag || AllowDeoptimization::IsAllowed(GetIsolate()));
|
2013-01-24 11:55:05 +00:00
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
|
|
|
|
int updated = MarkedForDeoptimizationField::update(previous, flag);
|
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 10:36:09 +00:00
|
|
|
bool Code::is_weak_stub() {
|
|
|
|
return CanBeWeakStub() && WeakStubField::decode(
|
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::mark_as_weak_stub() {
|
|
|
|
ASSERT(CanBeWeakStub());
|
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
|
|
|
|
int updated = WeakStubField::update(previous, true);
|
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Code::is_invalidated_weak_stub() {
|
|
|
|
return is_weak_stub() && InvalidatedWeakStubField::decode(
|
|
|
|
READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::mark_as_invalidated_weak_stub() {
|
|
|
|
ASSERT(is_inline_cache_stub());
|
|
|
|
int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
|
|
|
|
int updated = InvalidatedWeakStubField::update(previous, true);
|
|
|
|
WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool Code::is_inline_cache_stub() {
|
|
|
|
Kind kind = this->kind();
|
2013-06-25 09:09:25 +00:00
|
|
|
switch (kind) {
|
|
|
|
#define CASE(name) case name: return true;
|
|
|
|
IC_KIND_LIST(CASE)
|
|
|
|
#undef CASE
|
|
|
|
default: return false;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-01 09:44:35 +00:00
|
|
|
bool Code::is_keyed_stub() {
|
2014-01-31 16:52:17 +00:00
|
|
|
return is_keyed_load_stub() || is_keyed_store_stub();
|
2013-10-01 09:44:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-22 12:16:00 +00:00
|
|
|
bool Code::is_debug_stub() {
|
|
|
|
return ic_state() == DEBUG_STUB;
|
2013-01-10 14:15:12 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-30 11:23:59 +00:00
|
|
|
ConstantPoolArray* Code::constant_pool() {
|
|
|
|
return ConstantPoolArray::cast(READ_FIELD(this, kConstantPoolOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_constant_pool(Object* value) {
|
|
|
|
ASSERT(value->IsConstantPoolArray());
|
|
|
|
WRITE_FIELD(this, kConstantPoolOffset, value);
|
|
|
|
WRITE_BARRIER(GetHeap(), this, kConstantPoolOffset, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Code::Flags Code::ComputeFlags(Kind kind,
|
2008-07-30 08:49:36 +00:00
|
|
|
InlineCacheState ic_state,
|
2011-01-18 16:54:48 +00:00
|
|
|
ExtraICState extra_ic_state,
|
2012-06-25 11:35:23 +00:00
|
|
|
StubType type,
|
2010-07-02 14:15:04 +00:00
|
|
|
InlineCacheHolderFlag holder) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// Compute the bit mask.
|
2013-03-18 13:35:17 +00:00
|
|
|
unsigned int bits = KindField::encode(kind)
|
2011-09-12 10:50:50 +00:00
|
|
|
| ICStateField::encode(ic_state)
|
|
|
|
| TypeField::encode(type)
|
2014-02-11 15:01:44 +00:00
|
|
|
| ExtraICStateField::encode(extra_ic_state)
|
2011-09-12 10:50:50 +00:00
|
|
|
| CacheHolderField::encode(holder);
|
|
|
|
return static_cast<Flags>(bits);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Code::Flags Code::ComputeMonomorphicFlags(Kind kind,
|
2011-01-18 16:54:48 +00:00
|
|
|
ExtraICState extra_ic_state,
|
2013-11-14 16:25:31 +00:00
|
|
|
InlineCacheHolderFlag holder,
|
2014-02-14 15:15:08 +00:00
|
|
|
StubType type) {
|
|
|
|
return ComputeFlags(kind, MONOMORPHIC, extra_ic_state, type, holder);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Code::Flags Code::ComputeHandlerFlags(Kind handler_kind,
|
|
|
|
StubType type,
|
|
|
|
InlineCacheHolderFlag holder) {
|
2014-02-14 15:17:26 +00:00
|
|
|
return ComputeFlags(Code::HANDLER, MONOMORPHIC, handler_kind, type, holder);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Code::Kind Code::ExtractKindFromFlags(Flags flags) {
|
2011-09-12 10:50:50 +00:00
|
|
|
return KindField::decode(flags);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-30 08:49:36 +00:00
|
|
|
InlineCacheState Code::ExtractICStateFromFlags(Flags flags) {
|
2011-09-12 10:50:50 +00:00
|
|
|
return ICStateField::decode(flags);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-28 15:32:55 +00:00
|
|
|
ExtraICState Code::ExtractExtraICStateFromFlags(Flags flags) {
|
2011-09-12 10:50:50 +00:00
|
|
|
return ExtraICStateField::decode(flags);
|
2011-01-18 16:54:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-25 11:35:23 +00:00
|
|
|
Code::StubType Code::ExtractTypeFromFlags(Flags flags) {
|
2011-09-12 10:50:50 +00:00
|
|
|
return TypeField::decode(flags);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-02 14:15:04 +00:00
|
|
|
InlineCacheHolderFlag Code::ExtractCacheHolderFromFlags(Flags flags) {
|
2011-09-12 10:50:50 +00:00
|
|
|
return CacheHolderField::decode(flags);
|
2010-07-02 14:15:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Code::Flags Code::RemoveTypeFromFlags(Flags flags) {
|
2011-09-12 10:50:50 +00:00
|
|
|
int bits = flags & ~TypeField::kMask;
|
2008-07-03 15:10:15 +00:00
|
|
|
return static_cast<Flags>(bits);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-09 12:53:59 +00:00
|
|
|
Code* Code::GetCodeFromTargetAddress(Address address) {
|
|
|
|
HeapObject* code = HeapObject::FromAddress(address - Code::kHeaderSize);
|
|
|
|
// GetCodeFromTargetAddress might be called when marking objects during mark
|
|
|
|
// sweep. reinterpret_cast is therefore used instead of the more appropriate
|
|
|
|
// Code::cast. Code::cast does not work when the object's map is
|
|
|
|
// marked.
|
|
|
|
Code* result = reinterpret_cast<Code*>(code);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-20 07:10:18 +00:00
|
|
|
Object* Code::GetObjectFromEntryAddress(Address location_of_address) {
|
|
|
|
return HeapObject::
|
|
|
|
FromAddress(Memory::Address_at(location_of_address) - Code::kHeaderSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-19 14:03:48 +00:00
|
|
|
bool Code::IsWeakObjectInOptimizedCode(Object* object) {
|
2014-04-29 12:32:38 +00:00
|
|
|
if (!FLAG_collect_maps) return false;
|
2014-02-19 14:03:48 +00:00
|
|
|
if (object->IsMap()) {
|
|
|
|
return Map::cast(object)->CanTransition() &&
|
|
|
|
FLAG_weak_embedded_maps_in_optimized_code;
|
|
|
|
}
|
|
|
|
if (object->IsJSObject() ||
|
|
|
|
(object->IsCell() && Cell::cast(object)->value()->IsJSObject())) {
|
|
|
|
return FLAG_weak_embedded_objects_in_optimized_code;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-18 14:15:09 +00:00
|
|
|
class Code::FindAndReplacePattern {
|
|
|
|
public:
|
|
|
|
FindAndReplacePattern() : count_(0) { }
|
|
|
|
void Add(Handle<Map> map_to_find, Handle<Object> obj_to_replace) {
|
|
|
|
ASSERT(count_ < kMaxCount);
|
|
|
|
find_[count_] = map_to_find;
|
|
|
|
replace_[count_] = obj_to_replace;
|
|
|
|
++count_;
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
static const int kMaxCount = 4;
|
|
|
|
int count_;
|
|
|
|
Handle<Map> find_[kMaxCount];
|
|
|
|
Handle<Object> replace_[kMaxCount];
|
|
|
|
friend class Code;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2014-04-11 10:36:09 +00:00
|
|
|
bool Code::IsWeakObjectInIC(Object* object) {
|
|
|
|
return object->IsMap() && Map::cast(object)->CanTransition() &&
|
|
|
|
FLAG_collect_maps &&
|
|
|
|
FLAG_weak_embedded_maps_in_ic;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* Map::prototype() {
|
|
|
|
return READ_FIELD(this, kPrototypeOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-23 08:46:32 +00:00
|
|
|
void Map::set_prototype(Object* value, WriteBarrierMode mode) {
|
2011-06-03 10:15:49 +00:00
|
|
|
ASSERT(value->IsNull() || value->IsJSReceiver());
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_FIELD(this, kPrototypeOffset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kPrototypeOffset, value, mode);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-13 08:43:16 +00:00
|
|
|
// If the descriptor is using the empty transition array, install a new empty
|
|
|
|
// transition array that will have place for an element transition.
|
2014-04-11 12:13:53 +00:00
|
|
|
static void EnsureHasTransitionArray(Handle<Map> map) {
|
|
|
|
Handle<TransitionArray> transitions;
|
2012-10-11 12:01:19 +00:00
|
|
|
if (!map->HasTransitionArray()) {
|
2014-04-11 12:13:53 +00:00
|
|
|
transitions = TransitionArray::Allocate(map->GetIsolate(), 0);
|
2012-09-19 09:54:10 +00:00
|
|
|
transitions->set_back_pointer_storage(map->GetBackPointer());
|
2012-10-17 13:04:49 +00:00
|
|
|
} else if (!map->transitions()->IsFullTransitionArray()) {
|
2014-04-11 14:25:00 +00:00
|
|
|
transitions = TransitionArray::ExtendToFullTransitionArray(map);
|
2012-10-11 12:01:19 +00:00
|
|
|
} else {
|
2014-04-11 12:13:53 +00:00
|
|
|
return;
|
2012-09-19 09:54:10 +00:00
|
|
|
}
|
2014-04-11 12:13:53 +00:00
|
|
|
map->set_transitions(*transitions);
|
2012-08-13 08:43:16 +00:00
|
|
|
}
|
2012-07-10 13:31:36 +00:00
|
|
|
|
|
|
|
|
2012-10-17 13:04:49 +00:00
|
|
|
void Map::InitializeDescriptors(DescriptorArray* descriptors) {
|
2012-07-18 15:38:58 +00:00
|
|
|
int len = descriptors->number_of_descriptors();
|
2012-10-17 13:04:49 +00:00
|
|
|
set_instance_descriptors(descriptors);
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
SetNumberOfOwnDescriptors(len);
|
2012-07-18 15:38:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-10-17 13:04:49 +00:00
|
|
|
ACCESSORS(Map, instance_descriptors, DescriptorArray, kDescriptorsOffset)
|
2013-08-05 16:42:39 +00:00
|
|
|
|
|
|
|
|
|
|
|
void Map::set_bit_field3(uint32_t bits) {
|
|
|
|
// Ensure the upper 2 bits have the same value by sign extending it. This is
|
|
|
|
// necessary to be able to use the 31st bit.
|
|
|
|
int value = bits << 1;
|
|
|
|
WRITE_FIELD(this, kBitField3Offset, Smi::FromInt(value >> 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t Map::bit_field3() {
|
|
|
|
Object* value = READ_FIELD(this, kBitField3Offset);
|
|
|
|
return Smi::cast(value)->value();
|
|
|
|
}
|
2011-05-23 15:59:38 +00:00
|
|
|
|
|
|
|
|
2014-04-11 12:13:53 +00:00
|
|
|
void Map::AppendDescriptor(Descriptor* desc) {
|
2012-07-19 10:01:52 +00:00
|
|
|
DescriptorArray* descriptors = instance_descriptors();
|
2012-08-27 13:47:34 +00:00
|
|
|
int number_of_own_descriptors = NumberOfOwnDescriptors();
|
2012-09-18 13:25:12 +00:00
|
|
|
ASSERT(descriptors->number_of_descriptors() == number_of_own_descriptors);
|
2014-04-11 12:13:53 +00:00
|
|
|
descriptors->Append(desc);
|
2012-08-27 13:47:34 +00:00
|
|
|
SetNumberOfOwnDescriptors(number_of_own_descriptors + 1);
|
2012-07-19 10:01:52 +00:00
|
|
|
}
|
|
|
|
|
2011-05-23 15:59:38 +00:00
|
|
|
|
2012-05-09 07:29:18 +00:00
|
|
|
Object* Map::GetBackPointer() {
|
2012-08-13 08:43:16 +00:00
|
|
|
Object* object = READ_FIELD(this, kTransitionsOrBackPointerOffset);
|
2012-07-10 13:31:36 +00:00
|
|
|
if (object->IsDescriptorArray()) {
|
2012-08-13 08:43:16 +00:00
|
|
|
return TransitionArray::cast(object)->back_pointer_storage();
|
2012-07-10 13:31:36 +00:00
|
|
|
} else {
|
|
|
|
ASSERT(object->IsMap() || object->IsUndefined());
|
|
|
|
return object;
|
|
|
|
}
|
2012-05-09 07:29:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-05 13:54:20 +00:00
|
|
|
bool Map::HasElementsTransition() {
|
|
|
|
return HasTransitionArray() && transitions()->HasElementsTransition();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::HasTransitionArray() {
|
2012-08-13 08:43:16 +00:00
|
|
|
Object* object = READ_FIELD(this, kTransitionsOrBackPointerOffset);
|
|
|
|
return object->IsTransitionArray();
|
2012-07-05 13:54:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-11 06:59:56 +00:00
|
|
|
Map* Map::elements_transition_map() {
|
2013-07-31 17:08:50 +00:00
|
|
|
int index = transitions()->Search(GetHeap()->elements_transition_symbol());
|
|
|
|
return transitions()->GetTarget(index);
|
2012-06-11 06:59:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-26 08:27:20 +00:00
|
|
|
bool Map::CanHaveMoreTransitions() {
|
|
|
|
if (!HasTransitionArray()) return true;
|
|
|
|
return FixedArray::SizeFor(transitions()->length() +
|
|
|
|
TransitionArray::kTransitionSize)
|
2014-01-23 13:02:27 +00:00
|
|
|
<= Page::kMaxRegularHeapObjectSize;
|
2012-07-26 08:27:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
Sharing of descriptor arrays.
This CL adds multiple things:
Transition arrays do not directly point at their descriptor array anymore, but rather do so via an indirect pointer (a JSGlobalPropertyCell).
An ownership bit is added to maps indicating whether it owns its own descriptor array or not.
Maps owning a descriptor array can pass on ownership if a transition from that map is generated; but only if the descriptor array stays exactly the same; or if a descriptor is added.
Maps that don't have ownership get ownership back if their direct child to which ownership was passed is cleared in ClearNonLiveTransitions.
To detect which descriptors in an array are valid, each map knows its own NumberOfOwnDescriptors. Since the descriptors are sorted in order of addition, if we search and find a descriptor with index bigger than this number, it is not valid for the given map.
We currently still build up an enumeration cache (although this may disappear). The enumeration cache is always built for the entire descriptor array, even if not all descriptors are owned by the map. Once a descriptor array has an enumeration cache for a given map; this invariant will always be true, even if the descriptor array was extended. The extended array will inherit the enumeration cache from the smaller descriptor array. If a map with more descriptors needs an enumeration cache, it's EnumLength will still be set to invalid, so it will have to recompute the enumeration cache. This new cache will also be valid for smaller maps since they have their own enumlength; and use this to loop over the cache. If the EnumLength is still invalid, but there is already a cache present that is big enough; we just initialize the EnumLength field for the map.
When we apply ClearNonLiveTransitions and descriptor ownership is passed back to a parent map, the descriptor array is trimmed in-place and resorted. At the same time, the enumeration cache is trimmed in-place.
Only transition arrays contain descriptor arrays. If we transition to a map and pass ownership of the descriptor array along, the child map will not store the descriptor array it owns. Rather its parent will keep the pointer. So for every leaf-map, we find the descriptor array by following the back pointer, reading out the transition array, and fetching the descriptor array from the JSGlobalPropertyCell. If a map has a transition array, we fetch it from there. If a map has undefined as its back-pointer and has no transition array; it is considered to have an empty descriptor array.
When we modify properties, we cannot share the descriptor array. To accommodate this, the child map will get its own transition array; even if there are not necessarily any transitions leaving from the child map. This is necessary since it's the only way to store its own descriptor array.
Review URL: https://chromiumcodereview.appspot.com/10909007
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12492 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2012-09-12 16:43:57 +00:00
|
|
|
Map* Map::GetTransition(int transition_index) {
|
|
|
|
return transitions()->GetTarget(transition_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-04 04:49:07 +00:00
|
|
|
int Map::SearchTransition(Name* name) {
|
|
|
|
if (HasTransitionArray()) return transitions()->Search(name);
|
|
|
|
return TransitionArray::kNotFound;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-10 07:53:00 +00:00
|
|
|
FixedArray* Map::GetPrototypeTransitions() {
|
|
|
|
if (!HasTransitionArray()) return GetHeap()->empty_fixed_array();
|
|
|
|
if (!transitions()->HasPrototypeTransitions()) {
|
|
|
|
return GetHeap()->empty_fixed_array();
|
|
|
|
}
|
|
|
|
return transitions()->GetPrototypeTransitions();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-11 12:13:53 +00:00
|
|
|
void Map::SetPrototypeTransitions(
|
|
|
|
Handle<Map> map, Handle<FixedArray> proto_transitions) {
|
|
|
|
EnsureHasTransitionArray(map);
|
|
|
|
int old_number_of_transitions = map->NumberOfProtoTransitions();
|
2012-07-10 07:53:00 +00:00
|
|
|
#ifdef DEBUG
|
2014-04-11 12:13:53 +00:00
|
|
|
if (map->HasPrototypeTransitions()) {
|
|
|
|
ASSERT(map->GetPrototypeTransitions() != *proto_transitions);
|
|
|
|
map->ZapPrototypeTransitions();
|
2012-07-10 07:53:00 +00:00
|
|
|
}
|
|
|
|
#endif
|
2014-04-11 12:13:53 +00:00
|
|
|
map->transitions()->SetPrototypeTransitions(*proto_transitions);
|
|
|
|
map->SetNumberOfProtoTransitions(old_number_of_transitions);
|
2012-07-10 07:53:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Map::HasPrototypeTransitions() {
|
|
|
|
return HasTransitionArray() && transitions()->HasPrototypeTransitions();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-05 13:54:20 +00:00
|
|
|
TransitionArray* Map::transitions() {
|
2012-08-13 08:43:16 +00:00
|
|
|
ASSERT(HasTransitionArray());
|
|
|
|
Object* object = READ_FIELD(this, kTransitionsOrBackPointerOffset);
|
|
|
|
return TransitionArray::cast(object);
|
2012-07-05 13:54:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-13 08:43:16 +00:00
|
|
|
void Map::set_transitions(TransitionArray* transition_array,
|
|
|
|
WriteBarrierMode mode) {
|
2013-03-25 15:18:52 +00:00
|
|
|
// Transition arrays are not shared. When one is replaced, it should not
|
|
|
|
// keep referenced objects alive, so we zap it.
|
|
|
|
// When there is another reference to the array somewhere (e.g. a handle),
|
|
|
|
// not zapping turns from a waste of memory into a source of crashes.
|
|
|
|
if (HasTransitionArray()) {
|
2014-01-22 14:02:00 +00:00
|
|
|
#ifdef DEBUG
|
|
|
|
for (int i = 0; i < transitions()->number_of_transitions(); i++) {
|
|
|
|
Map* target = transitions()->GetTarget(i);
|
|
|
|
if (target->instance_descriptors() == instance_descriptors()) {
|
|
|
|
Name* key = transitions()->GetKey(i);
|
|
|
|
int new_target_index = transition_array->Search(key);
|
|
|
|
ASSERT(new_target_index != TransitionArray::kNotFound);
|
|
|
|
ASSERT(transition_array->GetTarget(new_target_index) == target);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2013-03-25 15:18:52 +00:00
|
|
|
ASSERT(transitions() != transition_array);
|
2012-07-05 13:54:20 +00:00
|
|
|
ZapTransitions();
|
|
|
|
}
|
2012-08-13 08:43:16 +00:00
|
|
|
|
|
|
|
WRITE_FIELD(this, kTransitionsOrBackPointerOffset, transition_array);
|
|
|
|
CONDITIONAL_WRITE_BARRIER(
|
|
|
|
GetHeap(), this, kTransitionsOrBackPointerOffset, transition_array, mode);
|
2012-06-11 06:59:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-07-10 07:53:00 +00:00
|
|
|
void Map::init_back_pointer(Object* undefined) {
|
|
|
|
ASSERT(undefined->IsUndefined());
|
2012-08-13 08:43:16 +00:00
|
|
|
WRITE_FIELD(this, kTransitionsOrBackPointerOffset, undefined);
|
2012-07-10 07:53:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-09 07:29:18 +00:00
|
|
|
void Map::SetBackPointer(Object* value, WriteBarrierMode mode) {
|
|
|
|
ASSERT(instance_type() >= FIRST_JS_RECEIVER_TYPE);
|
|
|
|
ASSERT((value->IsUndefined() && GetBackPointer()->IsMap()) ||
|
|
|
|
(value->IsMap() && GetBackPointer()->IsUndefined()));
|
2012-08-13 08:43:16 +00:00
|
|
|
Object* object = READ_FIELD(this, kTransitionsOrBackPointerOffset);
|
|
|
|
if (object->IsTransitionArray()) {
|
|
|
|
TransitionArray::cast(object)->set_back_pointer_storage(value);
|
2012-07-17 13:50:19 +00:00
|
|
|
} else {
|
2012-08-13 08:43:16 +00:00
|
|
|
WRITE_FIELD(this, kTransitionsOrBackPointerOffset, value);
|
2012-07-10 13:31:36 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(
|
2012-08-13 08:43:16 +00:00
|
|
|
GetHeap(), this, kTransitionsOrBackPointerOffset, value, mode);
|
2012-07-10 13:31:36 +00:00
|
|
|
}
|
2012-05-09 07:29:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-09 10:49:41 +00:00
|
|
|
ACCESSORS(Map, code_cache, Object, kCodeCacheOffset)
|
2013-02-20 11:49:54 +00:00
|
|
|
ACCESSORS(Map, dependent_code, DependentCode, kDependentCodeOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(Map, constructor, Object, kConstructorOffset)
|
|
|
|
|
|
|
|
ACCESSORS(JSFunction, shared, SharedFunctionInfo, kSharedFunctionInfoOffset)
|
2011-10-17 12:44:16 +00:00
|
|
|
ACCESSORS(JSFunction, literals_or_bindings, FixedArray, kLiteralsOffset)
|
2012-06-08 07:45:11 +00:00
|
|
|
ACCESSORS(JSFunction, next_function_link, Object, kNextFunctionLinkOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
ACCESSORS(GlobalObject, builtins, JSBuiltinsObject, kBuiltinsOffset)
|
2012-08-17 09:03:08 +00:00
|
|
|
ACCESSORS(GlobalObject, native_context, Context, kNativeContextOffset)
|
2012-08-28 11:25:08 +00:00
|
|
|
ACCESSORS(GlobalObject, global_context, Context, kGlobalContextOffset)
|
Split window support from V8.
Here is a description of the background and design of split window in Chrome and V8:
https://docs.google.com/a/google.com/Doc?id=chhjkpg_47fwddxbfr
This change list splits the window object into two parts: 1) an inner window object used as the global object of contexts; 2) an outer window object exposed to JavaScript and accessible by the name 'window'. Firefox did it awhile ago, here are some discussions: https://wiki.mozilla.org/Gecko:SplitWindow. One additional benefit of splitting window in Chrome is that accessing global variables don't need security checks anymore, it can improve applications that use many global variables.
V8 support of split window:
There are a small number of changes on V8 api to support split window:
Security context is removed from V8, so does related API functions;
A global object can be detached from its context and reused by a new context;
Access checks on an object template can be turned on/off by default;
An object can turn on its access checks later;
V8 has a new object type, ApiGlobalObject, which is the outer window object type. The existing JSGlobalObject becomes the inner window object type. Security checks are moved from JSGlobalObject to ApiGlobalObject. ApiGlobalObject is the one exposed to JavaScript, it is accessible through Context::Global(). ApiGlobalObject's prototype is set to JSGlobalObject so that property lookups are forwarded to JSGlobalObject. ApiGlobalObject forwards all other property access requests to JSGlobalObject, such as SetProperty, DeleteProperty, etc.
Security token is moved to a global context, and ApiGlobalObject has a reference to its global context. JSGlobalObject has a reference to its global context as well. When accessing properties on a global object in JavaScript, the domain security check is performed by comparing the security token of the lexical context (Top::global_context()) to the token of global object's context. The check is only needed when the receiver is a window object, such as 'window.document'. Accessing global variables, such as 'var foo = 3; foo' does not need checks because the receiver is the inner window object.
When an outer window is detached from its global context (when a frame navigates away from a page), it is completely detached from the inner window. A new context is created for the new page, and the outer global object is reused. At this point, the access check on the DOMWindow wrapper of the old context is turned on. The code in old context is still able to access DOMWindow properties, but it has to go through domain security checks.
It is debatable on how to implement the outer window object. Currently each property access function has to check if the receiver is ApiGlobalObject type. This approach might be error-prone that one may forget to check the receiver when adding new functions. It is unlikely a performance issue because accessing global variables are more common than 'window.foo' style coding.
I am still working on the ARM port, and I'd like to hear comments and suggestions on the best way to support it in V8.
Review URL: http://codereview.chromium.org/7366
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@540 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2008-10-21 19:07:58 +00:00
|
|
|
ACCESSORS(GlobalObject, global_receiver, JSObject, kGlobalReceiverOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-08-20 11:35:50 +00:00
|
|
|
ACCESSORS(JSGlobalProxy, native_context, Object, kNativeContextOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
ACCESSORS(AccessorInfo, name, Object, kNameOffset)
|
2012-03-14 16:16:46 +00:00
|
|
|
ACCESSORS_TO_SMI(AccessorInfo, flag, kFlagOffset)
|
2012-06-08 07:45:11 +00:00
|
|
|
ACCESSORS(AccessorInfo, expected_receiver_type, Object,
|
|
|
|
kExpectedReceiverTypeOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-03-07 11:42:58 +00:00
|
|
|
ACCESSORS(DeclaredAccessorDescriptor, serialized_data, ByteArray,
|
|
|
|
kSerializedDataOffset)
|
2013-02-12 14:33:08 +00:00
|
|
|
|
|
|
|
ACCESSORS(DeclaredAccessorInfo, descriptor, DeclaredAccessorDescriptor,
|
|
|
|
kDescriptorOffset)
|
|
|
|
|
|
|
|
ACCESSORS(ExecutableAccessorInfo, getter, Object, kGetterOffset)
|
|
|
|
ACCESSORS(ExecutableAccessorInfo, setter, Object, kSetterOffset)
|
|
|
|
ACCESSORS(ExecutableAccessorInfo, data, Object, kDataOffset)
|
|
|
|
|
2013-06-06 15:40:28 +00:00
|
|
|
ACCESSORS(Box, value, Object, kValueOffset)
|
|
|
|
|
2012-01-10 16:11:33 +00:00
|
|
|
ACCESSORS(AccessorPair, getter, Object, kGetterOffset)
|
|
|
|
ACCESSORS(AccessorPair, setter, Object, kSetterOffset)
|
2013-08-26 11:59:14 +00:00
|
|
|
ACCESSORS_TO_SMI(AccessorPair, access_flags, kAccessFlagsOffset)
|
2012-01-10 16:11:33 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(AccessCheckInfo, named_callback, Object, kNamedCallbackOffset)
|
|
|
|
ACCESSORS(AccessCheckInfo, indexed_callback, Object, kIndexedCallbackOffset)
|
|
|
|
ACCESSORS(AccessCheckInfo, data, Object, kDataOffset)
|
|
|
|
|
|
|
|
ACCESSORS(InterceptorInfo, getter, Object, kGetterOffset)
|
|
|
|
ACCESSORS(InterceptorInfo, setter, Object, kSetterOffset)
|
|
|
|
ACCESSORS(InterceptorInfo, query, Object, kQueryOffset)
|
|
|
|
ACCESSORS(InterceptorInfo, deleter, Object, kDeleterOffset)
|
|
|
|
ACCESSORS(InterceptorInfo, enumerator, Object, kEnumeratorOffset)
|
|
|
|
ACCESSORS(InterceptorInfo, data, Object, kDataOffset)
|
|
|
|
|
|
|
|
ACCESSORS(CallHandlerInfo, callback, Object, kCallbackOffset)
|
|
|
|
ACCESSORS(CallHandlerInfo, data, Object, kDataOffset)
|
|
|
|
|
|
|
|
ACCESSORS(TemplateInfo, tag, Object, kTagOffset)
|
|
|
|
ACCESSORS(TemplateInfo, property_list, Object, kPropertyListOffset)
|
2013-09-04 07:45:36 +00:00
|
|
|
ACCESSORS(TemplateInfo, property_accessors, Object, kPropertyAccessorsOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
ACCESSORS(FunctionTemplateInfo, serial_number, Object, kSerialNumberOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, call_code, Object, kCallCodeOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, prototype_template, Object,
|
|
|
|
kPrototypeTemplateOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, parent_template, Object, kParentTemplateOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, named_property_handler, Object,
|
|
|
|
kNamedPropertyHandlerOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, indexed_property_handler, Object,
|
|
|
|
kIndexedPropertyHandlerOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, instance_template, Object,
|
|
|
|
kInstanceTemplateOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, class_name, Object, kClassNameOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, signature, Object, kSignatureOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, instance_call_handler, Object,
|
|
|
|
kInstanceCallHandlerOffset)
|
|
|
|
ACCESSORS(FunctionTemplateInfo, access_check_info, Object,
|
|
|
|
kAccessCheckInfoOffset)
|
2012-03-14 16:16:46 +00:00
|
|
|
ACCESSORS_TO_SMI(FunctionTemplateInfo, flag, kFlagOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
ACCESSORS(ObjectTemplateInfo, constructor, Object, kConstructorOffset)
|
2008-07-16 07:07:30 +00:00
|
|
|
ACCESSORS(ObjectTemplateInfo, internal_field_count, Object,
|
|
|
|
kInternalFieldCountOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
ACCESSORS(SignatureInfo, receiver, Object, kReceiverOffset)
|
|
|
|
ACCESSORS(SignatureInfo, args, Object, kArgsOffset)
|
|
|
|
|
|
|
|
ACCESSORS(TypeSwitchInfo, types, Object, kTypesOffset)
|
|
|
|
|
2013-07-08 15:00:12 +00:00
|
|
|
ACCESSORS(AllocationSite, transition_info, Object, kTransitionInfoOffset)
|
2013-09-24 10:30:41 +00:00
|
|
|
ACCESSORS(AllocationSite, nested_site, Object, kNestedSiteOffset)
|
2014-01-13 10:28:01 +00:00
|
|
|
ACCESSORS_TO_SMI(AllocationSite, pretenure_data, kPretenureDataOffset)
|
|
|
|
ACCESSORS_TO_SMI(AllocationSite, pretenure_create_count,
|
|
|
|
kPretenureCreateCountOffset)
|
2013-09-19 14:13:34 +00:00
|
|
|
ACCESSORS(AllocationSite, dependent_code, DependentCode,
|
|
|
|
kDependentCodeOffset)
|
2013-07-17 11:50:24 +00:00
|
|
|
ACCESSORS(AllocationSite, weak_next, Object, kWeakNextOffset)
|
2013-07-19 13:30:49 +00:00
|
|
|
ACCESSORS(AllocationMemento, allocation_site, Object, kAllocationSiteOffset)
|
2013-01-08 09:03:16 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(Script, source, Object, kSourceOffset)
|
|
|
|
ACCESSORS(Script, name, Object, kNameOffset)
|
2013-06-25 14:57:47 +00:00
|
|
|
ACCESSORS(Script, id, Smi, kIdOffset)
|
2012-03-14 16:16:46 +00:00
|
|
|
ACCESSORS_TO_SMI(Script, line_offset, kLineOffsetOffset)
|
|
|
|
ACCESSORS_TO_SMI(Script, column_offset, kColumnOffsetOffset)
|
2009-05-06 08:52:48 +00:00
|
|
|
ACCESSORS(Script, context_data, Object, kContextOffset)
|
2011-05-19 11:47:34 +00:00
|
|
|
ACCESSORS(Script, wrapper, Foreign, kWrapperOffset)
|
2012-03-14 16:16:46 +00:00
|
|
|
ACCESSORS_TO_SMI(Script, type, kTypeOffset)
|
2009-11-27 14:10:48 +00:00
|
|
|
ACCESSORS(Script, line_ends, Object, kLineEndsOffset)
|
2009-12-01 14:36:45 +00:00
|
|
|
ACCESSORS(Script, eval_from_shared, Object, kEvalFromSharedOffset)
|
2012-03-14 16:16:46 +00:00
|
|
|
ACCESSORS_TO_SMI(Script, eval_from_instructions_offset,
|
|
|
|
kEvalFrominstructionsOffsetOffset)
|
2013-07-30 17:00:05 +00:00
|
|
|
ACCESSORS_TO_SMI(Script, flags, kFlagsOffset)
|
2013-07-30 17:05:50 +00:00
|
|
|
BOOL_ACCESSORS(Script, flags, is_shared_cross_origin, kIsSharedCrossOriginBit)
|
2013-07-30 17:00:05 +00:00
|
|
|
|
|
|
|
Script::CompilationType Script::compilation_type() {
|
|
|
|
return BooleanBit::get(flags(), kCompilationTypeBit) ?
|
|
|
|
COMPILATION_TYPE_EVAL : COMPILATION_TYPE_HOST;
|
|
|
|
}
|
|
|
|
void Script::set_compilation_type(CompilationType type) {
|
|
|
|
set_flags(BooleanBit::set(flags(), kCompilationTypeBit,
|
|
|
|
type == COMPILATION_TYPE_EVAL));
|
|
|
|
}
|
|
|
|
Script::CompilationState Script::compilation_state() {
|
|
|
|
return BooleanBit::get(flags(), kCompilationStateBit) ?
|
|
|
|
COMPILATION_STATE_COMPILED : COMPILATION_STATE_INITIAL;
|
|
|
|
}
|
|
|
|
void Script::set_compilation_state(CompilationState state) {
|
|
|
|
set_flags(BooleanBit::set(flags(), kCompilationStateBit,
|
|
|
|
state == COMPILATION_STATE_COMPILED));
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
ACCESSORS(DebugInfo, shared, SharedFunctionInfo, kSharedFunctionInfoIndex)
|
|
|
|
ACCESSORS(DebugInfo, original_code, Code, kOriginalCodeIndex)
|
|
|
|
ACCESSORS(DebugInfo, code, Code, kPatchedCodeIndex)
|
|
|
|
ACCESSORS(DebugInfo, break_points, FixedArray, kBreakPointsStateIndex)
|
|
|
|
|
2012-03-14 16:16:46 +00:00
|
|
|
ACCESSORS_TO_SMI(BreakPointInfo, code_position, kCodePositionIndex)
|
|
|
|
ACCESSORS_TO_SMI(BreakPointInfo, source_position, kSourcePositionIndex)
|
|
|
|
ACCESSORS_TO_SMI(BreakPointInfo, statement_position, kStatementPositionIndex)
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(BreakPointInfo, break_point_objects, Object, kBreakPointObjectsIndex)
|
|
|
|
|
|
|
|
ACCESSORS(SharedFunctionInfo, name, Object, kNameOffset)
|
2012-06-14 14:06:22 +00:00
|
|
|
ACCESSORS(SharedFunctionInfo, optimized_code_map, Object,
|
|
|
|
kOptimizedCodeMapOffset)
|
2011-10-05 08:12:36 +00:00
|
|
|
ACCESSORS(SharedFunctionInfo, construct_stub, Code, kConstructStubOffset)
|
2014-04-30 10:51:01 +00:00
|
|
|
ACCESSORS(SharedFunctionInfo, feedback_vector, FixedArray,
|
|
|
|
kFeedbackVectorOffset)
|
2011-10-05 08:12:36 +00:00
|
|
|
ACCESSORS(SharedFunctionInfo, initial_map, Object, kInitialMapOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(SharedFunctionInfo, instance_class_name, Object,
|
|
|
|
kInstanceClassNameOffset)
|
2010-03-11 16:24:31 +00:00
|
|
|
ACCESSORS(SharedFunctionInfo, function_data, Object, kFunctionDataOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(SharedFunctionInfo, script, Object, kScriptOffset)
|
|
|
|
ACCESSORS(SharedFunctionInfo, debug_info, Object, kDebugInfoOffset)
|
2009-04-14 00:51:59 +00:00
|
|
|
ACCESSORS(SharedFunctionInfo, inferred_name, String, kInferredNameOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-02-09 10:19:46 +00:00
|
|
|
|
2012-12-19 10:28:36 +00:00
|
|
|
SMI_ACCESSORS(FunctionTemplateInfo, length, kLengthOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
BOOL_ACCESSORS(FunctionTemplateInfo, flag, hidden_prototype,
|
|
|
|
kHiddenPrototypeBit)
|
|
|
|
BOOL_ACCESSORS(FunctionTemplateInfo, flag, undetectable, kUndetectableBit)
|
|
|
|
BOOL_ACCESSORS(FunctionTemplateInfo, flag, needs_access_check,
|
|
|
|
kNeedsAccessCheckBit)
|
2011-07-25 15:01:45 +00:00
|
|
|
BOOL_ACCESSORS(FunctionTemplateInfo, flag, read_only_prototype,
|
|
|
|
kReadOnlyPrototypeBit)
|
2013-08-26 17:40:03 +00:00
|
|
|
BOOL_ACCESSORS(FunctionTemplateInfo, flag, remove_prototype,
|
|
|
|
kRemovePrototypeBit)
|
2013-09-09 07:52:52 +00:00
|
|
|
BOOL_ACCESSORS(FunctionTemplateInfo, flag, do_not_cache,
|
|
|
|
kDoNotCacheBit)
|
2008-07-03 15:10:15 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, start_position_and_type, is_expression,
|
|
|
|
kIsExpressionBit)
|
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, start_position_and_type, is_toplevel,
|
|
|
|
kIsTopLevelBit)
|
2013-03-12 18:03:18 +00:00
|
|
|
|
2010-06-07 15:39:10 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo,
|
|
|
|
compiler_hints,
|
|
|
|
allows_lazy_compilation,
|
|
|
|
kAllowLazyCompilation)
|
2012-06-19 14:29:48 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo,
|
|
|
|
compiler_hints,
|
|
|
|
allows_lazy_compilation_without_context,
|
|
|
|
kAllowLazyCompilationWithoutContext)
|
2011-06-16 14:12:58 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo,
|
|
|
|
compiler_hints,
|
|
|
|
uses_arguments,
|
|
|
|
kUsesArguments)
|
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo,
|
|
|
|
compiler_hints,
|
|
|
|
has_duplicate_parameters,
|
|
|
|
kHasDuplicateParameters)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-08-17 11:44:01 +00:00
|
|
|
|
2010-05-27 12:30:45 +00:00
|
|
|
#if V8_HOST_ARCH_32_BIT
|
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, length, kLengthOffset)
|
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, formal_parameter_count,
|
2008-07-03 15:10:15 +00:00
|
|
|
kFormalParameterCountOffset)
|
2010-05-27 12:30:45 +00:00
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, expected_nof_properties,
|
2008-07-03 15:10:15 +00:00
|
|
|
kExpectedNofPropertiesOffset)
|
2010-05-27 12:30:45 +00:00
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, num_literals, kNumLiteralsOffset)
|
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, start_position_and_type,
|
2008-07-03 15:10:15 +00:00
|
|
|
kStartPositionAndTypeOffset)
|
2010-05-27 12:30:45 +00:00
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, end_position, kEndPositionOffset)
|
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, function_token_position,
|
2008-07-03 15:10:15 +00:00
|
|
|
kFunctionTokenPositionOffset)
|
2010-05-27 12:30:45 +00:00
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, compiler_hints,
|
2009-08-19 07:30:20 +00:00
|
|
|
kCompilerHintsOffset)
|
2013-09-05 13:20:51 +00:00
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, opt_count_and_bailout_reason,
|
|
|
|
kOptCountAndBailoutReasonOffset)
|
2012-06-11 16:57:27 +00:00
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, counters, kCountersOffset)
|
2014-04-23 07:07:54 +00:00
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, ast_node_count, kAstNodeCountOffset)
|
|
|
|
SMI_ACCESSORS(SharedFunctionInfo, profiler_ticks, kProfilerTicksOffset)
|
2013-07-18 08:12:01 +00:00
|
|
|
|
2010-05-27 12:30:45 +00:00
|
|
|
#else
|
|
|
|
|
|
|
|
#define PSEUDO_SMI_ACCESSORS_LO(holder, name, offset) \
|
2010-11-08 10:30:57 +00:00
|
|
|
STATIC_ASSERT(holder::offset % kPointerSize == 0); \
|
2010-05-27 12:30:45 +00:00
|
|
|
int holder::name() { \
|
|
|
|
int value = READ_INT_FIELD(this, offset); \
|
|
|
|
ASSERT(kHeapObjectTag == 1); \
|
|
|
|
ASSERT((value & kHeapObjectTag) == 0); \
|
|
|
|
return value >> 1; \
|
|
|
|
} \
|
|
|
|
void holder::set_##name(int value) { \
|
|
|
|
ASSERT(kHeapObjectTag == 1); \
|
|
|
|
ASSERT((value & 0xC0000000) == 0xC0000000 || \
|
|
|
|
(value & 0xC0000000) == 0x000000000); \
|
|
|
|
WRITE_INT_FIELD(this, \
|
|
|
|
offset, \
|
|
|
|
(value << 1) & ~kHeapObjectTag); \
|
|
|
|
}
|
|
|
|
|
2010-11-08 10:30:57 +00:00
|
|
|
#define PSEUDO_SMI_ACCESSORS_HI(holder, name, offset) \
|
|
|
|
STATIC_ASSERT(holder::offset % kPointerSize == kIntSize); \
|
2010-05-27 12:30:45 +00:00
|
|
|
INT_ACCESSORS(holder, name, offset)
|
|
|
|
|
|
|
|
|
|
|
|
PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo, length, kLengthOffset)
|
2010-11-08 10:30:57 +00:00
|
|
|
PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo,
|
|
|
|
formal_parameter_count,
|
|
|
|
kFormalParameterCountOffset)
|
2010-05-27 12:30:45 +00:00
|
|
|
|
2010-11-08 10:30:57 +00:00
|
|
|
PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo,
|
|
|
|
expected_nof_properties,
|
|
|
|
kExpectedNofPropertiesOffset)
|
2010-05-27 12:30:45 +00:00
|
|
|
PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo, num_literals, kNumLiteralsOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-11-08 10:30:57 +00:00
|
|
|
PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo, end_position, kEndPositionOffset)
|
|
|
|
PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo,
|
|
|
|
start_position_and_type,
|
|
|
|
kStartPositionAndTypeOffset)
|
|
|
|
|
|
|
|
PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo,
|
|
|
|
function_token_position,
|
|
|
|
kFunctionTokenPositionOffset)
|
|
|
|
PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo,
|
|
|
|
compiler_hints,
|
|
|
|
kCompilerHintsOffset)
|
|
|
|
|
2013-09-05 13:20:51 +00:00
|
|
|
PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo,
|
|
|
|
opt_count_and_bailout_reason,
|
|
|
|
kOptCountAndBailoutReasonOffset)
|
2013-06-05 08:43:25 +00:00
|
|
|
PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo, counters, kCountersOffset)
|
2013-07-18 08:12:01 +00:00
|
|
|
|
2014-04-23 07:07:54 +00:00
|
|
|
PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo,
|
|
|
|
ast_node_count,
|
|
|
|
kAstNodeCountOffset)
|
|
|
|
PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo,
|
|
|
|
profiler_ticks,
|
|
|
|
kProfilerTicksOffset)
|
|
|
|
|
2010-05-27 12:30:45 +00:00
|
|
|
#endif
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-09-23 09:15:26 +00:00
|
|
|
|
|
|
|
int SharedFunctionInfo::construction_count() {
|
|
|
|
return READ_BYTE_FIELD(this, kConstructionCountOffset);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SharedFunctionInfo::set_construction_count(int value) {
|
|
|
|
ASSERT(0 <= value && value < 256);
|
|
|
|
WRITE_BYTE_FIELD(this, kConstructionCountOffset, static_cast<byte>(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-16 14:12:58 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo,
|
|
|
|
compiler_hints,
|
|
|
|
live_objects_may_exist,
|
|
|
|
kLiveObjectsMayExist)
|
2010-09-23 09:15:26 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool SharedFunctionInfo::IsInobjectSlackTrackingInProgress() {
|
2012-02-23 09:12:57 +00:00
|
|
|
return initial_map() != GetHeap()->undefined_value();
|
2010-09-23 09:15:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-16 14:12:58 +00:00
|
|
|
BOOL_GETTER(SharedFunctionInfo,
|
|
|
|
compiler_hints,
|
|
|
|
optimization_disabled,
|
|
|
|
kOptimizationDisabled)
|
2010-12-07 11:31:57 +00:00
|
|
|
|
|
|
|
|
|
|
|
void SharedFunctionInfo::set_optimization_disabled(bool disable) {
|
|
|
|
set_compiler_hints(BooleanBit::set(compiler_hints(),
|
|
|
|
kOptimizationDisabled,
|
|
|
|
disable));
|
|
|
|
// If disabling optimizations we reflect that in the code object so
|
|
|
|
// it will not be counted as optimizable code.
|
|
|
|
if ((code()->kind() == Code::FUNCTION) && disable) {
|
|
|
|
code()->set_optimizable(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-11 14:41:22 +00:00
|
|
|
StrictMode SharedFunctionInfo::strict_mode() {
|
|
|
|
return BooleanBit::get(compiler_hints(), kStrictModeFunction)
|
|
|
|
? STRICT : SLOPPY;
|
2011-10-24 07:47:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-11 14:41:22 +00:00
|
|
|
void SharedFunctionInfo::set_strict_mode(StrictMode strict_mode) {
|
|
|
|
// We only allow mode transitions from sloppy to strict.
|
|
|
|
ASSERT(this->strict_mode() == SLOPPY || this->strict_mode() == strict_mode);
|
2011-11-24 15:17:04 +00:00
|
|
|
int hints = compiler_hints();
|
2014-03-11 14:41:22 +00:00
|
|
|
hints = BooleanBit::set(hints, kStrictModeFunction, strict_mode == STRICT);
|
2011-11-24 15:17:04 +00:00
|
|
|
set_compiler_hints(hints);
|
2011-10-24 07:47:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-08 16:14:46 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, native, kNative)
|
2013-11-12 14:43:18 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, inline_builtin,
|
|
|
|
kInlineBuiltin)
|
2011-08-08 16:14:46 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints,
|
|
|
|
name_should_print_as_anonymous,
|
|
|
|
kNameShouldPrintAsAnonymous)
|
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, bound, kBoundFunction)
|
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_anonymous, kIsAnonymous)
|
2012-02-14 14:14:51 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_function, kIsFunction)
|
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, dont_optimize,
|
|
|
|
kDontOptimize)
|
2012-02-08 09:56:33 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, dont_inline, kDontInline)
|
2012-07-09 08:59:03 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, dont_cache, kDontCache)
|
2013-05-13 10:59:00 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, dont_flush, kDontFlush)
|
2013-04-02 17:34:59 +00:00
|
|
|
BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_generator, kIsGenerator)
|
2011-06-15 10:47:37 +00:00
|
|
|
|
2012-06-14 14:06:22 +00:00
|
|
|
void SharedFunctionInfo::BeforeVisitingPointers() {
|
|
|
|
if (IsInobjectSlackTrackingInProgress()) DetachInitialMap();
|
2012-12-11 17:28:40 +00:00
|
|
|
}
|
|
|
|
|
2012-06-14 14:06:22 +00:00
|
|
|
|
2010-03-09 10:49:41 +00:00
|
|
|
ACCESSORS(CodeCache, default_cache, FixedArray, kDefaultCacheOffset)
|
|
|
|
ACCESSORS(CodeCache, normal_type_cache, Object, kNormalTypeCacheOffset)
|
|
|
|
|
2011-06-06 13:15:11 +00:00
|
|
|
ACCESSORS(PolymorphicCodeCache, cache, Object, kCacheOffset)
|
|
|
|
|
2009-10-07 12:20:02 +00:00
|
|
|
bool Script::HasValidSource() {
|
|
|
|
Object* src = this->source();
|
|
|
|
if (!src->IsString()) return true;
|
|
|
|
String* src_str = String::cast(src);
|
|
|
|
if (!StringShape(src_str).IsExternal()) return true;
|
2012-11-21 10:01:05 +00:00
|
|
|
if (src_str->IsOneByteRepresentation()) {
|
2009-10-07 12:20:02 +00:00
|
|
|
return ExternalAsciiString::cast(src)->resource() != NULL;
|
|
|
|
} else if (src_str->IsTwoByteRepresentation()) {
|
|
|
|
return ExternalTwoByteString::cast(src)->resource() != NULL;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-09-15 15:02:38 +00:00
|
|
|
void SharedFunctionInfo::DontAdaptArguments() {
|
2008-09-16 07:24:46 +00:00
|
|
|
ASSERT(code()->kind() == Code::BUILTIN);
|
2008-09-15 15:02:38 +00:00
|
|
|
set_formal_parameter_count(kDontAdaptArgumentsSentinel);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
int SharedFunctionInfo::start_position() {
|
|
|
|
return start_position_and_type() >> kStartPositionShift;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SharedFunctionInfo::set_start_position(int start_position) {
|
|
|
|
set_start_position_and_type((start_position << kStartPositionShift)
|
|
|
|
| (start_position_and_type() & ~kStartPositionMask));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Code* SharedFunctionInfo::code() {
|
|
|
|
return Code::cast(READ_FIELD(this, kCodeOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-23 08:46:32 +00:00
|
|
|
void SharedFunctionInfo::set_code(Code* value, WriteBarrierMode mode) {
|
2013-11-12 14:43:18 +00:00
|
|
|
ASSERT(value->kind() != Code::OPTIMIZED_FUNCTION);
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_FIELD(this, kCodeOffset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(value->GetHeap(), this, kCodeOffset, value, mode);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-11 13:13:11 +00:00
|
|
|
void SharedFunctionInfo::ReplaceCode(Code* value) {
|
|
|
|
// If the GC metadata field is already used then the function was
|
|
|
|
// enqueued as a code flushing candidate and we remove it now.
|
|
|
|
if (code()->gc_metadata() != NULL) {
|
|
|
|
CodeFlusher* flusher = GetHeap()->mark_compact_collector()->code_flusher();
|
|
|
|
flusher->EvictCandidate(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(code()->gc_metadata() == NULL && value->gc_metadata() == NULL);
|
2014-04-30 10:51:01 +00:00
|
|
|
|
2013-01-11 13:13:11 +00:00
|
|
|
set_code(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-03 10:36:55 +00:00
|
|
|
ScopeInfo* SharedFunctionInfo::scope_info() {
|
|
|
|
return reinterpret_cast<ScopeInfo*>(READ_FIELD(this, kScopeInfoOffset));
|
2010-07-14 11:18:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-11-03 10:36:55 +00:00
|
|
|
void SharedFunctionInfo::set_scope_info(ScopeInfo* value,
|
2010-07-14 11:18:09 +00:00
|
|
|
WriteBarrierMode mode) {
|
|
|
|
WRITE_FIELD(this, kScopeInfoOffset, reinterpret_cast<Object*>(value));
|
2011-09-19 18:36:47 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(),
|
|
|
|
this,
|
|
|
|
kScopeInfoOffset,
|
|
|
|
reinterpret_cast<Object*>(value),
|
|
|
|
mode);
|
2010-07-14 11:18:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool SharedFunctionInfo::is_compiled() {
|
2011-03-18 20:35:07 +00:00
|
|
|
return code() !=
|
2013-12-23 14:30:35 +00:00
|
|
|
GetIsolate()->builtins()->builtin(Builtins::kCompileUnoptimized);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-03-11 16:24:31 +00:00
|
|
|
bool SharedFunctionInfo::IsApiFunction() {
|
|
|
|
return function_data()->IsFunctionTemplateInfo();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
FunctionTemplateInfo* SharedFunctionInfo::get_api_func_data() {
|
|
|
|
ASSERT(IsApiFunction());
|
|
|
|
return FunctionTemplateInfo::cast(function_data());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-14 18:53:48 +00:00
|
|
|
bool SharedFunctionInfo::HasBuiltinFunctionId() {
|
2010-05-06 13:21:53 +00:00
|
|
|
return function_data()->IsSmi();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-14 18:53:48 +00:00
|
|
|
BuiltinFunctionId SharedFunctionInfo::builtin_function_id() {
|
|
|
|
ASSERT(HasBuiltinFunctionId());
|
|
|
|
return static_cast<BuiltinFunctionId>(Smi::cast(function_data())->value());
|
2010-03-11 16:24:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-11 16:57:27 +00:00
|
|
|
int SharedFunctionInfo::ic_age() {
|
|
|
|
return ICAgeBits::decode(counters());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SharedFunctionInfo::set_ic_age(int ic_age) {
|
|
|
|
set_counters(ICAgeBits::update(counters(), ic_age));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int SharedFunctionInfo::deopt_count() {
|
|
|
|
return DeoptCountBits::decode(counters());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SharedFunctionInfo::set_deopt_count(int deopt_count) {
|
|
|
|
set_counters(DeoptCountBits::update(counters(), deopt_count));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SharedFunctionInfo::increment_deopt_count() {
|
|
|
|
int value = counters();
|
|
|
|
int deopt_count = DeoptCountBits::decode(value);
|
|
|
|
deopt_count = (deopt_count + 1) & DeoptCountBits::kMax;
|
|
|
|
set_counters(DeoptCountBits::update(value, deopt_count));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int SharedFunctionInfo::opt_reenable_tries() {
|
|
|
|
return OptReenableTriesBits::decode(counters());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SharedFunctionInfo::set_opt_reenable_tries(int tries) {
|
|
|
|
set_counters(OptReenableTriesBits::update(counters(), tries));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-05 13:20:51 +00:00
|
|
|
int SharedFunctionInfo::opt_count() {
|
|
|
|
return OptCountBits::decode(opt_count_and_bailout_reason());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void SharedFunctionInfo::set_opt_count(int opt_count) {
|
|
|
|
set_opt_count_and_bailout_reason(
|
|
|
|
OptCountBits::update(opt_count_and_bailout_reason(), opt_count));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BailoutReason SharedFunctionInfo::DisableOptimizationReason() {
|
|
|
|
BailoutReason reason = static_cast<BailoutReason>(
|
|
|
|
DisabledOptimizationReasonBits::decode(opt_count_and_bailout_reason()));
|
|
|
|
return reason;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
bool SharedFunctionInfo::has_deoptimization_support() {
|
|
|
|
Code* code = this->code();
|
|
|
|
return code->kind() == Code::FUNCTION && code->has_deoptimization_support();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-11 16:57:27 +00:00
|
|
|
void SharedFunctionInfo::TryReenableOptimization() {
|
|
|
|
int tries = opt_reenable_tries();
|
|
|
|
set_opt_reenable_tries((tries + 1) & OptReenableTriesBits::kMax);
|
|
|
|
// We reenable optimization whenever the number of tries is a large
|
|
|
|
// enough power of 2.
|
|
|
|
if (tries >= 16 && (((tries - 1) & tries) == 0)) {
|
|
|
|
set_optimization_disabled(false);
|
|
|
|
set_opt_count(0);
|
|
|
|
set_deopt_count(0);
|
|
|
|
code()->set_optimizable(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-29 09:51:41 +00:00
|
|
|
bool JSFunction::IsBuiltin() {
|
2012-08-17 12:59:00 +00:00
|
|
|
return context()->global_object()->IsJSBuiltinsObject();
|
2009-07-29 09:51:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
bool JSFunction::NeedsArgumentsAdaption() {
|
|
|
|
return shared()->formal_parameter_count() !=
|
|
|
|
SharedFunctionInfo::kDontAdaptArgumentsSentinel;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool JSFunction::IsOptimized() {
|
|
|
|
return code()->kind() == Code::OPTIMIZED_FUNCTION;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-04-11 13:24:50 +00:00
|
|
|
bool JSFunction::IsOptimizable() {
|
|
|
|
return code()->kind() == Code::FUNCTION && code()->optimizable();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-23 14:30:35 +00:00
|
|
|
bool JSFunction::IsMarkedForOptimization() {
|
|
|
|
return code() == GetIsolate()->builtins()->builtin(
|
|
|
|
Builtins::kCompileOptimized);
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-23 14:30:35 +00:00
|
|
|
bool JSFunction::IsMarkedForConcurrentOptimization() {
|
2013-03-12 18:03:18 +00:00
|
|
|
return code() == GetIsolate()->builtins()->builtin(
|
2013-12-23 14:30:35 +00:00
|
|
|
Builtins::kCompileOptimizedConcurrent);
|
2012-07-19 18:58:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-23 14:30:35 +00:00
|
|
|
bool JSFunction::IsInOptimizationQueue() {
|
2012-07-19 18:58:23 +00:00
|
|
|
return code() == GetIsolate()->builtins()->builtin(
|
2013-12-23 14:30:35 +00:00
|
|
|
Builtins::kInOptimizationQueue);
|
2012-07-19 18:58:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Code* JSFunction::code() {
|
2013-06-26 14:04:25 +00:00
|
|
|
return Code::cast(
|
2010-08-20 07:10:18 +00:00
|
|
|
Code::GetObjectFromEntryAddress(FIELD_ADDR(this, kCodeEntryOffset)));
|
2010-08-17 11:44:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
void JSFunction::set_code(Code* value) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(!GetHeap()->InNewSpace(value));
|
2010-08-20 07:10:18 +00:00
|
|
|
Address entry = value->entry();
|
|
|
|
WRITE_INTPTR_FIELD(this, kCodeEntryOffset, reinterpret_cast<intptr_t>(entry));
|
2011-09-19 18:36:47 +00:00
|
|
|
GetHeap()->incremental_marking()->RecordWriteOfCodeEntry(
|
|
|
|
this,
|
|
|
|
HeapObject::RawField(this, kCodeEntryOffset),
|
|
|
|
value);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-12 18:03:18 +00:00
|
|
|
void JSFunction::set_code_no_write_barrier(Code* value) {
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(!GetHeap()->InNewSpace(value));
|
2013-03-12 18:03:18 +00:00
|
|
|
Address entry = value->entry();
|
|
|
|
WRITE_INTPTR_FIELD(this, kCodeEntryOffset, reinterpret_cast<intptr_t>(entry));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
void JSFunction::ReplaceCode(Code* code) {
|
|
|
|
bool was_optimized = IsOptimized();
|
|
|
|
bool is_optimized = code->kind() == Code::OPTIMIZED_FUNCTION;
|
|
|
|
|
2014-03-11 15:50:41 +00:00
|
|
|
if (was_optimized && is_optimized) {
|
|
|
|
shared()->EvictFromOptimizedCodeMap(this->code(),
|
|
|
|
"Replacing with another optimized code");
|
|
|
|
}
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
set_code(code);
|
|
|
|
|
|
|
|
// Add/remove the function from the list of optimized functions for this
|
|
|
|
// context based on the state change.
|
|
|
|
if (!was_optimized && is_optimized) {
|
2012-08-17 09:03:08 +00:00
|
|
|
context()->native_context()->AddOptimizedFunction(this);
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
if (was_optimized && !is_optimized) {
|
2013-09-04 13:53:24 +00:00
|
|
|
// TODO(titzer): linear in the number of optimized functions; fix!
|
2012-08-17 09:03:08 +00:00
|
|
|
context()->native_context()->RemoveOptimizedFunction(this);
|
2010-12-07 11:31:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Context* JSFunction::context() {
|
|
|
|
return Context::cast(READ_FIELD(this, kContextOffset));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSFunction::set_context(Object* value) {
|
2011-03-18 20:35:07 +00:00
|
|
|
ASSERT(value->IsUndefined() || value->IsContext());
|
2008-07-03 15:10:15 +00:00
|
|
|
WRITE_FIELD(this, kContextOffset, value);
|
2011-09-19 18:36:47 +00:00
|
|
|
WRITE_BARRIER(GetHeap(), this, kContextOffset, value);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ACCESSORS(JSFunction, prototype_or_initial_map, Object,
|
|
|
|
kPrototypeOrInitialMapOffset)
|
|
|
|
|
|
|
|
|
|
|
|
Map* JSFunction::initial_map() {
|
|
|
|
return Map::cast(prototype_or_initial_map());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSFunction::set_initial_map(Map* value) {
|
|
|
|
set_prototype_or_initial_map(value);
|
2012-01-26 21:47:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool JSFunction::has_initial_map() {
|
|
|
|
return prototype_or_initial_map()->IsMap();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool JSFunction::has_instance_prototype() {
|
|
|
|
return has_initial_map() || !prototype_or_initial_map()->IsTheHole();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool JSFunction::has_prototype() {
|
|
|
|
return map()->has_non_instance_prototype() || has_instance_prototype();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object* JSFunction::instance_prototype() {
|
|
|
|
ASSERT(has_instance_prototype());
|
|
|
|
if (has_initial_map()) return initial_map()->prototype();
|
|
|
|
// When there is no initial map and the prototype is a JSObject, the
|
|
|
|
// initial map field is used for the prototype field.
|
|
|
|
return prototype_or_initial_map();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object* JSFunction::prototype() {
|
|
|
|
ASSERT(has_prototype());
|
|
|
|
// If the function's prototype property has been set to a non-JSObject
|
|
|
|
// value, that value is stored in the constructor field of the map.
|
|
|
|
if (map()->has_non_instance_prototype()) return map()->constructor();
|
|
|
|
return instance_prototype();
|
|
|
|
}
|
|
|
|
|
2012-06-06 10:17:26 +00:00
|
|
|
|
2010-04-28 12:05:40 +00:00
|
|
|
bool JSFunction::should_have_prototype() {
|
|
|
|
return map()->function_with_prototype();
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
bool JSFunction::is_compiled() {
|
2013-12-23 14:30:35 +00:00
|
|
|
return code() !=
|
|
|
|
GetIsolate()->builtins()->builtin(Builtins::kCompileUnoptimized);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-17 12:44:16 +00:00
|
|
|
FixedArray* JSFunction::literals() {
|
|
|
|
ASSERT(!shared()->bound());
|
|
|
|
return literals_or_bindings();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSFunction::set_literals(FixedArray* literals) {
|
|
|
|
ASSERT(!shared()->bound());
|
|
|
|
set_literals_or_bindings(literals);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
FixedArray* JSFunction::function_bindings() {
|
|
|
|
ASSERT(shared()->bound());
|
|
|
|
return literals_or_bindings();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSFunction::set_function_bindings(FixedArray* bindings) {
|
|
|
|
ASSERT(shared()->bound());
|
|
|
|
// Bound function literal may be initialized to the empty fixed array
|
|
|
|
// before the bindings are set.
|
|
|
|
ASSERT(bindings == GetHeap()->empty_fixed_array() ||
|
|
|
|
bindings->map() == GetHeap()->fixed_cow_array_map());
|
|
|
|
set_literals_or_bindings(bindings);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-23 14:55:45 +00:00
|
|
|
int JSFunction::NumberOfLiterals() {
|
2011-10-17 12:44:16 +00:00
|
|
|
ASSERT(!shared()->bound());
|
2008-10-23 14:55:45 +00:00
|
|
|
return literals()->length();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Object* JSBuiltinsObject::javascript_builtin(Builtins::JavaScript id) {
|
2011-01-12 11:56:41 +00:00
|
|
|
ASSERT(id < kJSBuiltinsCount); // id is unsigned.
|
2010-04-14 20:16:19 +00:00
|
|
|
return READ_FIELD(this, OffsetOfFunctionWithId(id));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSBuiltinsObject::set_javascript_builtin(Builtins::JavaScript id,
|
|
|
|
Object* value) {
|
2011-01-12 11:56:41 +00:00
|
|
|
ASSERT(id < kJSBuiltinsCount); // id is unsigned.
|
2010-04-14 20:16:19 +00:00
|
|
|
WRITE_FIELD(this, OffsetOfFunctionWithId(id), value);
|
2011-09-19 18:36:47 +00:00
|
|
|
WRITE_BARRIER(GetHeap(), this, OffsetOfFunctionWithId(id), value);
|
2010-04-14 20:16:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Code* JSBuiltinsObject::javascript_builtin_code(Builtins::JavaScript id) {
|
2011-01-12 11:56:41 +00:00
|
|
|
ASSERT(id < kJSBuiltinsCount); // id is unsigned.
|
2010-04-14 20:16:19 +00:00
|
|
|
return Code::cast(READ_FIELD(this, OffsetOfCodeWithId(id)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSBuiltinsObject::set_javascript_builtin_code(Builtins::JavaScript id,
|
|
|
|
Code* value) {
|
2011-01-12 11:56:41 +00:00
|
|
|
ASSERT(id < kJSBuiltinsCount); // id is unsigned.
|
2010-04-14 20:16:19 +00:00
|
|
|
WRITE_FIELD(this, OffsetOfCodeWithId(id), value);
|
2013-09-11 07:14:41 +00:00
|
|
|
ASSERT(!GetHeap()->InNewSpace(value));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-13 10:58:25 +00:00
|
|
|
ACCESSORS(JSProxy, handler, Object, kHandlerOffset)
|
2011-09-22 13:54:53 +00:00
|
|
|
ACCESSORS(JSProxy, hash, Object, kHashOffset)
|
2011-09-13 11:42:57 +00:00
|
|
|
ACCESSORS(JSFunctionProxy, call_trap, Object, kCallTrapOffset)
|
|
|
|
ACCESSORS(JSFunctionProxy, construct_trap, Object, kConstructTrapOffset)
|
|
|
|
|
|
|
|
|
|
|
|
void JSProxy::InitializeBody(int object_size, Object* value) {
|
|
|
|
ASSERT(!value->IsHeapObject() || !GetHeap()->InNewSpace(value));
|
|
|
|
for (int offset = kHeaderSize; offset < object_size; offset += kPointerSize) {
|
|
|
|
WRITE_FIELD(this, offset, value);
|
|
|
|
}
|
|
|
|
}
|
2011-05-13 10:58:25 +00:00
|
|
|
|
|
|
|
|
2011-10-25 14:14:56 +00:00
|
|
|
ACCESSORS(JSSet, table, Object, kTableOffset)
|
|
|
|
ACCESSORS(JSMap, table, Object, kTableOffset)
|
2014-04-17 17:45:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
#define ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(name, type, offset) \
|
|
|
|
template<class Derived, class TableType> \
|
|
|
|
type* OrderedHashTableIterator<Derived, TableType>::name() { \
|
|
|
|
return type::cast(READ_FIELD(this, offset)); \
|
|
|
|
} \
|
|
|
|
template<class Derived, class TableType> \
|
|
|
|
void OrderedHashTableIterator<Derived, TableType>::set_##name( \
|
|
|
|
type* value, WriteBarrierMode mode) { \
|
|
|
|
WRITE_FIELD(this, offset, value); \
|
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, offset, value, mode); \
|
|
|
|
}
|
|
|
|
|
|
|
|
ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(table, Object, kTableOffset)
|
|
|
|
ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(index, Smi, kIndexOffset)
|
|
|
|
ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(count, Smi, kCountOffset)
|
|
|
|
ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(kind, Smi, kKindOffset)
|
|
|
|
ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(next_iterator, Object,
|
|
|
|
kNextIteratorOffset)
|
|
|
|
ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(previous_iterator, Object,
|
|
|
|
kPreviousIteratorOffset)
|
|
|
|
|
|
|
|
#undef ORDERED_HASH_TABLE_ITERATOR_ACCESSORS
|
|
|
|
|
|
|
|
|
2013-07-22 08:32:24 +00:00
|
|
|
ACCESSORS(JSWeakCollection, table, Object, kTableOffset)
|
|
|
|
ACCESSORS(JSWeakCollection, next, Object, kNextOffset)
|
2011-08-03 12:48:30 +00:00
|
|
|
|
|
|
|
|
2011-10-28 12:37:29 +00:00
|
|
|
Address Foreign::foreign_address() {
|
|
|
|
return AddressFrom<Address>(READ_INTPTR_FIELD(this, kForeignAddressOffset));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-10-28 12:37:29 +00:00
|
|
|
void Foreign::set_foreign_address(Address value) {
|
|
|
|
WRITE_INTPTR_FIELD(this, kForeignAddressOffset, OffsetFrom(value));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-04-15 12:29:44 +00:00
|
|
|
ACCESSORS(JSGeneratorObject, function, JSFunction, kFunctionOffset)
|
2013-04-19 14:11:23 +00:00
|
|
|
ACCESSORS(JSGeneratorObject, context, Context, kContextOffset)
|
2013-04-25 10:59:09 +00:00
|
|
|
ACCESSORS(JSGeneratorObject, receiver, Object, kReceiverOffset)
|
2013-04-15 12:29:44 +00:00
|
|
|
SMI_ACCESSORS(JSGeneratorObject, continuation, kContinuationOffset)
|
|
|
|
ACCESSORS(JSGeneratorObject, operand_stack, FixedArray, kOperandStackOffset)
|
2013-05-08 08:08:23 +00:00
|
|
|
SMI_ACCESSORS(JSGeneratorObject, stack_handler_index, kStackHandlerIndexOffset)
|
2013-04-15 12:29:44 +00:00
|
|
|
|
2014-05-05 14:31:51 +00:00
|
|
|
bool JSGeneratorObject::is_suspended() {
|
|
|
|
ASSERT_LT(kGeneratorExecuting, kGeneratorClosed);
|
|
|
|
ASSERT_EQ(kGeneratorClosed, 0);
|
|
|
|
return continuation() > 0;
|
|
|
|
}
|
2013-04-15 12:29:44 +00:00
|
|
|
|
|
|
|
JSGeneratorObject* JSGeneratorObject::cast(Object* obj) {
|
|
|
|
ASSERT(obj->IsJSGeneratorObject());
|
|
|
|
ASSERT(HeapObject::cast(obj)->Size() == JSGeneratorObject::kSize);
|
|
|
|
return reinterpret_cast<JSGeneratorObject*>(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-04-16 14:43:27 +00:00
|
|
|
ACCESSORS(JSModule, context, Object, kContextOffset)
|
2012-07-09 08:59:03 +00:00
|
|
|
ACCESSORS(JSModule, scope_info, ScopeInfo, kScopeInfoOffset)
|
2012-04-16 14:43:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
JSModule* JSModule::cast(Object* obj) {
|
|
|
|
ASSERT(obj->IsJSModule());
|
|
|
|
ASSERT(HeapObject::cast(obj)->Size() == JSModule::kSize);
|
|
|
|
return reinterpret_cast<JSModule*>(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
ACCESSORS(JSValue, value, Object, kValueOffset)
|
|
|
|
|
|
|
|
|
|
|
|
JSValue* JSValue::cast(Object* obj) {
|
|
|
|
ASSERT(obj->IsJSValue());
|
|
|
|
ASSERT(HeapObject::cast(obj)->Size() == JSValue::kSize);
|
|
|
|
return reinterpret_cast<JSValue*>(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-09 11:11:55 +00:00
|
|
|
ACCESSORS(JSDate, value, Object, kValueOffset)
|
2012-03-09 12:07:29 +00:00
|
|
|
ACCESSORS(JSDate, cache_stamp, Object, kCacheStampOffset)
|
2012-03-09 11:11:55 +00:00
|
|
|
ACCESSORS(JSDate, year, Object, kYearOffset)
|
|
|
|
ACCESSORS(JSDate, month, Object, kMonthOffset)
|
|
|
|
ACCESSORS(JSDate, day, Object, kDayOffset)
|
2012-03-09 12:07:29 +00:00
|
|
|
ACCESSORS(JSDate, weekday, Object, kWeekdayOffset)
|
2012-03-09 11:11:55 +00:00
|
|
|
ACCESSORS(JSDate, hour, Object, kHourOffset)
|
|
|
|
ACCESSORS(JSDate, min, Object, kMinOffset)
|
|
|
|
ACCESSORS(JSDate, sec, Object, kSecOffset)
|
|
|
|
|
|
|
|
|
|
|
|
JSDate* JSDate::cast(Object* obj) {
|
|
|
|
ASSERT(obj->IsJSDate());
|
|
|
|
ASSERT(HeapObject::cast(obj)->Size() == JSDate::kSize);
|
|
|
|
return reinterpret_cast<JSDate*>(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-02-02 13:31:52 +00:00
|
|
|
ACCESSORS(JSMessageObject, type, String, kTypeOffset)
|
|
|
|
ACCESSORS(JSMessageObject, arguments, JSArray, kArgumentsOffset)
|
|
|
|
ACCESSORS(JSMessageObject, script, Object, kScriptOffset)
|
|
|
|
ACCESSORS(JSMessageObject, stack_frames, Object, kStackFramesOffset)
|
|
|
|
SMI_ACCESSORS(JSMessageObject, start_position, kStartPositionOffset)
|
|
|
|
SMI_ACCESSORS(JSMessageObject, end_position, kEndPositionOffset)
|
|
|
|
|
|
|
|
|
|
|
|
JSMessageObject* JSMessageObject::cast(Object* obj) {
|
|
|
|
ASSERT(obj->IsJSMessageObject());
|
|
|
|
ASSERT(HeapObject::cast(obj)->Size() == JSMessageObject::kSize);
|
|
|
|
return reinterpret_cast<JSMessageObject*>(obj);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
INT_ACCESSORS(Code, instruction_size, kInstructionSizeOffset)
|
2012-11-29 07:38:00 +00:00
|
|
|
INT_ACCESSORS(Code, prologue_offset, kPrologueOffset)
|
2010-07-05 11:45:11 +00:00
|
|
|
ACCESSORS(Code, relocation_info, ByteArray, kRelocationInfoOffset)
|
2011-11-11 13:48:14 +00:00
|
|
|
ACCESSORS(Code, handler_table, FixedArray, kHandlerTableOffset)
|
2010-12-07 11:31:57 +00:00
|
|
|
ACCESSORS(Code, deoptimization_data, FixedArray, kDeoptimizationDataOffset)
|
2013-11-05 10:05:03 +00:00
|
|
|
ACCESSORS(Code, raw_type_feedback_info, Object, kTypeFeedbackInfoOffset)
|
2014-03-26 15:14:51 +00:00
|
|
|
ACCESSORS(Code, next_code_link, Object, kNextCodeLinkOffset)
|
2012-11-14 15:59:45 +00:00
|
|
|
|
|
|
|
|
2013-11-05 10:14:48 +00:00
|
|
|
void Code::WipeOutHeader() {
|
|
|
|
WRITE_FIELD(this, kRelocationInfoOffset, NULL);
|
|
|
|
WRITE_FIELD(this, kHandlerTableOffset, NULL);
|
|
|
|
WRITE_FIELD(this, kDeoptimizationDataOffset, NULL);
|
2014-03-11 20:52:00 +00:00
|
|
|
WRITE_FIELD(this, kConstantPoolOffset, NULL);
|
2013-11-05 10:14:48 +00:00
|
|
|
// Do not wipe out e.g. a minor key.
|
|
|
|
if (!READ_FIELD(this, kTypeFeedbackInfoOffset)->IsSmi()) {
|
|
|
|
WRITE_FIELD(this, kTypeFeedbackInfoOffset, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-14 15:59:45 +00:00
|
|
|
Object* Code::type_feedback_info() {
|
|
|
|
ASSERT(kind() == FUNCTION);
|
2013-11-05 10:05:03 +00:00
|
|
|
return raw_type_feedback_info();
|
2012-11-14 15:59:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_type_feedback_info(Object* value, WriteBarrierMode mode) {
|
|
|
|
ASSERT(kind() == FUNCTION);
|
2013-11-05 10:05:03 +00:00
|
|
|
set_raw_type_feedback_info(value, mode);
|
2012-11-14 15:59:45 +00:00
|
|
|
CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kTypeFeedbackInfoOffset,
|
|
|
|
value, mode);
|
2014-03-13 15:10:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-14 15:59:45 +00:00
|
|
|
int Code::stub_info() {
|
2013-04-24 11:32:17 +00:00
|
|
|
ASSERT(kind() == COMPARE_IC || kind() == COMPARE_NIL_IC ||
|
2014-04-30 14:33:35 +00:00
|
|
|
kind() == BINARY_OP_IC || kind() == LOAD_IC || kind() == CALL_IC);
|
2013-11-05 10:05:03 +00:00
|
|
|
return Smi::cast(raw_type_feedback_info())->value();
|
2012-11-14 15:59:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Code::set_stub_info(int value) {
|
2013-01-21 14:53:29 +00:00
|
|
|
ASSERT(kind() == COMPARE_IC ||
|
2013-04-24 11:32:17 +00:00
|
|
|
kind() == COMPARE_NIL_IC ||
|
2013-01-21 14:53:29 +00:00
|
|
|
kind() == BINARY_OP_IC ||
|
2013-03-05 17:38:35 +00:00
|
|
|
kind() == STUB ||
|
2013-01-21 14:53:29 +00:00
|
|
|
kind() == LOAD_IC ||
|
2014-04-30 14:33:35 +00:00
|
|
|
kind() == CALL_IC ||
|
2013-01-25 11:55:29 +00:00
|
|
|
kind() == KEYED_LOAD_IC ||
|
|
|
|
kind() == STORE_IC ||
|
|
|
|
kind() == KEYED_STORE_IC);
|
2013-11-05 10:05:03 +00:00
|
|
|
set_raw_type_feedback_info(Smi::FromInt(value));
|
2012-11-14 15:59:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-25 15:11:59 +00:00
|
|
|
ACCESSORS(Code, gc_metadata, Object, kGCMetadataOffset)
|
2012-03-23 13:33:11 +00:00
|
|
|
INT_ACCESSORS(Code, ic_age, kICAgeOffset)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-11-14 15:59:45 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
byte* Code::instruction_start() {
|
|
|
|
return FIELD_ADDR(this, kHeaderSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-07-05 11:45:11 +00:00
|
|
|
byte* Code::instruction_end() {
|
|
|
|
return instruction_start() + instruction_size();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
int Code::body_size() {
|
2010-07-05 11:45:11 +00:00
|
|
|
return RoundUp(instruction_size(), kObjectAlignment);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ByteArray* Code::unchecked_relocation_info() {
|
|
|
|
return reinterpret_cast<ByteArray*>(READ_FIELD(this, kRelocationInfoOffset));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
byte* Code::relocation_start() {
|
2010-07-05 11:45:11 +00:00
|
|
|
return unchecked_relocation_info()->GetDataStartAddress();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Code::relocation_size() {
|
|
|
|
return unchecked_relocation_info()->length();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
byte* Code::entry() {
|
|
|
|
return instruction_start();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-20 10:08:39 +00:00
|
|
|
bool Code::contains(byte* inner_pointer) {
|
|
|
|
return (address() <= inner_pointer) && (inner_pointer <= address() + Size());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ACCESSORS(JSArray, length, Object, kLengthOffset)
|
|
|
|
|
|
|
|
|
2013-03-28 12:50:18 +00:00
|
|
|
void* JSArrayBuffer::backing_store() {
|
|
|
|
intptr_t ptr = READ_INTPTR_FIELD(this, kBackingStoreOffset);
|
|
|
|
return reinterpret_cast<void*>(ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSArrayBuffer::set_backing_store(void* value, WriteBarrierMode mode) {
|
|
|
|
intptr_t ptr = reinterpret_cast<intptr_t>(value);
|
|
|
|
WRITE_INTPTR_FIELD(this, kBackingStoreOffset, ptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ACCESSORS(JSArrayBuffer, byte_length, Object, kByteLengthOffset)
|
2013-05-23 10:01:42 +00:00
|
|
|
ACCESSORS_TO_SMI(JSArrayBuffer, flag, kFlagOffset)
|
|
|
|
|
|
|
|
|
|
|
|
bool JSArrayBuffer::is_external() {
|
|
|
|
return BooleanBit::get(flag(), kIsExternalBit);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSArrayBuffer::set_is_external(bool value) {
|
|
|
|
set_flag(BooleanBit::set(flag(), kIsExternalBit, value));
|
|
|
|
}
|
2013-03-28 12:50:18 +00:00
|
|
|
|
|
|
|
|
2013-11-22 11:35:39 +00:00
|
|
|
bool JSArrayBuffer::should_be_freed() {
|
|
|
|
return BooleanBit::get(flag(), kShouldBeFreed);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void JSArrayBuffer::set_should_be_freed(bool value) {
|
|
|
|
set_flag(BooleanBit::set(flag(), kShouldBeFreed, value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-06-07 10:52:11 +00:00
|
|
|
ACCESSORS(JSArrayBuffer, weak_next, Object, kWeakNextOffset)
|
2013-06-21 13:02:38 +00:00
|
|
|
ACCESSORS(JSArrayBuffer, weak_first_view, Object, kWeakFirstViewOffset)
|
2013-06-07 10:52:11 +00:00
|
|
|
|
|
|
|
|
2013-06-21 13:02:38 +00:00
|
|
|
ACCESSORS(JSArrayBufferView, buffer, Object, kBufferOffset)
|
|
|
|
ACCESSORS(JSArrayBufferView, byte_offset, Object, kByteOffsetOffset)
|
|
|
|
ACCESSORS(JSArrayBufferView, byte_length, Object, kByteLengthOffset)
|
|
|
|
ACCESSORS(JSArrayBufferView, weak_next, Object, kWeakNextOffset)
|
2013-04-16 14:16:30 +00:00
|
|
|
ACCESSORS(JSTypedArray, length, Object, kLengthOffset)
|
|
|
|
|
2008-09-23 11:45:43 +00:00
|
|
|
ACCESSORS(JSRegExp, data, Object, kDataOffset)
|
|
|
|
|
|
|
|
|
2008-10-24 08:40:02 +00:00
|
|
|
JSRegExp::Type JSRegExp::TypeTag() {
|
|
|
|
Object* data = this->data();
|
|
|
|
if (data->IsUndefined()) return JSRegExp::NOT_COMPILED;
|
|
|
|
Smi* smi = Smi::cast(FixedArray::cast(data)->get(kTagIndex));
|
|
|
|
return static_cast<JSRegExp::Type>(smi->value());
|
2008-09-23 11:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-13 10:22:38 +00:00
|
|
|
int JSRegExp::CaptureCount() {
|
|
|
|
switch (TypeTag()) {
|
|
|
|
case ATOM:
|
|
|
|
return 0;
|
|
|
|
case IRREGEXP:
|
|
|
|
return Smi::cast(DataAt(kIrregexpCaptureCountIndex))->value();
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-11-25 11:07:48 +00:00
|
|
|
JSRegExp::Flags JSRegExp::GetFlags() {
|
|
|
|
ASSERT(this->data()->IsFixedArray());
|
|
|
|
Object* data = this->data();
|
|
|
|
Smi* smi = Smi::cast(FixedArray::cast(data)->get(kFlagsIndex));
|
|
|
|
return Flags(smi->value());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String* JSRegExp::Pattern() {
|
|
|
|
ASSERT(this->data()->IsFixedArray());
|
|
|
|
Object* data = this->data();
|
|
|
|
String* pattern= String::cast(FixedArray::cast(data)->get(kSourceIndex));
|
|
|
|
return pattern;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-24 08:40:02 +00:00
|
|
|
Object* JSRegExp::DataAt(int index) {
|
|
|
|
ASSERT(TypeTag() != NOT_COMPILED);
|
|
|
|
return FixedArray::cast(data())->get(index);
|
2008-09-23 11:45:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-11 14:00:55 +00:00
|
|
|
void JSRegExp::SetDataAt(int index, Object* value) {
|
|
|
|
ASSERT(TypeTag() != NOT_COMPILED);
|
|
|
|
ASSERT(index >= kDataIndex); // Only implementation data can be set this way.
|
|
|
|
FixedArray::cast(data())->set(index, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-09 09:35:57 +00:00
|
|
|
ElementsKind JSObject::GetElementsKind() {
|
2011-06-03 07:41:37 +00:00
|
|
|
ElementsKind kind = map()->elements_kind();
|
2011-09-22 11:30:04 +00:00
|
|
|
#if DEBUG
|
|
|
|
FixedArrayBase* fixed_array =
|
|
|
|
reinterpret_cast<FixedArrayBase*>(READ_FIELD(this, kElementsOffset));
|
2013-10-25 12:26:47 +00:00
|
|
|
|
|
|
|
// If a GC was caused while constructing this object, the elements
|
|
|
|
// pointer may point to a one pointer filler map.
|
|
|
|
if (ElementsAreSafeToExamine()) {
|
|
|
|
Map* map = fixed_array->map();
|
|
|
|
ASSERT((IsFastSmiOrObjectElementsKind(kind) &&
|
|
|
|
(map == GetHeap()->fixed_array_map() ||
|
|
|
|
map == GetHeap()->fixed_cow_array_map())) ||
|
|
|
|
(IsFastDoubleElementsKind(kind) &&
|
|
|
|
(fixed_array->IsFixedDoubleArray() ||
|
|
|
|
fixed_array == GetHeap()->empty_fixed_array())) ||
|
|
|
|
(kind == DICTIONARY_ELEMENTS &&
|
2011-10-18 11:32:57 +00:00
|
|
|
fixed_array->IsFixedArray() &&
|
2013-10-25 12:26:47 +00:00
|
|
|
fixed_array->IsDictionary()) ||
|
|
|
|
(kind > DICTIONARY_ELEMENTS));
|
2014-03-11 14:39:08 +00:00
|
|
|
ASSERT((kind != SLOPPY_ARGUMENTS_ELEMENTS) ||
|
2013-10-25 12:26:47 +00:00
|
|
|
(elements()->IsFixedArray() && elements()->length() >= 2));
|
|
|
|
}
|
2011-09-22 11:30:04 +00:00
|
|
|
#endif
|
2011-06-03 07:41:37 +00:00
|
|
|
return kind;
|
2009-07-28 08:43:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-08-03 11:12:46 +00:00
|
|
|
ElementsAccessor* JSObject::GetElementsAccessor() {
|
|
|
|
return ElementsAccessor::ForKind(GetElementsKind());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-23 14:24:29 +00:00
|
|
|
bool JSObject::HasFastObjectElements() {
|
|
|
|
return IsFastObjectElementsKind(GetElementsKind());
|
2009-07-28 08:43:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-23 14:24:29 +00:00
|
|
|
bool JSObject::HasFastSmiElements() {
|
|
|
|
return IsFastSmiElementsKind(GetElementsKind());
|
2011-09-22 11:30:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-23 14:24:29 +00:00
|
|
|
bool JSObject::HasFastSmiOrObjectElements() {
|
|
|
|
return IsFastSmiOrObjectElementsKind(GetElementsKind());
|
2011-09-22 11:30:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-09 10:03:35 +00:00
|
|
|
bool JSObject::HasFastDoubleElements() {
|
2012-05-23 14:24:29 +00:00
|
|
|
return IsFastDoubleElementsKind(GetElementsKind());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool JSObject::HasFastHoleyElements() {
|
|
|
|
return IsFastHoleyElementsKind(GetElementsKind());
|
2011-06-09 10:03:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-11-15 12:19:14 +00:00
|
|
|
bool JSObject::HasFastElements() {
|
|
|
|
return IsFastElementsKind(GetElementsKind());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-28 08:43:51 +00:00
|
|
|
bool JSObject::HasDictionaryElements() {
|
|
|
|
return GetElementsKind() == DICTIONARY_ELEMENTS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-11 14:39:08 +00:00
|
|
|
bool JSObject::HasSloppyArgumentsElements() {
|
|
|
|
return GetElementsKind() == SLOPPY_ARGUMENTS_ELEMENTS;
|
2011-09-22 11:30:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-10-20 15:26:17 +00:00
|
|
|
bool JSObject::HasExternalArrayElements() {
|
2011-03-09 15:01:16 +00:00
|
|
|
HeapObject* array = elements();
|
|
|
|
ASSERT(array != NULL);
|
|
|
|
return array->IsExternalArray();
|
2009-10-20 15:26:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
#define EXTERNAL_ELEMENTS_CHECK(Type, type, TYPE, ctype, size) \
|
|
|
|
bool JSObject::HasExternal##Type##Elements() { \
|
|
|
|
HeapObject* array = elements(); \
|
|
|
|
ASSERT(array != NULL); \
|
|
|
|
if (!array->IsHeapObject()) \
|
|
|
|
return false; \
|
|
|
|
return array->map()->instance_type() == EXTERNAL_##TYPE##_ARRAY_TYPE; \
|
2009-10-20 15:26:17 +00:00
|
|
|
}
|
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
TYPED_ARRAYS(EXTERNAL_ELEMENTS_CHECK)
|
2009-10-20 15:26:17 +00:00
|
|
|
|
2014-01-24 16:01:15 +00:00
|
|
|
#undef EXTERNAL_ELEMENTS_CHECK
|
2009-10-20 15:26:17 +00:00
|
|
|
|
|
|
|
|
2014-01-16 17:08:45 +00:00
|
|
|
bool JSObject::HasFixedTypedArrayElements() {
|
|
|
|
HeapObject* array = elements();
|
|
|
|
ASSERT(array != NULL);
|
|
|
|
return array->IsFixedTypedArrayBase();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-26 12:50:13 +00:00
|
|
|
#define FIXED_TYPED_ELEMENTS_CHECK(Type, type, TYPE, ctype, size) \
|
|
|
|
bool JSObject::HasFixed##Type##Elements() { \
|
|
|
|
HeapObject* array = elements(); \
|
|
|
|
ASSERT(array != NULL); \
|
|
|
|
if (!array->IsHeapObject()) \
|
|
|
|
return false; \
|
|
|
|
return array->map()->instance_type() == FIXED_##TYPE##_ARRAY_TYPE; \
|
|
|
|
}
|
|
|
|
|
|
|
|
TYPED_ARRAYS(FIXED_TYPED_ELEMENTS_CHECK)
|
|
|
|
|
|
|
|
#undef FIXED_TYPED_ELEMENTS_CHECK
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool JSObject::HasNamedInterceptor() {
|
|
|
|
return map()->has_named_interceptor();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool JSObject::HasIndexedInterceptor() {
|
|
|
|
return map()->has_indexed_interceptor();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-04 15:00:57 +00:00
|
|
|
NameDictionary* JSObject::property_dictionary() {
|
2008-07-03 15:10:15 +00:00
|
|
|
ASSERT(!HasFastProperties());
|
2013-03-04 15:00:57 +00:00
|
|
|
return NameDictionary::cast(properties());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-16 09:44:35 +00:00
|
|
|
SeededNumberDictionary* JSObject::element_dictionary() {
|
2009-07-28 08:43:51 +00:00
|
|
|
ASSERT(HasDictionaryElements());
|
2012-01-16 09:44:35 +00:00
|
|
|
return SeededNumberDictionary::cast(elements());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-17 17:45:32 +00:00
|
|
|
Handle<JSSetIterator> JSSetIterator::Create(
|
|
|
|
Handle<OrderedHashSet> table,
|
|
|
|
int kind) {
|
|
|
|
return CreateInternal(table->GetIsolate()->set_iterator_map(), table, kind);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Handle<JSMapIterator> JSMapIterator::Create(
|
|
|
|
Handle<OrderedHashMap> table,
|
|
|
|
int kind) {
|
|
|
|
return CreateInternal(table->GetIsolate()->map_iterator_map(), table, kind);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-01 10:34:31 +00:00
|
|
|
bool Name::IsHashFieldComputed(uint32_t field) {
|
2010-05-27 12:30:45 +00:00
|
|
|
return (field & kHashNotComputedMask) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-01 10:34:31 +00:00
|
|
|
bool Name::HasHashCode() {
|
2010-05-27 12:30:45 +00:00
|
|
|
return IsHashFieldComputed(hash_field());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-01 10:34:31 +00:00
|
|
|
uint32_t Name::Hash() {
|
2008-07-03 15:10:15 +00:00
|
|
|
// Fast case: has hash code already been computed?
|
2009-11-24 14:10:06 +00:00
|
|
|
uint32_t field = hash_field();
|
2010-05-27 12:30:45 +00:00
|
|
|
if (IsHashFieldComputed(field)) return field >> kHashShift;
|
2013-03-01 10:34:31 +00:00
|
|
|
// Slow case: compute hash code and set it. Has to be a string.
|
|
|
|
return String::cast(this)->ComputeAndSetHash();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-04 15:12:15 +00:00
|
|
|
StringHasher::StringHasher(int length, uint32_t seed)
|
2008-10-07 10:10:03 +00:00
|
|
|
: length_(length),
|
2012-01-04 15:12:15 +00:00
|
|
|
raw_running_hash_(seed),
|
2008-10-07 10:10:03 +00:00
|
|
|
array_index_(0),
|
2010-06-07 09:36:30 +00:00
|
|
|
is_array_index_(0 < length_ && length_ <= String::kMaxArrayIndexSize),
|
2012-09-24 14:23:46 +00:00
|
|
|
is_first_char_(true) {
|
2012-01-10 13:24:18 +00:00
|
|
|
ASSERT(FLAG_randomize_hashes || raw_running_hash_ == 0);
|
2012-01-04 15:12:15 +00:00
|
|
|
}
|
2008-10-07 10:10:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
bool StringHasher::has_trivial_hash() {
|
2009-11-24 14:10:06 +00:00
|
|
|
return length_ > String::kMaxHashCalcLength;
|
2008-10-07 10:10:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-19 13:27:20 +00:00
|
|
|
uint32_t StringHasher::AddCharacterCore(uint32_t running_hash, uint16_t c) {
|
2012-09-24 14:23:46 +00:00
|
|
|
running_hash += c;
|
|
|
|
running_hash += (running_hash << 10);
|
|
|
|
running_hash ^= (running_hash >> 6);
|
|
|
|
return running_hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t StringHasher::GetHashCore(uint32_t running_hash) {
|
|
|
|
running_hash += (running_hash << 3);
|
|
|
|
running_hash ^= (running_hash >> 11);
|
|
|
|
running_hash += (running_hash << 15);
|
|
|
|
if ((running_hash & String::kHashBitMask) == 0) {
|
2012-10-22 10:38:40 +00:00
|
|
|
return kZeroHash;
|
2012-09-24 14:23:46 +00:00
|
|
|
}
|
|
|
|
return running_hash;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-19 13:27:20 +00:00
|
|
|
void StringHasher::AddCharacter(uint16_t c) {
|
2008-10-23 06:20:57 +00:00
|
|
|
// Use the Jenkins one-at-a-time hash function to update the hash
|
|
|
|
// for the given character.
|
2012-09-24 14:23:46 +00:00
|
|
|
raw_running_hash_ = AddCharacterCore(raw_running_hash_, c);
|
2008-10-07 10:10:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-19 13:27:20 +00:00
|
|
|
bool StringHasher::UpdateIndex(uint16_t c) {
|
|
|
|
ASSERT(is_array_index_);
|
|
|
|
if (c < '0' || c > '9') {
|
|
|
|
is_array_index_ = false;
|
|
|
|
return false;
|
2012-03-12 12:35:28 +00:00
|
|
|
}
|
2012-12-19 13:27:20 +00:00
|
|
|
int d = c - '0';
|
|
|
|
if (is_first_char_) {
|
|
|
|
is_first_char_ = false;
|
|
|
|
if (c == '0' && length_ > 1) {
|
|
|
|
is_array_index_ = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (array_index_ > 429496729U - ((d + 2) >> 3)) {
|
|
|
|
is_array_index_ = false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
array_index_ = array_index_ * 10 + d;
|
|
|
|
return true;
|
2008-10-07 10:10:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-12-19 13:27:20 +00:00
|
|
|
template<typename Char>
|
|
|
|
inline void StringHasher::AddCharacters(const Char* chars, int length) {
|
|
|
|
ASSERT(sizeof(Char) == 1 || sizeof(Char) == 2);
|
|
|
|
int i = 0;
|
|
|
|
if (is_array_index_) {
|
|
|
|
for (; i < length; i++) {
|
|
|
|
AddCharacter(chars[i]);
|
|
|
|
if (!UpdateIndex(chars[i])) {
|
|
|
|
i++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (; i < length; i++) {
|
|
|
|
ASSERT(!is_array_index_);
|
|
|
|
AddCharacter(chars[i]);
|
|
|
|
}
|
2008-10-07 10:10:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-10 12:05:31 +00:00
|
|
|
template <typename schar>
|
2012-12-19 13:27:20 +00:00
|
|
|
uint32_t StringHasher::HashSequentialString(const schar* chars,
|
|
|
|
int length,
|
|
|
|
uint32_t seed) {
|
2012-01-04 15:12:15 +00:00
|
|
|
StringHasher hasher(length, seed);
|
2012-12-19 13:27:20 +00:00
|
|
|
if (!hasher.has_trivial_hash()) hasher.AddCharacters(chars, length);
|
2011-03-10 12:05:31 +00:00
|
|
|
return hasher.GetHashField();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-03-04 15:00:57 +00:00
|
|
|
bool Name::AsArrayIndex(uint32_t* index) {
|
|
|
|
return IsString() && String::cast(this)->AsArrayIndex(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool String::AsArrayIndex(uint32_t* index) {
|
2009-11-24 14:10:06 +00:00
|
|
|
uint32_t field = hash_field();
|
2010-06-07 09:36:30 +00:00
|
|
|
if (IsHashFieldComputed(field) && (field & kIsNotArrayIndexMask)) {
|
|
|
|
return false;
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
return SlowAsArrayIndex(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Implement set trap for proxies, and revamp class hierarchy in preparation:
- Introduce a class JSReceiver, that is a common superclass of JSObject and
JSProxy. Use JSReceiver where appropriate (probably lots of places that we
still have to migrate, but we will find those later with proxy test suite).
- Move appropriate methods to JSReceiver class (SetProperty,
GetPropertyAttribute, Get/SetPrototype, Lookup, and so on).
- Introduce new JSFunctionProxy subclass of JSProxy. Currently only a stub.
- Overhaul enum InstanceType:
* Introduce FIRST/LAST_SPEC_OBJECT_TYPE that ranges over all types that
represent JS objects, and use that consistently to check language types.
* Rename FIRST/LAST_JS_OBJECT_TYPE and FIRST/LAST_FUNCTION_CLASS_TYPE
to FIRST/LAST_[NON]CALLABLE_SPEC_OBJECT_TYPE for clarity.
* Eliminate the overlap over JS_REGEXP_TYPE.
* Also replace FIRST_JS_OBJECT with FIRST_JS_RECEIVER, but only use it where
we exclusively talk about the internal representation type.
* Insert JS_PROXY and JS_FUNCTION_PROXY in the appropriate places.
- Fix all checks concerning classification, especially for functions, to
use the CALLABLE_SPEC_OBJECT range (that includes funciton proxies).
- Handle proxies in SetProperty (that was the easiest part :) ).
- A few simple test cases.
R=kmillikin@chromium.org
Review URL: http://codereview.chromium.org/6992072
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8126 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-05-31 16:38:40 +00:00
|
|
|
Object* JSReceiver::GetPrototype() {
|
2012-07-05 13:13:33 +00:00
|
|
|
return map()->prototype();
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-06-28 16:29:53 +00:00
|
|
|
Object* JSReceiver::GetConstructor() {
|
|
|
|
return map()->constructor();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-17 11:34:20 +00:00
|
|
|
bool JSReceiver::HasProperty(Handle<JSReceiver> object,
|
|
|
|
Handle<Name> name) {
|
|
|
|
if (object->IsJSProxy()) {
|
|
|
|
Handle<JSProxy> proxy = Handle<JSProxy>::cast(object);
|
|
|
|
return JSProxy::HasPropertyWithHandler(proxy, name);
|
2011-07-19 09:38:59 +00:00
|
|
|
}
|
2014-03-13 11:55:31 +00:00
|
|
|
return GetPropertyAttribute(object, name) != ABSENT;
|
2011-07-19 09:38:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-17 11:34:20 +00:00
|
|
|
bool JSReceiver::HasLocalProperty(Handle<JSReceiver> object,
|
|
|
|
Handle<Name> name) {
|
|
|
|
if (object->IsJSProxy()) {
|
|
|
|
Handle<JSProxy> proxy = Handle<JSProxy>::cast(object);
|
|
|
|
return JSProxy::HasPropertyWithHandler(proxy, name);
|
2011-07-19 09:38:59 +00:00
|
|
|
}
|
2014-03-13 11:55:31 +00:00
|
|
|
return GetLocalPropertyAttribute(object, name) != ABSENT;
|
2011-07-19 09:38:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-13 11:55:31 +00:00
|
|
|
PropertyAttributes JSReceiver::GetPropertyAttribute(Handle<JSReceiver> object,
|
|
|
|
Handle<Name> key) {
|
2012-11-16 13:28:34 +00:00
|
|
|
uint32_t index;
|
2014-03-13 11:55:31 +00:00
|
|
|
if (object->IsJSObject() && key->AsArrayIndex(&index)) {
|
|
|
|
return GetElementAttribute(object, index);
|
2012-11-16 13:28:34 +00:00
|
|
|
}
|
2014-03-13 11:55:31 +00:00
|
|
|
return GetPropertyAttributeWithReceiver(object, object, key);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2012-11-07 14:14:50 +00:00
|
|
|
|
2014-03-13 11:55:31 +00:00
|
|
|
PropertyAttributes JSReceiver::GetElementAttribute(Handle<JSReceiver> object,
|
|
|
|
uint32_t index) {
|
|
|
|
if (object->IsJSProxy()) {
|
|
|
|
return JSProxy::GetElementAttributeWithHandler(
|
|
|
|
Handle<JSProxy>::cast(object), object, index);
|
2012-11-12 11:08:34 +00:00
|
|
|
}
|
2014-03-13 11:55:31 +00:00
|
|
|
return JSObject::GetElementAttributeWithReceiver(
|
|
|
|
Handle<JSObject>::cast(object), object, index, true);
|
2012-11-08 12:58:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-12-11 13:51:48 +00:00
|
|
|
bool JSGlobalObject::IsDetached() {
|
|
|
|
return JSGlobalProxy::cast(global_receiver())->IsDetachedFrom(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool JSGlobalProxy::IsDetachedFrom(GlobalObject* global) {
|
|
|
|
return GetPrototype() != global;
|
2009-11-12 16:34:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-05 11:47:11 +00:00
|
|
|
Handle<Object> JSReceiver::GetOrCreateIdentityHash(Handle<JSReceiver> object) {
|
|
|
|
return object->IsJSProxy()
|
|
|
|
? JSProxy::GetOrCreateIdentityHash(Handle<JSProxy>::cast(object))
|
|
|
|
: JSObject::GetOrCreateIdentityHash(Handle<JSObject>::cast(object));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Object* JSReceiver::GetIdentityHash() {
|
2011-09-22 13:54:53 +00:00
|
|
|
return IsJSProxy()
|
2013-11-05 11:47:11 +00:00
|
|
|
? JSProxy::cast(this)->GetIdentityHash()
|
|
|
|
: JSObject::cast(this)->GetIdentityHash();
|
2011-09-22 13:54:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-17 11:34:20 +00:00
|
|
|
bool JSReceiver::HasElement(Handle<JSReceiver> object, uint32_t index) {
|
|
|
|
if (object->IsJSProxy()) {
|
|
|
|
Handle<JSProxy> proxy = Handle<JSProxy>::cast(object);
|
|
|
|
return JSProxy::HasElementWithHandler(proxy, index);
|
2011-09-22 10:45:37 +00:00
|
|
|
}
|
2014-03-13 11:55:31 +00:00
|
|
|
return JSObject::GetElementAttributeWithReceiver(
|
|
|
|
Handle<JSObject>::cast(object), object, index, true) != ABSENT;
|
2012-11-08 12:58:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-17 11:34:20 +00:00
|
|
|
bool JSReceiver::HasLocalElement(Handle<JSReceiver> object, uint32_t index) {
|
|
|
|
if (object->IsJSProxy()) {
|
|
|
|
Handle<JSProxy> proxy = Handle<JSProxy>::cast(object);
|
|
|
|
return JSProxy::HasElementWithHandler(proxy, index);
|
2012-11-08 12:58:08 +00:00
|
|
|
}
|
2014-03-13 11:55:31 +00:00
|
|
|
return JSObject::GetElementAttributeWithReceiver(
|
|
|
|
Handle<JSObject>::cast(object), object, index, false) != ABSENT;
|
2012-11-08 12:58:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-13 11:55:31 +00:00
|
|
|
PropertyAttributes JSReceiver::GetLocalElementAttribute(
|
|
|
|
Handle<JSReceiver> object, uint32_t index) {
|
|
|
|
if (object->IsJSProxy()) {
|
|
|
|
return JSProxy::GetElementAttributeWithHandler(
|
|
|
|
Handle<JSProxy>::cast(object), object, index);
|
2012-11-08 12:58:08 +00:00
|
|
|
}
|
2014-03-13 11:55:31 +00:00
|
|
|
return JSObject::GetElementAttributeWithReceiver(
|
|
|
|
Handle<JSObject>::cast(object), object, index, false);
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool AccessorInfo::all_can_read() {
|
|
|
|
return BooleanBit::get(flag(), kAllCanReadBit);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void AccessorInfo::set_all_can_read(bool value) {
|
|
|
|
set_flag(BooleanBit::set(flag(), kAllCanReadBit, value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool AccessorInfo::all_can_write() {
|
|
|
|
return BooleanBit::get(flag(), kAllCanWriteBit);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void AccessorInfo::set_all_can_write(bool value) {
|
|
|
|
set_flag(BooleanBit::set(flag(), kAllCanWriteBit, value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-30 12:51:06 +00:00
|
|
|
bool AccessorInfo::prohibits_overwriting() {
|
|
|
|
return BooleanBit::get(flag(), kProhibitsOverwritingBit);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void AccessorInfo::set_prohibits_overwriting(bool value) {
|
|
|
|
set_flag(BooleanBit::set(flag(), kProhibitsOverwritingBit, value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
PropertyAttributes AccessorInfo::property_attributes() {
|
|
|
|
return AttributesField::decode(static_cast<uint32_t>(flag()->value()));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void AccessorInfo::set_property_attributes(PropertyAttributes attributes) {
|
2011-09-12 10:50:50 +00:00
|
|
|
set_flag(Smi::FromInt(AttributesField::update(flag()->value(), attributes)));
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
2011-04-11 11:38:34 +00:00
|
|
|
|
2012-06-08 07:45:11 +00:00
|
|
|
bool AccessorInfo::IsCompatibleReceiver(Object* receiver) {
|
|
|
|
Object* function_template = expected_receiver_type();
|
|
|
|
if (!function_template->IsFunctionTemplateInfo()) return true;
|
2013-11-19 13:38:15 +00:00
|
|
|
return FunctionTemplateInfo::cast(function_template)->IsTemplateFor(receiver);
|
2012-06-08 07:45:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-08-26 11:59:14 +00:00
|
|
|
void AccessorPair::set_access_flags(v8::AccessControl access_control) {
|
|
|
|
int current = access_flags()->value();
|
|
|
|
current = BooleanBit::set(current,
|
|
|
|
kProhibitsOverwritingBit,
|
|
|
|
access_control & PROHIBITS_OVERWRITING);
|
|
|
|
current = BooleanBit::set(current,
|
|
|
|
kAllCanReadBit,
|
|
|
|
access_control & ALL_CAN_READ);
|
|
|
|
current = BooleanBit::set(current,
|
|
|
|
kAllCanWriteBit,
|
|
|
|
access_control & ALL_CAN_WRITE);
|
|
|
|
set_access_flags(Smi::FromInt(current));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool AccessorPair::all_can_read() {
|
|
|
|
return BooleanBit::get(access_flags(), kAllCanReadBit);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool AccessorPair::all_can_write() {
|
|
|
|
return BooleanBit::get(access_flags(), kAllCanWriteBit);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool AccessorPair::prohibits_overwriting() {
|
|
|
|
return BooleanBit::get(access_flags(), kProhibitsOverwritingBit);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-14 15:56:57 +00:00
|
|
|
template<typename Derived, typename Shape, typename Key>
|
|
|
|
void Dictionary<Derived, Shape, Key>::SetEntry(int entry,
|
2014-04-25 13:21:16 +00:00
|
|
|
Handle<Object> key,
|
|
|
|
Handle<Object> value) {
|
2011-04-11 11:38:34 +00:00
|
|
|
SetEntry(entry, key, value, PropertyDetails(Smi::FromInt(0)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-14 15:56:57 +00:00
|
|
|
template<typename Derived, typename Shape, typename Key>
|
|
|
|
void Dictionary<Derived, Shape, Key>::SetEntry(int entry,
|
2014-04-25 13:21:16 +00:00
|
|
|
Handle<Object> key,
|
|
|
|
Handle<Object> value,
|
2014-04-14 15:56:57 +00:00
|
|
|
PropertyDetails details) {
|
2013-03-04 15:00:57 +00:00
|
|
|
ASSERT(!key->IsName() ||
|
2012-08-27 13:47:34 +00:00
|
|
|
details.IsDeleted() ||
|
|
|
|
details.dictionary_index() > 0);
|
2014-04-14 15:56:57 +00:00
|
|
|
int index = DerivedHashTable::EntryToIndex(entry);
|
2013-06-03 15:32:22 +00:00
|
|
|
DisallowHeapAllocation no_gc;
|
2010-01-29 11:46:55 +00:00
|
|
|
WriteBarrierMode mode = FixedArray::GetWriteBarrierMode(no_gc);
|
2014-04-25 13:21:16 +00:00
|
|
|
FixedArray::set(index, *key, mode);
|
|
|
|
FixedArray::set(index+1, *value, mode);
|
2011-10-21 10:32:38 +00:00
|
|
|
FixedArray::set(index+2, details.AsSmi());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-18 20:35:07 +00:00
|
|
|
bool NumberDictionaryShape::IsMatch(uint32_t key, Object* other) {
|
|
|
|
ASSERT(other->IsNumber());
|
|
|
|
return key == static_cast<uint32_t>(other->Number());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-16 09:44:35 +00:00
|
|
|
uint32_t UnseededNumberDictionaryShape::Hash(uint32_t key) {
|
|
|
|
return ComputeIntegerHash(key, 0);
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-16 09:44:35 +00:00
|
|
|
uint32_t UnseededNumberDictionaryShape::HashForObject(uint32_t key,
|
|
|
|
Object* other) {
|
|
|
|
ASSERT(other->IsNumber());
|
|
|
|
return ComputeIntegerHash(static_cast<uint32_t>(other->Number()), 0);
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
2014-04-25 13:50:19 +00:00
|
|
|
|
2012-01-16 09:44:35 +00:00
|
|
|
uint32_t SeededNumberDictionaryShape::SeededHash(uint32_t key, uint32_t seed) {
|
2012-01-10 12:58:41 +00:00
|
|
|
return ComputeIntegerHash(key, seed);
|
|
|
|
}
|
|
|
|
|
2014-04-25 13:50:19 +00:00
|
|
|
|
2012-01-16 09:44:35 +00:00
|
|
|
uint32_t SeededNumberDictionaryShape::SeededHashForObject(uint32_t key,
|
|
|
|
uint32_t seed,
|
|
|
|
Object* other) {
|
2012-01-10 12:58:41 +00:00
|
|
|
ASSERT(other->IsNumber());
|
|
|
|
return ComputeIntegerHash(static_cast<uint32_t>(other->Number()), seed);
|
|
|
|
}
|
2011-03-18 20:35:07 +00:00
|
|
|
|
|
|
|
|
2014-04-24 08:55:31 +00:00
|
|
|
Handle<Object> NumberDictionaryShape::AsHandle(Isolate* isolate, uint32_t key) {
|
|
|
|
return isolate->factory()->NewNumberFromUint(key);
|
|
|
|
}
|
|
|
|
|
2011-03-18 20:35:07 +00:00
|
|
|
|
2014-04-24 14:59:09 +00:00
|
|
|
bool NameDictionaryShape::IsMatch(Handle<Name> key, Object* other) {
|
2011-03-18 20:35:07 +00:00
|
|
|
// We know that all entries in a hash table had their hash keys created.
|
|
|
|
// Use that knowledge to have fast failure.
|
2013-03-04 15:00:57 +00:00
|
|
|
if (key->Hash() != Name::cast(other)->Hash()) return false;
|
|
|
|
return key->Equals(Name::cast(other));
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 14:59:09 +00:00
|
|
|
uint32_t NameDictionaryShape::Hash(Handle<Name> key) {
|
2011-03-18 20:35:07 +00:00
|
|
|
return key->Hash();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 14:59:09 +00:00
|
|
|
uint32_t NameDictionaryShape::HashForObject(Handle<Name> key, Object* other) {
|
2013-03-04 15:00:57 +00:00
|
|
|
return Name::cast(other)->Hash();
|
2011-03-18 20:35:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 14:59:09 +00:00
|
|
|
Handle<Object> NameDictionaryShape::AsHandle(Isolate* isolate,
|
|
|
|
Handle<Name> key) {
|
2014-04-24 10:47:13 +00:00
|
|
|
ASSERT(key->IsUniqueName());
|
2014-04-24 14:59:09 +00:00
|
|
|
return key;
|
2014-04-24 10:47:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-24 15:33:40 +00:00
|
|
|
void NameDictionary::DoGenerateNewEnumerationIndices(
|
|
|
|
Handle<NameDictionary> dictionary) {
|
|
|
|
DerivedDictionary::GenerateNewEnumerationIndices(dictionary);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-25 13:35:03 +00:00
|
|
|
bool ObjectHashTableShape::IsMatch(Handle<Object> key, Object* other) {
|
2011-10-25 14:14:56 +00:00
|
|
|
return key->SameValue(other);
|
2011-07-28 17:21:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-25 13:35:03 +00:00
|
|
|
uint32_t ObjectHashTableShape::Hash(Handle<Object> key) {
|
2013-11-05 11:47:11 +00:00
|
|
|
return Smi::cast(key->GetHash())->value();
|
2011-07-28 17:21:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-25 13:35:03 +00:00
|
|
|
uint32_t ObjectHashTableShape::HashForObject(Handle<Object> key,
|
|
|
|
Object* other) {
|
2013-11-05 11:47:11 +00:00
|
|
|
return Smi::cast(other->GetHash())->value();
|
2011-07-28 17:21:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-25 13:35:03 +00:00
|
|
|
Handle<Object> ObjectHashTableShape::AsHandle(Isolate* isolate,
|
|
|
|
Handle<Object> key) {
|
2011-07-28 17:21:22 +00:00
|
|
|
return key;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-14 15:56:57 +00:00
|
|
|
Handle<ObjectHashTable> ObjectHashTable::Shrink(
|
|
|
|
Handle<ObjectHashTable> table, Handle<Object> key) {
|
2014-04-25 13:35:03 +00:00
|
|
|
return DerivedHashTable::Shrink(table, key);
|
2014-04-14 15:56:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-10-04 07:25:24 +00:00
|
|
|
template <int entrysize>
|
2014-04-25 13:35:03 +00:00
|
|
|
bool WeakHashTableShape<entrysize>::IsMatch(Handle<Object> key, Object* other) {
|
2013-10-04 07:25:24 +00:00
|
|
|
return key->SameValue(other);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template <int entrysize>
|
2014-04-25 13:35:03 +00:00
|
|
|
uint32_t WeakHashTableShape<entrysize>::Hash(Handle<Object> key) {
|
|
|
|
intptr_t hash = reinterpret_cast<intptr_t>(*key);
|
2013-10-04 07:25:24 +00:00
|
|
|
return (uint32_t)(hash & 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template <int entrysize>
|
2014-04-25 13:35:03 +00:00
|
|
|
uint32_t WeakHashTableShape<entrysize>::HashForObject(Handle<Object> key,
|
2013-10-04 07:25:24 +00:00
|
|
|
Object* other) {
|
|
|
|
intptr_t hash = reinterpret_cast<intptr_t>(other);
|
|
|
|
return (uint32_t)(hash & 0xFFFFFFFF);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template <int entrysize>
|
2014-04-25 13:35:03 +00:00
|
|
|
Handle<Object> WeakHashTableShape<entrysize>::AsHandle(Isolate* isolate,
|
|
|
|
Handle<Object> key) {
|
2013-10-04 07:25:24 +00:00
|
|
|
return key;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-18 20:35:07 +00:00
|
|
|
void Map::ClearCodeCache(Heap* heap) {
|
2008-07-03 15:10:15 +00:00
|
|
|
// No write barrier is needed since empty_fixed_array is not in new space.
|
|
|
|
// Please note this function is used during marking:
|
|
|
|
// - MarkCompactCollector::MarkUnmarkedObject
|
2012-03-23 13:33:11 +00:00
|
|
|
// - IncrementalMarking::Step
|
2012-10-25 11:52:37 +00:00
|
|
|
ASSERT(!heap->InNewSpace(heap->empty_fixed_array()));
|
|
|
|
WRITE_FIELD(this, kCodeCacheOffset, heap->empty_fixed_array());
|
2008-07-03 15:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-25 09:51:13 +00:00
|
|
|
void JSArray::EnsureSize(Handle<JSArray> array, int required_size) {
|
|
|
|
ASSERT(array->HasFastSmiOrObjectElements());
|
|
|
|
Handle<FixedArray> elts = handle(FixedArray::cast(array->elements()));
|
2009-11-20 10:11:45 +00:00
|
|
|
const int kArraySizeThatFitsComfortablyInNewSpace = 128;
|
|
|
|
if (elts->length() < required_size) {
|
|
|
|
// Doubling in size would be overkill, but leave some slack to avoid
|
|
|
|
// constantly growing.
|
2014-03-25 09:51:13 +00:00
|
|
|
Expand(array, required_size + (required_size >> 3));
|
2009-11-20 10:11:45 +00:00
|
|
|
// It's a performance benefit to keep a frequently used array in new-space.
|
2014-03-25 09:51:13 +00:00
|
|
|
} else if (!array->GetHeap()->new_space()->Contains(*elts) &&
|
2009-11-20 10:11:45 +00:00
|
|
|
required_size < kArraySizeThatFitsComfortablyInNewSpace) {
|
|
|
|
// Expand will allocate a new backing store in new space even if the size
|
|
|
|
// we asked for isn't larger than what we had before.
|
2014-03-25 09:51:13 +00:00
|
|
|
Expand(array, required_size);
|
2009-11-20 10:11:45 +00:00
|
|
|
}
|
2009-06-17 10:55:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-01-29 11:46:55 +00:00
|
|
|
void JSArray::set_length(Smi* length) {
|
2011-09-19 18:36:47 +00:00
|
|
|
// Don't need a write barrier for a Smi.
|
2010-01-29 11:46:55 +00:00
|
|
|
set_length(static_cast<Object*>(length), SKIP_WRITE_BARRIER);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-14 12:46:32 +00:00
|
|
|
bool JSArray::AllowsSetElementsLength() {
|
|
|
|
bool result = elements()->IsFixedArray() || elements()->IsFixedDoubleArray();
|
|
|
|
ASSERT(result == !HasExternalArrayElements());
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-21 08:30:42 +00:00
|
|
|
void JSArray::SetContent(Handle<JSArray> array,
|
|
|
|
Handle<FixedArrayBase> storage) {
|
|
|
|
EnsureCanContainElements(array, storage, storage->length(),
|
|
|
|
ALLOW_COPIED_DOUBLE_ELEMENTS);
|
|
|
|
|
|
|
|
ASSERT((storage->map() == array->GetHeap()->fixed_double_array_map() &&
|
|
|
|
IsFastDoubleElementsKind(array->GetElementsKind())) ||
|
|
|
|
((storage->map() != array->GetHeap()->fixed_double_array_map()) &&
|
|
|
|
(IsFastObjectElementsKind(array->GetElementsKind()) ||
|
|
|
|
(IsFastSmiElementsKind(array->GetElementsKind()) &&
|
|
|
|
Handle<FixedArray>::cast(storage)->ContainsOnlySmisOrHoles()))));
|
|
|
|
array->set_elements(*storage);
|
|
|
|
array->set_length(Smi::FromInt(storage->length()));
|
2008-10-10 10:27:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-10 21:38:17 +00:00
|
|
|
Handle<Object> TypeFeedbackInfo::UninitializedSentinel(Isolate* isolate) {
|
2014-03-07 09:10:18 +00:00
|
|
|
return isolate->factory()->uninitialized_symbol();
|
2012-01-27 13:03:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-10 21:38:17 +00:00
|
|
|
Handle<Object> TypeFeedbackInfo::MegamorphicSentinel(Isolate* isolate) {
|
2014-03-07 09:10:18 +00:00
|
|
|
return isolate->factory()->megamorphic_symbol();
|
2012-01-27 13:03:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-10 21:38:17 +00:00
|
|
|
Handle<Object> TypeFeedbackInfo::MonomorphicArraySentinel(Isolate* isolate,
|
2013-03-01 16:06:34 +00:00
|
|
|
ElementsKind elements_kind) {
|
|
|
|
return Handle<Object>(Smi::FromInt(static_cast<int>(elements_kind)), isolate);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-02-10 21:38:17 +00:00
|
|
|
Object* TypeFeedbackInfo::RawUninitializedSentinel(Heap* heap) {
|
2014-03-07 09:10:18 +00:00
|
|
|
return heap->uninitialized_symbol();
|
2012-01-27 13:03:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-23 21:08:58 +00:00
|
|
|
int TypeFeedbackInfo::ic_total_count() {
|
|
|
|
int current = Smi::cast(READ_FIELD(this, kStorage1Offset))->value();
|
|
|
|
return ICTotalCountField::decode(current);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TypeFeedbackInfo::set_ic_total_count(int count) {
|
|
|
|
int value = Smi::cast(READ_FIELD(this, kStorage1Offset))->value();
|
|
|
|
value = ICTotalCountField::update(value,
|
|
|
|
ICTotalCountField::decode(count));
|
|
|
|
WRITE_FIELD(this, kStorage1Offset, Smi::FromInt(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int TypeFeedbackInfo::ic_with_type_info_count() {
|
|
|
|
int current = Smi::cast(READ_FIELD(this, kStorage2Offset))->value();
|
|
|
|
return ICsWithTypeInfoCountField::decode(current);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TypeFeedbackInfo::change_ic_with_type_info_count(int delta) {
|
|
|
|
int value = Smi::cast(READ_FIELD(this, kStorage2Offset))->value();
|
2012-08-27 15:17:14 +00:00
|
|
|
int new_count = ICsWithTypeInfoCountField::decode(value) + delta;
|
|
|
|
// We can get negative count here when the type-feedback info is
|
|
|
|
// shared between two code objects. The can only happen when
|
|
|
|
// the debugger made a shallow copy of code object (see Heap::CopyCode).
|
|
|
|
// Since we do not optimize when the debugger is active, we can skip
|
|
|
|
// this counter update.
|
|
|
|
if (new_count >= 0) {
|
|
|
|
new_count &= ICsWithTypeInfoCountField::kMask;
|
|
|
|
value = ICsWithTypeInfoCountField::update(value, new_count);
|
|
|
|
WRITE_FIELD(this, kStorage2Offset, Smi::FromInt(value));
|
|
|
|
}
|
2012-08-23 21:08:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TypeFeedbackInfo::initialize_storage() {
|
|
|
|
WRITE_FIELD(this, kStorage1Offset, Smi::FromInt(0));
|
|
|
|
WRITE_FIELD(this, kStorage2Offset, Smi::FromInt(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TypeFeedbackInfo::change_own_type_change_checksum() {
|
|
|
|
int value = Smi::cast(READ_FIELD(this, kStorage1Offset))->value();
|
|
|
|
int checksum = OwnTypeChangeChecksum::decode(value);
|
|
|
|
checksum = (checksum + 1) % (1 << kTypeChangeChecksumBits);
|
|
|
|
value = OwnTypeChangeChecksum::update(value, checksum);
|
|
|
|
// Ensure packed bit field is in Smi range.
|
|
|
|
if (value > Smi::kMaxValue) value |= Smi::kMinValue;
|
|
|
|
if (value < Smi::kMinValue) value &= ~Smi::kMinValue;
|
|
|
|
WRITE_FIELD(this, kStorage1Offset, Smi::FromInt(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TypeFeedbackInfo::set_inlined_type_change_checksum(int checksum) {
|
|
|
|
int value = Smi::cast(READ_FIELD(this, kStorage2Offset))->value();
|
|
|
|
int mask = (1 << kTypeChangeChecksumBits) - 1;
|
|
|
|
value = InlinedTypeChangeChecksum::update(value, checksum & mask);
|
|
|
|
// Ensure packed bit field is in Smi range.
|
|
|
|
if (value > Smi::kMaxValue) value |= Smi::kMinValue;
|
|
|
|
if (value < Smi::kMinValue) value &= ~Smi::kMinValue;
|
|
|
|
WRITE_FIELD(this, kStorage2Offset, Smi::FromInt(value));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int TypeFeedbackInfo::own_type_change_checksum() {
|
|
|
|
int value = Smi::cast(READ_FIELD(this, kStorage1Offset))->value();
|
|
|
|
return OwnTypeChangeChecksum::decode(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool TypeFeedbackInfo::matches_inlined_type_change_checksum(int checksum) {
|
|
|
|
int value = Smi::cast(READ_FIELD(this, kStorage2Offset))->value();
|
|
|
|
int mask = (1 << kTypeChangeChecksumBits) - 1;
|
|
|
|
return InlinedTypeChangeChecksum::decode(value) == (checksum & mask);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-24 14:34:01 +00:00
|
|
|
SMI_ACCESSORS(AliasedArgumentsEntry, aliased_context_slot, kAliasedContextSlot)
|
|
|
|
|
|
|
|
|
2011-03-18 20:35:07 +00:00
|
|
|
Relocatable::Relocatable(Isolate* isolate) {
|
|
|
|
isolate_ = isolate;
|
|
|
|
prev_ = isolate->relocatable_top();
|
|
|
|
isolate->set_relocatable_top(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Relocatable::~Relocatable() {
|
|
|
|
ASSERT_EQ(isolate_->relocatable_top(), this);
|
|
|
|
isolate_->set_relocatable_top(prev_);
|
2008-10-20 06:35:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-08-11 14:30:14 +00:00
|
|
|
int JSObject::BodyDescriptor::SizeOf(Map* map, HeapObject* object) {
|
|
|
|
return map->instance_size();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-05-19 11:47:34 +00:00
|
|
|
void Foreign::ForeignIterateBody(ObjectVisitor* v) {
|
2010-08-11 14:30:14 +00:00
|
|
|
v->VisitExternalReference(
|
2011-10-28 12:37:29 +00:00
|
|
|
reinterpret_cast<Address*>(FIELD_ADDR(this, kForeignAddressOffset)));
|
2010-08-11 14:30:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<typename StaticVisitor>
|
2011-05-19 11:47:34 +00:00
|
|
|
void Foreign::ForeignIterateBody() {
|
2010-08-11 14:30:14 +00:00
|
|
|
StaticVisitor::VisitExternalReference(
|
2011-10-28 12:37:29 +00:00
|
|
|
reinterpret_cast<Address*>(FIELD_ADDR(this, kForeignAddressOffset)));
|
2010-08-11 14:30:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExternalAsciiString::ExternalAsciiStringIterateBody(ObjectVisitor* v) {
|
|
|
|
typedef v8::String::ExternalAsciiStringResource Resource;
|
|
|
|
v->VisitExternalAsciiString(
|
|
|
|
reinterpret_cast<Resource**>(FIELD_ADDR(this, kResourceOffset)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<typename StaticVisitor>
|
|
|
|
void ExternalAsciiString::ExternalAsciiStringIterateBody() {
|
|
|
|
typedef v8::String::ExternalAsciiStringResource Resource;
|
|
|
|
StaticVisitor::VisitExternalAsciiString(
|
|
|
|
reinterpret_cast<Resource**>(FIELD_ADDR(this, kResourceOffset)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExternalTwoByteString::ExternalTwoByteStringIterateBody(ObjectVisitor* v) {
|
|
|
|
typedef v8::String::ExternalStringResource Resource;
|
|
|
|
v->VisitExternalTwoByteString(
|
|
|
|
reinterpret_cast<Resource**>(FIELD_ADDR(this, kResourceOffset)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<typename StaticVisitor>
|
|
|
|
void ExternalTwoByteString::ExternalTwoByteStringIterateBody() {
|
|
|
|
typedef v8::String::ExternalStringResource Resource;
|
|
|
|
StaticVisitor::VisitExternalTwoByteString(
|
|
|
|
reinterpret_cast<Resource**>(FIELD_ADDR(this, kResourceOffset)));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<int start_offset, int end_offset, int size>
|
|
|
|
void FixedBodyDescriptor<start_offset, end_offset, size>::IterateBody(
|
|
|
|
HeapObject* obj,
|
|
|
|
ObjectVisitor* v) {
|
2012-08-09 12:25:03 +00:00
|
|
|
v->VisitPointers(HeapObject::RawField(obj, start_offset),
|
|
|
|
HeapObject::RawField(obj, end_offset));
|
2010-08-11 14:30:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
template<int start_offset>
|
|
|
|
void FlexibleBodyDescriptor<start_offset>::IterateBody(HeapObject* obj,
|
|
|
|
int object_size,
|
|
|
|
ObjectVisitor* v) {
|
2012-08-09 12:25:03 +00:00
|
|
|
v->VisitPointers(HeapObject::RawField(obj, start_offset),
|
|
|
|
HeapObject::RawField(obj, object_size));
|
2010-08-11 14:30:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-14 16:16:46 +00:00
|
|
|
#undef TYPE_CHECKER
|
2008-07-03 15:10:15 +00:00
|
|
|
#undef CAST_ACCESSOR
|
|
|
|
#undef INT_ACCESSORS
|
|
|
|
#undef ACCESSORS
|
2012-03-14 16:16:46 +00:00
|
|
|
#undef ACCESSORS_TO_SMI
|
|
|
|
#undef SMI_ACCESSORS
|
2014-04-09 09:50:25 +00:00
|
|
|
#undef SYNCHRONIZED_SMI_ACCESSORS
|
|
|
|
#undef NOBARRIER_SMI_ACCESSORS
|
2012-03-14 16:16:46 +00:00
|
|
|
#undef BOOL_GETTER
|
|
|
|
#undef BOOL_ACCESSORS
|
2008-07-03 15:10:15 +00:00
|
|
|
#undef FIELD_ADDR
|
|
|
|
#undef READ_FIELD
|
2014-04-09 09:50:25 +00:00
|
|
|
#undef NOBARRIER_READ_FIELD
|
2008-07-03 15:10:15 +00:00
|
|
|
#undef WRITE_FIELD
|
2014-04-09 09:50:25 +00:00
|
|
|
#undef NOBARRIER_WRITE_FIELD
|
2008-07-03 15:10:15 +00:00
|
|
|
#undef WRITE_BARRIER
|
2008-10-23 08:46:32 +00:00
|
|
|
#undef CONDITIONAL_WRITE_BARRIER
|
2008-07-03 15:10:15 +00:00
|
|
|
#undef READ_DOUBLE_FIELD
|
|
|
|
#undef WRITE_DOUBLE_FIELD
|
|
|
|
#undef READ_INT_FIELD
|
|
|
|
#undef WRITE_INT_FIELD
|
2012-03-14 16:16:46 +00:00
|
|
|
#undef READ_INTPTR_FIELD
|
|
|
|
#undef WRITE_INTPTR_FIELD
|
|
|
|
#undef READ_UINT32_FIELD
|
|
|
|
#undef WRITE_UINT32_FIELD
|
2008-07-03 15:10:15 +00:00
|
|
|
#undef READ_SHORT_FIELD
|
|
|
|
#undef WRITE_SHORT_FIELD
|
|
|
|
#undef READ_BYTE_FIELD
|
|
|
|
#undef WRITE_BYTE_FIELD
|
2014-04-09 09:50:25 +00:00
|
|
|
#undef NOBARRIER_READ_BYTE_FIELD
|
|
|
|
#undef NOBARRIER_WRITE_BYTE_FIELD
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
} } // namespace v8::internal
|
|
|
|
|
|
|
|
#endif // V8_OBJECTS_INL_H_
|