blob: a64d9ff132b8cea48389f852d1066dc5c6605f2a [file] [log] [blame]
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001// Copyright 2012 the V8 project authors. All rights reserved.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
Steve Blocka7e24c12009-10-30 11:49:00 +00004//
5// Review notes:
6//
7// - The use of macros in these inline functions may seem superfluous
8// but it is absolutely needed to make sure gcc generates optimal
9// code. gcc is not happy when attempting to inline too deep.
10//
11
12#ifndef V8_OBJECTS_INL_H_
13#define V8_OBJECTS_INL_H_
14
Ben Murdochb8a8cc12014-11-26 15:28:44 +000015#include "src/base/atomicops.h"
16#include "src/base/bits.h"
Ben Murdoch4a90d5f2016-03-22 12:00:34 +000017#include "src/contexts-inl.h"
Ben Murdochb8a8cc12014-11-26 15:28:44 +000018#include "src/conversions-inl.h"
Ben Murdochb8a8cc12014-11-26 15:28:44 +000019#include "src/factory.h"
20#include "src/field-index-inl.h"
Ben Murdochc5610432016-08-08 18:44:38 +010021#include "src/field-type.h"
Ben Murdoch097c5b22016-05-18 11:27:45 +010022#include "src/handles-inl.h"
Ben Murdochb8a8cc12014-11-26 15:28:44 +000023#include "src/heap/heap-inl.h"
24#include "src/heap/heap.h"
Ben Murdochb8a8cc12014-11-26 15:28:44 +000025#include "src/isolate.h"
Ben Murdochda12d292016-06-02 14:46:10 +010026#include "src/isolate-inl.h"
Emily Bernierd0a1eb72015-03-24 16:35:39 -040027#include "src/layout-descriptor-inl.h"
Ben Murdochb8a8cc12014-11-26 15:28:44 +000028#include "src/lookup.h"
29#include "src/objects.h"
30#include "src/property.h"
31#include "src/prototype.h"
32#include "src/transitions-inl.h"
33#include "src/type-feedback-vector-inl.h"
34#include "src/v8memory.h"
Ben Murdoch592a9fc2012-03-05 11:04:45 +000035
Steve Blocka7e24c12009-10-30 11:49:00 +000036namespace v8 {
37namespace internal {
38
39PropertyDetails::PropertyDetails(Smi* smi) {
40 value_ = smi->value();
41}
42
43
Ben Murdochb8a8cc12014-11-26 15:28:44 +000044Smi* PropertyDetails::AsSmi() const {
45 // Ensure the upper 2 bits have the same value by sign extending it. This is
46 // necessary to be able to use the 31st bit of the property details.
47 int value = value_ << 1;
48 return Smi::FromInt(value >> 1);
Steve Blocka7e24c12009-10-30 11:49:00 +000049}
50
51
Emily Bernierd0a1eb72015-03-24 16:35:39 -040052int PropertyDetails::field_width_in_words() const {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +000053 DCHECK(location() == kField);
Emily Bernierd0a1eb72015-03-24 16:35:39 -040054 if (!FLAG_unbox_double_fields) return 1;
55 if (kDoubleSize == kPointerSize) return 1;
56 return representation().IsDouble() ? kDoubleSize / kPointerSize : 1;
57}
58
Ben Murdoch097c5b22016-05-18 11:27:45 +010059#define TYPE_CHECKER(type, instancetype) \
60 bool HeapObject::Is##type() const { \
61 return map()->instance_type() == instancetype; \
Ben Murdoch3ef787d2012-04-12 10:51:47 +010062 }
63
Ben Murdochb8a8cc12014-11-26 15:28:44 +000064#define CAST_ACCESSOR(type) \
65 type* type::cast(Object* object) { \
66 SLOW_DCHECK(object->Is##type()); \
67 return reinterpret_cast<type*>(object); \
68 } \
69 const type* type::cast(const Object* object) { \
70 SLOW_DCHECK(object->Is##type()); \
71 return reinterpret_cast<const type*>(object); \
Steve Blocka7e24c12009-10-30 11:49:00 +000072 }
73
74
Ben Murdochb8a8cc12014-11-26 15:28:44 +000075#define INT_ACCESSORS(holder, name, offset) \
76 int holder::name() const { return READ_INT_FIELD(this, offset); } \
Steve Blocka7e24c12009-10-30 11:49:00 +000077 void holder::set_##name(int value) { WRITE_INT_FIELD(this, offset, value); }
78
79
Ben Murdochb8a8cc12014-11-26 15:28:44 +000080#define ACCESSORS(holder, name, type, offset) \
81 type* holder::name() const { return type::cast(READ_FIELD(this, offset)); } \
82 void holder::set_##name(type* value, WriteBarrierMode mode) { \
83 WRITE_FIELD(this, offset, value); \
84 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, offset, value, mode); \
Steve Blocka7e24c12009-10-30 11:49:00 +000085 }
86
87
Ben Murdoch3ef787d2012-04-12 10:51:47 +010088// Getter that returns a Smi as an int and writes an int as a Smi.
Steve Blocka7e24c12009-10-30 11:49:00 +000089#define SMI_ACCESSORS(holder, name, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +000090 int holder::name() const { \
Steve Blocka7e24c12009-10-30 11:49:00 +000091 Object* value = READ_FIELD(this, offset); \
92 return Smi::cast(value)->value(); \
93 } \
94 void holder::set_##name(int value) { \
95 WRITE_FIELD(this, offset, Smi::FromInt(value)); \
96 }
97
Ben Murdochb8a8cc12014-11-26 15:28:44 +000098#define SYNCHRONIZED_SMI_ACCESSORS(holder, name, offset) \
99 int holder::synchronized_##name() const { \
100 Object* value = ACQUIRE_READ_FIELD(this, offset); \
101 return Smi::cast(value)->value(); \
102 } \
103 void holder::synchronized_set_##name(int value) { \
104 RELEASE_WRITE_FIELD(this, offset, Smi::FromInt(value)); \
105 }
106
107#define NOBARRIER_SMI_ACCESSORS(holder, name, offset) \
108 int holder::nobarrier_##name() const { \
109 Object* value = NOBARRIER_READ_FIELD(this, offset); \
110 return Smi::cast(value)->value(); \
111 } \
112 void holder::nobarrier_set_##name(int value) { \
113 NOBARRIER_WRITE_FIELD(this, offset, Smi::FromInt(value)); \
114 }
Steve Blocka7e24c12009-10-30 11:49:00 +0000115
116#define BOOL_GETTER(holder, field, name, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000117 bool holder::name() const { \
Steve Blocka7e24c12009-10-30 11:49:00 +0000118 return BooleanBit::get(field(), offset); \
119 } \
120
121
122#define BOOL_ACCESSORS(holder, field, name, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000123 bool holder::name() const { \
Steve Blocka7e24c12009-10-30 11:49:00 +0000124 return BooleanBit::get(field(), offset); \
125 } \
126 void holder::set_##name(bool value) { \
127 set_##field(BooleanBit::set(field(), offset, value)); \
128 }
129
Ben Murdoch097c5b22016-05-18 11:27:45 +0100130bool HeapObject::IsFixedArrayBase() const {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000131 return IsFixedArray() || IsFixedDoubleArray() || IsFixedTypedArrayBase();
132}
133
Ben Murdoch097c5b22016-05-18 11:27:45 +0100134bool HeapObject::IsFixedArray() const {
135 InstanceType instance_type = map()->instance_type();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000136 return instance_type == FIXED_ARRAY_TYPE ||
137 instance_type == TRANSITION_ARRAY_TYPE;
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100138}
139
140
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000141// External objects are not extensible, so the map check is enough.
Ben Murdoch097c5b22016-05-18 11:27:45 +0100142bool HeapObject::IsExternal() const {
143 return map() == GetHeap()->external_map();
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100144}
145
146
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100147TYPE_CHECKER(HeapNumber, HEAP_NUMBER_TYPE)
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000148TYPE_CHECKER(MutableHeapNumber, MUTABLE_HEAP_NUMBER_TYPE)
149TYPE_CHECKER(Symbol, SYMBOL_TYPE)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000150TYPE_CHECKER(Simd128Value, SIMD128_VALUE_TYPE)
151
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000152#define SIMD128_TYPE_CHECKER(TYPE, Type, type, lane_count, lane_type) \
Ben Murdoch097c5b22016-05-18 11:27:45 +0100153 bool HeapObject::Is##Type() const { return map() == GetHeap()->type##_map(); }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000154SIMD128_TYPES(SIMD128_TYPE_CHECKER)
155#undef SIMD128_TYPE_CHECKER
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100156
Ben Murdoch097c5b22016-05-18 11:27:45 +0100157#define IS_TYPE_FUNCTION_DEF(type_) \
158 bool Object::Is##type_() const { \
159 return IsHeapObject() && HeapObject::cast(this)->Is##type_(); \
160 }
161HEAP_OBJECT_TYPE_LIST(IS_TYPE_FUNCTION_DEF)
162ODDBALL_LIST(IS_TYPE_FUNCTION_DEF)
163#undef IS_TYPE_FUNCTION_DEF
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100164
Ben Murdoch097c5b22016-05-18 11:27:45 +0100165bool HeapObject::IsString() const {
166 return map()->instance_type() < FIRST_NONSTRING_TYPE;
Steve Blocka7e24c12009-10-30 11:49:00 +0000167}
168
Ben Murdoch097c5b22016-05-18 11:27:45 +0100169bool HeapObject::IsName() const {
170 return map()->instance_type() <= LAST_NAME_TYPE;
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000171}
172
Ben Murdoch097c5b22016-05-18 11:27:45 +0100173bool HeapObject::IsUniqueName() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000174 return IsInternalizedString() || IsSymbol();
175}
176
Ben Murdoch097c5b22016-05-18 11:27:45 +0100177bool Name::IsUniqueName() const {
178 uint32_t type = map()->instance_type();
179 return (type & (kIsNotStringMask | kIsNotInternalizedMask)) !=
180 (kStringTag | kNotInternalizedTag);
181}
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000182
Ben Murdoch097c5b22016-05-18 11:27:45 +0100183bool HeapObject::IsFunction() const {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000184 STATIC_ASSERT(LAST_FUNCTION_TYPE == LAST_TYPE);
Ben Murdoch097c5b22016-05-18 11:27:45 +0100185 return map()->instance_type() >= FIRST_FUNCTION_TYPE;
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000186}
187
Ben Murdoch097c5b22016-05-18 11:27:45 +0100188bool HeapObject::IsCallable() const { return map()->is_callable(); }
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000189
Ben Murdoch097c5b22016-05-18 11:27:45 +0100190bool HeapObject::IsConstructor() const { return map()->is_constructor(); }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000191
Ben Murdoch097c5b22016-05-18 11:27:45 +0100192bool HeapObject::IsTemplateInfo() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000193 return IsObjectTemplateInfo() || IsFunctionTemplateInfo();
Steve Blocka7e24c12009-10-30 11:49:00 +0000194}
195
Ben Murdoch097c5b22016-05-18 11:27:45 +0100196bool HeapObject::IsInternalizedString() const {
197 uint32_t type = map()->instance_type();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000198 STATIC_ASSERT(kNotInternalizedTag != 0);
199 return (type & (kIsNotStringMask | kIsNotInternalizedMask)) ==
200 (kStringTag | kInternalizedTag);
201}
202
Ben Murdoch097c5b22016-05-18 11:27:45 +0100203bool HeapObject::IsConsString() const {
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000204 if (!IsString()) return false;
205 return StringShape(String::cast(this)).IsCons();
206}
207
Ben Murdoch097c5b22016-05-18 11:27:45 +0100208bool HeapObject::IsSlicedString() const {
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000209 if (!IsString()) return false;
210 return StringShape(String::cast(this)).IsSliced();
Steve Blocka7e24c12009-10-30 11:49:00 +0000211}
212
Ben Murdoch097c5b22016-05-18 11:27:45 +0100213bool HeapObject::IsSeqString() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000214 if (!IsString()) return false;
215 return StringShape(String::cast(this)).IsSequential();
216}
217
Ben Murdoch097c5b22016-05-18 11:27:45 +0100218bool HeapObject::IsSeqOneByteString() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000219 if (!IsString()) return false;
220 return StringShape(String::cast(this)).IsSequential() &&
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000221 String::cast(this)->IsOneByteRepresentation();
Steve Blocka7e24c12009-10-30 11:49:00 +0000222}
223
Ben Murdoch097c5b22016-05-18 11:27:45 +0100224bool HeapObject::IsSeqTwoByteString() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000225 if (!IsString()) return false;
226 return StringShape(String::cast(this)).IsSequential() &&
227 String::cast(this)->IsTwoByteRepresentation();
228}
229
Ben Murdoch097c5b22016-05-18 11:27:45 +0100230bool HeapObject::IsExternalString() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000231 if (!IsString()) return false;
232 return StringShape(String::cast(this)).IsExternal();
233}
234
Ben Murdoch097c5b22016-05-18 11:27:45 +0100235bool HeapObject::IsExternalOneByteString() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000236 if (!IsString()) return false;
237 return StringShape(String::cast(this)).IsExternal() &&
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000238 String::cast(this)->IsOneByteRepresentation();
Steve Blocka7e24c12009-10-30 11:49:00 +0000239}
240
Ben Murdoch097c5b22016-05-18 11:27:45 +0100241bool HeapObject::IsExternalTwoByteString() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000242 if (!IsString()) return false;
243 return StringShape(String::cast(this)).IsExternal() &&
244 String::cast(this)->IsTwoByteRepresentation();
245}
246
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000247
Ben Murdoch3fb3ca82011-12-02 17:19:32 +0000248bool Object::HasValidElements() {
249 // Dictionary is covered under FixedArray.
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000250 return IsFixedArray() || IsFixedDoubleArray() || IsFixedTypedArrayBase();
251}
252
253
254bool Object::KeyEquals(Object* second) {
255 Object* first = this;
256 if (second->IsNumber()) {
257 if (first->IsNumber()) return first->Number() == second->Number();
258 Object* temp = first;
259 first = second;
260 second = temp;
261 }
262 if (first->IsNumber()) {
263 DCHECK_LE(0, first->Number());
264 uint32_t expected = static_cast<uint32_t>(first->Number());
265 uint32_t index;
266 return Name::cast(second)->AsArrayIndex(&index) && index == expected;
267 }
268 return Name::cast(first)->Equals(Name::cast(second));
269}
270
271
272bool Object::FilterKey(PropertyFilter filter) {
273 if (IsSymbol()) {
274 if (filter & SKIP_SYMBOLS) return true;
275 if (Symbol::cast(this)->is_private()) return true;
276 } else {
277 if (filter & SKIP_STRINGS) return true;
278 }
279 return false;
Ben Murdoch3fb3ca82011-12-02 17:19:32 +0000280}
Steve Blocka7e24c12009-10-30 11:49:00 +0000281
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000282
283Handle<Object> Object::NewStorageFor(Isolate* isolate,
284 Handle<Object> object,
285 Representation representation) {
286 if (representation.IsSmi() && object->IsUninitialized()) {
287 return handle(Smi::FromInt(0), isolate);
288 }
289 if (!representation.IsDouble()) return object;
290 double value;
291 if (object->IsUninitialized()) {
292 value = 0;
293 } else if (object->IsMutableHeapNumber()) {
294 value = HeapNumber::cast(*object)->value();
295 } else {
296 value = object->Number();
297 }
298 return isolate->factory()->NewHeapNumber(value, MUTABLE);
299}
300
301
302Handle<Object> Object::WrapForRead(Isolate* isolate,
303 Handle<Object> object,
304 Representation representation) {
305 DCHECK(!object->IsUninitialized());
306 if (!representation.IsDouble()) {
307 DCHECK(object->FitsRepresentation(representation));
308 return object;
309 }
310 return isolate->factory()->NewHeapNumber(HeapNumber::cast(*object)->value());
311}
312
313
314StringShape::StringShape(const String* str)
Steve Blocka7e24c12009-10-30 11:49:00 +0000315 : type_(str->map()->instance_type()) {
316 set_valid();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000317 DCHECK((type_ & kIsNotStringMask) == kStringTag);
Steve Blocka7e24c12009-10-30 11:49:00 +0000318}
319
320
321StringShape::StringShape(Map* map)
322 : type_(map->instance_type()) {
323 set_valid();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000324 DCHECK((type_ & kIsNotStringMask) == kStringTag);
Steve Blocka7e24c12009-10-30 11:49:00 +0000325}
326
327
328StringShape::StringShape(InstanceType t)
329 : type_(static_cast<uint32_t>(t)) {
330 set_valid();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000331 DCHECK((type_ & kIsNotStringMask) == kStringTag);
Steve Blocka7e24c12009-10-30 11:49:00 +0000332}
333
334
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000335bool StringShape::IsInternalized() {
336 DCHECK(valid());
337 STATIC_ASSERT(kNotInternalizedTag != 0);
338 return (type_ & (kIsNotStringMask | kIsNotInternalizedMask)) ==
339 (kStringTag | kInternalizedTag);
Steve Blocka7e24c12009-10-30 11:49:00 +0000340}
341
342
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000343bool String::IsOneByteRepresentation() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000344 uint32_t type = map()->instance_type();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000345 return (type & kStringEncodingMask) == kOneByteStringTag;
Steve Blocka7e24c12009-10-30 11:49:00 +0000346}
347
348
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000349bool String::IsTwoByteRepresentation() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000350 uint32_t type = map()->instance_type();
Steve Blocka7e24c12009-10-30 11:49:00 +0000351 return (type & kStringEncodingMask) == kTwoByteStringTag;
352}
353
354
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000355bool String::IsOneByteRepresentationUnderneath() {
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000356 uint32_t type = map()->instance_type();
357 STATIC_ASSERT(kIsIndirectStringTag != 0);
358 STATIC_ASSERT((kIsIndirectStringMask & kStringEncodingMask) == 0);
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000359 DCHECK(IsFlat());
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000360 switch (type & (kIsIndirectStringMask | kStringEncodingMask)) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000361 case kOneByteStringTag:
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000362 return true;
363 case kTwoByteStringTag:
364 return false;
365 default: // Cons or sliced string. Need to go deeper.
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000366 return GetUnderlying()->IsOneByteRepresentation();
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000367 }
368}
369
370
371bool String::IsTwoByteRepresentationUnderneath() {
372 uint32_t type = map()->instance_type();
373 STATIC_ASSERT(kIsIndirectStringTag != 0);
374 STATIC_ASSERT((kIsIndirectStringMask & kStringEncodingMask) == 0);
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000375 DCHECK(IsFlat());
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000376 switch (type & (kIsIndirectStringMask | kStringEncodingMask)) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000377 case kOneByteStringTag:
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000378 return false;
379 case kTwoByteStringTag:
380 return true;
381 default: // Cons or sliced string. Need to go deeper.
382 return GetUnderlying()->IsTwoByteRepresentation();
383 }
384}
385
386
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000387bool String::HasOnlyOneByteChars() {
Kristian Monsen9dcf7e22010-06-28 14:14:28 +0100388 uint32_t type = map()->instance_type();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000389 return (type & kOneByteDataHintMask) == kOneByteDataHintTag ||
390 IsOneByteRepresentation();
Steve Block6ded16b2010-05-10 14:33:55 +0100391}
392
393
Steve Blocka7e24c12009-10-30 11:49:00 +0000394bool StringShape::IsCons() {
395 return (type_ & kStringRepresentationMask) == kConsStringTag;
396}
397
398
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000399bool StringShape::IsSliced() {
400 return (type_ & kStringRepresentationMask) == kSlicedStringTag;
401}
402
403
404bool StringShape::IsIndirect() {
405 return (type_ & kIsIndirectStringMask) == kIsIndirectStringTag;
406}
407
408
Steve Blocka7e24c12009-10-30 11:49:00 +0000409bool StringShape::IsExternal() {
410 return (type_ & kStringRepresentationMask) == kExternalStringTag;
411}
412
413
414bool StringShape::IsSequential() {
415 return (type_ & kStringRepresentationMask) == kSeqStringTag;
416}
417
418
419StringRepresentationTag StringShape::representation_tag() {
420 uint32_t tag = (type_ & kStringRepresentationMask);
421 return static_cast<StringRepresentationTag>(tag);
422}
423
424
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000425uint32_t StringShape::encoding_tag() {
426 return type_ & kStringEncodingMask;
427}
428
429
Steve Blocka7e24c12009-10-30 11:49:00 +0000430uint32_t StringShape::full_representation_tag() {
431 return (type_ & (kStringRepresentationMask | kStringEncodingMask));
432}
433
434
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000435STATIC_ASSERT((kStringRepresentationMask | kStringEncodingMask) ==
Steve Blocka7e24c12009-10-30 11:49:00 +0000436 Internals::kFullStringRepresentationMask);
437
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000438STATIC_ASSERT(static_cast<uint32_t>(kStringEncodingMask) ==
439 Internals::kStringEncodingMask);
Steve Blocka7e24c12009-10-30 11:49:00 +0000440
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000441
442bool StringShape::IsSequentialOneByte() {
443 return full_representation_tag() == (kSeqStringTag | kOneByteStringTag);
Steve Blocka7e24c12009-10-30 11:49:00 +0000444}
445
446
447bool StringShape::IsSequentialTwoByte() {
448 return full_representation_tag() == (kSeqStringTag | kTwoByteStringTag);
449}
450
451
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000452bool StringShape::IsExternalOneByte() {
453 return full_representation_tag() == (kExternalStringTag | kOneByteStringTag);
Steve Blocka7e24c12009-10-30 11:49:00 +0000454}
455
456
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000457STATIC_ASSERT((kExternalStringTag | kOneByteStringTag) ==
458 Internals::kExternalOneByteRepresentationTag);
459
460STATIC_ASSERT(v8::String::ONE_BYTE_ENCODING == kOneByteStringTag);
461
462
Steve Blocka7e24c12009-10-30 11:49:00 +0000463bool StringShape::IsExternalTwoByte() {
464 return full_representation_tag() == (kExternalStringTag | kTwoByteStringTag);
465}
466
467
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000468STATIC_ASSERT((kExternalStringTag | kTwoByteStringTag) ==
Steve Blocka7e24c12009-10-30 11:49:00 +0000469 Internals::kExternalTwoByteRepresentationTag);
470
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000471STATIC_ASSERT(v8::String::TWO_BYTE_ENCODING == kTwoByteStringTag);
Steve Blocka7e24c12009-10-30 11:49:00 +0000472
Emily Bernierd0a1eb72015-03-24 16:35:39 -0400473
Steve Blocka7e24c12009-10-30 11:49:00 +0000474uc32 FlatStringReader::Get(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000475 if (is_one_byte_) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -0400476 return Get<uint8_t>(index);
Steve Blocka7e24c12009-10-30 11:49:00 +0000477 } else {
Emily Bernierd0a1eb72015-03-24 16:35:39 -0400478 return Get<uc16>(index);
479 }
480}
481
482
483template <typename Char>
484Char FlatStringReader::Get(int index) {
485 DCHECK_EQ(is_one_byte_, sizeof(Char) == 1);
486 DCHECK(0 <= index && index <= length_);
487 if (sizeof(Char) == 1) {
488 return static_cast<Char>(static_cast<const uint8_t*>(start_)[index]);
489 } else {
490 return static_cast<Char>(static_cast<const uc16*>(start_)[index]);
Steve Blocka7e24c12009-10-30 11:49:00 +0000491 }
492}
493
494
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000495Handle<Object> StringTableShape::AsHandle(Isolate* isolate, HashTableKey* key) {
496 return key->AsHandle(isolate);
497}
498
499
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000500Handle<Object> CompilationCacheShape::AsHandle(Isolate* isolate,
501 HashTableKey* key) {
502 return key->AsHandle(isolate);
503}
504
505
506Handle<Object> CodeCacheHashTableShape::AsHandle(Isolate* isolate,
507 HashTableKey* key) {
508 return key->AsHandle(isolate);
509}
510
511template <typename Char>
512class SequentialStringKey : public HashTableKey {
513 public:
514 explicit SequentialStringKey(Vector<const Char> string, uint32_t seed)
515 : string_(string), hash_field_(0), seed_(seed) { }
516
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000517 uint32_t Hash() override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000518 hash_field_ = StringHasher::HashSequentialString<Char>(string_.start(),
519 string_.length(),
520 seed_);
521
522 uint32_t result = hash_field_ >> String::kHashShift;
523 DCHECK(result != 0); // Ensure that the hash value of 0 is never computed.
524 return result;
525 }
526
527
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000528 uint32_t HashForObject(Object* other) override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000529 return String::cast(other)->Hash();
530 }
531
532 Vector<const Char> string_;
533 uint32_t hash_field_;
534 uint32_t seed_;
535};
536
537
538class OneByteStringKey : public SequentialStringKey<uint8_t> {
539 public:
540 OneByteStringKey(Vector<const uint8_t> str, uint32_t seed)
541 : SequentialStringKey<uint8_t>(str, seed) { }
542
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000543 bool IsMatch(Object* string) override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000544 return String::cast(string)->IsOneByteEqualTo(string_);
545 }
546
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000547 Handle<Object> AsHandle(Isolate* isolate) override;
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000548};
549
550
551class SeqOneByteSubStringKey : public HashTableKey {
552 public:
553 SeqOneByteSubStringKey(Handle<SeqOneByteString> string, int from, int length)
554 : string_(string), from_(from), length_(length) {
555 DCHECK(string_->IsSeqOneByteString());
556 }
557
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000558 uint32_t Hash() override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000559 DCHECK(length_ >= 0);
560 DCHECK(from_ + length_ <= string_->length());
561 const uint8_t* chars = string_->GetChars() + from_;
562 hash_field_ = StringHasher::HashSequentialString(
563 chars, length_, string_->GetHeap()->HashSeed());
564 uint32_t result = hash_field_ >> String::kHashShift;
565 DCHECK(result != 0); // Ensure that the hash value of 0 is never computed.
566 return result;
567 }
568
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000569 uint32_t HashForObject(Object* other) override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000570 return String::cast(other)->Hash();
571 }
572
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000573 bool IsMatch(Object* string) override;
574 Handle<Object> AsHandle(Isolate* isolate) override;
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000575
576 private:
577 Handle<SeqOneByteString> string_;
578 int from_;
579 int length_;
580 uint32_t hash_field_;
581};
582
583
584class TwoByteStringKey : public SequentialStringKey<uc16> {
585 public:
586 explicit TwoByteStringKey(Vector<const uc16> str, uint32_t seed)
587 : SequentialStringKey<uc16>(str, seed) { }
588
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000589 bool IsMatch(Object* string) override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000590 return String::cast(string)->IsTwoByteEqualTo(string_);
591 }
592
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000593 Handle<Object> AsHandle(Isolate* isolate) override;
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000594};
595
596
597// Utf8StringKey carries a vector of chars as key.
598class Utf8StringKey : public HashTableKey {
599 public:
600 explicit Utf8StringKey(Vector<const char> string, uint32_t seed)
601 : string_(string), hash_field_(0), seed_(seed) { }
602
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000603 bool IsMatch(Object* string) override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000604 return String::cast(string)->IsUtf8EqualTo(string_);
605 }
606
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000607 uint32_t Hash() override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000608 if (hash_field_ != 0) return hash_field_ >> String::kHashShift;
609 hash_field_ = StringHasher::ComputeUtf8Hash(string_, seed_, &chars_);
610 uint32_t result = hash_field_ >> String::kHashShift;
611 DCHECK(result != 0); // Ensure that the hash value of 0 is never computed.
612 return result;
613 }
614
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000615 uint32_t HashForObject(Object* other) override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000616 return String::cast(other)->Hash();
617 }
618
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000619 Handle<Object> AsHandle(Isolate* isolate) override {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000620 if (hash_field_ == 0) Hash();
621 return isolate->factory()->NewInternalizedStringFromUtf8(
622 string_, chars_, hash_field_);
623 }
624
625 Vector<const char> string_;
626 uint32_t hash_field_;
627 int chars_; // Caches the number of characters when computing the hash code.
628 uint32_t seed_;
629};
630
631
632bool Object::IsNumber() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000633 return IsSmi() || IsHeapNumber();
634}
635
636
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100637TYPE_CHECKER(ByteArray, BYTE_ARRAY_TYPE)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000638TYPE_CHECKER(BytecodeArray, BYTECODE_ARRAY_TYPE)
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100639TYPE_CHECKER(FreeSpace, FREE_SPACE_TYPE)
640
Ben Murdoch097c5b22016-05-18 11:27:45 +0100641bool HeapObject::IsFiller() const {
642 InstanceType instance_type = map()->instance_type();
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100643 return instance_type == FREE_SPACE_TYPE || instance_type == FILLER_TYPE;
Steve Blocka7e24c12009-10-30 11:49:00 +0000644}
645
646
Steve Block3ce2e202009-11-05 08:53:23 +0000647
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000648#define TYPED_ARRAY_TYPE_CHECKER(Type, type, TYPE, ctype, size) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000649 TYPE_CHECKER(Fixed##Type##Array, FIXED_##TYPE##_ARRAY_TYPE)
650
651TYPED_ARRAYS(TYPED_ARRAY_TYPE_CHECKER)
652#undef TYPED_ARRAY_TYPE_CHECKER
Ben Murdoch257744e2011-11-30 15:57:28 +0000653
Ben Murdoch097c5b22016-05-18 11:27:45 +0100654bool HeapObject::IsFixedTypedArrayBase() const {
655 InstanceType instance_type = map()->instance_type();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000656 return (instance_type >= FIRST_FIXED_TYPED_ARRAY_TYPE &&
657 instance_type <= LAST_FIXED_TYPED_ARRAY_TYPE);
Steve Blocka7e24c12009-10-30 11:49:00 +0000658}
659
Ben Murdoch097c5b22016-05-18 11:27:45 +0100660bool HeapObject::IsJSReceiver() const {
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100661 STATIC_ASSERT(LAST_JS_RECEIVER_TYPE == LAST_TYPE);
Ben Murdoch097c5b22016-05-18 11:27:45 +0100662 return map()->instance_type() >= FIRST_JS_RECEIVER_TYPE;
Ben Murdoch3fb3ca82011-12-02 17:19:32 +0000663}
664
Ben Murdoch097c5b22016-05-18 11:27:45 +0100665bool HeapObject::IsJSObject() const {
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100666 STATIC_ASSERT(LAST_JS_OBJECT_TYPE == LAST_TYPE);
Ben Murdoch097c5b22016-05-18 11:27:45 +0100667 return map()->IsJSObjectMap();
Ben Murdoch3fb3ca82011-12-02 17:19:32 +0000668}
669
Ben Murdoch097c5b22016-05-18 11:27:45 +0100670bool HeapObject::IsJSProxy() const { return map()->IsJSProxyMap(); }
Ben Murdoch3fb3ca82011-12-02 17:19:32 +0000671
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100672TYPE_CHECKER(JSSet, JS_SET_TYPE)
673TYPE_CHECKER(JSMap, JS_MAP_TYPE)
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000674TYPE_CHECKER(JSSetIterator, JS_SET_ITERATOR_TYPE)
675TYPE_CHECKER(JSMapIterator, JS_MAP_ITERATOR_TYPE)
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100676TYPE_CHECKER(JSWeakMap, JS_WEAK_MAP_TYPE)
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000677TYPE_CHECKER(JSWeakSet, JS_WEAK_SET_TYPE)
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100678TYPE_CHECKER(JSContextExtensionObject, JS_CONTEXT_EXTENSION_OBJECT_TYPE)
679TYPE_CHECKER(Map, MAP_TYPE)
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100680TYPE_CHECKER(FixedDoubleArray, FIXED_DOUBLE_ARRAY_TYPE)
Emily Bernierd0a1eb72015-03-24 16:35:39 -0400681TYPE_CHECKER(WeakFixedArray, FIXED_ARRAY_TYPE)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000682TYPE_CHECKER(TransitionArray, TRANSITION_ARRAY_TYPE)
Ben Murdoch3fb3ca82011-12-02 17:19:32 +0000683
Ben Murdoch097c5b22016-05-18 11:27:45 +0100684bool HeapObject::IsJSWeakCollection() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000685 return IsJSWeakMap() || IsJSWeakSet();
686}
687
Ben Murdoch097c5b22016-05-18 11:27:45 +0100688bool HeapObject::IsDescriptorArray() const { return IsFixedArray(); }
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000689
Ben Murdoch097c5b22016-05-18 11:27:45 +0100690bool HeapObject::IsArrayList() const { return IsFixedArray(); }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000691
Emily Bernierd0a1eb72015-03-24 16:35:39 -0400692bool Object::IsLayoutDescriptor() const {
693 return IsSmi() || IsFixedTypedArrayBase();
694}
695
Ben Murdoch097c5b22016-05-18 11:27:45 +0100696bool HeapObject::IsTypeFeedbackVector() const { return IsFixedArray(); }
Emily Bernierd0a1eb72015-03-24 16:35:39 -0400697
Ben Murdoch097c5b22016-05-18 11:27:45 +0100698bool HeapObject::IsTypeFeedbackMetadata() const { return IsFixedArray(); }
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000699
Ben Murdoch097c5b22016-05-18 11:27:45 +0100700bool HeapObject::IsLiteralsArray() const { return IsFixedArray(); }
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000701
Ben Murdoch097c5b22016-05-18 11:27:45 +0100702bool HeapObject::IsDeoptimizationInputData() const {
Ben Murdochb0fe1622011-05-05 13:52:32 +0100703 // Must be a fixed array.
704 if (!IsFixedArray()) return false;
705
706 // There's no sure way to detect the difference between a fixed array and
707 // a deoptimization data array. Since this is used for asserts we can
708 // check that the length is zero or else the fixed size plus a multiple of
709 // the entry size.
710 int length = FixedArray::cast(this)->length();
711 if (length == 0) return true;
712
713 length -= DeoptimizationInputData::kFirstDeoptEntryIndex;
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000714 return length >= 0 && length % DeoptimizationInputData::kDeoptEntrySize == 0;
Ben Murdochb0fe1622011-05-05 13:52:32 +0100715}
716
Ben Murdoch097c5b22016-05-18 11:27:45 +0100717bool HeapObject::IsDeoptimizationOutputData() const {
Ben Murdochb0fe1622011-05-05 13:52:32 +0100718 if (!IsFixedArray()) return false;
719 // There's actually no way to see the difference between a fixed array and
720 // a deoptimization data array. Since this is used for asserts we can check
721 // that the length is plausible though.
722 if (FixedArray::cast(this)->length() % 2 != 0) return false;
723 return true;
724}
725
Ben Murdoch097c5b22016-05-18 11:27:45 +0100726bool HeapObject::IsHandlerTable() const {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000727 if (!IsFixedArray()) return false;
728 // There's actually no way to see the difference between a fixed array and
729 // a handler table array.
730 return true;
731}
732
Ben Murdoch097c5b22016-05-18 11:27:45 +0100733bool HeapObject::IsDependentCode() const {
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100734 if (!IsFixedArray()) return false;
735 // There's actually no way to see the difference between a fixed array and
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000736 // a dependent codes array.
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100737 return true;
738}
739
Ben Murdoch097c5b22016-05-18 11:27:45 +0100740bool HeapObject::IsContext() const {
741 Map* map = this->map();
742 Heap* heap = GetHeap();
Ben Murdochda12d292016-06-02 14:46:10 +0100743 return (
744 map == heap->function_context_map() || map == heap->catch_context_map() ||
745 map == heap->with_context_map() || map == heap->native_context_map() ||
746 map == heap->block_context_map() || map == heap->module_context_map() ||
747 map == heap->script_context_map() ||
748 map == heap->debug_evaluate_context_map());
Steve Blocka7e24c12009-10-30 11:49:00 +0000749}
750
Ben Murdoch097c5b22016-05-18 11:27:45 +0100751bool HeapObject::IsNativeContext() const {
752 return map() == GetHeap()->native_context_map();
Steve Blocka7e24c12009-10-30 11:49:00 +0000753}
754
Ben Murdoch097c5b22016-05-18 11:27:45 +0100755bool HeapObject::IsScriptContextTable() const {
756 return map() == GetHeap()->script_context_table_map();
Emily Bernierd0a1eb72015-03-24 16:35:39 -0400757}
758
Ben Murdoch097c5b22016-05-18 11:27:45 +0100759bool HeapObject::IsScopeInfo() const {
760 return map() == GetHeap()->scope_info_map();
Ben Murdoch69a99ed2011-11-30 16:03:39 +0000761}
762
763
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000764TYPE_CHECKER(JSBoundFunction, JS_BOUND_FUNCTION_TYPE)
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100765TYPE_CHECKER(JSFunction, JS_FUNCTION_TYPE)
Steve Blocka7e24c12009-10-30 11:49:00 +0000766
767
768template <> inline bool Is<JSFunction>(Object* obj) {
769 return obj->IsJSFunction();
770}
771
772
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100773TYPE_CHECKER(Code, CODE_TYPE)
774TYPE_CHECKER(Oddball, ODDBALL_TYPE)
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000775TYPE_CHECKER(Cell, CELL_TYPE)
776TYPE_CHECKER(PropertyCell, PROPERTY_CELL_TYPE)
Emily Bernierd0a1eb72015-03-24 16:35:39 -0400777TYPE_CHECKER(WeakCell, WEAK_CELL_TYPE)
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100778TYPE_CHECKER(SharedFunctionInfo, SHARED_FUNCTION_INFO_TYPE)
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000779TYPE_CHECKER(JSGeneratorObject, JS_GENERATOR_OBJECT_TYPE)
780TYPE_CHECKER(JSModule, JS_MODULE_TYPE)
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100781TYPE_CHECKER(JSValue, JS_VALUE_TYPE)
782TYPE_CHECKER(JSDate, JS_DATE_TYPE)
783TYPE_CHECKER(JSMessageObject, JS_MESSAGE_OBJECT_TYPE)
Steve Blocka7e24c12009-10-30 11:49:00 +0000784
Ben Murdoch097c5b22016-05-18 11:27:45 +0100785bool HeapObject::IsAbstractCode() const {
786 return IsBytecodeArray() || IsCode();
787}
Steve Blocka7e24c12009-10-30 11:49:00 +0000788
Ben Murdoch097c5b22016-05-18 11:27:45 +0100789bool HeapObject::IsStringWrapper() const {
Steve Blocka7e24c12009-10-30 11:49:00 +0000790 return IsJSValue() && JSValue::cast(this)->value()->IsString();
791}
792
793
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100794TYPE_CHECKER(Foreign, FOREIGN_TYPE)
Steve Blocka7e24c12009-10-30 11:49:00 +0000795
Ben Murdoch097c5b22016-05-18 11:27:45 +0100796bool HeapObject::IsBoolean() const {
Steve Block44f0eee2011-05-26 01:26:41 +0100797 return IsOddball() &&
798 ((Oddball::cast(this)->kind() & Oddball::kNotBooleanMask) == 0);
Steve Blocka7e24c12009-10-30 11:49:00 +0000799}
800
801
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100802TYPE_CHECKER(JSArray, JS_ARRAY_TYPE)
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000803TYPE_CHECKER(JSArrayBuffer, JS_ARRAY_BUFFER_TYPE)
804TYPE_CHECKER(JSTypedArray, JS_TYPED_ARRAY_TYPE)
805TYPE_CHECKER(JSDataView, JS_DATA_VIEW_TYPE)
806
Ben Murdoch097c5b22016-05-18 11:27:45 +0100807bool HeapObject::IsJSArrayBufferView() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000808 return IsJSDataView() || IsJSTypedArray();
809}
810
811
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100812TYPE_CHECKER(JSRegExp, JS_REGEXP_TYPE)
Steve Blocka7e24c12009-10-30 11:49:00 +0000813
814
815template <> inline bool Is<JSArray>(Object* obj) {
816 return obj->IsJSArray();
817}
818
Ben Murdoch097c5b22016-05-18 11:27:45 +0100819bool HeapObject::IsHashTable() const {
820 return map() == GetHeap()->hash_table_map();
Steve Blocka7e24c12009-10-30 11:49:00 +0000821}
822
Ben Murdoch097c5b22016-05-18 11:27:45 +0100823bool HeapObject::IsWeakHashTable() const { return IsHashTable(); }
Steve Blocka7e24c12009-10-30 11:49:00 +0000824
Ben Murdoch097c5b22016-05-18 11:27:45 +0100825bool HeapObject::IsDictionary() const {
826 return IsHashTable() && this != GetHeap()->string_table();
Steve Blocka7e24c12009-10-30 11:49:00 +0000827}
828
829
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000830bool Object::IsNameDictionary() const {
831 return IsDictionary();
Steve Blocka7e24c12009-10-30 11:49:00 +0000832}
833
834
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000835bool Object::IsGlobalDictionary() const { return IsDictionary(); }
836
837
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000838bool Object::IsSeededNumberDictionary() const {
839 return IsDictionary();
840}
841
842
843bool Object::IsUnseededNumberDictionary() const {
844 return IsDictionary();
845}
846
Ben Murdoch097c5b22016-05-18 11:27:45 +0100847bool HeapObject::IsStringTable() const { return IsHashTable(); }
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000848
Ben Murdochda12d292016-06-02 14:46:10 +0100849bool HeapObject::IsStringSet() const { return IsHashTable(); }
850
Ben Murdoch097c5b22016-05-18 11:27:45 +0100851bool HeapObject::IsNormalizedMapCache() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000852 return NormalizedMapCache::IsNormalizedMapCache(this);
853}
854
855
856int NormalizedMapCache::GetIndex(Handle<Map> map) {
857 return map->Hash() % NormalizedMapCache::kEntries;
858}
859
Ben Murdoch097c5b22016-05-18 11:27:45 +0100860bool NormalizedMapCache::IsNormalizedMapCache(const HeapObject* obj) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000861 if (!obj->IsFixedArray()) return false;
862 if (FixedArray::cast(obj)->length() != NormalizedMapCache::kEntries) {
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100863 return false;
864 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000865#ifdef VERIFY_HEAP
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100866 if (FLAG_verify_heap) {
Ben Murdoch097c5b22016-05-18 11:27:45 +0100867 reinterpret_cast<NormalizedMapCache*>(const_cast<HeapObject*>(obj))
868 ->NormalizedMapCacheVerify();
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100869 }
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100870#endif
871 return true;
872}
873
Ben Murdoch097c5b22016-05-18 11:27:45 +0100874bool HeapObject::IsCompilationCacheTable() const { return IsHashTable(); }
Kristian Monsen80d68ea2010-09-08 11:05:35 +0100875
Ben Murdoch097c5b22016-05-18 11:27:45 +0100876bool HeapObject::IsCodeCacheHashTable() const { return IsHashTable(); }
877
Ben Murdoch097c5b22016-05-18 11:27:45 +0100878bool HeapObject::IsMapCache() const { return IsHashTable(); }
Steve Blocka7e24c12009-10-30 11:49:00 +0000879
Ben Murdoch097c5b22016-05-18 11:27:45 +0100880bool HeapObject::IsObjectHashTable() const { return IsHashTable(); }
Steve Block6ded16b2010-05-10 14:33:55 +0100881
Ben Murdoch097c5b22016-05-18 11:27:45 +0100882bool HeapObject::IsOrderedHashTable() const {
883 return map() == GetHeap()->ordered_hash_table_map();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000884}
885
886
887bool Object::IsOrderedHashSet() const {
888 return IsOrderedHashTable();
889}
890
891
892bool Object::IsOrderedHashMap() const {
893 return IsOrderedHashTable();
894}
895
896
897bool Object::IsPrimitive() const {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000898 return IsSmi() || HeapObject::cast(this)->map()->IsPrimitiveMap();
Steve Blocka7e24c12009-10-30 11:49:00 +0000899}
900
Ben Murdoch097c5b22016-05-18 11:27:45 +0100901bool HeapObject::IsJSGlobalProxy() const {
902 bool result = map()->instance_type() == JS_GLOBAL_PROXY_TYPE;
903 DCHECK(!result || map()->is_access_check_needed());
Steve Blocka7e24c12009-10-30 11:49:00 +0000904 return result;
905}
906
907
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100908TYPE_CHECKER(JSGlobalObject, JS_GLOBAL_OBJECT_TYPE)
Steve Blocka7e24c12009-10-30 11:49:00 +0000909
Ben Murdochda12d292016-06-02 14:46:10 +0100910bool HeapObject::IsUndetectable() const { return map()->is_undetectable(); }
Steve Blocka7e24c12009-10-30 11:49:00 +0000911
Ben Murdoch097c5b22016-05-18 11:27:45 +0100912bool HeapObject::IsAccessCheckNeeded() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000913 if (IsJSGlobalProxy()) {
914 const JSGlobalProxy* proxy = JSGlobalProxy::cast(this);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000915 JSGlobalObject* global = proxy->GetIsolate()->context()->global_object();
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000916 return proxy->IsDetachedFrom(global);
917 }
Ben Murdoch097c5b22016-05-18 11:27:45 +0100918 return map()->is_access_check_needed();
Steve Blocka7e24c12009-10-30 11:49:00 +0000919}
920
Ben Murdoch097c5b22016-05-18 11:27:45 +0100921bool HeapObject::IsStruct() const {
922 switch (map()->instance_type()) {
Steve Blocka7e24c12009-10-30 11:49:00 +0000923#define MAKE_STRUCT_CASE(NAME, Name, name) case NAME##_TYPE: return true;
924 STRUCT_LIST(MAKE_STRUCT_CASE)
925#undef MAKE_STRUCT_CASE
926 default: return false;
927 }
928}
929
Ben Murdoch097c5b22016-05-18 11:27:45 +0100930#define MAKE_STRUCT_PREDICATE(NAME, Name, name) \
931 bool Object::Is##Name() const { \
932 return IsHeapObject() && HeapObject::cast(this)->Is##Name(); \
933 } \
934 bool HeapObject::Is##Name() const { \
935 return map()->instance_type() == NAME##_TYPE; \
Steve Blocka7e24c12009-10-30 11:49:00 +0000936 }
Ben Murdoch097c5b22016-05-18 11:27:45 +0100937STRUCT_LIST(MAKE_STRUCT_PREDICATE)
Steve Blocka7e24c12009-10-30 11:49:00 +0000938#undef MAKE_STRUCT_PREDICATE
939
Ben Murdoch097c5b22016-05-18 11:27:45 +0100940#define MAKE_ODDBALL_PREDICATE(Name) \
941 bool HeapObject::Is##Name() const { \
942 return IsOddball() && Oddball::cast(this)->kind() == Oddball::k##Name; \
943 }
944ODDBALL_LIST(MAKE_ODDBALL_PREDICATE)
Steve Blocka7e24c12009-10-30 11:49:00 +0000945
Ben Murdoch097c5b22016-05-18 11:27:45 +0100946#undef MAKE_ODDBALL_PREDICATE
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000947double Object::Number() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000948 DCHECK(IsNumber());
Steve Blocka7e24c12009-10-30 11:49:00 +0000949 return IsSmi()
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000950 ? static_cast<double>(reinterpret_cast<const Smi*>(this)->value())
951 : reinterpret_cast<const HeapNumber*>(this)->value();
Steve Blocka7e24c12009-10-30 11:49:00 +0000952}
953
954
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000955bool Object::IsNaN() const {
956 return this->IsHeapNumber() && std::isnan(HeapNumber::cast(this)->value());
Ben Murdoch3ef787d2012-04-12 10:51:47 +0100957}
958
959
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000960bool Object::IsMinusZero() const {
961 return this->IsHeapNumber() &&
962 i::IsMinusZero(HeapNumber::cast(this)->value());
963}
964
965
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000966Representation Object::OptimalRepresentation() {
967 if (!FLAG_track_fields) return Representation::Tagged();
968 if (IsSmi()) {
969 return Representation::Smi();
970 } else if (FLAG_track_double_fields && IsHeapNumber()) {
971 return Representation::Double();
972 } else if (FLAG_track_computed_fields && IsUninitialized()) {
973 return Representation::None();
974 } else if (FLAG_track_heap_object_fields) {
975 DCHECK(IsHeapObject());
976 return Representation::HeapObject();
977 } else {
978 return Representation::Tagged();
Steve Blocka7e24c12009-10-30 11:49:00 +0000979 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +0000980}
981
982
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000983ElementsKind Object::OptimalElementsKind() {
984 if (IsSmi()) return FAST_SMI_ELEMENTS;
985 if (IsNumber()) return FAST_DOUBLE_ELEMENTS;
986 return FAST_ELEMENTS;
987}
988
989
990bool Object::FitsRepresentation(Representation representation) {
Ben Murdoch097c5b22016-05-18 11:27:45 +0100991 if (FLAG_track_fields && representation.IsSmi()) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000992 return IsSmi();
993 } else if (FLAG_track_double_fields && representation.IsDouble()) {
994 return IsMutableHeapNumber() || IsNumber();
995 } else if (FLAG_track_heap_object_fields && representation.IsHeapObject()) {
996 return IsHeapObject();
Ben Murdoch097c5b22016-05-18 11:27:45 +0100997 } else if (FLAG_track_fields && representation.IsNone()) {
998 return false;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +0000999 }
1000 return true;
1001}
1002
Ben Murdochda12d292016-06-02 14:46:10 +01001003bool Object::ToUint32(uint32_t* value) {
1004 if (IsSmi()) {
1005 int num = Smi::cast(this)->value();
1006 if (num < 0) return false;
1007 *value = static_cast<uint32_t>(num);
1008 return true;
1009 }
1010 if (IsHeapNumber()) {
1011 double num = HeapNumber::cast(this)->value();
1012 if (num < 0) return false;
1013 uint32_t uint_value = FastD2UI(num);
1014 if (FastUI2D(uint_value) == num) {
1015 *value = uint_value;
1016 return true;
1017 }
1018 }
1019 return false;
1020}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001021
1022// static
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001023MaybeHandle<JSReceiver> Object::ToObject(Isolate* isolate,
1024 Handle<Object> object) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01001025 if (object->IsJSReceiver()) return Handle<JSReceiver>::cast(object);
1026 return ToObject(isolate, object, isolate->native_context());
Steve Blocka7e24c12009-10-30 11:49:00 +00001027}
1028
1029
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001030// static
Ben Murdochda12d292016-06-02 14:46:10 +01001031MaybeHandle<Name> Object::ToName(Isolate* isolate, Handle<Object> input) {
1032 if (input->IsName()) return Handle<Name>::cast(input);
1033 return ConvertToName(isolate, input);
1034}
1035
1036// static
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001037MaybeHandle<Object> Object::ToPrimitive(Handle<Object> input,
1038 ToPrimitiveHint hint) {
1039 if (input->IsPrimitive()) return input;
1040 return JSReceiver::ToPrimitive(Handle<JSReceiver>::cast(input), hint);
1041}
1042
1043
Steve Blocka7e24c12009-10-30 11:49:00 +00001044bool Object::HasSpecificClassOf(String* name) {
1045 return this->IsJSObject() && (JSObject::cast(this)->class_name() == name);
1046}
1047
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001048MaybeHandle<Object> Object::GetProperty(Handle<Object> object,
Ben Murdoch097c5b22016-05-18 11:27:45 +01001049 Handle<Name> name) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001050 LookupIterator it(object, name);
Ben Murdochda12d292016-06-02 14:46:10 +01001051 if (!it.IsFound()) return it.factory()->undefined_value();
Ben Murdoch097c5b22016-05-18 11:27:45 +01001052 return GetProperty(&it);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001053}
1054
Ben Murdochda12d292016-06-02 14:46:10 +01001055MaybeHandle<Object> JSReceiver::GetProperty(Handle<JSReceiver> receiver,
1056 Handle<Name> name) {
1057 LookupIterator it(receiver, name, receiver);
1058 if (!it.IsFound()) return it.factory()->undefined_value();
1059 return Object::GetProperty(&it);
1060}
1061
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001062MaybeHandle<Object> Object::GetElement(Isolate* isolate, Handle<Object> object,
Ben Murdoch097c5b22016-05-18 11:27:45 +01001063 uint32_t index) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001064 LookupIterator it(isolate, object, index);
Ben Murdochda12d292016-06-02 14:46:10 +01001065 if (!it.IsFound()) return it.factory()->undefined_value();
Ben Murdoch097c5b22016-05-18 11:27:45 +01001066 return GetProperty(&it);
Steve Blocka7e24c12009-10-30 11:49:00 +00001067}
1068
Ben Murdochda12d292016-06-02 14:46:10 +01001069MaybeHandle<Object> JSReceiver::GetElement(Isolate* isolate,
1070 Handle<JSReceiver> receiver,
1071 uint32_t index) {
1072 LookupIterator it(isolate, receiver, index, receiver);
1073 if (!it.IsFound()) return it.factory()->undefined_value();
1074 return Object::GetProperty(&it);
1075}
1076
1077Handle<Object> JSReceiver::GetDataProperty(Handle<JSReceiver> object,
1078 Handle<Name> name) {
1079 LookupIterator it(object, name, object,
1080 LookupIterator::PROTOTYPE_CHAIN_SKIP_INTERCEPTOR);
1081 if (!it.IsFound()) return it.factory()->undefined_value();
1082 return GetDataProperty(&it);
1083}
Steve Blocka7e24c12009-10-30 11:49:00 +00001084
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001085MaybeHandle<Object> Object::SetElement(Isolate* isolate, Handle<Object> object,
1086 uint32_t index, Handle<Object> value,
1087 LanguageMode language_mode) {
1088 LookupIterator it(isolate, object, index);
1089 MAYBE_RETURN_NULL(
1090 SetProperty(&it, value, language_mode, MAY_BE_STORE_FROM_KEYED));
1091 return value;
1092}
1093
Ben Murdoch097c5b22016-05-18 11:27:45 +01001094MaybeHandle<Object> JSReceiver::GetPrototype(Isolate* isolate,
1095 Handle<JSReceiver> receiver) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001096 // We don't expect access checks to be needed on JSProxy objects.
1097 DCHECK(!receiver->IsAccessCheckNeeded() || receiver->IsJSObject());
1098 PrototypeIterator iter(isolate, receiver,
Ben Murdoch097c5b22016-05-18 11:27:45 +01001099 PrototypeIterator::START_AT_RECEIVER,
1100 PrototypeIterator::END_AT_NON_HIDDEN);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001101 do {
1102 if (!iter.AdvanceFollowingProxies()) return MaybeHandle<Object>();
Ben Murdoch097c5b22016-05-18 11:27:45 +01001103 } while (!iter.IsAtEnd());
Emily Bernierd0a1eb72015-03-24 16:35:39 -04001104 return PrototypeIterator::GetCurrent(iter);
1105}
1106
Ben Murdochda12d292016-06-02 14:46:10 +01001107MaybeHandle<Object> JSReceiver::GetProperty(Isolate* isolate,
1108 Handle<JSReceiver> receiver,
1109 const char* name) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001110 Handle<String> str = isolate->factory()->InternalizeUtf8String(name);
Ben Murdochda12d292016-06-02 14:46:10 +01001111 return GetProperty(receiver, str);
Steve Blocka7e24c12009-10-30 11:49:00 +00001112}
1113
1114
1115#define FIELD_ADDR(p, offset) \
1116 (reinterpret_cast<byte*>(p) + offset - kHeapObjectTag)
1117
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001118#define FIELD_ADDR_CONST(p, offset) \
1119 (reinterpret_cast<const byte*>(p) + offset - kHeapObjectTag)
1120
Steve Blocka7e24c12009-10-30 11:49:00 +00001121#define READ_FIELD(p, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001122 (*reinterpret_cast<Object* const*>(FIELD_ADDR_CONST(p, offset)))
1123
1124#define ACQUIRE_READ_FIELD(p, offset) \
1125 reinterpret_cast<Object*>(base::Acquire_Load( \
1126 reinterpret_cast<const base::AtomicWord*>(FIELD_ADDR_CONST(p, offset))))
1127
1128#define NOBARRIER_READ_FIELD(p, offset) \
1129 reinterpret_cast<Object*>(base::NoBarrier_Load( \
1130 reinterpret_cast<const base::AtomicWord*>(FIELD_ADDR_CONST(p, offset))))
Steve Blocka7e24c12009-10-30 11:49:00 +00001131
1132#define WRITE_FIELD(p, offset, value) \
1133 (*reinterpret_cast<Object**>(FIELD_ADDR(p, offset)) = value)
1134
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001135#define RELEASE_WRITE_FIELD(p, offset, value) \
1136 base::Release_Store( \
1137 reinterpret_cast<base::AtomicWord*>(FIELD_ADDR(p, offset)), \
1138 reinterpret_cast<base::AtomicWord>(value));
1139
1140#define NOBARRIER_WRITE_FIELD(p, offset, value) \
1141 base::NoBarrier_Store( \
1142 reinterpret_cast<base::AtomicWord*>(FIELD_ADDR(p, offset)), \
1143 reinterpret_cast<base::AtomicWord>(value));
1144
Ben Murdoch097c5b22016-05-18 11:27:45 +01001145#define WRITE_BARRIER(heap, object, offset, value) \
1146 heap->incremental_marking()->RecordWrite( \
1147 object, HeapObject::RawField(object, offset), value); \
1148 heap->RecordWrite(object, offset, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00001149
Ben Murdochc5610432016-08-08 18:44:38 +01001150#define FIXED_ARRAY_ELEMENTS_WRITE_BARRIER(heap, array, start, length) \
1151 do { \
1152 heap->RecordFixedArrayElements(array, start, length); \
1153 heap->incremental_marking()->IterateBlackObject(array); \
1154 } while (false)
1155
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001156#define CONDITIONAL_WRITE_BARRIER(heap, object, offset, value, mode) \
1157 if (mode != SKIP_WRITE_BARRIER) { \
1158 if (mode == UPDATE_WRITE_BARRIER) { \
1159 heap->incremental_marking()->RecordWrite( \
1160 object, HeapObject::RawField(object, offset), value); \
1161 } \
Ben Murdoch097c5b22016-05-18 11:27:45 +01001162 heap->RecordWrite(object, offset, value); \
Steve Blocka7e24c12009-10-30 11:49:00 +00001163 }
1164
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001165#define READ_DOUBLE_FIELD(p, offset) \
1166 ReadDoubleValue(FIELD_ADDR_CONST(p, offset))
Steve Blocka7e24c12009-10-30 11:49:00 +00001167
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001168#define WRITE_DOUBLE_FIELD(p, offset, value) \
1169 WriteDoubleValue(FIELD_ADDR(p, offset), value)
Steve Blocka7e24c12009-10-30 11:49:00 +00001170
1171#define READ_INT_FIELD(p, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001172 (*reinterpret_cast<const int*>(FIELD_ADDR_CONST(p, offset)))
Steve Blocka7e24c12009-10-30 11:49:00 +00001173
1174#define WRITE_INT_FIELD(p, offset, value) \
1175 (*reinterpret_cast<int*>(FIELD_ADDR(p, offset)) = value)
1176
1177#define READ_INTPTR_FIELD(p, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001178 (*reinterpret_cast<const intptr_t*>(FIELD_ADDR_CONST(p, offset)))
Steve Blocka7e24c12009-10-30 11:49:00 +00001179
1180#define WRITE_INTPTR_FIELD(p, offset, value) \
1181 (*reinterpret_cast<intptr_t*>(FIELD_ADDR(p, offset)) = value)
1182
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001183#define READ_UINT8_FIELD(p, offset) \
1184 (*reinterpret_cast<const uint8_t*>(FIELD_ADDR_CONST(p, offset)))
1185
1186#define WRITE_UINT8_FIELD(p, offset, value) \
1187 (*reinterpret_cast<uint8_t*>(FIELD_ADDR(p, offset)) = value)
1188
1189#define READ_INT8_FIELD(p, offset) \
1190 (*reinterpret_cast<const int8_t*>(FIELD_ADDR_CONST(p, offset)))
1191
1192#define WRITE_INT8_FIELD(p, offset, value) \
1193 (*reinterpret_cast<int8_t*>(FIELD_ADDR(p, offset)) = value)
1194
1195#define READ_UINT16_FIELD(p, offset) \
1196 (*reinterpret_cast<const uint16_t*>(FIELD_ADDR_CONST(p, offset)))
1197
1198#define WRITE_UINT16_FIELD(p, offset, value) \
1199 (*reinterpret_cast<uint16_t*>(FIELD_ADDR(p, offset)) = value)
1200
1201#define READ_INT16_FIELD(p, offset) \
1202 (*reinterpret_cast<const int16_t*>(FIELD_ADDR_CONST(p, offset)))
1203
1204#define WRITE_INT16_FIELD(p, offset, value) \
1205 (*reinterpret_cast<int16_t*>(FIELD_ADDR(p, offset)) = value)
1206
Steve Blocka7e24c12009-10-30 11:49:00 +00001207#define READ_UINT32_FIELD(p, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001208 (*reinterpret_cast<const uint32_t*>(FIELD_ADDR_CONST(p, offset)))
Steve Blocka7e24c12009-10-30 11:49:00 +00001209
1210#define WRITE_UINT32_FIELD(p, offset, value) \
1211 (*reinterpret_cast<uint32_t*>(FIELD_ADDR(p, offset)) = value)
1212
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001213#define READ_INT32_FIELD(p, offset) \
1214 (*reinterpret_cast<const int32_t*>(FIELD_ADDR_CONST(p, offset)))
1215
1216#define WRITE_INT32_FIELD(p, offset, value) \
1217 (*reinterpret_cast<int32_t*>(FIELD_ADDR(p, offset)) = value)
1218
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001219#define READ_FLOAT_FIELD(p, offset) \
1220 (*reinterpret_cast<const float*>(FIELD_ADDR_CONST(p, offset)))
1221
1222#define WRITE_FLOAT_FIELD(p, offset, value) \
1223 (*reinterpret_cast<float*>(FIELD_ADDR(p, offset)) = value)
1224
1225#define READ_UINT64_FIELD(p, offset) \
1226 (*reinterpret_cast<const uint64_t*>(FIELD_ADDR_CONST(p, offset)))
1227
1228#define WRITE_UINT64_FIELD(p, offset, value) \
1229 (*reinterpret_cast<uint64_t*>(FIELD_ADDR(p, offset)) = value)
1230
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001231#define READ_INT64_FIELD(p, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001232 (*reinterpret_cast<const int64_t*>(FIELD_ADDR_CONST(p, offset)))
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001233
1234#define WRITE_INT64_FIELD(p, offset, value) \
1235 (*reinterpret_cast<int64_t*>(FIELD_ADDR(p, offset)) = value)
1236
Steve Blocka7e24c12009-10-30 11:49:00 +00001237#define READ_BYTE_FIELD(p, offset) \
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001238 (*reinterpret_cast<const byte*>(FIELD_ADDR_CONST(p, offset)))
1239
1240#define NOBARRIER_READ_BYTE_FIELD(p, offset) \
1241 static_cast<byte>(base::NoBarrier_Load( \
1242 reinterpret_cast<base::Atomic8*>(FIELD_ADDR(p, offset))))
Steve Blocka7e24c12009-10-30 11:49:00 +00001243
1244#define WRITE_BYTE_FIELD(p, offset, value) \
1245 (*reinterpret_cast<byte*>(FIELD_ADDR(p, offset)) = value)
1246
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001247#define NOBARRIER_WRITE_BYTE_FIELD(p, offset, value) \
1248 base::NoBarrier_Store( \
1249 reinterpret_cast<base::Atomic8*>(FIELD_ADDR(p, offset)), \
1250 static_cast<base::Atomic8>(value));
Steve Blocka7e24c12009-10-30 11:49:00 +00001251
1252Object** HeapObject::RawField(HeapObject* obj, int byte_offset) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001253 return reinterpret_cast<Object**>(FIELD_ADDR(obj, byte_offset));
Steve Blocka7e24c12009-10-30 11:49:00 +00001254}
1255
1256
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001257MapWord MapWord::FromMap(const Map* map) {
Steve Blocka7e24c12009-10-30 11:49:00 +00001258 return MapWord(reinterpret_cast<uintptr_t>(map));
1259}
1260
1261
1262Map* MapWord::ToMap() {
1263 return reinterpret_cast<Map*>(value_);
1264}
1265
Ben Murdochc5610432016-08-08 18:44:38 +01001266bool MapWord::IsForwardingAddress() const {
Steve Blocka7e24c12009-10-30 11:49:00 +00001267 return HAS_SMI_TAG(reinterpret_cast<Object*>(value_));
1268}
1269
1270
1271MapWord MapWord::FromForwardingAddress(HeapObject* object) {
1272 Address raw = reinterpret_cast<Address>(object) - kHeapObjectTag;
1273 return MapWord(reinterpret_cast<uintptr_t>(raw));
1274}
1275
1276
1277HeapObject* MapWord::ToForwardingAddress() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001278 DCHECK(IsForwardingAddress());
Steve Blocka7e24c12009-10-30 11:49:00 +00001279 return HeapObject::FromAddress(reinterpret_cast<Address>(value_));
1280}
1281
1282
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001283#ifdef VERIFY_HEAP
Steve Blocka7e24c12009-10-30 11:49:00 +00001284void HeapObject::VerifyObjectField(int offset) {
1285 VerifyPointer(READ_FIELD(this, offset));
1286}
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01001287
1288void HeapObject::VerifySmiField(int offset) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001289 CHECK(READ_FIELD(this, offset)->IsSmi());
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01001290}
Steve Blocka7e24c12009-10-30 11:49:00 +00001291#endif
1292
1293
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001294Heap* HeapObject::GetHeap() const {
Ben Murdoch097c5b22016-05-18 11:27:45 +01001295 Heap* heap = MemoryChunk::FromAddress(
1296 reinterpret_cast<Address>(const_cast<HeapObject*>(this)))
1297 ->heap();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001298 SLOW_DCHECK(heap != NULL);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001299 return heap;
Steve Block44f0eee2011-05-26 01:26:41 +01001300}
1301
1302
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001303Isolate* HeapObject::GetIsolate() const {
Steve Block44f0eee2011-05-26 01:26:41 +01001304 return GetHeap()->isolate();
1305}
1306
1307
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001308Map* HeapObject::map() const {
1309#ifdef DEBUG
1310 // Clear mark potentially added by PathTracer.
1311 uintptr_t raw_value =
1312 map_word().ToRawValue() & ~static_cast<uintptr_t>(PathTracer::kMarkTag);
1313 return MapWord::FromRawValue(raw_value).ToMap();
1314#else
Steve Blocka7e24c12009-10-30 11:49:00 +00001315 return map_word().ToMap();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001316#endif
Steve Blocka7e24c12009-10-30 11:49:00 +00001317}
1318
1319
1320void HeapObject::set_map(Map* value) {
1321 set_map_word(MapWord::FromMap(value));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001322 if (value != NULL) {
1323 // TODO(1600) We are passing NULL as a slot because maps can never be on
1324 // evacuation candidate.
1325 value->GetHeap()->incremental_marking()->RecordWrite(this, NULL, value);
1326 }
1327}
1328
1329
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001330Map* HeapObject::synchronized_map() {
1331 return synchronized_map_word().ToMap();
1332}
1333
1334
1335void HeapObject::synchronized_set_map(Map* value) {
1336 synchronized_set_map_word(MapWord::FromMap(value));
1337 if (value != NULL) {
1338 // TODO(1600) We are passing NULL as a slot because maps can never be on
1339 // evacuation candidate.
1340 value->GetHeap()->incremental_marking()->RecordWrite(this, NULL, value);
1341 }
1342}
1343
1344
1345void HeapObject::synchronized_set_map_no_write_barrier(Map* value) {
1346 synchronized_set_map_word(MapWord::FromMap(value));
1347}
1348
1349
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001350// Unsafe accessor omitting write barrier.
1351void HeapObject::set_map_no_write_barrier(Map* value) {
1352 set_map_word(MapWord::FromMap(value));
Steve Blocka7e24c12009-10-30 11:49:00 +00001353}
1354
1355
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001356MapWord HeapObject::map_word() const {
1357 return MapWord(
1358 reinterpret_cast<uintptr_t>(NOBARRIER_READ_FIELD(this, kMapOffset)));
Steve Blocka7e24c12009-10-30 11:49:00 +00001359}
1360
1361
1362void HeapObject::set_map_word(MapWord map_word) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001363 NOBARRIER_WRITE_FIELD(
1364 this, kMapOffset, reinterpret_cast<Object*>(map_word.value_));
1365}
1366
1367
1368MapWord HeapObject::synchronized_map_word() const {
1369 return MapWord(
1370 reinterpret_cast<uintptr_t>(ACQUIRE_READ_FIELD(this, kMapOffset)));
1371}
1372
1373
1374void HeapObject::synchronized_set_map_word(MapWord map_word) {
1375 RELEASE_WRITE_FIELD(
1376 this, kMapOffset, reinterpret_cast<Object*>(map_word.value_));
Steve Blocka7e24c12009-10-30 11:49:00 +00001377}
1378
1379
Steve Blocka7e24c12009-10-30 11:49:00 +00001380int HeapObject::Size() {
1381 return SizeFromMap(map());
1382}
1383
1384
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001385double HeapNumber::value() const {
Steve Blocka7e24c12009-10-30 11:49:00 +00001386 return READ_DOUBLE_FIELD(this, kValueOffset);
1387}
1388
1389
1390void HeapNumber::set_value(double value) {
1391 WRITE_DOUBLE_FIELD(this, kValueOffset, value);
1392}
1393
1394
Steve Block6ded16b2010-05-10 14:33:55 +01001395int HeapNumber::get_exponent() {
1396 return ((READ_INT_FIELD(this, kExponentOffset) & kExponentMask) >>
1397 kExponentShift) - kExponentBias;
1398}
1399
1400
1401int HeapNumber::get_sign() {
1402 return READ_INT_FIELD(this, kExponentOffset) & kSignMask;
1403}
1404
1405
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001406bool Simd128Value::Equals(Simd128Value* that) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01001407 // TODO(bmeurer): This doesn't match the SIMD.js specification, but it seems
1408 // to be consistent with what the CompareICStub does, and what is tested in
1409 // the current SIMD.js testsuite.
1410 if (this == that) return true;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001411#define SIMD128_VALUE(TYPE, Type, type, lane_count, lane_type) \
1412 if (this->Is##Type()) { \
1413 if (!that->Is##Type()) return false; \
1414 return Type::cast(this)->Equals(Type::cast(that)); \
1415 }
1416 SIMD128_TYPES(SIMD128_VALUE)
1417#undef SIMD128_VALUE
1418 return false;
1419}
1420
1421
1422// static
1423bool Simd128Value::Equals(Handle<Simd128Value> one, Handle<Simd128Value> two) {
1424 return one->Equals(*two);
1425}
1426
1427
1428#define SIMD128_VALUE_EQUALS(TYPE, Type, type, lane_count, lane_type) \
1429 bool Type::Equals(Type* that) { \
1430 for (int lane = 0; lane < lane_count; ++lane) { \
1431 if (this->get_lane(lane) != that->get_lane(lane)) return false; \
1432 } \
1433 return true; \
1434 }
1435SIMD128_TYPES(SIMD128_VALUE_EQUALS)
1436#undef SIMD128_VALUE_EQUALS
1437
1438
1439#if defined(V8_TARGET_LITTLE_ENDIAN)
1440#define SIMD128_READ_LANE(lane_type, lane_count, field_type, field_size) \
1441 lane_type value = \
1442 READ_##field_type##_FIELD(this, kValueOffset + lane * field_size);
1443#elif defined(V8_TARGET_BIG_ENDIAN)
1444#define SIMD128_READ_LANE(lane_type, lane_count, field_type, field_size) \
1445 lane_type value = READ_##field_type##_FIELD( \
1446 this, kValueOffset + (lane_count - lane - 1) * field_size);
1447#else
1448#error Unknown byte ordering
1449#endif
1450
1451#if defined(V8_TARGET_LITTLE_ENDIAN)
1452#define SIMD128_WRITE_LANE(lane_count, field_type, field_size, value) \
1453 WRITE_##field_type##_FIELD(this, kValueOffset + lane * field_size, value);
1454#elif defined(V8_TARGET_BIG_ENDIAN)
1455#define SIMD128_WRITE_LANE(lane_count, field_type, field_size, value) \
1456 WRITE_##field_type##_FIELD( \
1457 this, kValueOffset + (lane_count - lane - 1) * field_size, value);
1458#else
1459#error Unknown byte ordering
1460#endif
1461
1462#define SIMD128_NUMERIC_LANE_FNS(type, lane_type, lane_count, field_type, \
1463 field_size) \
1464 lane_type type::get_lane(int lane) const { \
1465 DCHECK(lane < lane_count && lane >= 0); \
1466 SIMD128_READ_LANE(lane_type, lane_count, field_type, field_size) \
1467 return value; \
1468 } \
1469 \
1470 void type::set_lane(int lane, lane_type value) { \
1471 DCHECK(lane < lane_count && lane >= 0); \
1472 SIMD128_WRITE_LANE(lane_count, field_type, field_size, value) \
1473 }
1474
1475SIMD128_NUMERIC_LANE_FNS(Float32x4, float, 4, FLOAT, kFloatSize)
1476SIMD128_NUMERIC_LANE_FNS(Int32x4, int32_t, 4, INT32, kInt32Size)
1477SIMD128_NUMERIC_LANE_FNS(Uint32x4, uint32_t, 4, UINT32, kInt32Size)
1478SIMD128_NUMERIC_LANE_FNS(Int16x8, int16_t, 8, INT16, kShortSize)
1479SIMD128_NUMERIC_LANE_FNS(Uint16x8, uint16_t, 8, UINT16, kShortSize)
1480SIMD128_NUMERIC_LANE_FNS(Int8x16, int8_t, 16, INT8, kCharSize)
1481SIMD128_NUMERIC_LANE_FNS(Uint8x16, uint8_t, 16, UINT8, kCharSize)
1482#undef SIMD128_NUMERIC_LANE_FNS
1483
1484
1485#define SIMD128_BOOLEAN_LANE_FNS(type, lane_type, lane_count, field_type, \
1486 field_size) \
1487 bool type::get_lane(int lane) const { \
1488 DCHECK(lane < lane_count && lane >= 0); \
1489 SIMD128_READ_LANE(lane_type, lane_count, field_type, field_size) \
1490 DCHECK(value == 0 || value == -1); \
1491 return value != 0; \
1492 } \
1493 \
1494 void type::set_lane(int lane, bool value) { \
1495 DCHECK(lane < lane_count && lane >= 0); \
1496 int32_t int_val = value ? -1 : 0; \
1497 SIMD128_WRITE_LANE(lane_count, field_type, field_size, int_val) \
1498 }
1499
1500SIMD128_BOOLEAN_LANE_FNS(Bool32x4, int32_t, 4, INT32, kInt32Size)
1501SIMD128_BOOLEAN_LANE_FNS(Bool16x8, int16_t, 8, INT16, kShortSize)
1502SIMD128_BOOLEAN_LANE_FNS(Bool8x16, int8_t, 16, INT8, kCharSize)
1503#undef SIMD128_BOOLEAN_LANE_FNS
1504
1505#undef SIMD128_READ_LANE
1506#undef SIMD128_WRITE_LANE
1507
1508
1509ACCESSORS(JSReceiver, properties, FixedArray, kPropertiesOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00001510
1511
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001512Object** FixedArray::GetFirstElementAddress() {
1513 return reinterpret_cast<Object**>(FIELD_ADDR(this, OffsetOfElementAt(0)));
1514}
1515
1516
1517bool FixedArray::ContainsOnlySmisOrHoles() {
1518 Object* the_hole = GetHeap()->the_hole_value();
1519 Object** current = GetFirstElementAddress();
1520 for (int i = 0; i < length(); ++i) {
1521 Object* candidate = *current++;
1522 if (!candidate->IsSmi() && candidate != the_hole) return false;
1523 }
1524 return true;
1525}
1526
1527
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001528FixedArrayBase* JSObject::elements() const {
Steve Blocka7e24c12009-10-30 11:49:00 +00001529 Object* array = READ_FIELD(this, kElementsOffset);
Ben Murdoch69a99ed2011-11-30 16:03:39 +00001530 return static_cast<FixedArrayBase*>(array);
Steve Blocka7e24c12009-10-30 11:49:00 +00001531}
1532
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001533
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001534void AllocationSite::Initialize() {
1535 set_transition_info(Smi::FromInt(0));
1536 SetElementsKind(GetInitialFastElementsKind());
1537 set_nested_site(Smi::FromInt(0));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001538 set_pretenure_data(0);
1539 set_pretenure_create_count(0);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001540 set_dependent_code(DependentCode::cast(GetHeap()->empty_fixed_array()),
1541 SKIP_WRITE_BARRIER);
1542}
1543
1544
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001545bool AllocationSite::IsZombie() { return pretenure_decision() == kZombie; }
1546
1547
1548bool AllocationSite::IsMaybeTenure() {
1549 return pretenure_decision() == kMaybeTenure;
1550}
1551
1552
1553bool AllocationSite::PretenuringDecisionMade() {
1554 return pretenure_decision() != kUndecided;
1555}
1556
1557
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001558void AllocationSite::MarkZombie() {
1559 DCHECK(!IsZombie());
1560 Initialize();
1561 set_pretenure_decision(kZombie);
1562}
1563
1564
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001565ElementsKind AllocationSite::GetElementsKind() {
1566 DCHECK(!SitePointsToLiteral());
1567 int value = Smi::cast(transition_info())->value();
1568 return ElementsKindBits::decode(value);
1569}
1570
1571
1572void AllocationSite::SetElementsKind(ElementsKind kind) {
1573 int value = Smi::cast(transition_info())->value();
1574 set_transition_info(Smi::FromInt(ElementsKindBits::update(value, kind)),
1575 SKIP_WRITE_BARRIER);
1576}
1577
1578
1579bool AllocationSite::CanInlineCall() {
1580 int value = Smi::cast(transition_info())->value();
1581 return DoNotInlineBit::decode(value) == 0;
1582}
1583
1584
1585void AllocationSite::SetDoNotInlineCall() {
1586 int value = Smi::cast(transition_info())->value();
1587 set_transition_info(Smi::FromInt(DoNotInlineBit::update(value, true)),
1588 SKIP_WRITE_BARRIER);
1589}
1590
1591
1592bool AllocationSite::SitePointsToLiteral() {
1593 // If transition_info is a smi, then it represents an ElementsKind
1594 // for a constructed array. Otherwise, it must be a boilerplate
1595 // for an object or array literal.
1596 return transition_info()->IsJSArray() || transition_info()->IsJSObject();
1597}
1598
1599
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001600// Heuristic: We only need to create allocation site info if the boilerplate
1601// elements kind is the initial elements kind.
1602AllocationSiteMode AllocationSite::GetMode(
1603 ElementsKind boilerplate_elements_kind) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001604 if (IsFastSmiElementsKind(boilerplate_elements_kind)) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001605 return TRACK_ALLOCATION_SITE;
1606 }
1607
1608 return DONT_TRACK_ALLOCATION_SITE;
1609}
1610
1611
1612AllocationSiteMode AllocationSite::GetMode(ElementsKind from,
1613 ElementsKind to) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001614 if (IsFastSmiElementsKind(from) &&
1615 IsMoreGeneralElementsKindTransition(from, to)) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001616 return TRACK_ALLOCATION_SITE;
1617 }
1618
1619 return DONT_TRACK_ALLOCATION_SITE;
1620}
1621
1622
1623inline bool AllocationSite::CanTrack(InstanceType type) {
1624 if (FLAG_allocation_site_pretenuring) {
1625 return type == JS_ARRAY_TYPE ||
1626 type == JS_OBJECT_TYPE ||
1627 type < FIRST_NONSTRING_TYPE;
1628 }
1629 return type == JS_ARRAY_TYPE;
1630}
1631
1632
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001633AllocationSite::PretenureDecision AllocationSite::pretenure_decision() {
1634 int value = pretenure_data();
1635 return PretenureDecisionBits::decode(value);
1636}
1637
1638
1639void AllocationSite::set_pretenure_decision(PretenureDecision decision) {
1640 int value = pretenure_data();
1641 set_pretenure_data(PretenureDecisionBits::update(value, decision));
1642}
1643
1644
1645bool AllocationSite::deopt_dependent_code() {
1646 int value = pretenure_data();
1647 return DeoptDependentCodeBit::decode(value);
1648}
1649
1650
1651void AllocationSite::set_deopt_dependent_code(bool deopt) {
1652 int value = pretenure_data();
1653 set_pretenure_data(DeoptDependentCodeBit::update(value, deopt));
1654}
1655
1656
1657int AllocationSite::memento_found_count() {
1658 int value = pretenure_data();
1659 return MementoFoundCountBits::decode(value);
1660}
1661
1662
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001663inline void AllocationSite::set_memento_found_count(int count) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001664 int value = pretenure_data();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001665 // Verify that we can count more mementos than we can possibly find in one
1666 // new space collection.
1667 DCHECK((GetHeap()->MaxSemiSpaceSize() /
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001668 (Heap::kMinObjectSizeInWords * kPointerSize +
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001669 AllocationMemento::kSize)) < MementoFoundCountBits::kMax);
1670 DCHECK(count < MementoFoundCountBits::kMax);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001671 set_pretenure_data(MementoFoundCountBits::update(value, count));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001672}
1673
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001674
1675int AllocationSite::memento_create_count() { return pretenure_create_count(); }
1676
1677
1678void AllocationSite::set_memento_create_count(int count) {
1679 set_pretenure_create_count(count);
1680}
1681
1682
1683bool AllocationSite::IncrementMementoFoundCount(int increment) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001684 if (IsZombie()) return false;
1685
1686 int value = memento_found_count();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001687 set_memento_found_count(value + increment);
1688 return memento_found_count() >= kPretenureMinimumCreated;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001689}
1690
1691
1692inline void AllocationSite::IncrementMementoCreateCount() {
1693 DCHECK(FLAG_allocation_site_pretenuring);
1694 int value = memento_create_count();
1695 set_memento_create_count(value + 1);
1696}
1697
1698
1699inline bool AllocationSite::MakePretenureDecision(
1700 PretenureDecision current_decision,
1701 double ratio,
1702 bool maximum_size_scavenge) {
1703 // Here we just allow state transitions from undecided or maybe tenure
1704 // to don't tenure, maybe tenure, or tenure.
1705 if ((current_decision == kUndecided || current_decision == kMaybeTenure)) {
1706 if (ratio >= kPretenureRatio) {
1707 // We just transition into tenure state when the semi-space was at
1708 // maximum capacity.
1709 if (maximum_size_scavenge) {
1710 set_deopt_dependent_code(true);
1711 set_pretenure_decision(kTenure);
1712 // Currently we just need to deopt when we make a state transition to
1713 // tenure.
1714 return true;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001715 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001716 set_pretenure_decision(kMaybeTenure);
1717 } else {
1718 set_pretenure_decision(kDontTenure);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001719 }
1720 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001721 return false;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001722}
1723
1724
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001725inline bool AllocationSite::DigestPretenuringFeedback(
1726 bool maximum_size_scavenge) {
1727 bool deopt = false;
1728 int create_count = memento_create_count();
1729 int found_count = memento_found_count();
1730 bool minimum_mementos_created = create_count >= kPretenureMinimumCreated;
1731 double ratio =
1732 minimum_mementos_created || FLAG_trace_pretenuring_statistics ?
1733 static_cast<double>(found_count) / create_count : 0.0;
1734 PretenureDecision current_decision = pretenure_decision();
1735
1736 if (minimum_mementos_created) {
1737 deopt = MakePretenureDecision(
1738 current_decision, ratio, maximum_size_scavenge);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001739 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001740
1741 if (FLAG_trace_pretenuring_statistics) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001742 PrintIsolate(GetIsolate(),
1743 "pretenuring: AllocationSite(%p): (created, found, ratio) "
1744 "(%d, %d, %f) %s => %s\n",
1745 this, create_count, found_count, ratio,
1746 PretenureDecisionName(current_decision),
1747 PretenureDecisionName(pretenure_decision()));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001748 }
1749
1750 // Clear feedback calculation fields until the next gc.
1751 set_memento_found_count(0);
1752 set_memento_create_count(0);
1753 return deopt;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001754}
1755
1756
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001757bool AllocationMemento::IsValid() {
1758 return allocation_site()->IsAllocationSite() &&
1759 !AllocationSite::cast(allocation_site())->IsZombie();
1760}
1761
1762
1763AllocationSite* AllocationMemento::GetAllocationSite() {
1764 DCHECK(IsValid());
1765 return AllocationSite::cast(allocation_site());
1766}
1767
Ben Murdoch097c5b22016-05-18 11:27:45 +01001768Address AllocationMemento::GetAllocationSiteUnchecked() {
1769 return reinterpret_cast<Address>(allocation_site());
1770}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001771
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001772void JSObject::EnsureCanContainHeapObjectElements(Handle<JSObject> object) {
1773 JSObject::ValidateElements(object);
1774 ElementsKind elements_kind = object->map()->elements_kind();
1775 if (!IsFastObjectElementsKind(elements_kind)) {
1776 if (IsFastHoleyElementsKind(elements_kind)) {
1777 TransitionElementsKind(object, FAST_HOLEY_ELEMENTS);
1778 } else {
1779 TransitionElementsKind(object, FAST_ELEMENTS);
1780 }
1781 }
1782}
1783
1784
1785void JSObject::EnsureCanContainElements(Handle<JSObject> object,
1786 Object** objects,
1787 uint32_t count,
1788 EnsureElementsMode mode) {
Ben Murdochc5610432016-08-08 18:44:38 +01001789 ElementsKind current_kind = object->GetElementsKind();
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001790 ElementsKind target_kind = current_kind;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001791 {
1792 DisallowHeapAllocation no_allocation;
1793 DCHECK(mode != ALLOW_COPIED_DOUBLE_ELEMENTS);
1794 bool is_holey = IsFastHoleyElementsKind(current_kind);
1795 if (current_kind == FAST_HOLEY_ELEMENTS) return;
1796 Heap* heap = object->GetHeap();
1797 Object* the_hole = heap->the_hole_value();
1798 for (uint32_t i = 0; i < count; ++i) {
1799 Object* current = *objects++;
1800 if (current == the_hole) {
1801 is_holey = true;
1802 target_kind = GetHoleyElementsKind(target_kind);
1803 } else if (!current->IsSmi()) {
1804 if (mode == ALLOW_CONVERTED_DOUBLE_ELEMENTS && current->IsNumber()) {
1805 if (IsFastSmiElementsKind(target_kind)) {
1806 if (is_holey) {
1807 target_kind = FAST_HOLEY_DOUBLE_ELEMENTS;
1808 } else {
1809 target_kind = FAST_DOUBLE_ELEMENTS;
1810 }
1811 }
1812 } else if (is_holey) {
1813 target_kind = FAST_HOLEY_ELEMENTS;
1814 break;
1815 } else {
1816 target_kind = FAST_ELEMENTS;
1817 }
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001818 }
1819 }
1820 }
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001821 if (target_kind != current_kind) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001822 TransitionElementsKind(object, target_kind);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001823 }
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001824}
1825
1826
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001827void JSObject::EnsureCanContainElements(Handle<JSObject> object,
1828 Handle<FixedArrayBase> elements,
1829 uint32_t length,
1830 EnsureElementsMode mode) {
1831 Heap* heap = object->GetHeap();
1832 if (elements->map() != heap->fixed_double_array_map()) {
1833 DCHECK(elements->map() == heap->fixed_array_map() ||
1834 elements->map() == heap->fixed_cow_array_map());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001835 if (mode == ALLOW_COPIED_DOUBLE_ELEMENTS) {
1836 mode = DONT_ALLOW_DOUBLE_ELEMENTS;
1837 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001838 Object** objects =
1839 Handle<FixedArray>::cast(elements)->GetFirstElementAddress();
1840 EnsureCanContainElements(object, objects, length, mode);
1841 return;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001842 }
1843
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001844 DCHECK(mode == ALLOW_COPIED_DOUBLE_ELEMENTS);
1845 if (object->GetElementsKind() == FAST_HOLEY_SMI_ELEMENTS) {
1846 TransitionElementsKind(object, FAST_HOLEY_DOUBLE_ELEMENTS);
1847 } else if (object->GetElementsKind() == FAST_SMI_ELEMENTS) {
1848 Handle<FixedDoubleArray> double_array =
1849 Handle<FixedDoubleArray>::cast(elements);
1850 for (uint32_t i = 0; i < length; ++i) {
1851 if (double_array->is_the_hole(i)) {
1852 TransitionElementsKind(object, FAST_HOLEY_DOUBLE_ELEMENTS);
1853 return;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001854 }
1855 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001856 TransitionElementsKind(object, FAST_DOUBLE_ELEMENTS);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001857 }
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001858}
1859
1860
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001861void JSObject::SetMapAndElements(Handle<JSObject> object,
1862 Handle<Map> new_map,
1863 Handle<FixedArrayBase> value) {
1864 JSObject::MigrateToMap(object, new_map);
1865 DCHECK((object->map()->has_fast_smi_or_object_elements() ||
Ben Murdoch097c5b22016-05-18 11:27:45 +01001866 (*value == object->GetHeap()->empty_fixed_array()) ||
1867 object->map()->has_fast_string_wrapper_elements()) ==
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001868 (value->map() == object->GetHeap()->fixed_array_map() ||
1869 value->map() == object->GetHeap()->fixed_cow_array_map()));
1870 DCHECK((*value == object->GetHeap()->empty_fixed_array()) ||
1871 (object->map()->has_fast_double_elements() ==
1872 value->IsFixedDoubleArray()));
1873 object->set_elements(*value);
1874}
1875
1876
1877void JSObject::set_elements(FixedArrayBase* value, WriteBarrierMode mode) {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001878 WRITE_FIELD(this, kElementsOffset, value);
1879 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kElementsOffset, value, mode);
1880}
1881
Steve Blocka7e24c12009-10-30 11:49:00 +00001882
Steve Blocka7e24c12009-10-30 11:49:00 +00001883void JSObject::initialize_elements() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001884 FixedArrayBase* elements = map()->GetInitialElements();
1885 WRITE_FIELD(this, kElementsOffset, elements);
Steve Blocka7e24c12009-10-30 11:49:00 +00001886}
1887
1888
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001889InterceptorInfo* JSObject::GetIndexedInterceptor() {
Ben Murdochda12d292016-06-02 14:46:10 +01001890 return map()->GetIndexedInterceptor();
1891}
1892
1893InterceptorInfo* JSObject::GetNamedInterceptor() {
1894 return map()->GetNamedInterceptor();
1895}
1896
1897InterceptorInfo* Map::GetNamedInterceptor() {
1898 DCHECK(has_named_interceptor());
1899 JSFunction* constructor = JSFunction::cast(GetConstructor());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001900 DCHECK(constructor->shared()->IsApiFunction());
Ben Murdochda12d292016-06-02 14:46:10 +01001901 return InterceptorInfo::cast(
1902 constructor->shared()->get_api_func_data()->named_property_handler());
1903}
1904
1905InterceptorInfo* Map::GetIndexedInterceptor() {
1906 DCHECK(has_indexed_interceptor());
1907 JSFunction* constructor = JSFunction::cast(GetConstructor());
1908 DCHECK(constructor->shared()->IsApiFunction());
1909 return InterceptorInfo::cast(
1910 constructor->shared()->get_api_func_data()->indexed_property_handler());
Steve Block8defd9f2010-07-08 12:39:36 +01001911}
1912
Ben Murdochc5610432016-08-08 18:44:38 +01001913double Oddball::to_number_raw() const {
1914 return READ_DOUBLE_FIELD(this, kToNumberRawOffset);
1915}
1916
1917void Oddball::set_to_number_raw(double value) {
1918 WRITE_DOUBLE_FIELD(this, kToNumberRawOffset, value);
1919}
Steve Block8defd9f2010-07-08 12:39:36 +01001920
Steve Blocka7e24c12009-10-30 11:49:00 +00001921ACCESSORS(Oddball, to_string, String, kToStringOffset)
1922ACCESSORS(Oddball, to_number, Object, kToNumberOffset)
Ben Murdochda12d292016-06-02 14:46:10 +01001923ACCESSORS(Oddball, to_boolean, Oddball, kToBooleanOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001924ACCESSORS(Oddball, type_of, String, kTypeOfOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00001925
1926
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001927byte Oddball::kind() const {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001928 return Smi::cast(READ_FIELD(this, kKindOffset))->value();
Steve Block44f0eee2011-05-26 01:26:41 +01001929}
1930
1931
1932void Oddball::set_kind(byte value) {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01001933 WRITE_FIELD(this, kKindOffset, Smi::FromInt(value));
Steve Block44f0eee2011-05-26 01:26:41 +01001934}
1935
1936
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001937// static
1938Handle<Object> Oddball::ToNumber(Handle<Oddball> input) {
1939 return handle(input->to_number(), input->GetIsolate());
Steve Blocka7e24c12009-10-30 11:49:00 +00001940}
1941
1942
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001943ACCESSORS(Cell, value, Object, kValueOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001944ACCESSORS(PropertyCell, dependent_code, DependentCode, kDependentCodeOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001945ACCESSORS(PropertyCell, property_details_raw, Object, kDetailsOffset)
1946ACCESSORS(PropertyCell, value, Object, kValueOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001947
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001948
1949PropertyDetails PropertyCell::property_details() {
1950 return PropertyDetails(Smi::cast(property_details_raw()));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001951}
1952
1953
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001954void PropertyCell::set_property_details(PropertyDetails details) {
1955 set_property_details_raw(details.AsSmi());
Ben Murdochb8a8cc12014-11-26 15:28:44 +00001956}
1957
Steve Blocka7e24c12009-10-30 11:49:00 +00001958
Emily Bernierd0a1eb72015-03-24 16:35:39 -04001959Object* WeakCell::value() const { return READ_FIELD(this, kValueOffset); }
1960
1961
1962void WeakCell::clear() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001963 // Either the garbage collector is clearing the cell or we are simply
1964 // initializing the root empty weak cell.
1965 DCHECK(GetHeap()->gc_state() == Heap::MARK_COMPACT ||
1966 this == GetHeap()->empty_weak_cell());
Emily Bernierd0a1eb72015-03-24 16:35:39 -04001967 WRITE_FIELD(this, kValueOffset, Smi::FromInt(0));
1968}
1969
1970
1971void WeakCell::initialize(HeapObject* val) {
1972 WRITE_FIELD(this, kValueOffset, val);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001973 // We just have to execute the generational barrier here because we never
1974 // mark through a weak cell and collect evacuation candidates when we process
1975 // all weak cells.
Ben Murdochda12d292016-06-02 14:46:10 +01001976 WriteBarrierMode mode =
1977 Page::FromAddress(this->address())->IsFlagSet(Page::BLACK_PAGE)
1978 ? UPDATE_WRITE_BARRIER
1979 : UPDATE_WEAK_WRITE_BARRIER;
1980 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kValueOffset, val, mode);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04001981}
1982
1983
1984bool WeakCell::cleared() const { return value() == Smi::FromInt(0); }
1985
1986
1987Object* WeakCell::next() const { return READ_FIELD(this, kNextOffset); }
1988
1989
1990void WeakCell::set_next(Object* val, WriteBarrierMode mode) {
1991 WRITE_FIELD(this, kNextOffset, val);
1992 if (mode == UPDATE_WRITE_BARRIER) {
1993 WRITE_BARRIER(GetHeap(), this, kNextOffset, val);
1994 }
1995}
1996
1997
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00001998void WeakCell::clear_next(Object* the_hole_value) {
1999 DCHECK_EQ(GetHeap()->the_hole_value(), the_hole_value);
2000 set_next(the_hole_value, SKIP_WRITE_BARRIER);
2001}
2002
2003
2004bool WeakCell::next_cleared() { return next()->IsTheHole(); }
2005
2006
2007int JSObject::GetHeaderSize() { return GetHeaderSize(map()->instance_type()); }
2008
2009
2010int JSObject::GetHeaderSize(InstanceType type) {
Steve Blocka7e24c12009-10-30 11:49:00 +00002011 // Check for the most common kind of JavaScript object before
2012 // falling into the generic switch. This speeds up the internal
2013 // field operations considerably on average.
2014 if (type == JS_OBJECT_TYPE) return JSObject::kHeaderSize;
2015 switch (type) {
Ben Murdochc5610432016-08-08 18:44:38 +01002016 case JS_API_OBJECT_TYPE:
Ben Murdochda12d292016-06-02 14:46:10 +01002017 case JS_SPECIAL_API_OBJECT_TYPE:
2018 return JSObject::kHeaderSize;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002019 case JS_GENERATOR_OBJECT_TYPE:
2020 return JSGeneratorObject::kSize;
2021 case JS_MODULE_TYPE:
2022 return JSModule::kSize;
Steve Blocka7e24c12009-10-30 11:49:00 +00002023 case JS_GLOBAL_PROXY_TYPE:
2024 return JSGlobalProxy::kSize;
2025 case JS_GLOBAL_OBJECT_TYPE:
2026 return JSGlobalObject::kSize;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002027 case JS_BOUND_FUNCTION_TYPE:
2028 return JSBoundFunction::kSize;
Steve Blocka7e24c12009-10-30 11:49:00 +00002029 case JS_FUNCTION_TYPE:
2030 return JSFunction::kSize;
2031 case JS_VALUE_TYPE:
2032 return JSValue::kSize;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002033 case JS_DATE_TYPE:
2034 return JSDate::kSize;
Steve Blocka7e24c12009-10-30 11:49:00 +00002035 case JS_ARRAY_TYPE:
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002036 return JSArray::kSize;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002037 case JS_ARRAY_BUFFER_TYPE:
2038 return JSArrayBuffer::kSize;
2039 case JS_TYPED_ARRAY_TYPE:
2040 return JSTypedArray::kSize;
2041 case JS_DATA_VIEW_TYPE:
2042 return JSDataView::kSize;
2043 case JS_SET_TYPE:
2044 return JSSet::kSize;
2045 case JS_MAP_TYPE:
2046 return JSMap::kSize;
2047 case JS_SET_ITERATOR_TYPE:
2048 return JSSetIterator::kSize;
2049 case JS_MAP_ITERATOR_TYPE:
2050 return JSMapIterator::kSize;
Ben Murdoch69a99ed2011-11-30 16:03:39 +00002051 case JS_WEAK_MAP_TYPE:
2052 return JSWeakMap::kSize;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002053 case JS_WEAK_SET_TYPE:
2054 return JSWeakSet::kSize;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002055 case JS_PROMISE_TYPE:
2056 return JSObject::kHeaderSize;
Steve Blocka7e24c12009-10-30 11:49:00 +00002057 case JS_REGEXP_TYPE:
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002058 return JSRegExp::kSize;
Steve Blocka7e24c12009-10-30 11:49:00 +00002059 case JS_CONTEXT_EXTENSION_OBJECT_TYPE:
2060 return JSObject::kHeaderSize;
Steve Block1e0659c2011-05-24 12:43:12 +01002061 case JS_MESSAGE_OBJECT_TYPE:
2062 return JSMessageObject::kSize;
Steve Blocka7e24c12009-10-30 11:49:00 +00002063 default:
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002064 UNREACHABLE();
Steve Blocka7e24c12009-10-30 11:49:00 +00002065 return 0;
2066 }
2067}
2068
2069
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002070int JSObject::GetInternalFieldCount(Map* map) {
2071 int instance_size = map->instance_size();
2072 if (instance_size == kVariableSizeSentinel) return 0;
2073 InstanceType instance_type = map->instance_type();
2074 return ((instance_size - GetHeaderSize(instance_type)) >> kPointerSizeLog2) -
2075 map->GetInObjectProperties();
Steve Blocka7e24c12009-10-30 11:49:00 +00002076}
2077
2078
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002079int JSObject::GetInternalFieldCount() { return GetInternalFieldCount(map()); }
2080
2081
Steve Block44f0eee2011-05-26 01:26:41 +01002082int JSObject::GetInternalFieldOffset(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002083 DCHECK(index < GetInternalFieldCount() && index >= 0);
Steve Block44f0eee2011-05-26 01:26:41 +01002084 return GetHeaderSize() + (kPointerSize * index);
2085}
2086
2087
Steve Blocka7e24c12009-10-30 11:49:00 +00002088Object* JSObject::GetInternalField(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002089 DCHECK(index < GetInternalFieldCount() && index >= 0);
Steve Blocka7e24c12009-10-30 11:49:00 +00002090 // Internal objects do follow immediately after the header, whereas in-object
2091 // properties are at the end of the object. Therefore there is no need
2092 // to adjust the index here.
2093 return READ_FIELD(this, GetHeaderSize() + (kPointerSize * index));
2094}
2095
2096
2097void JSObject::SetInternalField(int index, Object* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002098 DCHECK(index < GetInternalFieldCount() && index >= 0);
Steve Blocka7e24c12009-10-30 11:49:00 +00002099 // Internal objects do follow immediately after the header, whereas in-object
2100 // properties are at the end of the object. Therefore there is no need
2101 // to adjust the index here.
2102 int offset = GetHeaderSize() + (kPointerSize * index);
2103 WRITE_FIELD(this, offset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002104 WRITE_BARRIER(GetHeap(), this, offset, value);
2105}
2106
2107
2108void JSObject::SetInternalField(int index, Smi* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002109 DCHECK(index < GetInternalFieldCount() && index >= 0);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002110 // Internal objects do follow immediately after the header, whereas in-object
2111 // properties are at the end of the object. Therefore there is no need
2112 // to adjust the index here.
2113 int offset = GetHeaderSize() + (kPointerSize * index);
2114 WRITE_FIELD(this, offset, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00002115}
2116
2117
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002118bool JSObject::IsUnboxedDoubleField(FieldIndex index) {
2119 if (!FLAG_unbox_double_fields) return false;
2120 return map()->IsUnboxedDoubleField(index);
2121}
2122
2123
2124bool Map::IsUnboxedDoubleField(FieldIndex index) {
2125 if (!FLAG_unbox_double_fields) return false;
2126 if (index.is_hidden_field() || !index.is_inobject()) return false;
2127 return !layout_descriptor()->IsTagged(index.property_index());
2128}
2129
2130
Steve Blocka7e24c12009-10-30 11:49:00 +00002131// Access fast-case object properties at index. The use of these routines
2132// is needed to correctly distinguish between properties stored in-object and
2133// properties stored in the properties array.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002134Object* JSObject::RawFastPropertyAt(FieldIndex index) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002135 DCHECK(!IsUnboxedDoubleField(index));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002136 if (index.is_inobject()) {
2137 return READ_FIELD(this, index.offset());
Steve Blocka7e24c12009-10-30 11:49:00 +00002138 } else {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002139 return properties()->get(index.outobject_array_index());
Steve Blocka7e24c12009-10-30 11:49:00 +00002140 }
2141}
2142
2143
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002144double JSObject::RawFastDoublePropertyAt(FieldIndex index) {
2145 DCHECK(IsUnboxedDoubleField(index));
2146 return READ_DOUBLE_FIELD(this, index.offset());
2147}
2148
2149
2150void JSObject::RawFastPropertyAtPut(FieldIndex index, Object* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002151 if (index.is_inobject()) {
2152 int offset = index.offset();
Steve Blocka7e24c12009-10-30 11:49:00 +00002153 WRITE_FIELD(this, offset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002154 WRITE_BARRIER(GetHeap(), this, offset, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00002155 } else {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002156 properties()->set(index.outobject_array_index(), value);
Steve Blocka7e24c12009-10-30 11:49:00 +00002157 }
Steve Blocka7e24c12009-10-30 11:49:00 +00002158}
2159
2160
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002161void JSObject::RawFastDoublePropertyAtPut(FieldIndex index, double value) {
2162 WRITE_DOUBLE_FIELD(this, index.offset(), value);
2163}
2164
2165
2166void JSObject::FastPropertyAtPut(FieldIndex index, Object* value) {
2167 if (IsUnboxedDoubleField(index)) {
2168 DCHECK(value->IsMutableHeapNumber());
2169 RawFastDoublePropertyAtPut(index, HeapNumber::cast(value)->value());
2170 } else {
2171 RawFastPropertyAtPut(index, value);
2172 }
2173}
2174
Ben Murdoch097c5b22016-05-18 11:27:45 +01002175void JSObject::WriteToField(int descriptor, PropertyDetails details,
2176 Object* value) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002177 DCHECK(details.type() == DATA);
Ben Murdoch097c5b22016-05-18 11:27:45 +01002178 DisallowHeapAllocation no_gc;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002179 FieldIndex index = FieldIndex::ForDescriptor(map(), descriptor);
2180 if (details.representation().IsDouble()) {
2181 // Nothing more to be done.
2182 if (value->IsUninitialized()) return;
2183 if (IsUnboxedDoubleField(index)) {
2184 RawFastDoublePropertyAtPut(index, value->Number());
2185 } else {
2186 HeapNumber* box = HeapNumber::cast(RawFastPropertyAt(index));
2187 DCHECK(box->IsMutableHeapNumber());
2188 box->set_value(value->Number());
2189 }
2190 } else {
2191 RawFastPropertyAtPut(index, value);
2192 }
2193}
2194
Ben Murdoch097c5b22016-05-18 11:27:45 +01002195void JSObject::WriteToField(int descriptor, Object* value) {
2196 DescriptorArray* desc = map()->instance_descriptors();
2197 PropertyDetails details = desc->GetDetails(descriptor);
2198 WriteToField(descriptor, details, value);
2199}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002200
Steve Block44f0eee2011-05-26 01:26:41 +01002201int JSObject::GetInObjectPropertyOffset(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002202 return map()->GetInObjectPropertyOffset(index);
Steve Block44f0eee2011-05-26 01:26:41 +01002203}
2204
2205
Steve Blocka7e24c12009-10-30 11:49:00 +00002206Object* JSObject::InObjectPropertyAt(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002207 int offset = GetInObjectPropertyOffset(index);
Steve Blocka7e24c12009-10-30 11:49:00 +00002208 return READ_FIELD(this, offset);
2209}
2210
2211
2212Object* JSObject::InObjectPropertyAtPut(int index,
2213 Object* value,
2214 WriteBarrierMode mode) {
2215 // Adjust for the number of properties stored in the object.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002216 int offset = GetInObjectPropertyOffset(index);
Steve Blocka7e24c12009-10-30 11:49:00 +00002217 WRITE_FIELD(this, offset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002218 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, offset, value, mode);
Steve Blocka7e24c12009-10-30 11:49:00 +00002219 return value;
2220}
2221
2222
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002223void JSObject::InitializeBody(Map* map, int start_offset,
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002224 Object* pre_allocated_value,
2225 Object* filler_value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002226 DCHECK(!filler_value->IsHeapObject() ||
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002227 !GetHeap()->InNewSpace(filler_value));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002228 DCHECK(!pre_allocated_value->IsHeapObject() ||
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002229 !GetHeap()->InNewSpace(pre_allocated_value));
2230 int size = map->instance_size();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002231 int offset = start_offset;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002232 if (filler_value != pre_allocated_value) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002233 int end_of_pre_allocated_offset =
2234 size - (map->unused_property_fields() * kPointerSize);
2235 DCHECK_LE(kHeaderSize, end_of_pre_allocated_offset);
2236 while (offset < end_of_pre_allocated_offset) {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002237 WRITE_FIELD(this, offset, pre_allocated_value);
2238 offset += kPointerSize;
2239 }
2240 }
2241 while (offset < size) {
2242 WRITE_FIELD(this, offset, filler_value);
2243 offset += kPointerSize;
Steve Blocka7e24c12009-10-30 11:49:00 +00002244 }
2245}
2246
2247
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002248bool Map::TooManyFastProperties(StoreFromKeyed store_mode) {
2249 if (unused_property_fields() != 0) return false;
2250 if (is_prototype_map()) return false;
2251 int minimum = store_mode == CERTAINLY_NOT_STORE_FROM_KEYED ? 128 : 12;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002252 int limit = Max(minimum, GetInObjectProperties());
2253 int external = NumberOfFields() - GetInObjectProperties();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002254 return external > limit;
Steve Block8defd9f2010-07-08 12:39:36 +01002255}
2256
2257
Steve Blocka7e24c12009-10-30 11:49:00 +00002258void Struct::InitializeBody(int object_size) {
Steve Block44f0eee2011-05-26 01:26:41 +01002259 Object* value = GetHeap()->undefined_value();
Steve Blocka7e24c12009-10-30 11:49:00 +00002260 for (int offset = kHeaderSize; offset < object_size; offset += kPointerSize) {
2261 WRITE_FIELD(this, offset, value);
2262 }
2263}
2264
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002265bool Object::ToArrayLength(uint32_t* index) { return Object::ToUint32(index); }
2266
2267
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01002268bool Object::ToArrayIndex(uint32_t* index) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002269 return Object::ToUint32(index) && *index != kMaxUInt32;
Steve Blocka7e24c12009-10-30 11:49:00 +00002270}
2271
2272
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002273void Object::VerifyApiCallResultType() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002274#if DEBUG
2275 if (!(IsSmi() || IsString() || IsSymbol() || IsJSReceiver() ||
2276 IsHeapNumber() || IsSimd128Value() || IsUndefined() || IsTrue() ||
2277 IsFalse() || IsNull())) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002278 FATAL("API call returned invalid object");
2279 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002280#endif // DEBUG
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002281}
2282
2283
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002284Object* FixedArray::get(int index) const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002285 SLOW_DCHECK(index >= 0 && index < this->length());
Steve Blocka7e24c12009-10-30 11:49:00 +00002286 return READ_FIELD(this, kHeaderSize + index * kPointerSize);
2287}
2288
Ben Murdoch097c5b22016-05-18 11:27:45 +01002289Handle<Object> FixedArray::get(FixedArray* array, int index, Isolate* isolate) {
2290 return handle(array->get(index), isolate);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002291}
2292
2293
2294bool FixedArray::is_the_hole(int index) {
2295 return get(index) == GetHeap()->the_hole_value();
2296}
2297
2298
Steve Blocka7e24c12009-10-30 11:49:00 +00002299void FixedArray::set(int index, Smi* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002300 DCHECK(map() != GetHeap()->fixed_cow_array_map());
2301 DCHECK(index >= 0 && index < this->length());
2302 DCHECK(reinterpret_cast<Object*>(value)->IsSmi());
Steve Blocka7e24c12009-10-30 11:49:00 +00002303 int offset = kHeaderSize + index * kPointerSize;
2304 WRITE_FIELD(this, offset, value);
2305}
2306
2307
2308void FixedArray::set(int index, Object* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002309 DCHECK_NE(GetHeap()->fixed_cow_array_map(), map());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002310 DCHECK(IsFixedArray());
Ben Murdoch097c5b22016-05-18 11:27:45 +01002311 DCHECK_GE(index, 0);
2312 DCHECK_LT(index, this->length());
Steve Blocka7e24c12009-10-30 11:49:00 +00002313 int offset = kHeaderSize + index * kPointerSize;
2314 WRITE_FIELD(this, offset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002315 WRITE_BARRIER(GetHeap(), this, offset, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00002316}
2317
2318
Ben Murdoch69a99ed2011-11-30 16:03:39 +00002319double FixedDoubleArray::get_scalar(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002320 DCHECK(map() != GetHeap()->fixed_cow_array_map() &&
2321 map() != GetHeap()->fixed_array_map());
2322 DCHECK(index >= 0 && index < this->length());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002323 DCHECK(!is_the_hole(index));
2324 return READ_DOUBLE_FIELD(this, kHeaderSize + index * kDoubleSize);
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002325}
2326
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002327
2328uint64_t FixedDoubleArray::get_representation(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002329 DCHECK(map() != GetHeap()->fixed_cow_array_map() &&
2330 map() != GetHeap()->fixed_array_map());
2331 DCHECK(index >= 0 && index < this->length());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002332 int offset = kHeaderSize + index * kDoubleSize;
2333 return READ_UINT64_FIELD(this, offset);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002334}
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002335
Ben Murdoch097c5b22016-05-18 11:27:45 +01002336Handle<Object> FixedDoubleArray::get(FixedDoubleArray* array, int index,
2337 Isolate* isolate) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002338 if (array->is_the_hole(index)) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002339 return isolate->factory()->the_hole_value();
Ben Murdoch69a99ed2011-11-30 16:03:39 +00002340 } else {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002341 return isolate->factory()->NewNumber(array->get_scalar(index));
Ben Murdoch69a99ed2011-11-30 16:03:39 +00002342 }
2343}
2344
2345
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002346void FixedDoubleArray::set(int index, double value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002347 DCHECK(map() != GetHeap()->fixed_cow_array_map() &&
2348 map() != GetHeap()->fixed_array_map());
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002349 int offset = kHeaderSize + index * kDoubleSize;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002350 if (std::isnan(value)) {
2351 WRITE_DOUBLE_FIELD(this, offset, std::numeric_limits<double>::quiet_NaN());
2352 } else {
2353 WRITE_DOUBLE_FIELD(this, offset, value);
2354 }
2355 DCHECK(!is_the_hole(index));
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002356}
2357
2358
2359void FixedDoubleArray::set_the_hole(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002360 DCHECK(map() != GetHeap()->fixed_cow_array_map() &&
2361 map() != GetHeap()->fixed_array_map());
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002362 int offset = kHeaderSize + index * kDoubleSize;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002363 WRITE_UINT64_FIELD(this, offset, kHoleNanInt64);
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002364}
2365
2366
2367bool FixedDoubleArray::is_the_hole(int index) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002368 return get_representation(index) == kHoleNanInt64;
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00002369}
2370
2371
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002372double* FixedDoubleArray::data_start() {
2373 return reinterpret_cast<double*>(FIELD_ADDR(this, kHeaderSize));
2374}
2375
2376
2377void FixedDoubleArray::FillWithHoles(int from, int to) {
2378 for (int i = from; i < to; i++) {
2379 set_the_hole(i);
2380 }
2381}
2382
2383
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002384Object* WeakFixedArray::Get(int index) const {
2385 Object* raw = FixedArray::cast(this)->get(index + kFirstIndex);
2386 if (raw->IsSmi()) return raw;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002387 DCHECK(raw->IsWeakCell());
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002388 return WeakCell::cast(raw)->value();
2389}
2390
2391
2392bool WeakFixedArray::IsEmptySlot(int index) const {
2393 DCHECK(index < Length());
2394 return Get(index)->IsSmi();
2395}
2396
2397
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002398void WeakFixedArray::Clear(int index) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002399 FixedArray::cast(this)->set(index + kFirstIndex, Smi::FromInt(0));
2400}
2401
2402
2403int WeakFixedArray::Length() const {
2404 return FixedArray::cast(this)->length() - kFirstIndex;
2405}
2406
2407
2408int WeakFixedArray::last_used_index() const {
2409 return Smi::cast(FixedArray::cast(this)->get(kLastUsedIndexIndex))->value();
2410}
2411
2412
2413void WeakFixedArray::set_last_used_index(int index) {
2414 FixedArray::cast(this)->set(kLastUsedIndexIndex, Smi::FromInt(index));
2415}
2416
2417
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002418template <class T>
2419T* WeakFixedArray::Iterator::Next() {
2420 if (list_ != NULL) {
2421 // Assert that list did not change during iteration.
2422 DCHECK_EQ(last_used_index_, list_->last_used_index());
2423 while (index_ < list_->Length()) {
2424 Object* item = list_->Get(index_++);
2425 if (item != Empty()) return T::cast(item);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002426 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002427 list_ = NULL;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002428 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002429 return NULL;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002430}
2431
2432
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002433int ArrayList::Length() {
2434 if (FixedArray::cast(this)->length() == 0) return 0;
2435 return Smi::cast(FixedArray::cast(this)->get(kLengthIndex))->value();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002436}
2437
2438
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002439void ArrayList::SetLength(int length) {
2440 return FixedArray::cast(this)->set(kLengthIndex, Smi::FromInt(length));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002441}
2442
2443
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002444Object* ArrayList::Get(int index) {
2445 return FixedArray::cast(this)->get(kFirstIndex + index);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002446}
2447
2448
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002449Object** ArrayList::Slot(int index) {
2450 return data_start() + kFirstIndex + index;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002451}
2452
2453
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002454void ArrayList::Set(int index, Object* obj) {
2455 FixedArray::cast(this)->set(kFirstIndex + index, obj);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002456}
2457
2458
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002459void ArrayList::Clear(int index, Object* undefined) {
2460 DCHECK(undefined->IsUndefined());
2461 FixedArray::cast(this)
2462 ->set(kFirstIndex + index, undefined, SKIP_WRITE_BARRIER);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002463}
2464
2465
2466WriteBarrierMode HeapObject::GetWriteBarrierMode(
2467 const DisallowHeapAllocation& promise) {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002468 Heap* heap = GetHeap();
2469 if (heap->incremental_marking()->IsMarking()) return UPDATE_WRITE_BARRIER;
2470 if (heap->InNewSpace(this)) return SKIP_WRITE_BARRIER;
Steve Blocka7e24c12009-10-30 11:49:00 +00002471 return UPDATE_WRITE_BARRIER;
2472}
2473
2474
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002475AllocationAlignment HeapObject::RequiredAlignment() {
2476#ifdef V8_HOST_ARCH_32_BIT
2477 if ((IsFixedFloat64Array() || IsFixedDoubleArray()) &&
2478 FixedArrayBase::cast(this)->length() != 0) {
2479 return kDoubleAligned;
2480 }
2481 if (IsHeapNumber()) return kDoubleUnaligned;
2482 if (IsSimd128Value()) return kSimd128Unaligned;
2483#endif // V8_HOST_ARCH_32_BIT
2484 return kWordAligned;
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002485}
2486
2487
Steve Blocka7e24c12009-10-30 11:49:00 +00002488void FixedArray::set(int index,
2489 Object* value,
2490 WriteBarrierMode mode) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002491 DCHECK(map() != GetHeap()->fixed_cow_array_map());
2492 DCHECK(index >= 0 && index < this->length());
Steve Blocka7e24c12009-10-30 11:49:00 +00002493 int offset = kHeaderSize + index * kPointerSize;
2494 WRITE_FIELD(this, offset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002495 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, offset, value, mode);
Steve Blocka7e24c12009-10-30 11:49:00 +00002496}
2497
2498
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002499void FixedArray::NoWriteBarrierSet(FixedArray* array,
2500 int index,
2501 Object* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002502 DCHECK(array->map() != array->GetHeap()->fixed_cow_array_map());
2503 DCHECK(index >= 0 && index < array->length());
2504 DCHECK(!array->GetHeap()->InNewSpace(value));
Steve Blocka7e24c12009-10-30 11:49:00 +00002505 WRITE_FIELD(array, kHeaderSize + index * kPointerSize, value);
2506}
2507
2508
2509void FixedArray::set_undefined(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002510 DCHECK(map() != GetHeap()->fixed_cow_array_map());
2511 DCHECK(index >= 0 && index < this->length());
2512 DCHECK(!GetHeap()->InNewSpace(GetHeap()->undefined_value()));
2513 WRITE_FIELD(this,
2514 kHeaderSize + index * kPointerSize,
2515 GetHeap()->undefined_value());
Steve Blocka7e24c12009-10-30 11:49:00 +00002516}
2517
2518
2519void FixedArray::set_null(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002520 DCHECK(index >= 0 && index < this->length());
2521 DCHECK(!GetHeap()->InNewSpace(GetHeap()->null_value()));
2522 WRITE_FIELD(this,
2523 kHeaderSize + index * kPointerSize,
2524 GetHeap()->null_value());
Steve Blocka7e24c12009-10-30 11:49:00 +00002525}
2526
2527
2528void FixedArray::set_the_hole(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002529 DCHECK(map() != GetHeap()->fixed_cow_array_map());
2530 DCHECK(index >= 0 && index < this->length());
2531 DCHECK(!GetHeap()->InNewSpace(GetHeap()->the_hole_value()));
Steve Block44f0eee2011-05-26 01:26:41 +01002532 WRITE_FIELD(this,
2533 kHeaderSize + index * kPointerSize,
2534 GetHeap()->the_hole_value());
Steve Blocka7e24c12009-10-30 11:49:00 +00002535}
2536
2537
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002538void FixedArray::FillWithHoles(int from, int to) {
2539 for (int i = from; i < to; i++) {
2540 set_the_hole(i);
2541 }
Iain Merrick75681382010-08-19 15:07:18 +01002542}
2543
2544
Steve Block6ded16b2010-05-10 14:33:55 +01002545Object** FixedArray::data_start() {
2546 return HeapObject::RawField(this, kHeaderSize);
2547}
2548
2549
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002550Object** FixedArray::RawFieldOfElementAt(int index) {
2551 return HeapObject::RawField(this, OffsetOfElementAt(index));
2552}
2553
2554
Steve Blocka7e24c12009-10-30 11:49:00 +00002555bool DescriptorArray::IsEmpty() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002556 DCHECK(length() >= kFirstIndex ||
2557 this == GetHeap()->empty_descriptor_array());
2558 return length() < kFirstIndex;
Ben Murdoch257744e2011-11-30 15:57:28 +00002559}
2560
2561
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002562int DescriptorArray::number_of_descriptors() {
2563 DCHECK(length() >= kFirstIndex || IsEmpty());
2564 int len = length();
2565 return len == 0 ? 0 : Smi::cast(get(kDescriptorLengthIndex))->value();
2566}
2567
2568
2569int DescriptorArray::number_of_descriptors_storage() {
2570 int len = length();
2571 return len == 0 ? 0 : (len - kFirstIndex) / kDescriptorSize;
2572}
2573
2574
2575int DescriptorArray::NumberOfSlackDescriptors() {
2576 return number_of_descriptors_storage() - number_of_descriptors();
2577}
2578
2579
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002580void DescriptorArray::SetNumberOfDescriptors(int number_of_descriptors) {
2581 WRITE_FIELD(
2582 this, kDescriptorLengthOffset, Smi::FromInt(number_of_descriptors));
Steve Blocka7e24c12009-10-30 11:49:00 +00002583}
2584
2585
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002586inline int DescriptorArray::number_of_entries() {
2587 return number_of_descriptors();
2588}
2589
2590
2591bool DescriptorArray::HasEnumCache() {
2592 return !IsEmpty() && !get(kEnumCacheIndex)->IsSmi();
2593}
2594
2595
2596void DescriptorArray::CopyEnumCacheFrom(DescriptorArray* array) {
2597 set(kEnumCacheIndex, array->get(kEnumCacheIndex));
2598}
2599
2600
2601FixedArray* DescriptorArray::GetEnumCache() {
2602 DCHECK(HasEnumCache());
2603 FixedArray* bridge = FixedArray::cast(get(kEnumCacheIndex));
2604 return FixedArray::cast(bridge->get(kEnumCacheBridgeCacheIndex));
2605}
2606
2607
2608bool DescriptorArray::HasEnumIndicesCache() {
2609 if (IsEmpty()) return false;
2610 Object* object = get(kEnumCacheIndex);
2611 if (object->IsSmi()) return false;
2612 FixedArray* bridge = FixedArray::cast(object);
2613 return !bridge->get(kEnumCacheBridgeIndicesCacheIndex)->IsSmi();
2614}
2615
2616
2617FixedArray* DescriptorArray::GetEnumIndicesCache() {
2618 DCHECK(HasEnumIndicesCache());
2619 FixedArray* bridge = FixedArray::cast(get(kEnumCacheIndex));
2620 return FixedArray::cast(bridge->get(kEnumCacheBridgeIndicesCacheIndex));
2621}
2622
2623
2624Object** DescriptorArray::GetEnumCacheSlot() {
2625 DCHECK(HasEnumCache());
2626 return HeapObject::RawField(reinterpret_cast<HeapObject*>(this),
2627 kEnumCacheOffset);
2628}
2629
Ben Murdoch097c5b22016-05-18 11:27:45 +01002630// Perform a binary search in a fixed array.
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002631template <SearchMode search_mode, typename T>
Ben Murdoch097c5b22016-05-18 11:27:45 +01002632int BinarySearch(T* array, Name* name, int valid_entries,
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002633 int* out_insertion_index) {
2634 DCHECK(search_mode == ALL_ENTRIES || out_insertion_index == NULL);
Ben Murdoch097c5b22016-05-18 11:27:45 +01002635 int low = 0;
2636 int high = array->number_of_entries() - 1;
2637 uint32_t hash = name->hash_field();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002638 int limit = high;
2639
2640 DCHECK(low <= high);
2641
2642 while (low != high) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002643 int mid = low + (high - low) / 2;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002644 Name* mid_name = array->GetSortedKey(mid);
Ben Murdoch097c5b22016-05-18 11:27:45 +01002645 uint32_t mid_hash = mid_name->hash_field();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002646
2647 if (mid_hash >= hash) {
2648 high = mid;
2649 } else {
2650 low = mid + 1;
2651 }
2652 }
2653
2654 for (; low <= limit; ++low) {
2655 int sort_index = array->GetSortedKeyIndex(low);
2656 Name* entry = array->GetKey(sort_index);
Ben Murdoch097c5b22016-05-18 11:27:45 +01002657 uint32_t current_hash = entry->hash_field();
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002658 if (current_hash != hash) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002659 if (search_mode == ALL_ENTRIES && out_insertion_index != nullptr) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002660 *out_insertion_index = sort_index + (current_hash > hash ? 0 : 1);
2661 }
2662 return T::kNotFound;
2663 }
Ben Murdoch097c5b22016-05-18 11:27:45 +01002664 if (entry == name) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002665 if (search_mode == ALL_ENTRIES || sort_index < valid_entries) {
2666 return sort_index;
2667 }
2668 return T::kNotFound;
2669 }
2670 }
2671
Ben Murdoch097c5b22016-05-18 11:27:45 +01002672 if (search_mode == ALL_ENTRIES && out_insertion_index != nullptr) {
2673 *out_insertion_index = limit + 1;
2674 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002675 return T::kNotFound;
Steve Blocka7e24c12009-10-30 11:49:00 +00002676}
2677
2678
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002679// Perform a linear search in this fixed array. len is the number of entry
2680// indices that are valid.
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002681template <SearchMode search_mode, typename T>
Ben Murdoch097c5b22016-05-18 11:27:45 +01002682int LinearSearch(T* array, Name* name, int valid_entries,
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002683 int* out_insertion_index) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002684 if (search_mode == ALL_ENTRIES && out_insertion_index != nullptr) {
2685 uint32_t hash = name->hash_field();
2686 int len = array->number_of_entries();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002687 for (int number = 0; number < len; number++) {
2688 int sorted_index = array->GetSortedKeyIndex(number);
2689 Name* entry = array->GetKey(sorted_index);
Ben Murdoch097c5b22016-05-18 11:27:45 +01002690 uint32_t current_hash = entry->hash_field();
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002691 if (current_hash > hash) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002692 *out_insertion_index = sorted_index;
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002693 return T::kNotFound;
2694 }
Ben Murdoch097c5b22016-05-18 11:27:45 +01002695 if (entry == name) return sorted_index;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002696 }
Ben Murdoch097c5b22016-05-18 11:27:45 +01002697 *out_insertion_index = len;
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002698 return T::kNotFound;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002699 } else {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002700 DCHECK_LE(valid_entries, array->number_of_entries());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002701 DCHECK_NULL(out_insertion_index); // Not supported here.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002702 for (int number = 0; number < valid_entries; number++) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002703 if (array->GetKey(number) == name) return number;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002704 }
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002705 return T::kNotFound;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002706 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002707}
Steve Blocka7e24c12009-10-30 11:49:00 +00002708
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002709
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002710template <SearchMode search_mode, typename T>
2711int Search(T* array, Name* name, int valid_entries, int* out_insertion_index) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002712 SLOW_DCHECK(array->IsSortedNoDuplicates());
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002713
Ben Murdoch097c5b22016-05-18 11:27:45 +01002714 if (valid_entries == 0) {
2715 if (search_mode == ALL_ENTRIES && out_insertion_index != nullptr) {
2716 *out_insertion_index = 0;
2717 }
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002718 return T::kNotFound;
2719 }
Steve Blocka7e24c12009-10-30 11:49:00 +00002720
2721 // Fast case: do linear search for small arrays.
2722 const int kMaxElementsForLinearSearch = 8;
Ben Murdoch097c5b22016-05-18 11:27:45 +01002723 if (valid_entries <= kMaxElementsForLinearSearch) {
2724 return LinearSearch<search_mode>(array, name, valid_entries,
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002725 out_insertion_index);
Steve Blocka7e24c12009-10-30 11:49:00 +00002726 }
2727
2728 // Slow case: perform binary search.
Ben Murdoch097c5b22016-05-18 11:27:45 +01002729 return BinarySearch<search_mode>(array, name, valid_entries,
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002730 out_insertion_index);
Steve Blocka7e24c12009-10-30 11:49:00 +00002731}
2732
2733
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002734int DescriptorArray::Search(Name* name, int valid_descriptors) {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002735 DCHECK(name->IsUniqueName());
Emily Bernierd0a1eb72015-03-24 16:35:39 -04002736 return internal::Search<VALID_ENTRIES>(this, name, valid_descriptors, NULL);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002737}
2738
Ben Murdoch097c5b22016-05-18 11:27:45 +01002739int DescriptorArray::SearchWithCache(Isolate* isolate, Name* name, Map* map) {
2740 DCHECK(name->IsUniqueName());
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002741 int number_of_own_descriptors = map->NumberOfOwnDescriptors();
2742 if (number_of_own_descriptors == 0) return kNotFound;
2743
Ben Murdoch097c5b22016-05-18 11:27:45 +01002744 DescriptorLookupCache* cache = isolate->descriptor_lookup_cache();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002745 int number = cache->Lookup(map, name);
2746
Iain Merrick75681382010-08-19 15:07:18 +01002747 if (number == DescriptorLookupCache::kAbsent) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002748 number = Search(name, number_of_own_descriptors);
2749 cache->Update(map, name, number);
Iain Merrick75681382010-08-19 15:07:18 +01002750 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002751
Iain Merrick75681382010-08-19 15:07:18 +01002752 return number;
2753}
2754
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002755PropertyDetails Map::GetLastDescriptorDetails() {
2756 return instance_descriptors()->GetDetails(LastAdded());
2757}
2758
2759
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002760int Map::LastAdded() {
2761 int number_of_own_descriptors = NumberOfOwnDescriptors();
2762 DCHECK(number_of_own_descriptors > 0);
2763 return number_of_own_descriptors - 1;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002764}
2765
2766
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002767int Map::NumberOfOwnDescriptors() {
2768 return NumberOfOwnDescriptorsBits::decode(bit_field3());
2769}
2770
2771
2772void Map::SetNumberOfOwnDescriptors(int number) {
2773 DCHECK(number <= instance_descriptors()->number_of_descriptors());
2774 set_bit_field3(NumberOfOwnDescriptorsBits::update(bit_field3(), number));
2775}
2776
2777
2778int Map::EnumLength() { return EnumLengthBits::decode(bit_field3()); }
2779
2780
2781void Map::SetEnumLength(int length) {
2782 if (length != kInvalidEnumCacheSentinel) {
2783 DCHECK(length >= 0);
2784 DCHECK(length == 0 || instance_descriptors()->HasEnumCache());
2785 DCHECK(length <= NumberOfOwnDescriptors());
2786 }
2787 set_bit_field3(EnumLengthBits::update(bit_field3(), length));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002788}
2789
2790
2791FixedArrayBase* Map::GetInitialElements() {
Ben Murdoch097c5b22016-05-18 11:27:45 +01002792 if (has_fast_elements() || has_fast_string_wrapper_elements()) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002793 DCHECK(!GetHeap()->InNewSpace(GetHeap()->empty_fixed_array()));
2794 return GetHeap()->empty_fixed_array();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002795 } else if (has_fixed_typed_array_elements()) {
2796 FixedTypedArrayBase* empty_array =
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002797 GetHeap()->EmptyFixedTypedArrayForMap(this);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002798 DCHECK(!GetHeap()->InNewSpace(empty_array));
2799 return empty_array;
2800 } else {
2801 UNREACHABLE();
2802 }
2803 return NULL;
2804}
2805
Ben Murdochc5610432016-08-08 18:44:38 +01002806// static
2807Handle<Map> Map::ReconfigureProperty(Handle<Map> map, int modify_index,
2808 PropertyKind new_kind,
2809 PropertyAttributes new_attributes,
2810 Representation new_representation,
2811 Handle<FieldType> new_field_type,
2812 StoreMode store_mode) {
2813 return Reconfigure(map, map->elements_kind(), modify_index, new_kind,
2814 new_attributes, new_representation, new_field_type,
2815 store_mode);
2816}
2817
2818// static
2819Handle<Map> Map::ReconfigureElementsKind(Handle<Map> map,
2820 ElementsKind new_elements_kind) {
2821 return Reconfigure(map, new_elements_kind, -1, kData, NONE,
2822 Representation::None(), FieldType::None(map->GetIsolate()),
2823 ALLOW_IN_DESCRIPTOR);
2824}
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002825
2826Object** DescriptorArray::GetKeySlot(int descriptor_number) {
2827 DCHECK(descriptor_number < number_of_descriptors());
2828 return RawFieldOfElementAt(ToKeyIndex(descriptor_number));
2829}
2830
2831
2832Object** DescriptorArray::GetDescriptorStartSlot(int descriptor_number) {
2833 return GetKeySlot(descriptor_number);
2834}
2835
2836
2837Object** DescriptorArray::GetDescriptorEndSlot(int descriptor_number) {
2838 return GetValueSlot(descriptor_number - 1) + 1;
2839}
2840
2841
2842Name* DescriptorArray::GetKey(int descriptor_number) {
2843 DCHECK(descriptor_number < number_of_descriptors());
2844 return Name::cast(get(ToKeyIndex(descriptor_number)));
2845}
2846
2847
2848int DescriptorArray::GetSortedKeyIndex(int descriptor_number) {
2849 return GetDetails(descriptor_number).pointer();
2850}
2851
2852
2853Name* DescriptorArray::GetSortedKey(int descriptor_number) {
2854 return GetKey(GetSortedKeyIndex(descriptor_number));
2855}
2856
2857
2858void DescriptorArray::SetSortedKey(int descriptor_index, int pointer) {
2859 PropertyDetails details = GetDetails(descriptor_index);
2860 set(ToDetailsIndex(descriptor_index), details.set_pointer(pointer).AsSmi());
2861}
2862
2863
2864void DescriptorArray::SetRepresentation(int descriptor_index,
2865 Representation representation) {
2866 DCHECK(!representation.IsNone());
2867 PropertyDetails details = GetDetails(descriptor_index);
2868 set(ToDetailsIndex(descriptor_index),
2869 details.CopyWithRepresentation(representation).AsSmi());
2870}
2871
2872
2873Object** DescriptorArray::GetValueSlot(int descriptor_number) {
2874 DCHECK(descriptor_number < number_of_descriptors());
2875 return RawFieldOfElementAt(ToValueIndex(descriptor_number));
2876}
2877
2878
2879int DescriptorArray::GetValueOffset(int descriptor_number) {
2880 return OffsetOfElementAt(ToValueIndex(descriptor_number));
Steve Blocka7e24c12009-10-30 11:49:00 +00002881}
2882
2883
2884Object* DescriptorArray::GetValue(int descriptor_number) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002885 DCHECK(descriptor_number < number_of_descriptors());
2886 return get(ToValueIndex(descriptor_number));
Steve Blocka7e24c12009-10-30 11:49:00 +00002887}
2888
2889
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002890void DescriptorArray::SetValue(int descriptor_index, Object* value) {
2891 set(ToValueIndex(descriptor_index), value);
2892}
2893
2894
2895PropertyDetails DescriptorArray::GetDetails(int descriptor_number) {
2896 DCHECK(descriptor_number < number_of_descriptors());
2897 Object* details = get(ToDetailsIndex(descriptor_number));
2898 return PropertyDetails(Smi::cast(details));
Steve Blocka7e24c12009-10-30 11:49:00 +00002899}
2900
2901
2902PropertyType DescriptorArray::GetType(int descriptor_number) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002903 return GetDetails(descriptor_number).type();
Steve Blocka7e24c12009-10-30 11:49:00 +00002904}
2905
2906
2907int DescriptorArray::GetFieldIndex(int descriptor_number) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002908 DCHECK(GetDetails(descriptor_number).location() == kField);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002909 return GetDetails(descriptor_number).field_index();
Steve Blocka7e24c12009-10-30 11:49:00 +00002910}
2911
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002912Object* DescriptorArray::GetConstant(int descriptor_number) {
2913 return GetValue(descriptor_number);
Steve Blocka7e24c12009-10-30 11:49:00 +00002914}
2915
2916
2917Object* DescriptorArray::GetCallbacksObject(int descriptor_number) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002918 DCHECK(GetType(descriptor_number) == ACCESSOR_CONSTANT);
Steve Blocka7e24c12009-10-30 11:49:00 +00002919 return GetValue(descriptor_number);
2920}
2921
2922
2923AccessorDescriptor* DescriptorArray::GetCallbacks(int descriptor_number) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002924 DCHECK(GetType(descriptor_number) == ACCESSOR_CONSTANT);
Ben Murdoch257744e2011-11-30 15:57:28 +00002925 Foreign* p = Foreign::cast(GetCallbacksObject(descriptor_number));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002926 return reinterpret_cast<AccessorDescriptor*>(p->foreign_address());
Steve Blocka7e24c12009-10-30 11:49:00 +00002927}
2928
2929
Steve Blocka7e24c12009-10-30 11:49:00 +00002930void DescriptorArray::Get(int descriptor_number, Descriptor* desc) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002931 desc->Init(handle(GetKey(descriptor_number), GetIsolate()),
2932 handle(GetValue(descriptor_number), GetIsolate()),
2933 GetDetails(descriptor_number));
Steve Blocka7e24c12009-10-30 11:49:00 +00002934}
2935
2936
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002937void DescriptorArray::SetDescriptor(int descriptor_number, Descriptor* desc) {
Steve Blocka7e24c12009-10-30 11:49:00 +00002938 // Range check.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002939 DCHECK(descriptor_number < number_of_descriptors());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002940 set(ToKeyIndex(descriptor_number), *desc->GetKey());
2941 set(ToValueIndex(descriptor_number), *desc->GetValue());
2942 set(ToDetailsIndex(descriptor_number), desc->GetDetails().AsSmi());
Steve Blocka7e24c12009-10-30 11:49:00 +00002943}
2944
2945
Ben Murdochb8a8cc12014-11-26 15:28:44 +00002946void DescriptorArray::Set(int descriptor_number, Descriptor* desc) {
2947 // Range check.
2948 DCHECK(descriptor_number < number_of_descriptors());
2949
2950 set(ToKeyIndex(descriptor_number), *desc->GetKey());
2951 set(ToValueIndex(descriptor_number), *desc->GetValue());
2952 set(ToDetailsIndex(descriptor_number), desc->GetDetails().AsSmi());
2953}
2954
2955
2956void DescriptorArray::Append(Descriptor* desc) {
2957 DisallowHeapAllocation no_gc;
2958 int descriptor_number = number_of_descriptors();
2959 SetNumberOfDescriptors(descriptor_number + 1);
2960 Set(descriptor_number, desc);
2961
2962 uint32_t hash = desc->GetKey()->Hash();
2963
2964 int insertion;
2965
2966 for (insertion = descriptor_number; insertion > 0; --insertion) {
2967 Name* key = GetSortedKey(insertion - 1);
2968 if (key->Hash() <= hash) break;
2969 SetSortedKey(insertion, GetSortedKeyIndex(insertion - 1));
2970 }
2971
2972 SetSortedKey(insertion, descriptor_number);
2973}
2974
2975
2976void DescriptorArray::SwapSortedKeys(int first, int second) {
2977 int first_key = GetSortedKeyIndex(first);
2978 SetSortedKey(first, GetSortedKeyIndex(second));
2979 SetSortedKey(second, first_key);
Ben Murdoch85b71792012-04-11 18:30:58 +01002980}
2981
2982
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002983PropertyType DescriptorArray::Entry::type() { return descs_->GetType(index_); }
2984
2985
2986Object* DescriptorArray::Entry::GetCallbackObject() {
2987 return descs_->GetValue(index_);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01002988}
2989
2990
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002991int HashTableBase::NumberOfElements() {
2992 return Smi::cast(get(kNumberOfElementsIndex))->value();
Steve Blocka7e24c12009-10-30 11:49:00 +00002993}
2994
2995
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00002996int HashTableBase::NumberOfDeletedElements() {
2997 return Smi::cast(get(kNumberOfDeletedElementsIndex))->value();
2998}
2999
3000
3001int HashTableBase::Capacity() {
3002 return Smi::cast(get(kCapacityIndex))->value();
3003}
3004
3005
3006void HashTableBase::ElementAdded() {
3007 SetNumberOfElements(NumberOfElements() + 1);
3008}
3009
3010
3011void HashTableBase::ElementRemoved() {
3012 SetNumberOfElements(NumberOfElements() - 1);
3013 SetNumberOfDeletedElements(NumberOfDeletedElements() + 1);
3014}
3015
3016
3017void HashTableBase::ElementsRemoved(int n) {
3018 SetNumberOfElements(NumberOfElements() - n);
3019 SetNumberOfDeletedElements(NumberOfDeletedElements() + n);
3020}
3021
3022
3023// static
3024int HashTableBase::ComputeCapacity(int at_least_space_for) {
3025 const int kMinCapacity = 4;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003026 int capacity = base::bits::RoundUpToPowerOfTwo32(at_least_space_for * 2);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003027 return Max(capacity, kMinCapacity);
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003028}
3029
Ben Murdochda12d292016-06-02 14:46:10 +01003030bool HashTableBase::IsKey(Heap* heap, Object* k) {
3031 return k != heap->the_hole_value() && k != heap->undefined_value();
3032}
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003033
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003034bool HashTableBase::IsKey(Object* k) {
3035 return !k->IsTheHole() && !k->IsUndefined();
3036}
3037
3038
3039void HashTableBase::SetNumberOfElements(int nof) {
3040 set(kNumberOfElementsIndex, Smi::FromInt(nof));
3041}
3042
3043
3044void HashTableBase::SetNumberOfDeletedElements(int nod) {
3045 set(kNumberOfDeletedElementsIndex, Smi::FromInt(nod));
3046}
3047
3048
3049template <typename Derived, typename Shape, typename Key>
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003050int HashTable<Derived, Shape, Key>::FindEntry(Key key) {
Steve Block44f0eee2011-05-26 01:26:41 +01003051 return FindEntry(GetIsolate(), key);
3052}
3053
3054
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003055template<typename Derived, typename Shape, typename Key>
3056int HashTable<Derived, Shape, Key>::FindEntry(Isolate* isolate, Key key) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003057 return FindEntry(isolate, key, HashTable::Hash(key));
3058}
3059
3060
3061// Find entry for key otherwise return kNotFound.
3062template <typename Derived, typename Shape, typename Key>
3063int HashTable<Derived, Shape, Key>::FindEntry(Isolate* isolate, Key key,
3064 int32_t hash) {
Steve Block44f0eee2011-05-26 01:26:41 +01003065 uint32_t capacity = Capacity();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003066 uint32_t entry = FirstProbe(hash, capacity);
Steve Block44f0eee2011-05-26 01:26:41 +01003067 uint32_t count = 1;
3068 // EnsureCapacity will guarantee the hash table is never full.
Ben Murdochc5610432016-08-08 18:44:38 +01003069 Object* undefined = isolate->heap()->undefined_value();
3070 Object* the_hole = isolate->heap()->the_hole_value();
Steve Block44f0eee2011-05-26 01:26:41 +01003071 while (true) {
3072 Object* element = KeyAt(entry);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003073 // Empty entry. Uses raw unchecked accessors because it is called by the
3074 // string table during bootstrapping.
Ben Murdochc5610432016-08-08 18:44:38 +01003075 if (element == undefined) break;
3076 if (element != the_hole && Shape::IsMatch(key, element)) return entry;
Steve Block44f0eee2011-05-26 01:26:41 +01003077 entry = NextProbe(entry, count++, capacity);
3078 }
3079 return kNotFound;
3080}
3081
Ben Murdochda12d292016-06-02 14:46:10 +01003082bool StringSetShape::IsMatch(String* key, Object* value) {
3083 return value->IsString() && key->Equals(String::cast(value));
3084}
3085
3086uint32_t StringSetShape::Hash(String* key) { return key->Hash(); }
3087
3088uint32_t StringSetShape::HashForObject(String* key, Object* object) {
3089 return object->IsString() ? String::cast(object)->Hash() : 0;
3090}
Steve Block44f0eee2011-05-26 01:26:41 +01003091
Ben Murdochc7cc0282012-03-05 14:35:55 +00003092bool SeededNumberDictionary::requires_slow_elements() {
Steve Blocka7e24c12009-10-30 11:49:00 +00003093 Object* max_index_object = get(kMaxNumberKeyIndex);
3094 if (!max_index_object->IsSmi()) return false;
3095 return 0 !=
3096 (Smi::cast(max_index_object)->value() & kRequiresSlowElementsMask);
3097}
3098
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003099
Ben Murdochc7cc0282012-03-05 14:35:55 +00003100uint32_t SeededNumberDictionary::max_number_key() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003101 DCHECK(!requires_slow_elements());
Steve Blocka7e24c12009-10-30 11:49:00 +00003102 Object* max_index_object = get(kMaxNumberKeyIndex);
3103 if (!max_index_object->IsSmi()) return 0;
3104 uint32_t value = static_cast<uint32_t>(Smi::cast(max_index_object)->value());
3105 return value >> kRequiresSlowElementsTagSize;
3106}
3107
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003108
Ben Murdochc7cc0282012-03-05 14:35:55 +00003109void SeededNumberDictionary::set_requires_slow_elements() {
Leon Clarke4515c472010-02-03 11:58:03 +00003110 set(kMaxNumberKeyIndex, Smi::FromInt(kRequiresSlowElementsMask));
Steve Blocka7e24c12009-10-30 11:49:00 +00003111}
3112
3113
3114// ------------------------------------
3115// Cast operations
3116
Ben Murdoch097c5b22016-05-18 11:27:45 +01003117CAST_ACCESSOR(AbstractCode)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003118CAST_ACCESSOR(ArrayList)
3119CAST_ACCESSOR(Bool16x8)
3120CAST_ACCESSOR(Bool32x4)
3121CAST_ACCESSOR(Bool8x16)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003122CAST_ACCESSOR(ByteArray)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003123CAST_ACCESSOR(BytecodeArray)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003124CAST_ACCESSOR(Cell)
3125CAST_ACCESSOR(Code)
3126CAST_ACCESSOR(CodeCacheHashTable)
3127CAST_ACCESSOR(CompilationCacheTable)
3128CAST_ACCESSOR(ConsString)
Ben Murdochb0fe1622011-05-05 13:52:32 +01003129CAST_ACCESSOR(DeoptimizationInputData)
3130CAST_ACCESSOR(DeoptimizationOutputData)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003131CAST_ACCESSOR(DependentCode)
3132CAST_ACCESSOR(DescriptorArray)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003133CAST_ACCESSOR(ExternalOneByteString)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003134CAST_ACCESSOR(ExternalString)
3135CAST_ACCESSOR(ExternalTwoByteString)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003136CAST_ACCESSOR(FixedArray)
3137CAST_ACCESSOR(FixedArrayBase)
3138CAST_ACCESSOR(FixedDoubleArray)
3139CAST_ACCESSOR(FixedTypedArrayBase)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003140CAST_ACCESSOR(Float32x4)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003141CAST_ACCESSOR(Foreign)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003142CAST_ACCESSOR(GlobalDictionary)
3143CAST_ACCESSOR(HandlerTable)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003144CAST_ACCESSOR(HeapObject)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003145CAST_ACCESSOR(Int16x8)
3146CAST_ACCESSOR(Int32x4)
3147CAST_ACCESSOR(Int8x16)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003148CAST_ACCESSOR(JSArray)
3149CAST_ACCESSOR(JSArrayBuffer)
3150CAST_ACCESSOR(JSArrayBufferView)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003151CAST_ACCESSOR(JSBoundFunction)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003152CAST_ACCESSOR(JSDataView)
3153CAST_ACCESSOR(JSDate)
3154CAST_ACCESSOR(JSFunction)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003155CAST_ACCESSOR(JSGeneratorObject)
3156CAST_ACCESSOR(JSGlobalObject)
3157CAST_ACCESSOR(JSGlobalProxy)
3158CAST_ACCESSOR(JSMap)
3159CAST_ACCESSOR(JSMapIterator)
3160CAST_ACCESSOR(JSMessageObject)
3161CAST_ACCESSOR(JSModule)
3162CAST_ACCESSOR(JSObject)
3163CAST_ACCESSOR(JSProxy)
3164CAST_ACCESSOR(JSReceiver)
3165CAST_ACCESSOR(JSRegExp)
3166CAST_ACCESSOR(JSSet)
3167CAST_ACCESSOR(JSSetIterator)
3168CAST_ACCESSOR(JSTypedArray)
3169CAST_ACCESSOR(JSValue)
3170CAST_ACCESSOR(JSWeakMap)
3171CAST_ACCESSOR(JSWeakSet)
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003172CAST_ACCESSOR(LayoutDescriptor)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003173CAST_ACCESSOR(Map)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003174CAST_ACCESSOR(Name)
3175CAST_ACCESSOR(NameDictionary)
3176CAST_ACCESSOR(NormalizedMapCache)
3177CAST_ACCESSOR(Object)
3178CAST_ACCESSOR(ObjectHashTable)
3179CAST_ACCESSOR(Oddball)
3180CAST_ACCESSOR(OrderedHashMap)
3181CAST_ACCESSOR(OrderedHashSet)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003182CAST_ACCESSOR(PropertyCell)
3183CAST_ACCESSOR(ScopeInfo)
3184CAST_ACCESSOR(SeededNumberDictionary)
3185CAST_ACCESSOR(SeqOneByteString)
3186CAST_ACCESSOR(SeqString)
3187CAST_ACCESSOR(SeqTwoByteString)
3188CAST_ACCESSOR(SharedFunctionInfo)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003189CAST_ACCESSOR(Simd128Value)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003190CAST_ACCESSOR(SlicedString)
3191CAST_ACCESSOR(Smi)
3192CAST_ACCESSOR(String)
Ben Murdochda12d292016-06-02 14:46:10 +01003193CAST_ACCESSOR(StringSet)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003194CAST_ACCESSOR(StringTable)
Steve Blocka7e24c12009-10-30 11:49:00 +00003195CAST_ACCESSOR(Struct)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003196CAST_ACCESSOR(Symbol)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003197CAST_ACCESSOR(Uint16x8)
3198CAST_ACCESSOR(Uint32x4)
3199CAST_ACCESSOR(Uint8x16)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003200CAST_ACCESSOR(UnseededNumberDictionary)
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003201CAST_ACCESSOR(WeakCell)
3202CAST_ACCESSOR(WeakFixedArray)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003203CAST_ACCESSOR(WeakHashTable)
3204
3205
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003206// static
3207template <class Traits>
3208STATIC_CONST_MEMBER_DEFINITION const InstanceType
3209 FixedTypedArray<Traits>::kInstanceType;
3210
3211
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003212template <class Traits>
3213FixedTypedArray<Traits>* FixedTypedArray<Traits>::cast(Object* object) {
3214 SLOW_DCHECK(object->IsHeapObject() &&
3215 HeapObject::cast(object)->map()->instance_type() ==
3216 Traits::kInstanceType);
3217 return reinterpret_cast<FixedTypedArray<Traits>*>(object);
3218}
3219
3220
3221template <class Traits>
3222const FixedTypedArray<Traits>*
3223FixedTypedArray<Traits>::cast(const Object* object) {
3224 SLOW_DCHECK(object->IsHeapObject() &&
3225 HeapObject::cast(object)->map()->instance_type() ==
3226 Traits::kInstanceType);
3227 return reinterpret_cast<FixedTypedArray<Traits>*>(object);
3228}
Steve Blocka7e24c12009-10-30 11:49:00 +00003229
3230
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003231#define DEFINE_DEOPT_ELEMENT_ACCESSORS(name, type) \
3232 type* DeoptimizationInputData::name() { \
3233 return type::cast(get(k##name##Index)); \
3234 } \
3235 void DeoptimizationInputData::Set##name(type* value) { \
3236 set(k##name##Index, value); \
3237 }
3238
3239DEFINE_DEOPT_ELEMENT_ACCESSORS(TranslationByteArray, ByteArray)
3240DEFINE_DEOPT_ELEMENT_ACCESSORS(InlinedFunctionCount, Smi)
3241DEFINE_DEOPT_ELEMENT_ACCESSORS(LiteralArray, FixedArray)
3242DEFINE_DEOPT_ELEMENT_ACCESSORS(OsrAstId, Smi)
3243DEFINE_DEOPT_ELEMENT_ACCESSORS(OsrPcOffset, Smi)
3244DEFINE_DEOPT_ELEMENT_ACCESSORS(OptimizationId, Smi)
3245DEFINE_DEOPT_ELEMENT_ACCESSORS(SharedFunctionInfo, Object)
3246DEFINE_DEOPT_ELEMENT_ACCESSORS(WeakCellCache, Object)
3247
3248#undef DEFINE_DEOPT_ELEMENT_ACCESSORS
3249
3250
3251#define DEFINE_DEOPT_ENTRY_ACCESSORS(name, type) \
3252 type* DeoptimizationInputData::name(int i) { \
3253 return type::cast(get(IndexForEntry(i) + k##name##Offset)); \
3254 } \
3255 void DeoptimizationInputData::Set##name(int i, type* value) { \
3256 set(IndexForEntry(i) + k##name##Offset, value); \
3257 }
3258
3259DEFINE_DEOPT_ENTRY_ACCESSORS(AstIdRaw, Smi)
3260DEFINE_DEOPT_ENTRY_ACCESSORS(TranslationIndex, Smi)
3261DEFINE_DEOPT_ENTRY_ACCESSORS(ArgumentsStackHeight, Smi)
3262DEFINE_DEOPT_ENTRY_ACCESSORS(Pc, Smi)
3263
3264#undef DEFINE_DEOPT_ENTRY_ACCESSORS
3265
3266
3267BailoutId DeoptimizationInputData::AstId(int i) {
3268 return BailoutId(AstIdRaw(i)->value());
3269}
3270
3271
3272void DeoptimizationInputData::SetAstId(int i, BailoutId value) {
3273 SetAstIdRaw(i, Smi::FromInt(value.ToInt()));
3274}
3275
3276
3277int DeoptimizationInputData::DeoptCount() {
3278 return (length() - kFirstDeoptEntryIndex) / kDeoptEntrySize;
3279}
3280
3281
3282int DeoptimizationOutputData::DeoptPoints() { return length() / 2; }
3283
3284
3285BailoutId DeoptimizationOutputData::AstId(int index) {
3286 return BailoutId(Smi::cast(get(index * 2))->value());
3287}
3288
3289
3290void DeoptimizationOutputData::SetAstId(int index, BailoutId id) {
3291 set(index * 2, Smi::FromInt(id.ToInt()));
3292}
3293
3294
3295Smi* DeoptimizationOutputData::PcAndState(int index) {
3296 return Smi::cast(get(1 + index * 2));
3297}
3298
3299
3300void DeoptimizationOutputData::SetPcAndState(int index, Smi* offset) {
3301 set(1 + index * 2, offset);
3302}
3303
3304
3305Object* LiteralsArray::get(int index) const { return FixedArray::get(index); }
3306
3307
3308void LiteralsArray::set(int index, Object* value) {
3309 FixedArray::set(index, value);
3310}
3311
3312
3313void LiteralsArray::set(int index, Smi* value) {
3314 FixedArray::set(index, value);
3315}
3316
3317
3318void LiteralsArray::set(int index, Object* value, WriteBarrierMode mode) {
3319 FixedArray::set(index, value, mode);
3320}
3321
3322
3323LiteralsArray* LiteralsArray::cast(Object* object) {
3324 SLOW_DCHECK(object->IsLiteralsArray());
3325 return reinterpret_cast<LiteralsArray*>(object);
3326}
3327
3328
3329TypeFeedbackVector* LiteralsArray::feedback_vector() const {
3330 return TypeFeedbackVector::cast(get(kVectorIndex));
3331}
3332
3333
3334void LiteralsArray::set_feedback_vector(TypeFeedbackVector* vector) {
3335 set(kVectorIndex, vector);
3336}
3337
3338
3339Object* LiteralsArray::literal(int literal_index) const {
3340 return get(kFirstLiteralIndex + literal_index);
3341}
3342
3343
3344void LiteralsArray::set_literal(int literal_index, Object* literal) {
3345 set(kFirstLiteralIndex + literal_index, literal);
3346}
3347
3348
3349int LiteralsArray::literals_count() const {
3350 return length() - kFirstLiteralIndex;
3351}
3352
Ben Murdoch097c5b22016-05-18 11:27:45 +01003353int HandlerTable::GetRangeStart(int index) const {
3354 return Smi::cast(get(index * kRangeEntrySize + kRangeStartIndex))->value();
3355}
3356
3357int HandlerTable::GetRangeEnd(int index) const {
3358 return Smi::cast(get(index * kRangeEntrySize + kRangeEndIndex))->value();
3359}
3360
3361int HandlerTable::GetRangeHandler(int index) const {
3362 return HandlerOffsetField::decode(
3363 Smi::cast(get(index * kRangeEntrySize + kRangeHandlerIndex))->value());
3364}
3365
3366int HandlerTable::GetRangeData(int index) const {
3367 return Smi::cast(get(index * kRangeEntrySize + kRangeDataIndex))->value();
3368}
3369
3370HandlerTable::CatchPrediction HandlerTable::GetRangePrediction(
3371 int index) const {
3372 return HandlerPredictionField::decode(
3373 Smi::cast(get(index * kRangeEntrySize + kRangeHandlerIndex))->value());
3374}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003375
3376void HandlerTable::SetRangeStart(int index, int value) {
3377 set(index * kRangeEntrySize + kRangeStartIndex, Smi::FromInt(value));
3378}
3379
3380
3381void HandlerTable::SetRangeEnd(int index, int value) {
3382 set(index * kRangeEntrySize + kRangeEndIndex, Smi::FromInt(value));
3383}
3384
3385
3386void HandlerTable::SetRangeHandler(int index, int offset,
3387 CatchPrediction prediction) {
3388 int value = HandlerOffsetField::encode(offset) |
3389 HandlerPredictionField::encode(prediction);
3390 set(index * kRangeEntrySize + kRangeHandlerIndex, Smi::FromInt(value));
3391}
3392
Ben Murdoch097c5b22016-05-18 11:27:45 +01003393void HandlerTable::SetRangeData(int index, int value) {
3394 set(index * kRangeEntrySize + kRangeDataIndex, Smi::FromInt(value));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003395}
3396
3397
3398void HandlerTable::SetReturnOffset(int index, int value) {
3399 set(index * kReturnEntrySize + kReturnOffsetIndex, Smi::FromInt(value));
3400}
3401
3402
3403void HandlerTable::SetReturnHandler(int index, int offset,
3404 CatchPrediction prediction) {
3405 int value = HandlerOffsetField::encode(offset) |
3406 HandlerPredictionField::encode(prediction);
3407 set(index * kReturnEntrySize + kReturnHandlerIndex, Smi::FromInt(value));
3408}
3409
Ben Murdoch097c5b22016-05-18 11:27:45 +01003410int HandlerTable::NumberOfRangeEntries() const {
3411 return length() / kRangeEntrySize;
3412}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003413
Steve Blocka7e24c12009-10-30 11:49:00 +00003414#define MAKE_STRUCT_CAST(NAME, Name, name) CAST_ACCESSOR(Name)
3415 STRUCT_LIST(MAKE_STRUCT_CAST)
3416#undef MAKE_STRUCT_CAST
3417
3418
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003419template <typename Derived, typename Shape, typename Key>
3420HashTable<Derived, Shape, Key>*
3421HashTable<Derived, Shape, Key>::cast(Object* obj) {
3422 SLOW_DCHECK(obj->IsHashTable());
Steve Blocka7e24c12009-10-30 11:49:00 +00003423 return reinterpret_cast<HashTable*>(obj);
3424}
3425
3426
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003427template <typename Derived, typename Shape, typename Key>
3428const HashTable<Derived, Shape, Key>*
3429HashTable<Derived, Shape, Key>::cast(const Object* obj) {
3430 SLOW_DCHECK(obj->IsHashTable());
3431 return reinterpret_cast<const HashTable*>(obj);
3432}
3433
3434
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00003435SMI_ACCESSORS(FixedArrayBase, length, kLengthOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003436SYNCHRONIZED_SMI_ACCESSORS(FixedArrayBase, length, kLengthOffset)
3437
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003438SMI_ACCESSORS(FreeSpace, size, kSizeOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003439NOBARRIER_SMI_ACCESSORS(FreeSpace, size, kSizeOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00003440
Steve Block6ded16b2010-05-10 14:33:55 +01003441SMI_ACCESSORS(String, length, kLengthOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003442SYNCHRONIZED_SMI_ACCESSORS(String, length, kLengthOffset)
Steve Blockd0582a62009-12-15 09:54:21 +00003443
3444
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003445int FreeSpace::Size() { return size(); }
3446
3447
3448FreeSpace* FreeSpace::next() {
3449 DCHECK(map() == GetHeap()->root(Heap::kFreeSpaceMapRootIndex) ||
3450 (!GetHeap()->deserialization_complete() && map() == NULL));
3451 DCHECK_LE(kNextOffset + kPointerSize, nobarrier_size());
3452 return reinterpret_cast<FreeSpace*>(
3453 Memory::Address_at(address() + kNextOffset));
3454}
3455
3456
3457void FreeSpace::set_next(FreeSpace* next) {
3458 DCHECK(map() == GetHeap()->root(Heap::kFreeSpaceMapRootIndex) ||
3459 (!GetHeap()->deserialization_complete() && map() == NULL));
3460 DCHECK_LE(kNextOffset + kPointerSize, nobarrier_size());
3461 base::NoBarrier_Store(
3462 reinterpret_cast<base::AtomicWord*>(address() + kNextOffset),
3463 reinterpret_cast<base::AtomicWord>(next));
3464}
3465
3466
3467FreeSpace* FreeSpace::cast(HeapObject* o) {
3468 SLOW_DCHECK(!o->GetHeap()->deserialization_complete() || o->IsFreeSpace());
3469 return reinterpret_cast<FreeSpace*>(o);
3470}
3471
3472
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003473uint32_t Name::hash_field() {
Steve Blockd0582a62009-12-15 09:54:21 +00003474 return READ_UINT32_FIELD(this, kHashFieldOffset);
3475}
3476
3477
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003478void Name::set_hash_field(uint32_t value) {
Steve Blockd0582a62009-12-15 09:54:21 +00003479 WRITE_UINT32_FIELD(this, kHashFieldOffset, value);
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01003480#if V8_HOST_ARCH_64_BIT
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003481#if V8_TARGET_LITTLE_ENDIAN
3482 WRITE_UINT32_FIELD(this, kHashFieldSlot + kIntSize, 0);
3483#else
3484 WRITE_UINT32_FIELD(this, kHashFieldSlot, 0);
3485#endif
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01003486#endif
Steve Blockd0582a62009-12-15 09:54:21 +00003487}
3488
3489
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003490bool Name::Equals(Name* other) {
3491 if (other == this) return true;
3492 if ((this->IsInternalizedString() && other->IsInternalizedString()) ||
3493 this->IsSymbol() || other->IsSymbol()) {
3494 return false;
3495 }
3496 return String::cast(this)->SlowEquals(String::cast(other));
3497}
3498
3499
3500bool Name::Equals(Handle<Name> one, Handle<Name> two) {
3501 if (one.is_identical_to(two)) return true;
3502 if ((one->IsInternalizedString() && two->IsInternalizedString()) ||
3503 one->IsSymbol() || two->IsSymbol()) {
3504 return false;
3505 }
3506 return String::SlowEquals(Handle<String>::cast(one),
3507 Handle<String>::cast(two));
3508}
3509
3510
3511ACCESSORS(Symbol, name, Object, kNameOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003512SMI_ACCESSORS(Symbol, flags, kFlagsOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003513BOOL_ACCESSORS(Symbol, flags, is_private, kPrivateBit)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003514BOOL_ACCESSORS(Symbol, flags, is_well_known_symbol, kWellKnownSymbolBit)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003515
3516
Steve Blocka7e24c12009-10-30 11:49:00 +00003517bool String::Equals(String* other) {
3518 if (other == this) return true;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003519 if (this->IsInternalizedString() && other->IsInternalizedString()) {
Steve Blocka7e24c12009-10-30 11:49:00 +00003520 return false;
3521 }
3522 return SlowEquals(other);
3523}
3524
3525
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003526bool String::Equals(Handle<String> one, Handle<String> two) {
3527 if (one.is_identical_to(two)) return true;
3528 if (one->IsInternalizedString() && two->IsInternalizedString()) {
3529 return false;
3530 }
3531 return SlowEquals(one, two);
Steve Blocka7e24c12009-10-30 11:49:00 +00003532}
3533
3534
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003535Handle<String> String::Flatten(Handle<String> string, PretenureFlag pretenure) {
3536 if (!string->IsConsString()) return string;
3537 Handle<ConsString> cons = Handle<ConsString>::cast(string);
3538 if (cons->IsFlat()) return handle(cons->first());
3539 return SlowFlatten(cons, pretenure);
Leon Clarkef7060e22010-06-03 12:02:55 +01003540}
3541
3542
Steve Blocka7e24c12009-10-30 11:49:00 +00003543uint16_t String::Get(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003544 DCHECK(index >= 0 && index < length());
Steve Blocka7e24c12009-10-30 11:49:00 +00003545 switch (StringShape(this).full_representation_tag()) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003546 case kSeqStringTag | kOneByteStringTag:
3547 return SeqOneByteString::cast(this)->SeqOneByteStringGet(index);
Steve Blocka7e24c12009-10-30 11:49:00 +00003548 case kSeqStringTag | kTwoByteStringTag:
3549 return SeqTwoByteString::cast(this)->SeqTwoByteStringGet(index);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003550 case kConsStringTag | kOneByteStringTag:
Steve Blocka7e24c12009-10-30 11:49:00 +00003551 case kConsStringTag | kTwoByteStringTag:
3552 return ConsString::cast(this)->ConsStringGet(index);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003553 case kExternalStringTag | kOneByteStringTag:
3554 return ExternalOneByteString::cast(this)->ExternalOneByteStringGet(index);
Steve Blocka7e24c12009-10-30 11:49:00 +00003555 case kExternalStringTag | kTwoByteStringTag:
3556 return ExternalTwoByteString::cast(this)->ExternalTwoByteStringGet(index);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003557 case kSlicedStringTag | kOneByteStringTag:
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003558 case kSlicedStringTag | kTwoByteStringTag:
3559 return SlicedString::cast(this)->SlicedStringGet(index);
Steve Blocka7e24c12009-10-30 11:49:00 +00003560 default:
3561 break;
3562 }
3563
3564 UNREACHABLE();
3565 return 0;
3566}
3567
3568
3569void String::Set(int index, uint16_t value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003570 DCHECK(index >= 0 && index < length());
3571 DCHECK(StringShape(this).IsSequential());
Steve Blocka7e24c12009-10-30 11:49:00 +00003572
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003573 return this->IsOneByteRepresentation()
3574 ? SeqOneByteString::cast(this)->SeqOneByteStringSet(index, value)
Steve Blocka7e24c12009-10-30 11:49:00 +00003575 : SeqTwoByteString::cast(this)->SeqTwoByteStringSet(index, value);
3576}
3577
3578
3579bool String::IsFlat() {
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003580 if (!StringShape(this).IsCons()) return true;
3581 return ConsString::cast(this)->second()->length() == 0;
3582}
3583
3584
3585String* String::GetUnderlying() {
3586 // Giving direct access to underlying string only makes sense if the
3587 // wrapping string is already flattened.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003588 DCHECK(this->IsFlat());
3589 DCHECK(StringShape(this).IsIndirect());
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003590 STATIC_ASSERT(ConsString::kFirstOffset == SlicedString::kParentOffset);
3591 const int kUnderlyingOffset = SlicedString::kParentOffset;
3592 return String::cast(READ_FIELD(this, kUnderlyingOffset));
Steve Blocka7e24c12009-10-30 11:49:00 +00003593}
3594
3595
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003596template<class Visitor>
3597ConsString* String::VisitFlat(Visitor* visitor,
3598 String* string,
3599 const int offset) {
3600 int slice_offset = offset;
3601 const int length = string->length();
3602 DCHECK(offset <= length);
3603 while (true) {
3604 int32_t type = string->map()->instance_type();
3605 switch (type & (kStringRepresentationMask | kStringEncodingMask)) {
3606 case kSeqStringTag | kOneByteStringTag:
3607 visitor->VisitOneByteString(
3608 SeqOneByteString::cast(string)->GetChars() + slice_offset,
3609 length - offset);
3610 return NULL;
3611
3612 case kSeqStringTag | kTwoByteStringTag:
3613 visitor->VisitTwoByteString(
3614 SeqTwoByteString::cast(string)->GetChars() + slice_offset,
3615 length - offset);
3616 return NULL;
3617
3618 case kExternalStringTag | kOneByteStringTag:
3619 visitor->VisitOneByteString(
3620 ExternalOneByteString::cast(string)->GetChars() + slice_offset,
3621 length - offset);
3622 return NULL;
3623
3624 case kExternalStringTag | kTwoByteStringTag:
3625 visitor->VisitTwoByteString(
3626 ExternalTwoByteString::cast(string)->GetChars() + slice_offset,
3627 length - offset);
3628 return NULL;
3629
3630 case kSlicedStringTag | kOneByteStringTag:
3631 case kSlicedStringTag | kTwoByteStringTag: {
3632 SlicedString* slicedString = SlicedString::cast(string);
3633 slice_offset += slicedString->offset();
3634 string = slicedString->parent();
3635 continue;
3636 }
3637
3638 case kConsStringTag | kOneByteStringTag:
3639 case kConsStringTag | kTwoByteStringTag:
3640 return ConsString::cast(string);
3641
3642 default:
3643 UNREACHABLE();
3644 return NULL;
3645 }
3646 }
3647}
3648
3649
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003650template <>
3651inline Vector<const uint8_t> String::GetCharVector() {
3652 String::FlatContent flat = GetFlatContent();
3653 DCHECK(flat.IsOneByte());
3654 return flat.ToOneByteVector();
3655}
3656
3657
3658template <>
3659inline Vector<const uc16> String::GetCharVector() {
3660 String::FlatContent flat = GetFlatContent();
3661 DCHECK(flat.IsTwoByte());
3662 return flat.ToUC16Vector();
3663}
3664
3665
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003666uint16_t SeqOneByteString::SeqOneByteStringGet(int index) {
3667 DCHECK(index >= 0 && index < length());
Steve Blocka7e24c12009-10-30 11:49:00 +00003668 return READ_BYTE_FIELD(this, kHeaderSize + index * kCharSize);
3669}
3670
3671
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003672void SeqOneByteString::SeqOneByteStringSet(int index, uint16_t value) {
3673 DCHECK(index >= 0 && index < length() && value <= kMaxOneByteCharCode);
Steve Blocka7e24c12009-10-30 11:49:00 +00003674 WRITE_BYTE_FIELD(this, kHeaderSize + index * kCharSize,
3675 static_cast<byte>(value));
3676}
3677
3678
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003679Address SeqOneByteString::GetCharsAddress() {
Steve Blocka7e24c12009-10-30 11:49:00 +00003680 return FIELD_ADDR(this, kHeaderSize);
3681}
3682
3683
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003684uint8_t* SeqOneByteString::GetChars() {
3685 return reinterpret_cast<uint8_t*>(GetCharsAddress());
Steve Blocka7e24c12009-10-30 11:49:00 +00003686}
3687
3688
3689Address SeqTwoByteString::GetCharsAddress() {
3690 return FIELD_ADDR(this, kHeaderSize);
3691}
3692
3693
3694uc16* SeqTwoByteString::GetChars() {
3695 return reinterpret_cast<uc16*>(FIELD_ADDR(this, kHeaderSize));
3696}
3697
3698
3699uint16_t SeqTwoByteString::SeqTwoByteStringGet(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003700 DCHECK(index >= 0 && index < length());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003701 return READ_UINT16_FIELD(this, kHeaderSize + index * kShortSize);
Steve Blocka7e24c12009-10-30 11:49:00 +00003702}
3703
3704
3705void SeqTwoByteString::SeqTwoByteStringSet(int index, uint16_t value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003706 DCHECK(index >= 0 && index < length());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003707 WRITE_UINT16_FIELD(this, kHeaderSize + index * kShortSize, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00003708}
3709
3710
3711int SeqTwoByteString::SeqTwoByteStringSize(InstanceType instance_type) {
Steve Block6ded16b2010-05-10 14:33:55 +01003712 return SizeFor(length());
Steve Blocka7e24c12009-10-30 11:49:00 +00003713}
3714
3715
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003716int SeqOneByteString::SeqOneByteStringSize(InstanceType instance_type) {
Steve Block6ded16b2010-05-10 14:33:55 +01003717 return SizeFor(length());
Steve Blocka7e24c12009-10-30 11:49:00 +00003718}
3719
3720
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003721String* SlicedString::parent() {
3722 return String::cast(READ_FIELD(this, kParentOffset));
3723}
3724
3725
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003726void SlicedString::set_parent(String* parent, WriteBarrierMode mode) {
3727 DCHECK(parent->IsSeqString() || parent->IsExternalString());
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003728 WRITE_FIELD(this, kParentOffset, parent);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003729 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kParentOffset, parent, mode);
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003730}
3731
3732
3733SMI_ACCESSORS(SlicedString, offset, kOffsetOffset)
3734
3735
Steve Blocka7e24c12009-10-30 11:49:00 +00003736String* ConsString::first() {
3737 return String::cast(READ_FIELD(this, kFirstOffset));
3738}
3739
3740
3741Object* ConsString::unchecked_first() {
3742 return READ_FIELD(this, kFirstOffset);
3743}
3744
3745
3746void ConsString::set_first(String* value, WriteBarrierMode mode) {
3747 WRITE_FIELD(this, kFirstOffset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003748 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kFirstOffset, value, mode);
Steve Blocka7e24c12009-10-30 11:49:00 +00003749}
3750
3751
3752String* ConsString::second() {
3753 return String::cast(READ_FIELD(this, kSecondOffset));
3754}
3755
3756
3757Object* ConsString::unchecked_second() {
3758 return READ_FIELD(this, kSecondOffset);
3759}
3760
3761
3762void ConsString::set_second(String* value, WriteBarrierMode mode) {
3763 WRITE_FIELD(this, kSecondOffset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003764 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kSecondOffset, value, mode);
Steve Blocka7e24c12009-10-30 11:49:00 +00003765}
3766
3767
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003768bool ExternalString::is_short() {
3769 InstanceType type = map()->instance_type();
3770 return (type & kShortExternalStringMask) == kShortExternalStringTag;
3771}
3772
3773
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003774const ExternalOneByteString::Resource* ExternalOneByteString::resource() {
Steve Blocka7e24c12009-10-30 11:49:00 +00003775 return *reinterpret_cast<Resource**>(FIELD_ADDR(this, kResourceOffset));
3776}
3777
3778
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003779void ExternalOneByteString::update_data_cache() {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003780 if (is_short()) return;
3781 const char** data_field =
3782 reinterpret_cast<const char**>(FIELD_ADDR(this, kResourceDataOffset));
3783 *data_field = resource()->data();
3784}
3785
3786
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003787void ExternalOneByteString::set_resource(
3788 const ExternalOneByteString::Resource* resource) {
3789 DCHECK(IsAligned(reinterpret_cast<intptr_t>(resource), kPointerSize));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003790 *reinterpret_cast<const Resource**>(
3791 FIELD_ADDR(this, kResourceOffset)) = resource;
3792 if (resource != NULL) update_data_cache();
Steve Blocka7e24c12009-10-30 11:49:00 +00003793}
3794
3795
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003796const uint8_t* ExternalOneByteString::GetChars() {
3797 return reinterpret_cast<const uint8_t*>(resource()->data());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003798}
3799
3800
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003801uint16_t ExternalOneByteString::ExternalOneByteStringGet(int index) {
3802 DCHECK(index >= 0 && index < length());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003803 return GetChars()[index];
3804}
3805
3806
3807const ExternalTwoByteString::Resource* ExternalTwoByteString::resource() {
Steve Blocka7e24c12009-10-30 11:49:00 +00003808 return *reinterpret_cast<Resource**>(FIELD_ADDR(this, kResourceOffset));
3809}
3810
3811
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003812void ExternalTwoByteString::update_data_cache() {
3813 if (is_short()) return;
3814 const uint16_t** data_field =
3815 reinterpret_cast<const uint16_t**>(FIELD_ADDR(this, kResourceDataOffset));
3816 *data_field = resource()->data();
3817}
3818
3819
Steve Blocka7e24c12009-10-30 11:49:00 +00003820void ExternalTwoByteString::set_resource(
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003821 const ExternalTwoByteString::Resource* resource) {
3822 *reinterpret_cast<const Resource**>(
3823 FIELD_ADDR(this, kResourceOffset)) = resource;
3824 if (resource != NULL) update_data_cache();
3825}
3826
3827
3828const uint16_t* ExternalTwoByteString::GetChars() {
3829 return resource()->data();
3830}
3831
3832
3833uint16_t ExternalTwoByteString::ExternalTwoByteStringGet(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003834 DCHECK(index >= 0 && index < length());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01003835 return GetChars()[index];
3836}
3837
3838
3839const uint16_t* ExternalTwoByteString::ExternalTwoByteStringGetData(
3840 unsigned start) {
3841 return GetChars() + start;
Steve Blocka7e24c12009-10-30 11:49:00 +00003842}
3843
3844
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003845int ConsStringIterator::OffsetForDepth(int depth) { return depth & kDepthMask; }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003846
3847
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003848void ConsStringIterator::PushLeft(ConsString* string) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003849 frames_[depth_++ & kDepthMask] = string;
3850}
3851
3852
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003853void ConsStringIterator::PushRight(ConsString* string) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003854 // Inplace update.
3855 frames_[(depth_-1) & kDepthMask] = string;
3856}
3857
3858
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003859void ConsStringIterator::AdjustMaximumDepth() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003860 if (depth_ > maximum_depth_) maximum_depth_ = depth_;
3861}
3862
3863
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003864void ConsStringIterator::Pop() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003865 DCHECK(depth_ > 0);
3866 DCHECK(depth_ <= maximum_depth_);
3867 depth_--;
3868}
3869
3870
3871uint16_t StringCharacterStream::GetNext() {
3872 DCHECK(buffer8_ != NULL && end_ != NULL);
3873 // Advance cursor if needed.
3874 if (buffer8_ == end_) HasMore();
3875 DCHECK(buffer8_ < end_);
3876 return is_one_byte_ ? *buffer8_++ : *buffer16_++;
3877}
3878
3879
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003880StringCharacterStream::StringCharacterStream(String* string, int offset)
3881 : is_one_byte_(false) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003882 Reset(string, offset);
3883}
3884
3885
3886void StringCharacterStream::Reset(String* string, int offset) {
3887 buffer8_ = NULL;
3888 end_ = NULL;
3889 ConsString* cons_string = String::VisitFlat(this, string, offset);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003890 iter_.Reset(cons_string, offset);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003891 if (cons_string != NULL) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003892 string = iter_.Next(&offset);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003893 if (string != NULL) String::VisitFlat(this, string, offset);
3894 }
3895}
3896
3897
3898bool StringCharacterStream::HasMore() {
3899 if (buffer8_ != end_) return true;
3900 int offset;
Emily Bernierd0a1eb72015-03-24 16:35:39 -04003901 String* string = iter_.Next(&offset);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003902 DCHECK_EQ(offset, 0);
3903 if (string == NULL) return false;
3904 String::VisitFlat(this, string);
3905 DCHECK(buffer8_ != end_);
3906 return true;
3907}
3908
3909
3910void StringCharacterStream::VisitOneByteString(
3911 const uint8_t* chars, int length) {
3912 is_one_byte_ = true;
3913 buffer8_ = chars;
3914 end_ = chars + length;
3915}
3916
3917
3918void StringCharacterStream::VisitTwoByteString(
3919 const uint16_t* chars, int length) {
3920 is_one_byte_ = false;
3921 buffer16_ = chars;
3922 end_ = reinterpret_cast<const uint8_t*>(chars + length);
3923}
3924
3925
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003926int ByteArray::Size() { return RoundUp(length() + kHeaderSize, kPointerSize); }
Ben Murdochb8e0da22011-05-16 14:20:40 +01003927
Steve Blocka7e24c12009-10-30 11:49:00 +00003928byte ByteArray::get(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003929 DCHECK(index >= 0 && index < this->length());
Steve Blocka7e24c12009-10-30 11:49:00 +00003930 return READ_BYTE_FIELD(this, kHeaderSize + index * kCharSize);
3931}
3932
3933
3934void ByteArray::set(int index, byte value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003935 DCHECK(index >= 0 && index < this->length());
Steve Blocka7e24c12009-10-30 11:49:00 +00003936 WRITE_BYTE_FIELD(this, kHeaderSize + index * kCharSize, value);
3937}
3938
Ben Murdochc5610432016-08-08 18:44:38 +01003939void ByteArray::copy_in(int index, const byte* buffer, int length) {
3940 DCHECK(index >= 0 && length >= 0 && index + length >= index &&
3941 index + length <= this->length());
3942 byte* dst_addr = FIELD_ADDR(this, kHeaderSize + index * kCharSize);
3943 memcpy(dst_addr, buffer, length);
3944}
3945
3946void ByteArray::copy_out(int index, byte* buffer, int length) {
3947 DCHECK(index >= 0 && length >= 0 && index + length >= index &&
3948 index + length <= this->length());
3949 const byte* src_addr = FIELD_ADDR(this, kHeaderSize + index * kCharSize);
3950 memcpy(buffer, src_addr, length);
3951}
Steve Blocka7e24c12009-10-30 11:49:00 +00003952
3953int ByteArray::get_int(int index) {
Ben Murdochc5610432016-08-08 18:44:38 +01003954 DCHECK(index >= 0 && index < this->length() / kIntSize);
Steve Blocka7e24c12009-10-30 11:49:00 +00003955 return READ_INT_FIELD(this, kHeaderSize + index * kIntSize);
3956}
3957
Ben Murdochc5610432016-08-08 18:44:38 +01003958void ByteArray::set_int(int index, int value) {
3959 DCHECK(index >= 0 && index < this->length() / kIntSize);
3960 WRITE_INT_FIELD(this, kHeaderSize + index * kIntSize, value);
3961}
Steve Blocka7e24c12009-10-30 11:49:00 +00003962
3963ByteArray* ByteArray::FromDataStartAddress(Address address) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00003964 DCHECK_TAG_ALIGNED(address);
Steve Blocka7e24c12009-10-30 11:49:00 +00003965 return reinterpret_cast<ByteArray*>(address - kHeaderSize + kHeapObjectTag);
3966}
3967
3968
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003969int ByteArray::ByteArraySize() { return SizeFor(this->length()); }
3970
3971
Steve Blocka7e24c12009-10-30 11:49:00 +00003972Address ByteArray::GetDataStartAddress() {
3973 return reinterpret_cast<Address>(this) - kHeapObjectTag + kHeaderSize;
3974}
3975
3976
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003977byte BytecodeArray::get(int index) {
3978 DCHECK(index >= 0 && index < this->length());
3979 return READ_BYTE_FIELD(this, kHeaderSize + index * kCharSize);
Steve Blocka7e24c12009-10-30 11:49:00 +00003980}
3981
3982
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003983void BytecodeArray::set(int index, byte value) {
3984 DCHECK(index >= 0 && index < this->length());
3985 WRITE_BYTE_FIELD(this, kHeaderSize + index * kCharSize, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00003986}
3987
3988
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003989void BytecodeArray::set_frame_size(int frame_size) {
3990 DCHECK_GE(frame_size, 0);
3991 DCHECK(IsAligned(frame_size, static_cast<unsigned>(kPointerSize)));
3992 WRITE_INT_FIELD(this, kFrameSizeOffset, frame_size);
Ben Murdoch69a99ed2011-11-30 16:03:39 +00003993}
3994
3995
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00003996int BytecodeArray::frame_size() const {
3997 return READ_INT_FIELD(this, kFrameSizeOffset);
Steve Blocka7e24c12009-10-30 11:49:00 +00003998}
3999
4000
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004001int BytecodeArray::register_count() const {
4002 return frame_size() / kPointerSize;
4003}
4004
4005
4006void BytecodeArray::set_parameter_count(int number_of_parameters) {
4007 DCHECK_GE(number_of_parameters, 0);
4008 // Parameter count is stored as the size on stack of the parameters to allow
4009 // it to be used directly by generated code.
4010 WRITE_INT_FIELD(this, kParameterSizeOffset,
4011 (number_of_parameters << kPointerSizeLog2));
4012}
4013
Ben Murdoch097c5b22016-05-18 11:27:45 +01004014int BytecodeArray::interrupt_budget() const {
4015 return READ_INT_FIELD(this, kInterruptBudgetOffset);
4016}
4017
4018void BytecodeArray::set_interrupt_budget(int interrupt_budget) {
4019 DCHECK_GE(interrupt_budget, 0);
4020 WRITE_INT_FIELD(this, kInterruptBudgetOffset, interrupt_budget);
4021}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004022
4023int BytecodeArray::parameter_count() const {
4024 // Parameter count is stored as the size on stack of the parameters to allow
4025 // it to be used directly by generated code.
4026 return READ_INT_FIELD(this, kParameterSizeOffset) >> kPointerSizeLog2;
4027}
4028
4029
4030ACCESSORS(BytecodeArray, constant_pool, FixedArray, kConstantPoolOffset)
Ben Murdoch097c5b22016-05-18 11:27:45 +01004031ACCESSORS(BytecodeArray, handler_table, FixedArray, kHandlerTableOffset)
Ben Murdochda12d292016-06-02 14:46:10 +01004032ACCESSORS(BytecodeArray, source_position_table, ByteArray,
Ben Murdoch097c5b22016-05-18 11:27:45 +01004033 kSourcePositionTableOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004034
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004035Address BytecodeArray::GetFirstBytecodeAddress() {
4036 return reinterpret_cast<Address>(this) - kHeapObjectTag + kHeaderSize;
4037}
4038
4039
4040int BytecodeArray::BytecodeArraySize() { return SizeFor(this->length()); }
4041
4042
4043ACCESSORS(FixedTypedArrayBase, base_pointer, Object, kBasePointerOffset)
4044
4045
4046void* FixedTypedArrayBase::external_pointer() const {
Steve Block3ce2e202009-11-05 08:53:23 +00004047 intptr_t ptr = READ_INTPTR_FIELD(this, kExternalPointerOffset);
4048 return reinterpret_cast<void*>(ptr);
4049}
4050
4051
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004052void FixedTypedArrayBase::set_external_pointer(void* value,
4053 WriteBarrierMode mode) {
Steve Block3ce2e202009-11-05 08:53:23 +00004054 intptr_t ptr = reinterpret_cast<intptr_t>(value);
4055 WRITE_INTPTR_FIELD(this, kExternalPointerOffset, ptr);
4056}
4057
4058
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004059void* FixedTypedArrayBase::DataPtr() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004060 return reinterpret_cast<void*>(
4061 reinterpret_cast<intptr_t>(base_pointer()) +
4062 reinterpret_cast<intptr_t>(external_pointer()));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004063}
4064
4065
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004066int FixedTypedArrayBase::ElementSize(InstanceType type) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004067 int element_size;
4068 switch (type) {
4069#define TYPED_ARRAY_CASE(Type, type, TYPE, ctype, size) \
4070 case FIXED_##TYPE##_ARRAY_TYPE: \
4071 element_size = size; \
4072 break;
4073
4074 TYPED_ARRAYS(TYPED_ARRAY_CASE)
4075#undef TYPED_ARRAY_CASE
4076 default:
4077 UNREACHABLE();
4078 return 0;
4079 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004080 return element_size;
4081}
4082
4083
4084int FixedTypedArrayBase::DataSize(InstanceType type) {
4085 if (base_pointer() == Smi::FromInt(0)) return 0;
4086 return length() * ElementSize(type);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004087}
4088
4089
4090int FixedTypedArrayBase::DataSize() {
4091 return DataSize(map()->instance_type());
4092}
4093
4094
4095int FixedTypedArrayBase::size() {
4096 return OBJECT_POINTER_ALIGN(kDataOffset + DataSize());
4097}
4098
4099
4100int FixedTypedArrayBase::TypedArraySize(InstanceType type) {
4101 return OBJECT_POINTER_ALIGN(kDataOffset + DataSize(type));
4102}
4103
4104
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004105int FixedTypedArrayBase::TypedArraySize(InstanceType type, int length) {
4106 return OBJECT_POINTER_ALIGN(kDataOffset + length * ElementSize(type));
4107}
4108
4109
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004110uint8_t Uint8ArrayTraits::defaultValue() { return 0; }
4111
4112
4113uint8_t Uint8ClampedArrayTraits::defaultValue() { return 0; }
4114
4115
4116int8_t Int8ArrayTraits::defaultValue() { return 0; }
4117
4118
4119uint16_t Uint16ArrayTraits::defaultValue() { return 0; }
4120
4121
4122int16_t Int16ArrayTraits::defaultValue() { return 0; }
4123
4124
4125uint32_t Uint32ArrayTraits::defaultValue() { return 0; }
4126
4127
4128int32_t Int32ArrayTraits::defaultValue() { return 0; }
4129
4130
4131float Float32ArrayTraits::defaultValue() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004132 return std::numeric_limits<float>::quiet_NaN();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004133}
4134
4135
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004136double Float64ArrayTraits::defaultValue() {
4137 return std::numeric_limits<double>::quiet_NaN();
4138}
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004139
4140
4141template <class Traits>
4142typename Traits::ElementType FixedTypedArray<Traits>::get_scalar(int index) {
4143 DCHECK((index >= 0) && (index < this->length()));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004144 ElementType* ptr = reinterpret_cast<ElementType*>(DataPtr());
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004145 return ptr[index];
4146}
4147
4148
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004149template <class Traits>
4150void FixedTypedArray<Traits>::set(int index, ElementType value) {
4151 DCHECK((index >= 0) && (index < this->length()));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004152 ElementType* ptr = reinterpret_cast<ElementType*>(DataPtr());
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004153 ptr[index] = value;
4154}
4155
4156
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004157template <class Traits>
4158typename Traits::ElementType FixedTypedArray<Traits>::from_int(int value) {
4159 return static_cast<ElementType>(value);
4160}
4161
4162
4163template <> inline
4164uint8_t FixedTypedArray<Uint8ClampedArrayTraits>::from_int(int value) {
4165 if (value < 0) return 0;
4166 if (value > 0xFF) return 0xFF;
4167 return static_cast<uint8_t>(value);
4168}
4169
4170
4171template <class Traits>
4172typename Traits::ElementType FixedTypedArray<Traits>::from_double(
4173 double value) {
4174 return static_cast<ElementType>(DoubleToInt32(value));
4175}
4176
4177
4178template<> inline
4179uint8_t FixedTypedArray<Uint8ClampedArrayTraits>::from_double(double value) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004180 // Handle NaNs and less than zero values which clamp to zero.
4181 if (!(value > 0)) return 0;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004182 if (value > 0xFF) return 0xFF;
4183 return static_cast<uint8_t>(lrint(value));
4184}
4185
4186
4187template<> inline
4188float FixedTypedArray<Float32ArrayTraits>::from_double(double value) {
4189 return static_cast<float>(value);
4190}
4191
4192
4193template<> inline
4194double FixedTypedArray<Float64ArrayTraits>::from_double(double value) {
4195 return value;
4196}
4197
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004198template <class Traits>
Ben Murdoch097c5b22016-05-18 11:27:45 +01004199Handle<Object> FixedTypedArray<Traits>::get(FixedTypedArray<Traits>* array,
4200 int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004201 return Traits::ToHandle(array->GetIsolate(), array->get_scalar(index));
4202}
4203
4204
4205template <class Traits>
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004206void FixedTypedArray<Traits>::SetValue(uint32_t index, Object* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004207 ElementType cast_value = Traits::defaultValue();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004208 if (value->IsSmi()) {
4209 int int_value = Smi::cast(value)->value();
4210 cast_value = from_int(int_value);
4211 } else if (value->IsHeapNumber()) {
4212 double double_value = HeapNumber::cast(value)->value();
4213 cast_value = from_double(double_value);
4214 } else {
4215 // Clamp undefined to the default value. All other types have been
4216 // converted to a number type further up in the call chain.
4217 DCHECK(value->IsUndefined());
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004218 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004219 set(index, cast_value);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004220}
4221
4222
4223Handle<Object> Uint8ArrayTraits::ToHandle(Isolate* isolate, uint8_t scalar) {
4224 return handle(Smi::FromInt(scalar), isolate);
4225}
4226
4227
4228Handle<Object> Uint8ClampedArrayTraits::ToHandle(Isolate* isolate,
4229 uint8_t scalar) {
4230 return handle(Smi::FromInt(scalar), isolate);
4231}
4232
4233
4234Handle<Object> Int8ArrayTraits::ToHandle(Isolate* isolate, int8_t scalar) {
4235 return handle(Smi::FromInt(scalar), isolate);
4236}
4237
4238
4239Handle<Object> Uint16ArrayTraits::ToHandle(Isolate* isolate, uint16_t scalar) {
4240 return handle(Smi::FromInt(scalar), isolate);
4241}
4242
4243
4244Handle<Object> Int16ArrayTraits::ToHandle(Isolate* isolate, int16_t scalar) {
4245 return handle(Smi::FromInt(scalar), isolate);
4246}
4247
4248
4249Handle<Object> Uint32ArrayTraits::ToHandle(Isolate* isolate, uint32_t scalar) {
4250 return isolate->factory()->NewNumberFromUint(scalar);
4251}
4252
4253
4254Handle<Object> Int32ArrayTraits::ToHandle(Isolate* isolate, int32_t scalar) {
4255 return isolate->factory()->NewNumberFromInt(scalar);
4256}
4257
4258
4259Handle<Object> Float32ArrayTraits::ToHandle(Isolate* isolate, float scalar) {
4260 return isolate->factory()->NewNumber(scalar);
4261}
4262
4263
4264Handle<Object> Float64ArrayTraits::ToHandle(Isolate* isolate, double scalar) {
4265 return isolate->factory()->NewNumber(scalar);
4266}
4267
4268
Iain Merrick9ac36c92010-09-13 15:29:50 +01004269int Map::visitor_id() {
4270 return READ_BYTE_FIELD(this, kVisitorIdOffset);
4271}
4272
4273
4274void Map::set_visitor_id(int id) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004275 DCHECK(0 <= id && id < 256);
Iain Merrick9ac36c92010-09-13 15:29:50 +01004276 WRITE_BYTE_FIELD(this, kVisitorIdOffset, static_cast<byte>(id));
4277}
4278
Steve Block3ce2e202009-11-05 08:53:23 +00004279
Steve Blocka7e24c12009-10-30 11:49:00 +00004280int Map::instance_size() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004281 return NOBARRIER_READ_BYTE_FIELD(
4282 this, kInstanceSizeOffset) << kPointerSizeLog2;
Steve Blocka7e24c12009-10-30 11:49:00 +00004283}
4284
4285
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004286int Map::inobject_properties_or_constructor_function_index() {
4287 return READ_BYTE_FIELD(this,
4288 kInObjectPropertiesOrConstructorFunctionIndexOffset);
Steve Blocka7e24c12009-10-30 11:49:00 +00004289}
4290
4291
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004292void Map::set_inobject_properties_or_constructor_function_index(int value) {
4293 DCHECK(0 <= value && value < 256);
4294 WRITE_BYTE_FIELD(this, kInObjectPropertiesOrConstructorFunctionIndexOffset,
4295 static_cast<byte>(value));
4296}
4297
4298
4299int Map::GetInObjectProperties() {
4300 DCHECK(IsJSObjectMap());
4301 return inobject_properties_or_constructor_function_index();
4302}
4303
4304
4305void Map::SetInObjectProperties(int value) {
4306 DCHECK(IsJSObjectMap());
4307 set_inobject_properties_or_constructor_function_index(value);
4308}
4309
4310
4311int Map::GetConstructorFunctionIndex() {
4312 DCHECK(IsPrimitiveMap());
4313 return inobject_properties_or_constructor_function_index();
4314}
4315
4316
4317void Map::SetConstructorFunctionIndex(int value) {
4318 DCHECK(IsPrimitiveMap());
4319 set_inobject_properties_or_constructor_function_index(value);
Steve Blocka7e24c12009-10-30 11:49:00 +00004320}
4321
4322
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004323int Map::GetInObjectPropertyOffset(int index) {
4324 // Adjust for the number of properties stored in the object.
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004325 index -= GetInObjectProperties();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004326 DCHECK(index <= 0);
4327 return instance_size() + (index * kPointerSize);
4328}
4329
4330
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004331Handle<Map> Map::AddMissingTransitionsForTesting(
4332 Handle<Map> split_map, Handle<DescriptorArray> descriptors,
4333 Handle<LayoutDescriptor> full_layout_descriptor) {
4334 return AddMissingTransitions(split_map, descriptors, full_layout_descriptor);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004335}
4336
4337
Steve Blocka7e24c12009-10-30 11:49:00 +00004338int HeapObject::SizeFromMap(Map* map) {
Steve Block791712a2010-08-27 10:21:07 +01004339 int instance_size = map->instance_size();
4340 if (instance_size != kVariableSizeSentinel) return instance_size;
Steve Blocka7e24c12009-10-30 11:49:00 +00004341 // Only inline the most frequent cases.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004342 InstanceType instance_type = map->instance_type();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004343 if (instance_type == FIXED_ARRAY_TYPE ||
4344 instance_type == TRANSITION_ARRAY_TYPE) {
4345 return FixedArray::SizeFor(
4346 reinterpret_cast<FixedArray*>(this)->synchronized_length());
Steve Blocka7e24c12009-10-30 11:49:00 +00004347 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004348 if (instance_type == ONE_BYTE_STRING_TYPE ||
4349 instance_type == ONE_BYTE_INTERNALIZED_STRING_TYPE) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004350 // Strings may get concurrently truncated, hence we have to access its
4351 // length synchronized.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004352 return SeqOneByteString::SizeFor(
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004353 reinterpret_cast<SeqOneByteString*>(this)->synchronized_length());
Steve Block791712a2010-08-27 10:21:07 +01004354 }
Steve Blocka7e24c12009-10-30 11:49:00 +00004355 if (instance_type == BYTE_ARRAY_TYPE) {
4356 return reinterpret_cast<ByteArray*>(this)->ByteArraySize();
4357 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004358 if (instance_type == BYTECODE_ARRAY_TYPE) {
4359 return reinterpret_cast<BytecodeArray*>(this)->BytecodeArraySize();
4360 }
Ben Murdoch3ef787d2012-04-12 10:51:47 +01004361 if (instance_type == FREE_SPACE_TYPE) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004362 return reinterpret_cast<FreeSpace*>(this)->nobarrier_size();
Ben Murdoch3ef787d2012-04-12 10:51:47 +01004363 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004364 if (instance_type == STRING_TYPE ||
4365 instance_type == INTERNALIZED_STRING_TYPE) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004366 // Strings may get concurrently truncated, hence we have to access its
4367 // length synchronized.
Steve Block791712a2010-08-27 10:21:07 +01004368 return SeqTwoByteString::SizeFor(
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004369 reinterpret_cast<SeqTwoByteString*>(this)->synchronized_length());
Steve Block791712a2010-08-27 10:21:07 +01004370 }
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00004371 if (instance_type == FIXED_DOUBLE_ARRAY_TYPE) {
4372 return FixedDoubleArray::SizeFor(
4373 reinterpret_cast<FixedDoubleArray*>(this)->length());
4374 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004375 if (instance_type >= FIRST_FIXED_TYPED_ARRAY_TYPE &&
4376 instance_type <= LAST_FIXED_TYPED_ARRAY_TYPE) {
4377 return reinterpret_cast<FixedTypedArrayBase*>(
4378 this)->TypedArraySize(instance_type);
4379 }
4380 DCHECK(instance_type == CODE_TYPE);
Steve Block791712a2010-08-27 10:21:07 +01004381 return reinterpret_cast<Code*>(this)->CodeSize();
Steve Blocka7e24c12009-10-30 11:49:00 +00004382}
4383
4384
4385void Map::set_instance_size(int value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004386 DCHECK_EQ(0, value & (kPointerSize - 1));
Steve Blocka7e24c12009-10-30 11:49:00 +00004387 value >>= kPointerSizeLog2;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004388 DCHECK(0 <= value && value < 256);
4389 NOBARRIER_WRITE_BYTE_FIELD(
4390 this, kInstanceSizeOffset, static_cast<byte>(value));
Steve Blocka7e24c12009-10-30 11:49:00 +00004391}
4392
4393
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004394void Map::clear_unused() { WRITE_BYTE_FIELD(this, kUnusedOffset, 0); }
Steve Blocka7e24c12009-10-30 11:49:00 +00004395
4396
4397InstanceType Map::instance_type() {
4398 return static_cast<InstanceType>(READ_BYTE_FIELD(this, kInstanceTypeOffset));
4399}
4400
4401
4402void Map::set_instance_type(InstanceType value) {
Steve Blocka7e24c12009-10-30 11:49:00 +00004403 WRITE_BYTE_FIELD(this, kInstanceTypeOffset, value);
4404}
4405
4406
4407int Map::unused_property_fields() {
4408 return READ_BYTE_FIELD(this, kUnusedPropertyFieldsOffset);
4409}
4410
4411
4412void Map::set_unused_property_fields(int value) {
4413 WRITE_BYTE_FIELD(this, kUnusedPropertyFieldsOffset, Min(value, 255));
4414}
4415
4416
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004417byte Map::bit_field() const { return READ_BYTE_FIELD(this, kBitFieldOffset); }
Steve Blocka7e24c12009-10-30 11:49:00 +00004418
4419
4420void Map::set_bit_field(byte value) {
4421 WRITE_BYTE_FIELD(this, kBitFieldOffset, value);
4422}
4423
4424
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004425byte Map::bit_field2() const { return READ_BYTE_FIELD(this, kBitField2Offset); }
Steve Blocka7e24c12009-10-30 11:49:00 +00004426
4427
4428void Map::set_bit_field2(byte value) {
4429 WRITE_BYTE_FIELD(this, kBitField2Offset, value);
4430}
4431
4432
4433void Map::set_non_instance_prototype(bool value) {
4434 if (value) {
4435 set_bit_field(bit_field() | (1 << kHasNonInstancePrototype));
4436 } else {
4437 set_bit_field(bit_field() & ~(1 << kHasNonInstancePrototype));
4438 }
4439}
4440
4441
4442bool Map::has_non_instance_prototype() {
4443 return ((1 << kHasNonInstancePrototype) & bit_field()) != 0;
4444}
4445
4446
Ben Murdoch097c5b22016-05-18 11:27:45 +01004447void Map::set_is_constructor(bool value) {
4448 if (value) {
4449 set_bit_field(bit_field() | (1 << kIsConstructor));
4450 } else {
4451 set_bit_field(bit_field() & ~(1 << kIsConstructor));
4452 }
Steve Block6ded16b2010-05-10 14:33:55 +01004453}
4454
4455
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004456bool Map::is_constructor() const {
4457 return ((1 << kIsConstructor) & bit_field()) != 0;
4458}
4459
Ben Murdoch097c5b22016-05-18 11:27:45 +01004460void Map::set_has_hidden_prototype(bool value) {
4461 set_bit_field3(HasHiddenPrototype::update(bit_field3(), value));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004462}
4463
Ben Murdoch097c5b22016-05-18 11:27:45 +01004464bool Map::has_hidden_prototype() const {
4465 return HasHiddenPrototype::decode(bit_field3());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004466}
4467
4468
4469void Map::set_has_indexed_interceptor() {
4470 set_bit_field(bit_field() | (1 << kHasIndexedInterceptor));
4471}
4472
4473
4474bool Map::has_indexed_interceptor() {
4475 return ((1 << kHasIndexedInterceptor) & bit_field()) != 0;
4476}
4477
4478
4479void Map::set_is_undetectable() {
4480 set_bit_field(bit_field() | (1 << kIsUndetectable));
4481}
4482
4483
4484bool Map::is_undetectable() {
4485 return ((1 << kIsUndetectable) & bit_field()) != 0;
4486}
4487
4488
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004489void Map::set_has_named_interceptor() {
4490 set_bit_field(bit_field() | (1 << kHasNamedInterceptor));
4491}
4492
4493
4494bool Map::has_named_interceptor() {
4495 return ((1 << kHasNamedInterceptor) & bit_field()) != 0;
Steve Block6ded16b2010-05-10 14:33:55 +01004496}
4497
4498
Steve Blocka7e24c12009-10-30 11:49:00 +00004499void Map::set_is_access_check_needed(bool access_check_needed) {
4500 if (access_check_needed) {
4501 set_bit_field(bit_field() | (1 << kIsAccessCheckNeeded));
4502 } else {
4503 set_bit_field(bit_field() & ~(1 << kIsAccessCheckNeeded));
4504 }
4505}
4506
4507
4508bool Map::is_access_check_needed() {
4509 return ((1 << kIsAccessCheckNeeded) & bit_field()) != 0;
4510}
4511
4512
Steve Block8defd9f2010-07-08 12:39:36 +01004513void Map::set_is_extensible(bool value) {
4514 if (value) {
4515 set_bit_field2(bit_field2() | (1 << kIsExtensible));
4516 } else {
4517 set_bit_field2(bit_field2() & ~(1 << kIsExtensible));
4518 }
4519}
4520
4521bool Map::is_extensible() {
4522 return ((1 << kIsExtensible) & bit_field2()) != 0;
4523}
4524
4525
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004526void Map::set_is_prototype_map(bool value) {
4527 set_bit_field2(IsPrototypeMapBits::update(bit_field2(), value));
Kristian Monsen0d5e1162010-09-30 15:31:59 +01004528}
4529
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004530bool Map::is_prototype_map() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004531 return IsPrototypeMapBits::decode(bit_field2());
Kristian Monsen0d5e1162010-09-30 15:31:59 +01004532}
4533
4534
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004535void Map::set_elements_kind(ElementsKind elements_kind) {
4536 DCHECK(static_cast<int>(elements_kind) < kElementsKindCount);
4537 DCHECK(kElementsKindCount <= (1 << Map::ElementsKindBits::kSize));
4538 set_bit_field2(Map::ElementsKindBits::update(bit_field2(), elements_kind));
4539 DCHECK(this->elements_kind() == elements_kind);
4540}
4541
4542
4543ElementsKind Map::elements_kind() {
4544 return Map::ElementsKindBits::decode(bit_field2());
4545}
4546
4547
4548bool Map::has_fast_smi_elements() {
4549 return IsFastSmiElementsKind(elements_kind());
4550}
4551
4552bool Map::has_fast_object_elements() {
4553 return IsFastObjectElementsKind(elements_kind());
4554}
4555
4556bool Map::has_fast_smi_or_object_elements() {
4557 return IsFastSmiOrObjectElementsKind(elements_kind());
4558}
4559
4560bool Map::has_fast_double_elements() {
4561 return IsFastDoubleElementsKind(elements_kind());
4562}
4563
4564bool Map::has_fast_elements() { return IsFastElementsKind(elements_kind()); }
4565
4566bool Map::has_sloppy_arguments_elements() {
4567 return IsSloppyArgumentsElements(elements_kind());
4568}
4569
Ben Murdoch097c5b22016-05-18 11:27:45 +01004570bool Map::has_fast_string_wrapper_elements() {
4571 return elements_kind() == FAST_STRING_WRAPPER_ELEMENTS;
4572}
4573
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004574bool Map::has_fixed_typed_array_elements() {
4575 return IsFixedTypedArrayElementsKind(elements_kind());
4576}
4577
4578bool Map::has_dictionary_elements() {
4579 return IsDictionaryElementsKind(elements_kind());
4580}
4581
4582
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004583void Map::set_dictionary_map(bool value) {
4584 uint32_t new_bit_field3 = DictionaryMap::update(bit_field3(), value);
4585 new_bit_field3 = IsUnstable::update(new_bit_field3, value);
4586 set_bit_field3(new_bit_field3);
Kristian Monsen0d5e1162010-09-30 15:31:59 +01004587}
4588
4589
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004590bool Map::is_dictionary_map() {
4591 return DictionaryMap::decode(bit_field3());
Kristian Monsen0d5e1162010-09-30 15:31:59 +01004592}
4593
Steve Block8defd9f2010-07-08 12:39:36 +01004594
Steve Blocka7e24c12009-10-30 11:49:00 +00004595Code::Flags Code::flags() {
4596 return static_cast<Flags>(READ_INT_FIELD(this, kFlagsOffset));
4597}
4598
4599
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004600void Map::set_owns_descriptors(bool owns_descriptors) {
4601 set_bit_field3(OwnsDescriptors::update(bit_field3(), owns_descriptors));
4602}
4603
4604
4605bool Map::owns_descriptors() {
4606 return OwnsDescriptors::decode(bit_field3());
4607}
4608
4609
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004610void Map::set_is_callable() { set_bit_field(bit_field() | (1 << kIsCallable)); }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004611
4612
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004613bool Map::is_callable() const {
4614 return ((1 << kIsCallable) & bit_field()) != 0;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004615}
4616
4617
4618void Map::deprecate() {
4619 set_bit_field3(Deprecated::update(bit_field3(), true));
4620}
4621
4622
4623bool Map::is_deprecated() {
4624 return Deprecated::decode(bit_field3());
4625}
4626
4627
4628void Map::set_migration_target(bool value) {
4629 set_bit_field3(IsMigrationTarget::update(bit_field3(), value));
4630}
4631
4632
4633bool Map::is_migration_target() {
4634 return IsMigrationTarget::decode(bit_field3());
4635}
4636
4637
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004638void Map::set_new_target_is_base(bool value) {
4639 set_bit_field3(NewTargetIsBase::update(bit_field3(), value));
4640}
4641
4642
4643bool Map::new_target_is_base() { return NewTargetIsBase::decode(bit_field3()); }
4644
4645
4646void Map::set_construction_counter(int value) {
4647 set_bit_field3(ConstructionCounter::update(bit_field3(), value));
4648}
4649
4650
4651int Map::construction_counter() {
4652 return ConstructionCounter::decode(bit_field3());
4653}
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004654
4655
4656void Map::mark_unstable() {
4657 set_bit_field3(IsUnstable::update(bit_field3(), true));
4658}
4659
4660
4661bool Map::is_stable() {
4662 return !IsUnstable::decode(bit_field3());
4663}
4664
4665
4666bool Map::has_code_cache() {
Ben Murdochc5610432016-08-08 18:44:38 +01004667 // Code caches are always fixed arrays. The empty fixed array is used as a
4668 // sentinel for an absent code cache.
4669 return FixedArray::cast(code_cache())->length() != 0;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004670}
4671
4672
4673bool Map::CanBeDeprecated() {
4674 int descriptor = LastAdded();
4675 for (int i = 0; i <= descriptor; i++) {
4676 PropertyDetails details = instance_descriptors()->GetDetails(i);
4677 if (details.representation().IsNone()) return true;
4678 if (details.representation().IsSmi()) return true;
4679 if (details.representation().IsDouble()) return true;
4680 if (details.representation().IsHeapObject()) return true;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004681 if (details.type() == DATA_CONSTANT) return true;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004682 }
4683 return false;
4684}
4685
4686
4687void Map::NotifyLeafMapLayoutChange() {
4688 if (is_stable()) {
4689 mark_unstable();
4690 dependent_code()->DeoptimizeDependentCodeGroup(
4691 GetIsolate(),
4692 DependentCode::kPrototypeCheckGroup);
4693 }
4694}
4695
4696
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004697bool Map::CanTransition() {
4698 // Only JSObject and subtypes have map transitions and back pointers.
4699 STATIC_ASSERT(LAST_TYPE == LAST_JS_OBJECT_TYPE);
4700 return instance_type() >= FIRST_JS_OBJECT_TYPE;
4701}
4702
4703
4704bool Map::IsBooleanMap() { return this == GetHeap()->boolean_map(); }
4705bool Map::IsPrimitiveMap() {
4706 STATIC_ASSERT(FIRST_PRIMITIVE_TYPE == FIRST_TYPE);
4707 return instance_type() <= LAST_PRIMITIVE_TYPE;
4708}
4709bool Map::IsJSReceiverMap() {
4710 STATIC_ASSERT(LAST_JS_RECEIVER_TYPE == LAST_TYPE);
4711 return instance_type() >= FIRST_JS_RECEIVER_TYPE;
4712}
4713bool Map::IsJSObjectMap() {
4714 STATIC_ASSERT(LAST_JS_OBJECT_TYPE == LAST_TYPE);
4715 return instance_type() >= FIRST_JS_OBJECT_TYPE;
4716}
4717bool Map::IsJSArrayMap() { return instance_type() == JS_ARRAY_TYPE; }
4718bool Map::IsJSFunctionMap() { return instance_type() == JS_FUNCTION_TYPE; }
4719bool Map::IsStringMap() { return instance_type() < FIRST_NONSTRING_TYPE; }
4720bool Map::IsJSProxyMap() { return instance_type() == JS_PROXY_TYPE; }
4721bool Map::IsJSGlobalProxyMap() {
4722 return instance_type() == JS_GLOBAL_PROXY_TYPE;
4723}
4724bool Map::IsJSGlobalObjectMap() {
4725 return instance_type() == JS_GLOBAL_OBJECT_TYPE;
4726}
4727bool Map::IsJSTypedArrayMap() { return instance_type() == JS_TYPED_ARRAY_TYPE; }
4728bool Map::IsJSDataViewMap() { return instance_type() == JS_DATA_VIEW_TYPE; }
4729
4730
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004731bool Map::CanOmitMapChecks() {
4732 return is_stable() && FLAG_omit_map_checks_for_leaf_maps;
4733}
4734
4735
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004736DependentCode* DependentCode::next_link() {
4737 return DependentCode::cast(get(kNextLinkIndex));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004738}
4739
4740
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004741void DependentCode::set_next_link(DependentCode* next) {
4742 set(kNextLinkIndex, next);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004743}
4744
4745
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004746int DependentCode::flags() { return Smi::cast(get(kFlagsIndex))->value(); }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004747
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004748
4749void DependentCode::set_flags(int flags) {
4750 set(kFlagsIndex, Smi::FromInt(flags));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004751}
4752
4753
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004754int DependentCode::count() { return CountField::decode(flags()); }
4755
4756void DependentCode::set_count(int value) {
4757 set_flags(CountField::update(flags(), value));
4758}
4759
4760
4761DependentCode::DependencyGroup DependentCode::group() {
4762 return static_cast<DependencyGroup>(GroupField::decode(flags()));
4763}
4764
4765
4766void DependentCode::set_group(DependentCode::DependencyGroup group) {
4767 set_flags(GroupField::update(flags(), static_cast<int>(group)));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004768}
4769
4770
4771void DependentCode::set_object_at(int i, Object* object) {
4772 set(kCodesStartIndex + i, object);
4773}
4774
4775
4776Object* DependentCode::object_at(int i) {
4777 return get(kCodesStartIndex + i);
4778}
4779
4780
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004781void DependentCode::clear_at(int i) {
4782 set_undefined(kCodesStartIndex + i);
4783}
4784
4785
4786void DependentCode::copy(int from, int to) {
4787 set(kCodesStartIndex + to, get(kCodesStartIndex + from));
4788}
4789
4790
Steve Blocka7e24c12009-10-30 11:49:00 +00004791void Code::set_flags(Code::Flags flags) {
Ben Murdoch589d6972011-11-30 16:04:58 +00004792 STATIC_ASSERT(Code::NUMBER_OF_KINDS <= KindField::kMax + 1);
Steve Blocka7e24c12009-10-30 11:49:00 +00004793 WRITE_INT_FIELD(this, kFlagsOffset, flags);
4794}
4795
4796
4797Code::Kind Code::kind() {
4798 return ExtractKindFromFlags(flags());
4799}
4800
4801
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004802bool Code::IsCodeStubOrIC() {
4803 return kind() == STUB || kind() == HANDLER || kind() == LOAD_IC ||
4804 kind() == KEYED_LOAD_IC || kind() == CALL_IC || kind() == STORE_IC ||
4805 kind() == KEYED_STORE_IC || kind() == BINARY_OP_IC ||
Ben Murdochda12d292016-06-02 14:46:10 +01004806 kind() == COMPARE_IC || kind() == TO_BOOLEAN_IC;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004807}
4808
4809
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004810bool Code::IsJavaScriptCode() {
4811 return kind() == FUNCTION || kind() == OPTIMIZED_FUNCTION ||
4812 is_interpreter_entry_trampoline();
4813}
4814
4815
Steve Blocka7e24c12009-10-30 11:49:00 +00004816InlineCacheState Code::ic_state() {
4817 InlineCacheState result = ExtractICStateFromFlags(flags());
4818 // Only allow uninitialized or debugger states for non-IC code
4819 // objects. This is used in the debugger to determine whether or not
4820 // a call to code object has been replaced with a debug break call.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004821 DCHECK(is_inline_cache_stub() ||
Steve Blocka7e24c12009-10-30 11:49:00 +00004822 result == UNINITIALIZED ||
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004823 result == DEBUG_STUB);
Steve Blocka7e24c12009-10-30 11:49:00 +00004824 return result;
4825}
4826
4827
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004828ExtraICState Code::extra_ic_state() {
4829 DCHECK(is_inline_cache_stub() || ic_state() == DEBUG_STUB);
Ben Murdochb8e0da22011-05-16 14:20:40 +01004830 return ExtractExtraICStateFromFlags(flags());
4831}
4832
4833
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004834// For initialization.
4835void Code::set_raw_kind_specific_flags1(int value) {
4836 WRITE_INT_FIELD(this, kKindSpecificFlags1Offset, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00004837}
4838
4839
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004840void Code::set_raw_kind_specific_flags2(int value) {
4841 WRITE_INT_FIELD(this, kKindSpecificFlags2Offset, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00004842}
4843
4844
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004845inline bool Code::is_crankshafted() {
4846 return IsCrankshaftedField::decode(
4847 READ_UINT32_FIELD(this, kKindSpecificFlags2Offset));
Steve Blocka7e24c12009-10-30 11:49:00 +00004848}
4849
4850
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004851inline bool Code::is_hydrogen_stub() {
4852 return is_crankshafted() && kind() != OPTIMIZED_FUNCTION;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01004853}
4854
4855
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004856inline bool Code::is_interpreter_entry_trampoline() {
4857 Handle<Code> interpreter_entry =
4858 GetIsolate()->builtins()->InterpreterEntryTrampoline();
4859 return interpreter_entry.location() != nullptr && *interpreter_entry == this;
4860}
4861
Ben Murdoch097c5b22016-05-18 11:27:45 +01004862inline bool Code::is_interpreter_enter_bytecode_dispatch() {
4863 Handle<Code> interpreter_handler =
4864 GetIsolate()->builtins()->InterpreterEnterBytecodeDispatch();
4865 return interpreter_handler.location() != nullptr &&
4866 *interpreter_handler == this;
4867}
4868
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004869inline void Code::set_is_crankshafted(bool value) {
4870 int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
4871 int updated = IsCrankshaftedField::update(previous, value);
4872 WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
4873}
4874
4875
4876inline bool Code::is_turbofanned() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004877 return IsTurbofannedField::decode(
4878 READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
4879}
4880
4881
4882inline void Code::set_is_turbofanned(bool value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004883 int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
4884 int updated = IsTurbofannedField::update(previous, value);
4885 WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01004886}
4887
4888
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004889inline bool Code::can_have_weak_objects() {
4890 DCHECK(kind() == OPTIMIZED_FUNCTION);
4891 return CanHaveWeakObjectsField::decode(
4892 READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
Ben Murdochb0fe1622011-05-05 13:52:32 +01004893}
4894
4895
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004896inline void Code::set_can_have_weak_objects(bool value) {
4897 DCHECK(kind() == OPTIMIZED_FUNCTION);
4898 int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
4899 int updated = CanHaveWeakObjectsField::update(previous, value);
4900 WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
Ben Murdochb0fe1622011-05-05 13:52:32 +01004901}
4902
4903
4904bool Code::has_deoptimization_support() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004905 DCHECK_EQ(FUNCTION, kind());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004906 unsigned flags = READ_UINT32_FIELD(this, kFullCodeFlags);
Ben Murdoch589d6972011-11-30 16:04:58 +00004907 return FullCodeFlagsHasDeoptimizationSupportField::decode(flags);
Ben Murdochb0fe1622011-05-05 13:52:32 +01004908}
4909
4910
4911void Code::set_has_deoptimization_support(bool value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004912 DCHECK_EQ(FUNCTION, kind());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004913 unsigned flags = READ_UINT32_FIELD(this, kFullCodeFlags);
Ben Murdoch589d6972011-11-30 16:04:58 +00004914 flags = FullCodeFlagsHasDeoptimizationSupportField::update(flags, value);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004915 WRITE_UINT32_FIELD(this, kFullCodeFlags, flags);
Ben Murdoch589d6972011-11-30 16:04:58 +00004916}
4917
4918
4919bool Code::has_debug_break_slots() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004920 DCHECK_EQ(FUNCTION, kind());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004921 unsigned flags = READ_UINT32_FIELD(this, kFullCodeFlags);
Ben Murdoch589d6972011-11-30 16:04:58 +00004922 return FullCodeFlagsHasDebugBreakSlotsField::decode(flags);
4923}
4924
4925
4926void Code::set_has_debug_break_slots(bool value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004927 DCHECK_EQ(FUNCTION, kind());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004928 unsigned flags = READ_UINT32_FIELD(this, kFullCodeFlags);
Ben Murdoch589d6972011-11-30 16:04:58 +00004929 flags = FullCodeFlagsHasDebugBreakSlotsField::update(flags, value);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004930 WRITE_UINT32_FIELD(this, kFullCodeFlags, flags);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01004931}
4932
4933
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004934bool Code::has_reloc_info_for_serialization() {
4935 DCHECK_EQ(FUNCTION, kind());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004936 unsigned flags = READ_UINT32_FIELD(this, kFullCodeFlags);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004937 return FullCodeFlagsHasRelocInfoForSerialization::decode(flags);
4938}
4939
4940
4941void Code::set_has_reloc_info_for_serialization(bool value) {
4942 DCHECK_EQ(FUNCTION, kind());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004943 unsigned flags = READ_UINT32_FIELD(this, kFullCodeFlags);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004944 flags = FullCodeFlagsHasRelocInfoForSerialization::update(flags, value);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004945 WRITE_UINT32_FIELD(this, kFullCodeFlags, flags);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04004946}
4947
4948
Ben Murdochb0fe1622011-05-05 13:52:32 +01004949int Code::allow_osr_at_loop_nesting_level() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004950 DCHECK_EQ(FUNCTION, kind());
4951 int fields = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
4952 return AllowOSRAtLoopNestingLevelField::decode(fields);
Ben Murdochb0fe1622011-05-05 13:52:32 +01004953}
4954
4955
4956void Code::set_allow_osr_at_loop_nesting_level(int level) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004957 DCHECK_EQ(FUNCTION, kind());
4958 DCHECK(level >= 0 && level <= kMaxLoopNestingMarker);
4959 int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
4960 int updated = AllowOSRAtLoopNestingLevelField::update(previous, level);
4961 WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
Ben Murdochb0fe1622011-05-05 13:52:32 +01004962}
4963
4964
Ben Murdoch8f9999f2012-04-23 10:39:17 +01004965int Code::profiler_ticks() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004966 DCHECK_EQ(FUNCTION, kind());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004967 return ProfilerTicksField::decode(
4968 READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
Ben Murdoch8f9999f2012-04-23 10:39:17 +01004969}
4970
4971
4972void Code::set_profiler_ticks(int ticks) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004973 if (kind() == FUNCTION) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00004974 unsigned previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
4975 unsigned updated = ProfilerTicksField::update(previous, ticks);
4976 WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004977 }
4978}
4979
Ben Murdochda12d292016-06-02 14:46:10 +01004980int Code::builtin_index() { return READ_INT_FIELD(this, kBuiltinIndexOffset); }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004981
4982void Code::set_builtin_index(int index) {
Ben Murdochda12d292016-06-02 14:46:10 +01004983 WRITE_INT_FIELD(this, kBuiltinIndexOffset, index);
Ben Murdoch8f9999f2012-04-23 10:39:17 +01004984}
4985
4986
Ben Murdochb0fe1622011-05-05 13:52:32 +01004987unsigned Code::stack_slots() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004988 DCHECK(is_crankshafted());
4989 return StackSlotsField::decode(
4990 READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
Ben Murdochb0fe1622011-05-05 13:52:32 +01004991}
4992
4993
4994void Code::set_stack_slots(unsigned slots) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00004995 CHECK(slots <= (1 << kStackSlotsBitCount));
4996 DCHECK(is_crankshafted());
4997 int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
4998 int updated = StackSlotsField::update(previous, slots);
4999 WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
Ben Murdochb0fe1622011-05-05 13:52:32 +01005000}
5001
5002
Steve Block1e0659c2011-05-24 12:43:12 +01005003unsigned Code::safepoint_table_offset() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005004 DCHECK(is_crankshafted());
5005 return SafepointTableOffsetField::decode(
5006 READ_UINT32_FIELD(this, kKindSpecificFlags2Offset));
Ben Murdochb0fe1622011-05-05 13:52:32 +01005007}
5008
5009
Steve Block1e0659c2011-05-24 12:43:12 +01005010void Code::set_safepoint_table_offset(unsigned offset) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005011 CHECK(offset <= (1 << kSafepointTableOffsetBitCount));
5012 DCHECK(is_crankshafted());
5013 DCHECK(IsAligned(offset, static_cast<unsigned>(kIntSize)));
5014 int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
5015 int updated = SafepointTableOffsetField::update(previous, offset);
5016 WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
Ben Murdochb0fe1622011-05-05 13:52:32 +01005017}
5018
5019
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005020unsigned Code::back_edge_table_offset() {
5021 DCHECK_EQ(FUNCTION, kind());
5022 return BackEdgeTableOffsetField::decode(
5023 READ_UINT32_FIELD(this, kKindSpecificFlags2Offset)) << kPointerSizeLog2;
Ben Murdochb0fe1622011-05-05 13:52:32 +01005024}
5025
5026
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005027void Code::set_back_edge_table_offset(unsigned offset) {
5028 DCHECK_EQ(FUNCTION, kind());
5029 DCHECK(IsAligned(offset, static_cast<unsigned>(kPointerSize)));
5030 offset = offset >> kPointerSizeLog2;
5031 int previous = READ_UINT32_FIELD(this, kKindSpecificFlags2Offset);
5032 int updated = BackEdgeTableOffsetField::update(previous, offset);
5033 WRITE_UINT32_FIELD(this, kKindSpecificFlags2Offset, updated);
Ben Murdochb0fe1622011-05-05 13:52:32 +01005034}
5035
5036
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005037bool Code::back_edges_patched_for_osr() {
5038 DCHECK_EQ(FUNCTION, kind());
5039 return allow_osr_at_loop_nesting_level() > 0;
Ben Murdochb0fe1622011-05-05 13:52:32 +01005040}
5041
5042
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005043uint16_t Code::to_boolean_state() { return extra_ic_state(); }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005044
5045
5046bool Code::marked_for_deoptimization() {
5047 DCHECK(kind() == OPTIMIZED_FUNCTION);
5048 return MarkedForDeoptimizationField::decode(
5049 READ_UINT32_FIELD(this, kKindSpecificFlags1Offset));
5050}
5051
5052
5053void Code::set_marked_for_deoptimization(bool flag) {
5054 DCHECK(kind() == OPTIMIZED_FUNCTION);
5055 DCHECK(!flag || AllowDeoptimization::IsAllowed(GetIsolate()));
5056 int previous = READ_UINT32_FIELD(this, kKindSpecificFlags1Offset);
5057 int updated = MarkedForDeoptimizationField::update(previous, flag);
5058 WRITE_UINT32_FIELD(this, kKindSpecificFlags1Offset, updated);
5059}
5060
5061
Steve Blocka7e24c12009-10-30 11:49:00 +00005062bool Code::is_inline_cache_stub() {
5063 Kind kind = this->kind();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005064 switch (kind) {
5065#define CASE(name) case name: return true;
5066 IC_KIND_LIST(CASE)
5067#undef CASE
5068 default: return false;
5069 }
Steve Blocka7e24c12009-10-30 11:49:00 +00005070}
5071
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005072bool Code::is_debug_stub() { return ic_state() == DEBUG_STUB; }
5073bool Code::is_handler() { return kind() == HANDLER; }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005074bool Code::is_call_stub() { return kind() == CALL_IC; }
5075bool Code::is_binary_op_stub() { return kind() == BINARY_OP_IC; }
5076bool Code::is_compare_ic_stub() { return kind() == COMPARE_IC; }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005077bool Code::is_to_boolean_ic_stub() { return kind() == TO_BOOLEAN_IC; }
5078bool Code::is_optimized_code() { return kind() == OPTIMIZED_FUNCTION; }
Ben Murdochda12d292016-06-02 14:46:10 +01005079bool Code::is_wasm_code() { return kind() == WASM_FUNCTION; }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005080
5081bool Code::embeds_maps_weakly() {
5082 Kind k = kind();
5083 return (k == LOAD_IC || k == STORE_IC || k == KEYED_LOAD_IC ||
Ben Murdochda12d292016-06-02 14:46:10 +01005084 k == KEYED_STORE_IC) &&
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005085 ic_state() == MONOMORPHIC;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005086}
5087
5088
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005089Address Code::constant_pool() {
5090 Address constant_pool = NULL;
5091 if (FLAG_enable_embedded_constant_pool) {
5092 int offset = constant_pool_offset();
5093 if (offset < instruction_size()) {
5094 constant_pool = FIELD_ADDR(this, kHeaderSize + offset);
5095 }
5096 }
5097 return constant_pool;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005098}
5099
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005100Code::Flags Code::ComputeFlags(Kind kind, InlineCacheState ic_state,
Ben Murdochc5610432016-08-08 18:44:38 +01005101 ExtraICState extra_ic_state,
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005102 CacheHolderFlag holder) {
Steve Blocka7e24c12009-10-30 11:49:00 +00005103 // Compute the bit mask.
Ben Murdochda12d292016-06-02 14:46:10 +01005104 unsigned int bits = KindField::encode(kind) | ICStateField::encode(ic_state) |
Ben Murdochda12d292016-06-02 14:46:10 +01005105 ExtraICStateField::encode(extra_ic_state) |
5106 CacheHolderField::encode(holder);
Ben Murdoch589d6972011-11-30 16:04:58 +00005107 return static_cast<Flags>(bits);
Steve Blocka7e24c12009-10-30 11:49:00 +00005108}
5109
Steve Blocka7e24c12009-10-30 11:49:00 +00005110Code::Flags Code::ComputeMonomorphicFlags(Kind kind,
Ben Murdochb8e0da22011-05-16 14:20:40 +01005111 ExtraICState extra_ic_state,
Ben Murdochc5610432016-08-08 18:44:38 +01005112 CacheHolderFlag holder) {
5113 return ComputeFlags(kind, MONOMORPHIC, extra_ic_state, holder);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005114}
5115
Ben Murdochc5610432016-08-08 18:44:38 +01005116Code::Flags Code::ComputeHandlerFlags(Kind handler_kind,
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005117 CacheHolderFlag holder) {
Ben Murdochc5610432016-08-08 18:44:38 +01005118 return ComputeFlags(Code::HANDLER, MONOMORPHIC, handler_kind, holder);
Steve Blocka7e24c12009-10-30 11:49:00 +00005119}
5120
5121
5122Code::Kind Code::ExtractKindFromFlags(Flags flags) {
Ben Murdoch589d6972011-11-30 16:04:58 +00005123 return KindField::decode(flags);
Steve Blocka7e24c12009-10-30 11:49:00 +00005124}
5125
5126
5127InlineCacheState Code::ExtractICStateFromFlags(Flags flags) {
Ben Murdoch589d6972011-11-30 16:04:58 +00005128 return ICStateField::decode(flags);
Steve Blocka7e24c12009-10-30 11:49:00 +00005129}
5130
5131
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005132ExtraICState Code::ExtractExtraICStateFromFlags(Flags flags) {
Ben Murdoch589d6972011-11-30 16:04:58 +00005133 return ExtraICStateField::decode(flags);
Steve Blocka7e24c12009-10-30 11:49:00 +00005134}
5135
5136
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005137CacheHolderFlag Code::ExtractCacheHolderFromFlags(Flags flags) {
Ben Murdoch589d6972011-11-30 16:04:58 +00005138 return CacheHolderField::decode(flags);
Steve Block8defd9f2010-07-08 12:39:36 +01005139}
5140
Ben Murdochc5610432016-08-08 18:44:38 +01005141Code::Flags Code::RemoveHolderFromFlags(Flags flags) {
5142 int bits = flags & ~CacheHolderField::kMask;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005143 return static_cast<Flags>(bits);
5144}
5145
5146
Steve Blocka7e24c12009-10-30 11:49:00 +00005147Code* Code::GetCodeFromTargetAddress(Address address) {
5148 HeapObject* code = HeapObject::FromAddress(address - Code::kHeaderSize);
5149 // GetCodeFromTargetAddress might be called when marking objects during mark
5150 // sweep. reinterpret_cast is therefore used instead of the more appropriate
5151 // Code::cast. Code::cast does not work when the object's map is
5152 // marked.
5153 Code* result = reinterpret_cast<Code*>(code);
5154 return result;
5155}
5156
5157
Steve Block791712a2010-08-27 10:21:07 +01005158Object* Code::GetObjectFromEntryAddress(Address location_of_address) {
5159 return HeapObject::
5160 FromAddress(Memory::Address_at(location_of_address) - Code::kHeaderSize);
5161}
5162
5163
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005164bool Code::CanContainWeakObjects() {
5165 return is_optimized_code() && can_have_weak_objects();
5166}
5167
5168
5169bool Code::IsWeakObject(Object* object) {
5170 return (CanContainWeakObjects() && IsWeakObjectInOptimizedCode(object));
5171}
5172
5173
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005174bool Code::IsWeakObjectInOptimizedCode(Object* object) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005175 if (object->IsMap()) {
5176 return Map::cast(object)->CanTransition() &&
5177 FLAG_weak_embedded_maps_in_optimized_code;
5178 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005179 if (object->IsCell()) {
5180 object = Cell::cast(object)->value();
5181 } else if (object->IsPropertyCell()) {
5182 object = PropertyCell::cast(object)->value();
5183 }
5184 if (object->IsJSReceiver()) {
5185 return FLAG_weak_embedded_objects_in_optimized_code;
5186 }
5187 if (object->IsContext()) {
5188 // Contexts of inlined functions are embedded in optimized code.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005189 return FLAG_weak_embedded_objects_in_optimized_code;
5190 }
5191 return false;
5192}
5193
5194
5195class Code::FindAndReplacePattern {
5196 public:
5197 FindAndReplacePattern() : count_(0) { }
5198 void Add(Handle<Map> map_to_find, Handle<Object> obj_to_replace) {
5199 DCHECK(count_ < kMaxCount);
5200 find_[count_] = map_to_find;
5201 replace_[count_] = obj_to_replace;
5202 ++count_;
5203 }
5204 private:
5205 static const int kMaxCount = 4;
5206 int count_;
5207 Handle<Map> find_[kMaxCount];
5208 Handle<Object> replace_[kMaxCount];
5209 friend class Code;
5210};
5211
Ben Murdochda12d292016-06-02 14:46:10 +01005212int AbstractCode::instruction_size() {
Ben Murdoch097c5b22016-05-18 11:27:45 +01005213 if (IsCode()) {
5214 return GetCode()->instruction_size();
5215 } else {
5216 return GetBytecodeArray()->length();
5217 }
5218}
5219
Ben Murdochda12d292016-06-02 14:46:10 +01005220int AbstractCode::ExecutableSize() {
5221 if (IsCode()) {
5222 return GetCode()->ExecutableSize();
5223 } else {
5224 return GetBytecodeArray()->BytecodeArraySize();
5225 }
5226}
5227
5228Address AbstractCode::instruction_start() {
5229 if (IsCode()) {
5230 return GetCode()->instruction_start();
5231 } else {
5232 return GetBytecodeArray()->GetFirstBytecodeAddress();
5233 }
5234}
5235
5236Address AbstractCode::instruction_end() {
5237 if (IsCode()) {
5238 return GetCode()->instruction_end();
5239 } else {
5240 return GetBytecodeArray()->GetFirstBytecodeAddress() +
5241 GetBytecodeArray()->length();
5242 }
5243}
5244
5245bool AbstractCode::contains(byte* inner_pointer) {
5246 return (address() <= inner_pointer) && (inner_pointer <= address() + Size());
5247}
5248
5249AbstractCode::Kind AbstractCode::kind() {
5250 if (IsCode()) {
5251 STATIC_ASSERT(AbstractCode::FUNCTION ==
5252 static_cast<AbstractCode::Kind>(Code::FUNCTION));
5253 return static_cast<AbstractCode::Kind>(GetCode()->kind());
5254 } else {
5255 return INTERPRETED_FUNCTION;
5256 }
5257}
5258
Ben Murdoch097c5b22016-05-18 11:27:45 +01005259Code* AbstractCode::GetCode() { return Code::cast(this); }
5260
5261BytecodeArray* AbstractCode::GetBytecodeArray() {
5262 return BytecodeArray::cast(this);
5263}
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005264
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005265Object* Map::prototype() const {
Steve Blocka7e24c12009-10-30 11:49:00 +00005266 return READ_FIELD(this, kPrototypeOffset);
5267}
5268
5269
5270void Map::set_prototype(Object* value, WriteBarrierMode mode) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005271 DCHECK(value->IsNull() || value->IsJSReceiver());
Steve Blocka7e24c12009-10-30 11:49:00 +00005272 WRITE_FIELD(this, kPrototypeOffset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005273 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kPrototypeOffset, value, mode);
Steve Block1e0659c2011-05-24 12:43:12 +01005274}
5275
5276
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005277LayoutDescriptor* Map::layout_descriptor_gc_safe() {
5278 Object* layout_desc = READ_FIELD(this, kLayoutDecriptorOffset);
5279 return LayoutDescriptor::cast_gc_safe(layout_desc);
5280}
5281
5282
5283bool Map::HasFastPointerLayout() const {
5284 Object* layout_desc = READ_FIELD(this, kLayoutDecriptorOffset);
5285 return LayoutDescriptor::IsFastPointerLayout(layout_desc);
5286}
5287
5288
5289void Map::UpdateDescriptors(DescriptorArray* descriptors,
5290 LayoutDescriptor* layout_desc) {
5291 set_instance_descriptors(descriptors);
5292 if (FLAG_unbox_double_fields) {
5293 if (layout_descriptor()->IsSlowLayout()) {
5294 set_layout_descriptor(layout_desc);
5295 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005296#ifdef VERIFY_HEAP
5297 // TODO(ishell): remove these checks from VERIFY_HEAP mode.
5298 if (FLAG_verify_heap) {
5299 CHECK(layout_descriptor()->IsConsistentWithMap(this));
5300 CHECK(visitor_id() == Heap::GetStaticVisitorIdForMap(this));
5301 }
5302#else
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005303 SLOW_DCHECK(layout_descriptor()->IsConsistentWithMap(this));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005304 DCHECK(visitor_id() == Heap::GetStaticVisitorIdForMap(this));
5305#endif
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005306 }
5307}
5308
5309
5310void Map::InitializeDescriptors(DescriptorArray* descriptors,
5311 LayoutDescriptor* layout_desc) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005312 int len = descriptors->number_of_descriptors();
5313 set_instance_descriptors(descriptors);
5314 SetNumberOfOwnDescriptors(len);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005315
5316 if (FLAG_unbox_double_fields) {
5317 set_layout_descriptor(layout_desc);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005318#ifdef VERIFY_HEAP
5319 // TODO(ishell): remove these checks from VERIFY_HEAP mode.
5320 if (FLAG_verify_heap) {
5321 CHECK(layout_descriptor()->IsConsistentWithMap(this));
5322 }
5323#else
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005324 SLOW_DCHECK(layout_descriptor()->IsConsistentWithMap(this));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005325#endif
5326 set_visitor_id(Heap::GetStaticVisitorIdForMap(this));
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005327 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005328}
5329
5330
5331ACCESSORS(Map, instance_descriptors, DescriptorArray, kDescriptorsOffset)
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005332ACCESSORS(Map, layout_descriptor, LayoutDescriptor, kLayoutDecriptorOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005333
5334
5335void Map::set_bit_field3(uint32_t bits) {
5336 if (kInt32Size != kPointerSize) {
5337 WRITE_UINT32_FIELD(this, kBitField3Offset + kInt32Size, 0);
5338 }
5339 WRITE_UINT32_FIELD(this, kBitField3Offset, bits);
5340}
5341
5342
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005343uint32_t Map::bit_field3() const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005344 return READ_UINT32_FIELD(this, kBitField3Offset);
5345}
5346
5347
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005348LayoutDescriptor* Map::GetLayoutDescriptor() {
5349 return FLAG_unbox_double_fields ? layout_descriptor()
5350 : LayoutDescriptor::FastPointerLayout();
5351}
5352
5353
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005354void Map::AppendDescriptor(Descriptor* desc) {
5355 DescriptorArray* descriptors = instance_descriptors();
5356 int number_of_own_descriptors = NumberOfOwnDescriptors();
5357 DCHECK(descriptors->number_of_descriptors() == number_of_own_descriptors);
5358 descriptors->Append(desc);
5359 SetNumberOfOwnDescriptors(number_of_own_descriptors + 1);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005360
5361// This function does not support appending double field descriptors and
5362// it should never try to (otherwise, layout descriptor must be updated too).
5363#ifdef DEBUG
5364 PropertyDetails details = desc->GetDetails();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005365 CHECK(details.type() != DATA || !details.representation().IsDouble());
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005366#endif
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005367}
5368
5369
5370Object* Map::GetBackPointer() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005371 Object* object = constructor_or_backpointer();
5372 if (object->IsMap()) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005373 return object;
5374 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005375 return GetIsolate()->heap()->undefined_value();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005376}
5377
5378
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005379Map* Map::ElementsTransitionMap() {
5380 return TransitionArray::SearchSpecial(
5381 this, GetHeap()->elements_transition_symbol());
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005382}
5383
5384
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005385ACCESSORS(Map, raw_transitions, Object, kTransitionsOrPrototypeInfoOffset)
5386
5387
5388Object* Map::prototype_info() const {
5389 DCHECK(is_prototype_map());
5390 return READ_FIELD(this, Map::kTransitionsOrPrototypeInfoOffset);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005391}
5392
5393
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005394void Map::set_prototype_info(Object* value, WriteBarrierMode mode) {
5395 DCHECK(is_prototype_map());
5396 WRITE_FIELD(this, Map::kTransitionsOrPrototypeInfoOffset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005397 CONDITIONAL_WRITE_BARRIER(
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005398 GetHeap(), this, Map::kTransitionsOrPrototypeInfoOffset, value, mode);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005399}
5400
5401
5402void Map::SetBackPointer(Object* value, WriteBarrierMode mode) {
5403 DCHECK(instance_type() >= FIRST_JS_RECEIVER_TYPE);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005404 DCHECK((value->IsMap() && GetBackPointer()->IsUndefined()));
5405 DCHECK(!value->IsMap() ||
5406 Map::cast(value)->GetConstructor() == constructor_or_backpointer());
5407 set_constructor_or_backpointer(value, mode);
Ben Murdoch257744e2011-11-30 15:57:28 +00005408}
5409
5410
Steve Block6ded16b2010-05-10 14:33:55 +01005411ACCESSORS(Map, code_cache, Object, kCodeCacheOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005412ACCESSORS(Map, dependent_code, DependentCode, kDependentCodeOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005413ACCESSORS(Map, weak_cell_cache, Object, kWeakCellCacheOffset)
5414ACCESSORS(Map, constructor_or_backpointer, Object,
5415 kConstructorOrBackPointerOffset)
5416
5417
5418Object* Map::GetConstructor() const {
5419 Object* maybe_constructor = constructor_or_backpointer();
5420 // Follow any back pointers.
5421 while (maybe_constructor->IsMap()) {
5422 maybe_constructor =
5423 Map::cast(maybe_constructor)->constructor_or_backpointer();
5424 }
5425 return maybe_constructor;
5426}
5427
5428
5429void Map::SetConstructor(Object* constructor, WriteBarrierMode mode) {
5430 // Never overwrite a back pointer with a constructor.
5431 DCHECK(!constructor_or_backpointer()->IsMap());
5432 set_constructor_or_backpointer(constructor, mode);
5433}
5434
5435
5436Handle<Map> Map::CopyInitialMap(Handle<Map> map) {
5437 return CopyInitialMap(map, map->instance_size(), map->GetInObjectProperties(),
5438 map->unused_property_fields());
5439}
5440
5441
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005442ACCESSORS(JSBoundFunction, bound_target_function, JSReceiver,
5443 kBoundTargetFunctionOffset)
5444ACCESSORS(JSBoundFunction, bound_this, Object, kBoundThisOffset)
5445ACCESSORS(JSBoundFunction, bound_arguments, FixedArray, kBoundArgumentsOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005446
5447ACCESSORS(JSFunction, shared, SharedFunctionInfo, kSharedFunctionInfoOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005448ACCESSORS(JSFunction, literals, LiteralsArray, kLiteralsOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005449ACCESSORS(JSFunction, next_function_link, Object, kNextFunctionLinkOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005450
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005451ACCESSORS(JSGlobalObject, native_context, Context, kNativeContextOffset)
5452ACCESSORS(JSGlobalObject, global_proxy, JSObject, kGlobalProxyOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005453
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005454ACCESSORS(JSGlobalProxy, native_context, Object, kNativeContextOffset)
5455ACCESSORS(JSGlobalProxy, hash, Object, kHashOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005456
Steve Blocka7e24c12009-10-30 11:49:00 +00005457ACCESSORS(AccessorInfo, name, Object, kNameOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005458SMI_ACCESSORS(AccessorInfo, flag, kFlagOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005459ACCESSORS(AccessorInfo, expected_receiver_type, Object,
5460 kExpectedReceiverTypeOffset)
5461
Ben Murdoch097c5b22016-05-18 11:27:45 +01005462ACCESSORS(AccessorInfo, getter, Object, kGetterOffset)
5463ACCESSORS(AccessorInfo, setter, Object, kSetterOffset)
Ben Murdochc5610432016-08-08 18:44:38 +01005464ACCESSORS(AccessorInfo, js_getter, Object, kJsGetterOffset)
Ben Murdoch097c5b22016-05-18 11:27:45 +01005465ACCESSORS(AccessorInfo, data, Object, kDataOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005466
5467ACCESSORS(Box, value, Object, kValueOffset)
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005468
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005469ACCESSORS(PrototypeInfo, prototype_users, Object, kPrototypeUsersOffset)
5470SMI_ACCESSORS(PrototypeInfo, registry_slot, kRegistrySlotOffset)
5471ACCESSORS(PrototypeInfo, validity_cell, Object, kValidityCellOffset)
5472
5473ACCESSORS(SloppyBlockWithEvalContextExtension, scope_info, ScopeInfo,
5474 kScopeInfoOffset)
5475ACCESSORS(SloppyBlockWithEvalContextExtension, extension, JSObject,
5476 kExtensionOffset)
5477
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005478ACCESSORS(AccessorPair, getter, Object, kGetterOffset)
5479ACCESSORS(AccessorPair, setter, Object, kSetterOffset)
Ben Murdochc7cc0282012-03-05 14:35:55 +00005480
Steve Blocka7e24c12009-10-30 11:49:00 +00005481ACCESSORS(AccessCheckInfo, named_callback, Object, kNamedCallbackOffset)
5482ACCESSORS(AccessCheckInfo, indexed_callback, Object, kIndexedCallbackOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005483ACCESSORS(AccessCheckInfo, callback, Object, kCallbackOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005484ACCESSORS(AccessCheckInfo, data, Object, kDataOffset)
5485
5486ACCESSORS(InterceptorInfo, getter, Object, kGetterOffset)
5487ACCESSORS(InterceptorInfo, setter, Object, kSetterOffset)
5488ACCESSORS(InterceptorInfo, query, Object, kQueryOffset)
5489ACCESSORS(InterceptorInfo, deleter, Object, kDeleterOffset)
5490ACCESSORS(InterceptorInfo, enumerator, Object, kEnumeratorOffset)
5491ACCESSORS(InterceptorInfo, data, Object, kDataOffset)
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005492SMI_ACCESSORS(InterceptorInfo, flags, kFlagsOffset)
5493BOOL_ACCESSORS(InterceptorInfo, flags, can_intercept_symbols,
5494 kCanInterceptSymbolsBit)
5495BOOL_ACCESSORS(InterceptorInfo, flags, all_can_read, kAllCanReadBit)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005496BOOL_ACCESSORS(InterceptorInfo, flags, non_masking, kNonMasking)
Steve Blocka7e24c12009-10-30 11:49:00 +00005497
5498ACCESSORS(CallHandlerInfo, callback, Object, kCallbackOffset)
5499ACCESSORS(CallHandlerInfo, data, Object, kDataOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005500ACCESSORS(CallHandlerInfo, fast_handler, Object, kFastHandlerOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005501
5502ACCESSORS(TemplateInfo, tag, Object, kTagOffset)
Ben Murdoch097c5b22016-05-18 11:27:45 +01005503ACCESSORS(TemplateInfo, serial_number, Object, kSerialNumberOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005504SMI_ACCESSORS(TemplateInfo, number_of_properties, kNumberOfProperties)
Steve Blocka7e24c12009-10-30 11:49:00 +00005505ACCESSORS(TemplateInfo, property_list, Object, kPropertyListOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005506ACCESSORS(TemplateInfo, property_accessors, Object, kPropertyAccessorsOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005507
Steve Blocka7e24c12009-10-30 11:49:00 +00005508ACCESSORS(FunctionTemplateInfo, call_code, Object, kCallCodeOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005509ACCESSORS(FunctionTemplateInfo, prototype_template, Object,
5510 kPrototypeTemplateOffset)
5511ACCESSORS(FunctionTemplateInfo, parent_template, Object, kParentTemplateOffset)
5512ACCESSORS(FunctionTemplateInfo, named_property_handler, Object,
5513 kNamedPropertyHandlerOffset)
5514ACCESSORS(FunctionTemplateInfo, indexed_property_handler, Object,
5515 kIndexedPropertyHandlerOffset)
5516ACCESSORS(FunctionTemplateInfo, instance_template, Object,
5517 kInstanceTemplateOffset)
5518ACCESSORS(FunctionTemplateInfo, class_name, Object, kClassNameOffset)
5519ACCESSORS(FunctionTemplateInfo, signature, Object, kSignatureOffset)
5520ACCESSORS(FunctionTemplateInfo, instance_call_handler, Object,
5521 kInstanceCallHandlerOffset)
5522ACCESSORS(FunctionTemplateInfo, access_check_info, Object,
5523 kAccessCheckInfoOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005524SMI_ACCESSORS(FunctionTemplateInfo, flag, kFlagOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005525
5526ACCESSORS(ObjectTemplateInfo, constructor, Object, kConstructorOffset)
5527ACCESSORS(ObjectTemplateInfo, internal_field_count, Object,
5528 kInternalFieldCountOffset)
5529
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005530ACCESSORS(AllocationSite, transition_info, Object, kTransitionInfoOffset)
5531ACCESSORS(AllocationSite, nested_site, Object, kNestedSiteOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005532SMI_ACCESSORS(AllocationSite, pretenure_data, kPretenureDataOffset)
5533SMI_ACCESSORS(AllocationSite, pretenure_create_count,
5534 kPretenureCreateCountOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005535ACCESSORS(AllocationSite, dependent_code, DependentCode,
5536 kDependentCodeOffset)
5537ACCESSORS(AllocationSite, weak_next, Object, kWeakNextOffset)
5538ACCESSORS(AllocationMemento, allocation_site, Object, kAllocationSiteOffset)
5539
Steve Blocka7e24c12009-10-30 11:49:00 +00005540ACCESSORS(Script, source, Object, kSourceOffset)
5541ACCESSORS(Script, name, Object, kNameOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005542SMI_ACCESSORS(Script, id, kIdOffset)
5543SMI_ACCESSORS(Script, line_offset, kLineOffsetOffset)
5544SMI_ACCESSORS(Script, column_offset, kColumnOffsetOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005545ACCESSORS(Script, context_data, Object, kContextOffset)
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005546ACCESSORS(Script, wrapper, HeapObject, kWrapperOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005547SMI_ACCESSORS(Script, type, kTypeOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005548ACCESSORS(Script, line_ends, Object, kLineEndsOffset)
Steve Blockd0582a62009-12-15 09:54:21 +00005549ACCESSORS(Script, eval_from_shared, Object, kEvalFromSharedOffset)
Ben Murdochc5610432016-08-08 18:44:38 +01005550SMI_ACCESSORS(Script, eval_from_position, kEvalFromPositionOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005551ACCESSORS(Script, shared_function_infos, Object, kSharedFunctionInfosOffset)
5552SMI_ACCESSORS(Script, flags, kFlagsOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005553ACCESSORS(Script, source_url, Object, kSourceUrlOffset)
5554ACCESSORS(Script, source_mapping_url, Object, kSourceMappingUrlOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005555
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005556Script::CompilationType Script::compilation_type() {
5557 return BooleanBit::get(flags(), kCompilationTypeBit) ?
5558 COMPILATION_TYPE_EVAL : COMPILATION_TYPE_HOST;
5559}
5560void Script::set_compilation_type(CompilationType type) {
5561 set_flags(BooleanBit::set(flags(), kCompilationTypeBit,
5562 type == COMPILATION_TYPE_EVAL));
5563}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005564bool Script::hide_source() { return BooleanBit::get(flags(), kHideSourceBit); }
5565void Script::set_hide_source(bool value) {
5566 set_flags(BooleanBit::set(flags(), kHideSourceBit, value));
5567}
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005568Script::CompilationState Script::compilation_state() {
5569 return BooleanBit::get(flags(), kCompilationStateBit) ?
5570 COMPILATION_STATE_COMPILED : COMPILATION_STATE_INITIAL;
5571}
5572void Script::set_compilation_state(CompilationState state) {
5573 set_flags(BooleanBit::set(flags(), kCompilationStateBit,
5574 state == COMPILATION_STATE_COMPILED));
5575}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005576ScriptOriginOptions Script::origin_options() {
5577 return ScriptOriginOptions((flags() & kOriginOptionsMask) >>
5578 kOriginOptionsShift);
5579}
5580void Script::set_origin_options(ScriptOriginOptions origin_options) {
5581 DCHECK(!(origin_options.Flags() & ~((1 << kOriginOptionsSize) - 1)));
5582 set_flags((flags() & ~kOriginOptionsMask) |
5583 (origin_options.Flags() << kOriginOptionsShift));
5584}
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005585
5586
Steve Blocka7e24c12009-10-30 11:49:00 +00005587ACCESSORS(DebugInfo, shared, SharedFunctionInfo, kSharedFunctionInfoIndex)
Ben Murdoch097c5b22016-05-18 11:27:45 +01005588ACCESSORS(DebugInfo, abstract_code, AbstractCode, kAbstractCodeIndex)
Steve Blocka7e24c12009-10-30 11:49:00 +00005589ACCESSORS(DebugInfo, break_points, FixedArray, kBreakPointsStateIndex)
5590
Ben Murdoch097c5b22016-05-18 11:27:45 +01005591BytecodeArray* DebugInfo::original_bytecode_array() {
5592 return shared()->bytecode_array();
5593}
5594
5595SMI_ACCESSORS(BreakPointInfo, code_offset, kCodeOffsetIndex)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005596SMI_ACCESSORS(BreakPointInfo, source_position, kSourcePositionIndex)
5597SMI_ACCESSORS(BreakPointInfo, statement_position, kStatementPositionIndex)
Steve Blocka7e24c12009-10-30 11:49:00 +00005598ACCESSORS(BreakPointInfo, break_point_objects, Object, kBreakPointObjectsIndex)
Steve Blocka7e24c12009-10-30 11:49:00 +00005599
Steve Blocka7e24c12009-10-30 11:49:00 +00005600ACCESSORS(SharedFunctionInfo, name, Object, kNameOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005601ACCESSORS(SharedFunctionInfo, optimized_code_map, FixedArray,
5602 kOptimizedCodeMapOffset)
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005603ACCESSORS(SharedFunctionInfo, construct_stub, Code, kConstructStubOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005604ACCESSORS(SharedFunctionInfo, feedback_vector, TypeFeedbackVector,
5605 kFeedbackVectorOffset)
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005606#if TRACE_MAPS
5607SMI_ACCESSORS(SharedFunctionInfo, unique_id, kUniqueIdOffset)
5608#endif
Steve Blocka7e24c12009-10-30 11:49:00 +00005609ACCESSORS(SharedFunctionInfo, instance_class_name, Object,
5610 kInstanceClassNameOffset)
Steve Block6ded16b2010-05-10 14:33:55 +01005611ACCESSORS(SharedFunctionInfo, function_data, Object, kFunctionDataOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005612ACCESSORS(SharedFunctionInfo, script, Object, kScriptOffset)
5613ACCESSORS(SharedFunctionInfo, debug_info, Object, kDebugInfoOffset)
Ben Murdochda12d292016-06-02 14:46:10 +01005614ACCESSORS(SharedFunctionInfo, function_identifier, Object,
5615 kFunctionIdentifierOffset)
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005616
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005617SMI_ACCESSORS(FunctionTemplateInfo, length, kLengthOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00005618BOOL_ACCESSORS(FunctionTemplateInfo, flag, hidden_prototype,
5619 kHiddenPrototypeBit)
5620BOOL_ACCESSORS(FunctionTemplateInfo, flag, undetectable, kUndetectableBit)
5621BOOL_ACCESSORS(FunctionTemplateInfo, flag, needs_access_check,
5622 kNeedsAccessCheckBit)
Ben Murdoch69a99ed2011-11-30 16:03:39 +00005623BOOL_ACCESSORS(FunctionTemplateInfo, flag, read_only_prototype,
5624 kReadOnlyPrototypeBit)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005625BOOL_ACCESSORS(FunctionTemplateInfo, flag, remove_prototype,
5626 kRemovePrototypeBit)
5627BOOL_ACCESSORS(FunctionTemplateInfo, flag, do_not_cache,
5628 kDoNotCacheBit)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005629BOOL_ACCESSORS(FunctionTemplateInfo, flag, instantiated, kInstantiatedBit)
5630BOOL_ACCESSORS(FunctionTemplateInfo, flag, accept_any_receiver,
5631 kAcceptAnyReceiver)
Ben Murdoch097c5b22016-05-18 11:27:45 +01005632BOOL_ACCESSORS(SharedFunctionInfo, start_position_and_type, is_named_expression,
5633 kIsNamedExpressionBit)
Steve Blocka7e24c12009-10-30 11:49:00 +00005634BOOL_ACCESSORS(SharedFunctionInfo, start_position_and_type, is_toplevel,
5635 kIsTopLevelBit)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005636
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005637BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, allows_lazy_compilation,
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005638 kAllowLazyCompilation)
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00005639BOOL_ACCESSORS(SharedFunctionInfo,
5640 compiler_hints,
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005641 allows_lazy_compilation_without_context,
5642 kAllowLazyCompilationWithoutContext)
5643BOOL_ACCESSORS(SharedFunctionInfo,
5644 compiler_hints,
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00005645 uses_arguments,
5646 kUsesArguments)
5647BOOL_ACCESSORS(SharedFunctionInfo,
5648 compiler_hints,
5649 has_duplicate_parameters,
5650 kHasDuplicateParameters)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005651BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, asm_function, kIsAsmFunction)
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005652BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, deserialized, kDeserialized)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005653BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, never_compiled,
5654 kNeverCompiled)
Ben Murdoch097c5b22016-05-18 11:27:45 +01005655BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_declaration,
5656 kIsDeclaration)
Iain Merrick75681382010-08-19 15:07:18 +01005657
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005658#if V8_HOST_ARCH_32_BIT
5659SMI_ACCESSORS(SharedFunctionInfo, length, kLengthOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005660SMI_ACCESSORS(SharedFunctionInfo, internal_formal_parameter_count,
Steve Blocka7e24c12009-10-30 11:49:00 +00005661 kFormalParameterCountOffset)
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005662SMI_ACCESSORS(SharedFunctionInfo, expected_nof_properties,
Steve Blocka7e24c12009-10-30 11:49:00 +00005663 kExpectedNofPropertiesOffset)
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005664SMI_ACCESSORS(SharedFunctionInfo, num_literals, kNumLiteralsOffset)
5665SMI_ACCESSORS(SharedFunctionInfo, start_position_and_type,
Steve Blocka7e24c12009-10-30 11:49:00 +00005666 kStartPositionAndTypeOffset)
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005667SMI_ACCESSORS(SharedFunctionInfo, end_position, kEndPositionOffset)
5668SMI_ACCESSORS(SharedFunctionInfo, function_token_position,
Steve Blocka7e24c12009-10-30 11:49:00 +00005669 kFunctionTokenPositionOffset)
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005670SMI_ACCESSORS(SharedFunctionInfo, compiler_hints,
Steve Blocka7e24c12009-10-30 11:49:00 +00005671 kCompilerHintsOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005672SMI_ACCESSORS(SharedFunctionInfo, opt_count_and_bailout_reason,
5673 kOptCountAndBailoutReasonOffset)
5674SMI_ACCESSORS(SharedFunctionInfo, counters, kCountersOffset)
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005675SMI_ACCESSORS(SharedFunctionInfo, ast_node_count, kAstNodeCountOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005676SMI_ACCESSORS(SharedFunctionInfo, profiler_ticks, kProfilerTicksOffset)
5677
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005678#else
Steve Blocka7e24c12009-10-30 11:49:00 +00005679
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005680#if V8_TARGET_LITTLE_ENDIAN
5681#define PSEUDO_SMI_LO_ALIGN 0
5682#define PSEUDO_SMI_HI_ALIGN kIntSize
5683#else
5684#define PSEUDO_SMI_LO_ALIGN kIntSize
5685#define PSEUDO_SMI_HI_ALIGN 0
5686#endif
5687
5688#define PSEUDO_SMI_ACCESSORS_LO(holder, name, offset) \
5689 STATIC_ASSERT(holder::offset % kPointerSize == PSEUDO_SMI_LO_ALIGN); \
5690 int holder::name() const { \
5691 int value = READ_INT_FIELD(this, offset); \
5692 DCHECK(kHeapObjectTag == 1); \
5693 DCHECK((value & kHeapObjectTag) == 0); \
5694 return value >> 1; \
5695 } \
5696 void holder::set_##name(int value) { \
5697 DCHECK(kHeapObjectTag == 1); \
5698 DCHECK((value & 0xC0000000) == 0xC0000000 || (value & 0xC0000000) == 0x0); \
5699 WRITE_INT_FIELD(this, offset, (value << 1) & ~kHeapObjectTag); \
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005700 }
5701
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005702#define PSEUDO_SMI_ACCESSORS_HI(holder, name, offset) \
5703 STATIC_ASSERT(holder::offset % kPointerSize == PSEUDO_SMI_HI_ALIGN); \
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005704 INT_ACCESSORS(holder, name, offset)
5705
5706
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005707PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo, length, kLengthOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005708PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo, internal_formal_parameter_count,
Teng-Hui Zhu3e5fa292010-11-09 16:16:48 -08005709 kFormalParameterCountOffset)
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005710
Teng-Hui Zhu3e5fa292010-11-09 16:16:48 -08005711PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo,
5712 expected_nof_properties,
5713 kExpectedNofPropertiesOffset)
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005714PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo, num_literals, kNumLiteralsOffset)
5715
Teng-Hui Zhu3e5fa292010-11-09 16:16:48 -08005716PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo, end_position, kEndPositionOffset)
5717PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo,
5718 start_position_and_type,
5719 kStartPositionAndTypeOffset)
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005720
Teng-Hui Zhu3e5fa292010-11-09 16:16:48 -08005721PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo,
5722 function_token_position,
5723 kFunctionTokenPositionOffset)
5724PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo,
5725 compiler_hints,
5726 kCompilerHintsOffset)
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005727
Teng-Hui Zhu3e5fa292010-11-09 16:16:48 -08005728PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo,
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005729 opt_count_and_bailout_reason,
5730 kOptCountAndBailoutReasonOffset)
5731PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo, counters, kCountersOffset)
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005732
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005733PSEUDO_SMI_ACCESSORS_LO(SharedFunctionInfo,
5734 ast_node_count,
5735 kAstNodeCountOffset)
5736PSEUDO_SMI_ACCESSORS_HI(SharedFunctionInfo,
5737 profiler_ticks,
5738 kProfilerTicksOffset)
5739
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01005740#endif
Steve Blocka7e24c12009-10-30 11:49:00 +00005741
Kristian Monsen0d5e1162010-09-30 15:31:59 +01005742
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00005743BOOL_GETTER(SharedFunctionInfo,
5744 compiler_hints,
5745 optimization_disabled,
5746 kOptimizationDisabled)
Ben Murdochb0fe1622011-05-05 13:52:32 +01005747
Ben Murdochda12d292016-06-02 14:46:10 +01005748AbstractCode* SharedFunctionInfo::abstract_code() {
5749 if (HasBytecodeArray()) {
5750 return AbstractCode::cast(bytecode_array());
5751 } else {
5752 return AbstractCode::cast(code());
5753 }
5754}
Ben Murdochb0fe1622011-05-05 13:52:32 +01005755
5756void SharedFunctionInfo::set_optimization_disabled(bool disable) {
5757 set_compiler_hints(BooleanBit::set(compiler_hints(),
5758 kOptimizationDisabled,
5759 disable));
Ben Murdochb0fe1622011-05-05 13:52:32 +01005760}
5761
5762
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005763LanguageMode SharedFunctionInfo::language_mode() {
5764 STATIC_ASSERT(LANGUAGE_END == 3);
5765 return construct_language_mode(
Ben Murdochda12d292016-06-02 14:46:10 +01005766 BooleanBit::get(compiler_hints(), kStrictModeFunction));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005767}
5768
5769
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005770void SharedFunctionInfo::set_language_mode(LanguageMode language_mode) {
5771 STATIC_ASSERT(LANGUAGE_END == 3);
5772 // We only allow language mode transitions that set the same language mode
5773 // again or go up in the chain:
5774 DCHECK(is_sloppy(this->language_mode()) || is_strict(language_mode));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005775 int hints = compiler_hints();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005776 hints = BooleanBit::set(hints, kStrictModeFunction, is_strict(language_mode));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005777 set_compiler_hints(hints);
5778}
5779
5780
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005781FunctionKind SharedFunctionInfo::kind() {
5782 return FunctionKindBits::decode(compiler_hints());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005783}
5784
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005785
5786void SharedFunctionInfo::set_kind(FunctionKind kind) {
5787 DCHECK(IsValidFunctionKind(kind));
5788 int hints = compiler_hints();
5789 hints = FunctionKindBits::update(hints, kind);
5790 set_compiler_hints(hints);
5791}
5792
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005793BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, needs_home_object,
5794 kNeedsHomeObject)
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00005795BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, native, kNative)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005796BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, force_inline, kForceInline)
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00005797BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints,
5798 name_should_print_as_anonymous,
5799 kNameShouldPrintAsAnonymous)
Ben Murdoch097c5b22016-05-18 11:27:45 +01005800BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_anonymous_expression,
5801 kIsAnonymousExpression)
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005802BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_function, kIsFunction)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005803BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, dont_crankshaft,
5804 kDontCrankshaft)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005805BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, dont_flush, kDontFlush)
5806BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_arrow, kIsArrow)
5807BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_generator, kIsGenerator)
Ben Murdochc5610432016-08-08 18:44:38 +01005808BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_async, kIsAsyncFunction)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005809BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_concise_method,
5810 kIsConciseMethod)
Ben Murdoch097c5b22016-05-18 11:27:45 +01005811BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_getter_function,
5812 kIsGetterFunction)
5813BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_setter_function,
5814 kIsSetterFunction)
Emily Bernierd0a1eb72015-03-24 16:35:39 -04005815BOOL_ACCESSORS(SharedFunctionInfo, compiler_hints, is_default_constructor,
5816 kIsDefaultConstructor)
Ben Murdoch257744e2011-11-30 15:57:28 +00005817
Ben Murdochc5610432016-08-08 18:44:38 +01005818inline bool SharedFunctionInfo::is_resumable() const {
5819 return is_generator() || is_async();
5820}
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00005821
Steve Block3ce2e202009-11-05 08:53:23 +00005822bool Script::HasValidSource() {
5823 Object* src = this->source();
5824 if (!src->IsString()) return true;
5825 String* src_str = String::cast(src);
5826 if (!StringShape(src_str).IsExternal()) return true;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005827 if (src_str->IsOneByteRepresentation()) {
5828 return ExternalOneByteString::cast(src)->resource() != NULL;
Steve Block3ce2e202009-11-05 08:53:23 +00005829 } else if (src_str->IsTwoByteRepresentation()) {
5830 return ExternalTwoByteString::cast(src)->resource() != NULL;
5831 }
5832 return true;
5833}
5834
5835
Steve Blocka7e24c12009-10-30 11:49:00 +00005836void SharedFunctionInfo::DontAdaptArguments() {
Ben Murdochda12d292016-06-02 14:46:10 +01005837 DCHECK(code()->kind() == Code::BUILTIN || code()->kind() == Code::STUB);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005838 set_internal_formal_parameter_count(kDontAdaptArgumentsSentinel);
Steve Blocka7e24c12009-10-30 11:49:00 +00005839}
5840
5841
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005842int SharedFunctionInfo::start_position() const {
Steve Blocka7e24c12009-10-30 11:49:00 +00005843 return start_position_and_type() >> kStartPositionShift;
5844}
5845
5846
5847void SharedFunctionInfo::set_start_position(int start_position) {
5848 set_start_position_and_type((start_position << kStartPositionShift)
5849 | (start_position_and_type() & ~kStartPositionMask));
5850}
5851
5852
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005853Code* SharedFunctionInfo::code() const {
Steve Blocka7e24c12009-10-30 11:49:00 +00005854 return Code::cast(READ_FIELD(this, kCodeOffset));
5855}
5856
5857
5858void SharedFunctionInfo::set_code(Code* value, WriteBarrierMode mode) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005859 DCHECK(value->kind() != Code::OPTIMIZED_FUNCTION);
Steve Blocka7e24c12009-10-30 11:49:00 +00005860 WRITE_FIELD(this, kCodeOffset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005861 CONDITIONAL_WRITE_BARRIER(value->GetHeap(), this, kCodeOffset, value, mode);
Steve Blocka7e24c12009-10-30 11:49:00 +00005862}
5863
5864
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005865void SharedFunctionInfo::ReplaceCode(Code* value) {
5866 // If the GC metadata field is already used then the function was
5867 // enqueued as a code flushing candidate and we remove it now.
5868 if (code()->gc_metadata() != NULL) {
5869 CodeFlusher* flusher = GetHeap()->mark_compact_collector()->code_flusher();
5870 flusher->EvictCandidate(this);
5871 }
5872
5873 DCHECK(code()->gc_metadata() == NULL && value->gc_metadata() == NULL);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005874#ifdef DEBUG
5875 Code::VerifyRecompiledCode(code(), value);
5876#endif // DEBUG
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005877
5878 set_code(value);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005879
5880 if (is_compiled()) set_never_compiled(false);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005881}
5882
5883
5884ScopeInfo* SharedFunctionInfo::scope_info() const {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005885 return reinterpret_cast<ScopeInfo*>(READ_FIELD(this, kScopeInfoOffset));
Ben Murdoch3bec4d22010-07-22 14:51:16 +01005886}
5887
5888
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005889void SharedFunctionInfo::set_scope_info(ScopeInfo* value,
Ben Murdoch3bec4d22010-07-22 14:51:16 +01005890 WriteBarrierMode mode) {
5891 WRITE_FIELD(this, kScopeInfoOffset, reinterpret_cast<Object*>(value));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01005892 CONDITIONAL_WRITE_BARRIER(GetHeap(),
5893 this,
5894 kScopeInfoOffset,
5895 reinterpret_cast<Object*>(value),
5896 mode);
Ben Murdoch3bec4d22010-07-22 14:51:16 +01005897}
5898
5899
Steve Blocka7e24c12009-10-30 11:49:00 +00005900bool SharedFunctionInfo::is_compiled() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005901 Builtins* builtins = GetIsolate()->builtins();
5902 DCHECK(code() != builtins->builtin(Builtins::kCompileOptimizedConcurrent));
5903 DCHECK(code() != builtins->builtin(Builtins::kCompileOptimized));
Ben Murdochc5610432016-08-08 18:44:38 +01005904 DCHECK(code() != builtins->builtin(Builtins::kCompileBaseline));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005905 return code() != builtins->builtin(Builtins::kCompileLazy);
5906}
5907
5908
5909bool SharedFunctionInfo::has_simple_parameters() {
5910 return scope_info()->HasSimpleParameters();
5911}
5912
5913
5914bool SharedFunctionInfo::HasDebugInfo() {
5915 bool has_debug_info = debug_info()->IsStruct();
5916 DCHECK(!has_debug_info || HasDebugCode());
5917 return has_debug_info;
5918}
5919
5920
5921DebugInfo* SharedFunctionInfo::GetDebugInfo() {
5922 DCHECK(HasDebugInfo());
5923 return DebugInfo::cast(debug_info());
5924}
5925
5926
5927bool SharedFunctionInfo::HasDebugCode() {
Ben Murdoch097c5b22016-05-18 11:27:45 +01005928 return HasBytecodeArray() ||
5929 (code()->kind() == Code::FUNCTION && code()->has_debug_break_slots());
Steve Blocka7e24c12009-10-30 11:49:00 +00005930}
5931
5932
Steve Block6ded16b2010-05-10 14:33:55 +01005933bool SharedFunctionInfo::IsApiFunction() {
5934 return function_data()->IsFunctionTemplateInfo();
5935}
5936
5937
5938FunctionTemplateInfo* SharedFunctionInfo::get_api_func_data() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005939 DCHECK(IsApiFunction());
Steve Block6ded16b2010-05-10 14:33:55 +01005940 return FunctionTemplateInfo::cast(function_data());
5941}
5942
Ben Murdochda12d292016-06-02 14:46:10 +01005943void SharedFunctionInfo::set_api_func_data(FunctionTemplateInfo* data) {
5944 DCHECK(function_data()->IsUndefined());
5945 set_function_data(data);
Kristian Monsen25f61362010-05-21 11:50:48 +01005946}
5947
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005948bool SharedFunctionInfo::HasBytecodeArray() {
5949 return function_data()->IsBytecodeArray();
5950}
5951
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005952BytecodeArray* SharedFunctionInfo::bytecode_array() {
5953 DCHECK(HasBytecodeArray());
5954 return BytecodeArray::cast(function_data());
5955}
5956
Ben Murdochda12d292016-06-02 14:46:10 +01005957void SharedFunctionInfo::set_bytecode_array(BytecodeArray* bytecode) {
5958 DCHECK(function_data()->IsUndefined());
5959 set_function_data(bytecode);
5960}
5961
5962void SharedFunctionInfo::ClearBytecodeArray() {
5963 DCHECK(function_data()->IsUndefined() || HasBytecodeArray());
5964 set_function_data(GetHeap()->undefined_value());
5965}
5966
5967bool SharedFunctionInfo::HasBuiltinFunctionId() {
5968 return function_identifier()->IsSmi();
5969}
5970
5971BuiltinFunctionId SharedFunctionInfo::builtin_function_id() {
5972 DCHECK(HasBuiltinFunctionId());
5973 return static_cast<BuiltinFunctionId>(
5974 Smi::cast(function_identifier())->value());
5975}
5976
5977void SharedFunctionInfo::set_builtin_function_id(BuiltinFunctionId id) {
5978 set_function_identifier(Smi::FromInt(id));
5979}
5980
5981bool SharedFunctionInfo::HasInferredName() {
5982 return function_identifier()->IsString();
5983}
5984
5985String* SharedFunctionInfo::inferred_name() {
5986 if (HasInferredName()) {
5987 return String::cast(function_identifier());
5988 }
5989 DCHECK(function_identifier()->IsUndefined() || HasBuiltinFunctionId());
5990 return GetIsolate()->heap()->empty_string();
5991}
5992
5993void SharedFunctionInfo::set_inferred_name(String* inferred_name) {
5994 DCHECK(function_identifier()->IsUndefined() || HasInferredName());
5995 set_function_identifier(inferred_name);
5996}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00005997
Ben Murdochb8a8cc12014-11-26 15:28:44 +00005998int SharedFunctionInfo::ic_age() {
5999 return ICAgeBits::decode(counters());
Iain Merrick75681382010-08-19 15:07:18 +01006000}
6001
6002
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006003void SharedFunctionInfo::set_ic_age(int ic_age) {
6004 set_counters(ICAgeBits::update(counters(), ic_age));
6005}
6006
6007
6008int SharedFunctionInfo::deopt_count() {
6009 return DeoptCountBits::decode(counters());
6010}
6011
6012
6013void SharedFunctionInfo::set_deopt_count(int deopt_count) {
6014 set_counters(DeoptCountBits::update(counters(), deopt_count));
6015}
6016
6017
6018void SharedFunctionInfo::increment_deopt_count() {
6019 int value = counters();
6020 int deopt_count = DeoptCountBits::decode(value);
6021 deopt_count = (deopt_count + 1) & DeoptCountBits::kMax;
6022 set_counters(DeoptCountBits::update(value, deopt_count));
6023}
6024
6025
6026int SharedFunctionInfo::opt_reenable_tries() {
6027 return OptReenableTriesBits::decode(counters());
6028}
6029
6030
6031void SharedFunctionInfo::set_opt_reenable_tries(int tries) {
6032 set_counters(OptReenableTriesBits::update(counters(), tries));
6033}
6034
6035
6036int SharedFunctionInfo::opt_count() {
6037 return OptCountBits::decode(opt_count_and_bailout_reason());
6038}
6039
6040
6041void SharedFunctionInfo::set_opt_count(int opt_count) {
6042 set_opt_count_and_bailout_reason(
6043 OptCountBits::update(opt_count_and_bailout_reason(), opt_count));
6044}
6045
6046
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006047BailoutReason SharedFunctionInfo::disable_optimization_reason() {
6048 return static_cast<BailoutReason>(
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006049 DisabledOptimizationReasonBits::decode(opt_count_and_bailout_reason()));
Iain Merrick75681382010-08-19 15:07:18 +01006050}
6051
6052
Ben Murdochb0fe1622011-05-05 13:52:32 +01006053bool SharedFunctionInfo::has_deoptimization_support() {
6054 Code* code = this->code();
6055 return code->kind() == Code::FUNCTION && code->has_deoptimization_support();
6056}
6057
6058
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006059void SharedFunctionInfo::TryReenableOptimization() {
6060 int tries = opt_reenable_tries();
6061 set_opt_reenable_tries((tries + 1) & OptReenableTriesBits::kMax);
6062 // We reenable optimization whenever the number of tries is a large
6063 // enough power of 2.
6064 if (tries >= 16 && (((tries - 1) & tries) == 0)) {
6065 set_optimization_disabled(false);
6066 set_opt_count(0);
6067 set_deopt_count(0);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006068 }
6069}
6070
6071
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006072void SharedFunctionInfo::set_disable_optimization_reason(BailoutReason reason) {
6073 set_opt_count_and_bailout_reason(DisabledOptimizationReasonBits::update(
6074 opt_count_and_bailout_reason(), reason));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006075}
6076
6077
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006078bool SharedFunctionInfo::IsBuiltin() {
6079 Object* script_obj = script();
6080 if (script_obj->IsUndefined()) return true;
6081 Script* script = Script::cast(script_obj);
6082 Script::Type type = static_cast<Script::Type>(script->type());
6083 return type != Script::TYPE_NORMAL;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006084}
6085
6086
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006087bool SharedFunctionInfo::IsSubjectToDebugging() { return !IsBuiltin(); }
6088
6089
6090bool SharedFunctionInfo::OptimizedCodeMapIsCleared() const {
6091 return optimized_code_map() == GetHeap()->cleared_optimized_code_map();
Steve Blocka7e24c12009-10-30 11:49:00 +00006092}
6093
6094
Ben Murdochb0fe1622011-05-05 13:52:32 +01006095bool JSFunction::IsOptimized() {
6096 return code()->kind() == Code::OPTIMIZED_FUNCTION;
6097}
6098
Ben Murdochc5610432016-08-08 18:44:38 +01006099bool JSFunction::IsMarkedForBaseline() {
6100 return code() ==
6101 GetIsolate()->builtins()->builtin(Builtins::kCompileBaseline);
6102}
Ben Murdochb0fe1622011-05-05 13:52:32 +01006103
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006104bool JSFunction::IsMarkedForOptimization() {
6105 return code() == GetIsolate()->builtins()->builtin(
6106 Builtins::kCompileOptimized);
6107}
6108
6109
6110bool JSFunction::IsMarkedForConcurrentOptimization() {
6111 return code() == GetIsolate()->builtins()->builtin(
6112 Builtins::kCompileOptimizedConcurrent);
6113}
6114
6115
6116bool JSFunction::IsInOptimizationQueue() {
6117 return code() == GetIsolate()->builtins()->builtin(
6118 Builtins::kInOptimizationQueue);
6119}
6120
6121
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006122void JSFunction::CompleteInobjectSlackTrackingIfActive() {
6123 if (has_initial_map() && initial_map()->IsInobjectSlackTrackingInProgress()) {
6124 initial_map()->CompleteInobjectSlackTracking();
6125 }
6126}
6127
6128
6129bool Map::IsInobjectSlackTrackingInProgress() {
6130 return construction_counter() != Map::kNoSlackTracking;
6131}
6132
6133
6134void Map::InobjectSlackTrackingStep() {
6135 if (!IsInobjectSlackTrackingInProgress()) return;
6136 int counter = construction_counter();
6137 set_construction_counter(counter - 1);
6138 if (counter == kSlackTrackingCounterEnd) {
6139 CompleteInobjectSlackTracking();
6140 }
Ben Murdochb0fe1622011-05-05 13:52:32 +01006141}
6142
Ben Murdochda12d292016-06-02 14:46:10 +01006143AbstractCode* JSFunction::abstract_code() {
6144 Code* code = this->code();
6145 if (code->is_interpreter_entry_trampoline()) {
6146 return AbstractCode::cast(shared()->bytecode_array());
6147 } else {
6148 return AbstractCode::cast(code);
6149 }
6150}
Ben Murdochb0fe1622011-05-05 13:52:32 +01006151
Steve Blocka7e24c12009-10-30 11:49:00 +00006152Code* JSFunction::code() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006153 return Code::cast(
Steve Block791712a2010-08-27 10:21:07 +01006154 Code::GetObjectFromEntryAddress(FIELD_ADDR(this, kCodeEntryOffset)));
Steve Blocka7e24c12009-10-30 11:49:00 +00006155}
6156
6157
6158void JSFunction::set_code(Code* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006159 DCHECK(!GetHeap()->InNewSpace(value));
Steve Block791712a2010-08-27 10:21:07 +01006160 Address entry = value->entry();
6161 WRITE_INTPTR_FIELD(this, kCodeEntryOffset, reinterpret_cast<intptr_t>(entry));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006162 GetHeap()->incremental_marking()->RecordWriteOfCodeEntry(
6163 this,
6164 HeapObject::RawField(this, kCodeEntryOffset),
6165 value);
Steve Blocka7e24c12009-10-30 11:49:00 +00006166}
6167
6168
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006169void JSFunction::set_code_no_write_barrier(Code* value) {
6170 DCHECK(!GetHeap()->InNewSpace(value));
6171 Address entry = value->entry();
6172 WRITE_INTPTR_FIELD(this, kCodeEntryOffset, reinterpret_cast<intptr_t>(entry));
6173}
6174
6175
Ben Murdochb0fe1622011-05-05 13:52:32 +01006176void JSFunction::ReplaceCode(Code* code) {
6177 bool was_optimized = IsOptimized();
6178 bool is_optimized = code->kind() == Code::OPTIMIZED_FUNCTION;
6179
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006180 if (was_optimized && is_optimized) {
6181 shared()->EvictFromOptimizedCodeMap(this->code(),
6182 "Replacing with another optimized code");
6183 }
6184
Ben Murdochb0fe1622011-05-05 13:52:32 +01006185 set_code(code);
6186
6187 // Add/remove the function from the list of optimized functions for this
6188 // context based on the state change.
6189 if (!was_optimized && is_optimized) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006190 context()->native_context()->AddOptimizedFunction(this);
Ben Murdochb0fe1622011-05-05 13:52:32 +01006191 }
6192 if (was_optimized && !is_optimized) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006193 // TODO(titzer): linear in the number of optimized functions; fix!
6194 context()->native_context()->RemoveOptimizedFunction(this);
Ben Murdochb0fe1622011-05-05 13:52:32 +01006195 }
6196}
6197
6198
Steve Blocka7e24c12009-10-30 11:49:00 +00006199Context* JSFunction::context() {
6200 return Context::cast(READ_FIELD(this, kContextOffset));
6201}
6202
6203
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006204JSObject* JSFunction::global_proxy() {
6205 return context()->global_proxy();
Iain Merrick75681382010-08-19 15:07:18 +01006206}
6207
6208
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006209Context* JSFunction::native_context() { return context()->native_context(); }
6210
6211
Steve Blocka7e24c12009-10-30 11:49:00 +00006212void JSFunction::set_context(Object* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006213 DCHECK(value->IsUndefined() || value->IsContext());
Steve Blocka7e24c12009-10-30 11:49:00 +00006214 WRITE_FIELD(this, kContextOffset, value);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006215 WRITE_BARRIER(GetHeap(), this, kContextOffset, value);
Steve Blocka7e24c12009-10-30 11:49:00 +00006216}
6217
6218ACCESSORS(JSFunction, prototype_or_initial_map, Object,
6219 kPrototypeOrInitialMapOffset)
6220
6221
6222Map* JSFunction::initial_map() {
6223 return Map::cast(prototype_or_initial_map());
6224}
6225
6226
Steve Blocka7e24c12009-10-30 11:49:00 +00006227bool JSFunction::has_initial_map() {
6228 return prototype_or_initial_map()->IsMap();
6229}
6230
6231
6232bool JSFunction::has_instance_prototype() {
6233 return has_initial_map() || !prototype_or_initial_map()->IsTheHole();
6234}
6235
6236
6237bool JSFunction::has_prototype() {
6238 return map()->has_non_instance_prototype() || has_instance_prototype();
6239}
6240
6241
6242Object* JSFunction::instance_prototype() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006243 DCHECK(has_instance_prototype());
Steve Blocka7e24c12009-10-30 11:49:00 +00006244 if (has_initial_map()) return initial_map()->prototype();
6245 // When there is no initial map and the prototype is a JSObject, the
6246 // initial map field is used for the prototype field.
6247 return prototype_or_initial_map();
6248}
6249
6250
6251Object* JSFunction::prototype() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006252 DCHECK(has_prototype());
Steve Blocka7e24c12009-10-30 11:49:00 +00006253 // If the function's prototype property has been set to a non-JSObject
6254 // value, that value is stored in the constructor field of the map.
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006255 if (map()->has_non_instance_prototype()) {
6256 Object* prototype = map()->GetConstructor();
6257 // The map must have a prototype in that field, not a back pointer.
6258 DCHECK(!prototype->IsMap());
6259 return prototype;
6260 }
Steve Blocka7e24c12009-10-30 11:49:00 +00006261 return instance_prototype();
6262}
6263
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006264
Steve Blocka7e24c12009-10-30 11:49:00 +00006265bool JSFunction::is_compiled() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006266 Builtins* builtins = GetIsolate()->builtins();
6267 return code() != builtins->builtin(Builtins::kCompileLazy) &&
Ben Murdochc5610432016-08-08 18:44:38 +01006268 code() != builtins->builtin(Builtins::kCompileBaseline) &&
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006269 code() != builtins->builtin(Builtins::kCompileOptimized) &&
6270 code() != builtins->builtin(Builtins::kCompileOptimizedConcurrent);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006271}
6272
6273
Steve Blocka7e24c12009-10-30 11:49:00 +00006274int JSFunction::NumberOfLiterals() {
6275 return literals()->length();
6276}
6277
6278
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006279ACCESSORS(JSProxy, target, JSReceiver, kTargetOffset)
Ben Murdoch257744e2011-11-30 15:57:28 +00006280ACCESSORS(JSProxy, handler, Object, kHandlerOffset)
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006281ACCESSORS(JSProxy, hash, Object, kHashOffset)
Ben Murdoch589d6972011-11-30 16:04:58 +00006282
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006283bool JSProxy::IsRevoked() const { return !handler()->IsJSReceiver(); }
Ben Murdoch257744e2011-11-30 15:57:28 +00006284
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006285ACCESSORS(JSCollection, table, Object, kTableOffset)
6286
6287
6288#define ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(name, type, offset) \
6289 template<class Derived, class TableType> \
6290 type* OrderedHashTableIterator<Derived, TableType>::name() const { \
6291 return type::cast(READ_FIELD(this, offset)); \
6292 } \
6293 template<class Derived, class TableType> \
6294 void OrderedHashTableIterator<Derived, TableType>::set_##name( \
6295 type* value, WriteBarrierMode mode) { \
6296 WRITE_FIELD(this, offset, value); \
6297 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, offset, value, mode); \
6298 }
6299
6300ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(table, Object, kTableOffset)
6301ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(index, Object, kIndexOffset)
6302ORDERED_HASH_TABLE_ITERATOR_ACCESSORS(kind, Object, kKindOffset)
6303
6304#undef ORDERED_HASH_TABLE_ITERATOR_ACCESSORS
6305
6306
6307ACCESSORS(JSWeakCollection, table, Object, kTableOffset)
6308ACCESSORS(JSWeakCollection, next, Object, kNextOffset)
Ben Murdoch69a99ed2011-11-30 16:03:39 +00006309
6310
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006311Address Foreign::foreign_address() {
6312 return AddressFrom<Address>(READ_INTPTR_FIELD(this, kForeignAddressOffset));
Ben Murdoch69a99ed2011-11-30 16:03:39 +00006313}
6314
6315
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006316void Foreign::set_foreign_address(Address value) {
6317 WRITE_INTPTR_FIELD(this, kForeignAddressOffset, OffsetFrom(value));
Steve Blocka7e24c12009-10-30 11:49:00 +00006318}
6319
6320
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006321ACCESSORS(JSGeneratorObject, function, JSFunction, kFunctionOffset)
6322ACCESSORS(JSGeneratorObject, context, Context, kContextOffset)
6323ACCESSORS(JSGeneratorObject, receiver, Object, kReceiverOffset)
Ben Murdoch097c5b22016-05-18 11:27:45 +01006324ACCESSORS(JSGeneratorObject, input, Object, kInputOffset)
Ben Murdochc5610432016-08-08 18:44:38 +01006325SMI_ACCESSORS(JSGeneratorObject, resume_mode, kResumeModeOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006326SMI_ACCESSORS(JSGeneratorObject, continuation, kContinuationOffset)
6327ACCESSORS(JSGeneratorObject, operand_stack, FixedArray, kOperandStackOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006328
6329bool JSGeneratorObject::is_suspended() {
Ben Murdochc5610432016-08-08 18:44:38 +01006330 DCHECK_LT(kGeneratorExecuting, 0);
6331 DCHECK_LT(kGeneratorClosed, 0);
6332 return continuation() >= 0;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006333}
6334
6335bool JSGeneratorObject::is_closed() {
6336 return continuation() == kGeneratorClosed;
6337}
6338
6339bool JSGeneratorObject::is_executing() {
6340 return continuation() == kGeneratorExecuting;
6341}
6342
6343ACCESSORS(JSModule, context, Object, kContextOffset)
6344ACCESSORS(JSModule, scope_info, ScopeInfo, kScopeInfoOffset)
6345
6346
Steve Blocka7e24c12009-10-30 11:49:00 +00006347ACCESSORS(JSValue, value, Object, kValueOffset)
6348
6349
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006350HeapNumber* HeapNumber::cast(Object* object) {
6351 SLOW_DCHECK(object->IsHeapNumber() || object->IsMutableHeapNumber());
6352 return reinterpret_cast<HeapNumber*>(object);
6353}
6354
6355
6356const HeapNumber* HeapNumber::cast(const Object* object) {
6357 SLOW_DCHECK(object->IsHeapNumber() || object->IsMutableHeapNumber());
6358 return reinterpret_cast<const HeapNumber*>(object);
Steve Blocka7e24c12009-10-30 11:49:00 +00006359}
6360
6361
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006362ACCESSORS(JSDate, value, Object, kValueOffset)
6363ACCESSORS(JSDate, cache_stamp, Object, kCacheStampOffset)
6364ACCESSORS(JSDate, year, Object, kYearOffset)
6365ACCESSORS(JSDate, month, Object, kMonthOffset)
6366ACCESSORS(JSDate, day, Object, kDayOffset)
6367ACCESSORS(JSDate, weekday, Object, kWeekdayOffset)
6368ACCESSORS(JSDate, hour, Object, kHourOffset)
6369ACCESSORS(JSDate, min, Object, kMinOffset)
6370ACCESSORS(JSDate, sec, Object, kSecOffset)
6371
6372
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006373SMI_ACCESSORS(JSMessageObject, type, kTypeOffset)
6374ACCESSORS(JSMessageObject, argument, Object, kArgumentsOffset)
Steve Block1e0659c2011-05-24 12:43:12 +01006375ACCESSORS(JSMessageObject, script, Object, kScriptOffset)
Steve Block1e0659c2011-05-24 12:43:12 +01006376ACCESSORS(JSMessageObject, stack_frames, Object, kStackFramesOffset)
6377SMI_ACCESSORS(JSMessageObject, start_position, kStartPositionOffset)
6378SMI_ACCESSORS(JSMessageObject, end_position, kEndPositionOffset)
6379
6380
Steve Blocka7e24c12009-10-30 11:49:00 +00006381INT_ACCESSORS(Code, instruction_size, kInstructionSizeOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006382INT_ACCESSORS(Code, prologue_offset, kPrologueOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006383INT_ACCESSORS(Code, constant_pool_offset, kConstantPoolOffset)
Leon Clarkeac952652010-07-15 11:15:24 +01006384ACCESSORS(Code, relocation_info, ByteArray, kRelocationInfoOffset)
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006385ACCESSORS(Code, handler_table, FixedArray, kHandlerTableOffset)
Ben Murdochb0fe1622011-05-05 13:52:32 +01006386ACCESSORS(Code, deoptimization_data, FixedArray, kDeoptimizationDataOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006387ACCESSORS(Code, raw_type_feedback_info, Object, kTypeFeedbackInfoOffset)
6388ACCESSORS(Code, next_code_link, Object, kNextCodeLinkOffset)
6389
6390
6391void Code::WipeOutHeader() {
6392 WRITE_FIELD(this, kRelocationInfoOffset, NULL);
6393 WRITE_FIELD(this, kHandlerTableOffset, NULL);
6394 WRITE_FIELD(this, kDeoptimizationDataOffset, NULL);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006395 // Do not wipe out major/minor keys on a code stub or IC
6396 if (!READ_FIELD(this, kTypeFeedbackInfoOffset)->IsSmi()) {
6397 WRITE_FIELD(this, kTypeFeedbackInfoOffset, NULL);
6398 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006399 WRITE_FIELD(this, kNextCodeLinkOffset, NULL);
6400 WRITE_FIELD(this, kGCMetadataOffset, NULL);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006401}
6402
6403
6404Object* Code::type_feedback_info() {
6405 DCHECK(kind() == FUNCTION);
6406 return raw_type_feedback_info();
6407}
6408
6409
6410void Code::set_type_feedback_info(Object* value, WriteBarrierMode mode) {
6411 DCHECK(kind() == FUNCTION);
6412 set_raw_type_feedback_info(value, mode);
6413 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kTypeFeedbackInfoOffset,
6414 value, mode);
6415}
6416
6417
6418uint32_t Code::stub_key() {
6419 DCHECK(IsCodeStubOrIC());
6420 Smi* smi_key = Smi::cast(raw_type_feedback_info());
6421 return static_cast<uint32_t>(smi_key->value());
6422}
6423
6424
6425void Code::set_stub_key(uint32_t key) {
6426 DCHECK(IsCodeStubOrIC());
6427 set_raw_type_feedback_info(Smi::FromInt(key));
6428}
6429
6430
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006431ACCESSORS(Code, gc_metadata, Object, kGCMetadataOffset)
6432INT_ACCESSORS(Code, ic_age, kICAgeOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00006433
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006434
Steve Blocka7e24c12009-10-30 11:49:00 +00006435byte* Code::instruction_start() {
6436 return FIELD_ADDR(this, kHeaderSize);
6437}
6438
6439
Leon Clarkeac952652010-07-15 11:15:24 +01006440byte* Code::instruction_end() {
6441 return instruction_start() + instruction_size();
6442}
6443
6444
Steve Blocka7e24c12009-10-30 11:49:00 +00006445int Code::body_size() {
Leon Clarkeac952652010-07-15 11:15:24 +01006446 return RoundUp(instruction_size(), kObjectAlignment);
6447}
6448
6449
6450ByteArray* Code::unchecked_relocation_info() {
6451 return reinterpret_cast<ByteArray*>(READ_FIELD(this, kRelocationInfoOffset));
Steve Blocka7e24c12009-10-30 11:49:00 +00006452}
6453
6454
6455byte* Code::relocation_start() {
Leon Clarkeac952652010-07-15 11:15:24 +01006456 return unchecked_relocation_info()->GetDataStartAddress();
6457}
6458
6459
6460int Code::relocation_size() {
6461 return unchecked_relocation_info()->length();
Steve Blocka7e24c12009-10-30 11:49:00 +00006462}
6463
6464
6465byte* Code::entry() {
6466 return instruction_start();
6467}
6468
6469
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006470bool Code::contains(byte* inner_pointer) {
6471 return (address() <= inner_pointer) && (inner_pointer <= address() + Size());
Steve Blocka7e24c12009-10-30 11:49:00 +00006472}
6473
6474
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006475int Code::ExecutableSize() {
6476 // Check that the assumptions about the layout of the code object holds.
6477 DCHECK_EQ(static_cast<int>(instruction_start() - address()),
6478 Code::kHeaderSize);
6479 return instruction_size() + Code::kHeaderSize;
6480}
6481
6482
6483int Code::CodeSize() { return SizeFor(body_size()); }
6484
6485
Steve Blocka7e24c12009-10-30 11:49:00 +00006486ACCESSORS(JSArray, length, Object, kLengthOffset)
6487
6488
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006489void* JSArrayBuffer::backing_store() const {
6490 intptr_t ptr = READ_INTPTR_FIELD(this, kBackingStoreOffset);
6491 return reinterpret_cast<void*>(ptr);
6492}
6493
6494
6495void JSArrayBuffer::set_backing_store(void* value, WriteBarrierMode mode) {
6496 intptr_t ptr = reinterpret_cast<intptr_t>(value);
6497 WRITE_INTPTR_FIELD(this, kBackingStoreOffset, ptr);
6498}
6499
6500
6501ACCESSORS(JSArrayBuffer, byte_length, Object, kByteLengthOffset)
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006502
6503
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006504void JSArrayBuffer::set_bit_field(uint32_t bits) {
6505 if (kInt32Size != kPointerSize) {
6506#if V8_TARGET_LITTLE_ENDIAN
6507 WRITE_UINT32_FIELD(this, kBitFieldSlot + kInt32Size, 0);
6508#else
6509 WRITE_UINT32_FIELD(this, kBitFieldSlot, 0);
6510#endif
6511 }
6512 WRITE_UINT32_FIELD(this, kBitFieldOffset, bits);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006513}
6514
6515
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006516uint32_t JSArrayBuffer::bit_field() const {
6517 return READ_UINT32_FIELD(this, kBitFieldOffset);
6518}
6519
6520
6521bool JSArrayBuffer::is_external() { return IsExternal::decode(bit_field()); }
6522
6523
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006524void JSArrayBuffer::set_is_external(bool value) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006525 set_bit_field(IsExternal::update(bit_field(), value));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006526}
6527
6528
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006529bool JSArrayBuffer::is_neuterable() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006530 return IsNeuterable::decode(bit_field());
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006531}
6532
6533
6534void JSArrayBuffer::set_is_neuterable(bool value) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006535 set_bit_field(IsNeuterable::update(bit_field(), value));
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006536}
6537
6538
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006539bool JSArrayBuffer::was_neutered() { return WasNeutered::decode(bit_field()); }
6540
6541
6542void JSArrayBuffer::set_was_neutered(bool value) {
6543 set_bit_field(WasNeutered::update(bit_field(), value));
6544}
6545
6546
6547bool JSArrayBuffer::is_shared() { return IsShared::decode(bit_field()); }
6548
6549
6550void JSArrayBuffer::set_is_shared(bool value) {
6551 set_bit_field(IsShared::update(bit_field(), value));
6552}
6553
6554
6555Object* JSArrayBufferView::byte_offset() const {
6556 if (WasNeutered()) return Smi::FromInt(0);
6557 return Object::cast(READ_FIELD(this, kByteOffsetOffset));
6558}
6559
6560
6561void JSArrayBufferView::set_byte_offset(Object* value, WriteBarrierMode mode) {
6562 WRITE_FIELD(this, kByteOffsetOffset, value);
6563 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kByteOffsetOffset, value, mode);
6564}
6565
6566
6567Object* JSArrayBufferView::byte_length() const {
6568 if (WasNeutered()) return Smi::FromInt(0);
6569 return Object::cast(READ_FIELD(this, kByteLengthOffset));
6570}
6571
6572
6573void JSArrayBufferView::set_byte_length(Object* value, WriteBarrierMode mode) {
6574 WRITE_FIELD(this, kByteLengthOffset, value);
6575 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kByteLengthOffset, value, mode);
6576}
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006577
6578
6579ACCESSORS(JSArrayBufferView, buffer, Object, kBufferOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006580#ifdef VERIFY_HEAP
6581ACCESSORS(JSArrayBufferView, raw_byte_offset, Object, kByteOffsetOffset)
6582ACCESSORS(JSArrayBufferView, raw_byte_length, Object, kByteLengthOffset)
6583#endif
6584
6585
6586bool JSArrayBufferView::WasNeutered() const {
6587 return JSArrayBuffer::cast(buffer())->was_neutered();
6588}
6589
6590
6591Object* JSTypedArray::length() const {
6592 if (WasNeutered()) return Smi::FromInt(0);
6593 return Object::cast(READ_FIELD(this, kLengthOffset));
6594}
6595
6596
6597uint32_t JSTypedArray::length_value() const {
6598 if (WasNeutered()) return 0;
6599 uint32_t index = 0;
6600 CHECK(Object::cast(READ_FIELD(this, kLengthOffset))->ToArrayLength(&index));
6601 return index;
6602}
6603
6604
6605void JSTypedArray::set_length(Object* value, WriteBarrierMode mode) {
6606 WRITE_FIELD(this, kLengthOffset, value);
6607 CONDITIONAL_WRITE_BARRIER(GetHeap(), this, kLengthOffset, value, mode);
6608}
6609
6610
6611#ifdef VERIFY_HEAP
6612ACCESSORS(JSTypedArray, raw_length, Object, kLengthOffset)
6613#endif
6614
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006615
Steve Blocka7e24c12009-10-30 11:49:00 +00006616ACCESSORS(JSRegExp, data, Object, kDataOffset)
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006617ACCESSORS(JSRegExp, flags, Object, kFlagsOffset)
6618ACCESSORS(JSRegExp, source, Object, kSourceOffset)
Steve Blocka7e24c12009-10-30 11:49:00 +00006619
6620
6621JSRegExp::Type JSRegExp::TypeTag() {
6622 Object* data = this->data();
6623 if (data->IsUndefined()) return JSRegExp::NOT_COMPILED;
6624 Smi* smi = Smi::cast(FixedArray::cast(data)->get(kTagIndex));
6625 return static_cast<JSRegExp::Type>(smi->value());
6626}
6627
6628
6629int JSRegExp::CaptureCount() {
6630 switch (TypeTag()) {
6631 case ATOM:
6632 return 0;
6633 case IRREGEXP:
6634 return Smi::cast(DataAt(kIrregexpCaptureCountIndex))->value();
6635 default:
6636 UNREACHABLE();
6637 return -1;
6638 }
6639}
6640
6641
6642JSRegExp::Flags JSRegExp::GetFlags() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006643 DCHECK(this->data()->IsFixedArray());
Steve Blocka7e24c12009-10-30 11:49:00 +00006644 Object* data = this->data();
6645 Smi* smi = Smi::cast(FixedArray::cast(data)->get(kFlagsIndex));
6646 return Flags(smi->value());
6647}
6648
6649
6650String* JSRegExp::Pattern() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006651 DCHECK(this->data()->IsFixedArray());
Steve Blocka7e24c12009-10-30 11:49:00 +00006652 Object* data = this->data();
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006653 String* pattern = String::cast(FixedArray::cast(data)->get(kSourceIndex));
Steve Blocka7e24c12009-10-30 11:49:00 +00006654 return pattern;
6655}
6656
6657
6658Object* JSRegExp::DataAt(int index) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006659 DCHECK(TypeTag() != NOT_COMPILED);
Steve Blocka7e24c12009-10-30 11:49:00 +00006660 return FixedArray::cast(data())->get(index);
6661}
6662
6663
6664void JSRegExp::SetDataAt(int index, Object* value) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006665 DCHECK(TypeTag() != NOT_COMPILED);
6666 DCHECK(index >= kDataIndex); // Only implementation data can be set this way.
Steve Blocka7e24c12009-10-30 11:49:00 +00006667 FixedArray::cast(data())->set(index, value);
6668}
6669
6670
Ben Murdoch589d6972011-11-30 16:04:58 +00006671ElementsKind JSObject::GetElementsKind() {
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00006672 ElementsKind kind = map()->elements_kind();
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006673#if VERIFY_HEAP && DEBUG
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006674 FixedArrayBase* fixed_array =
6675 reinterpret_cast<FixedArrayBase*>(READ_FIELD(this, kElementsOffset));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006676
6677 // If a GC was caused while constructing this object, the elements
6678 // pointer may point to a one pointer filler map.
6679 if (ElementsAreSafeToExamine()) {
6680 Map* map = fixed_array->map();
6681 DCHECK((IsFastSmiOrObjectElementsKind(kind) &&
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006682 (map == GetHeap()->fixed_array_map() ||
6683 map == GetHeap()->fixed_cow_array_map())) ||
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006684 (IsFastDoubleElementsKind(kind) &&
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006685 (fixed_array->IsFixedDoubleArray() ||
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006686 fixed_array == GetHeap()->empty_fixed_array())) ||
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006687 (kind == DICTIONARY_ELEMENTS &&
6688 fixed_array->IsFixedArray() &&
6689 fixed_array->IsDictionary()) ||
6690 (kind > DICTIONARY_ELEMENTS));
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006691 DCHECK(!IsSloppyArgumentsElements(kind) ||
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006692 (elements()->IsFixedArray() && elements()->length() >= 2));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006693 }
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006694#endif
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00006695 return kind;
Steve Blocka7e24c12009-10-30 11:49:00 +00006696}
6697
6698
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006699bool JSObject::HasFastObjectElements() {
6700 return IsFastObjectElementsKind(GetElementsKind());
Steve Blocka7e24c12009-10-30 11:49:00 +00006701}
6702
6703
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006704bool JSObject::HasFastSmiElements() {
6705 return IsFastSmiElementsKind(GetElementsKind());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006706}
6707
6708
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006709bool JSObject::HasFastSmiOrObjectElements() {
6710 return IsFastSmiOrObjectElementsKind(GetElementsKind());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006711}
6712
6713
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00006714bool JSObject::HasFastDoubleElements() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006715 return IsFastDoubleElementsKind(GetElementsKind());
6716}
6717
6718
6719bool JSObject::HasFastHoleyElements() {
6720 return IsFastHoleyElementsKind(GetElementsKind());
6721}
6722
6723
6724bool JSObject::HasFastElements() {
6725 return IsFastElementsKind(GetElementsKind());
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00006726}
6727
6728
Steve Blocka7e24c12009-10-30 11:49:00 +00006729bool JSObject::HasDictionaryElements() {
6730 return GetElementsKind() == DICTIONARY_ELEMENTS;
6731}
6732
6733
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006734bool JSObject::HasFastArgumentsElements() {
6735 return GetElementsKind() == FAST_SLOPPY_ARGUMENTS_ELEMENTS;
6736}
6737
6738
6739bool JSObject::HasSlowArgumentsElements() {
6740 return GetElementsKind() == SLOW_SLOPPY_ARGUMENTS_ELEMENTS;
6741}
6742
6743
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006744bool JSObject::HasSloppyArgumentsElements() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006745 return IsSloppyArgumentsElements(GetElementsKind());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006746}
6747
Ben Murdoch097c5b22016-05-18 11:27:45 +01006748bool JSObject::HasStringWrapperElements() {
6749 return IsStringWrapperElementsKind(GetElementsKind());
6750}
6751
6752bool JSObject::HasFastStringWrapperElements() {
6753 return GetElementsKind() == FAST_STRING_WRAPPER_ELEMENTS;
6754}
6755
6756bool JSObject::HasSlowStringWrapperElements() {
6757 return GetElementsKind() == SLOW_STRING_WRAPPER_ELEMENTS;
6758}
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006759
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006760bool JSObject::HasFixedTypedArrayElements() {
Ben Murdoch097c5b22016-05-18 11:27:45 +01006761 DCHECK_NOT_NULL(elements());
6762 return map()->has_fixed_typed_array_elements();
Steve Block3ce2e202009-11-05 08:53:23 +00006763}
6764
Ben Murdoch097c5b22016-05-18 11:27:45 +01006765#define FIXED_TYPED_ELEMENTS_CHECK(Type, type, TYPE, ctype, size) \
6766 bool JSObject::HasFixed##Type##Elements() { \
6767 HeapObject* array = elements(); \
6768 DCHECK(array != NULL); \
6769 if (!array->IsHeapObject()) return false; \
6770 return array->map()->instance_type() == FIXED_##TYPE##_ARRAY_TYPE; \
6771 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006772
6773TYPED_ARRAYS(FIXED_TYPED_ELEMENTS_CHECK)
6774
6775#undef FIXED_TYPED_ELEMENTS_CHECK
Steve Block3ce2e202009-11-05 08:53:23 +00006776
6777
Steve Blocka7e24c12009-10-30 11:49:00 +00006778bool JSObject::HasNamedInterceptor() {
6779 return map()->has_named_interceptor();
6780}
6781
6782
6783bool JSObject::HasIndexedInterceptor() {
6784 return map()->has_indexed_interceptor();
6785}
6786
6787
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006788GlobalDictionary* JSObject::global_dictionary() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006789 DCHECK(!HasFastProperties());
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006790 DCHECK(IsJSGlobalObject());
6791 return GlobalDictionary::cast(properties());
Steve Blocka7e24c12009-10-30 11:49:00 +00006792}
6793
6794
Ben Murdochc7cc0282012-03-05 14:35:55 +00006795SeededNumberDictionary* JSObject::element_dictionary() {
Ben Murdoch097c5b22016-05-18 11:27:45 +01006796 DCHECK(HasDictionaryElements() || HasSlowStringWrapperElements());
Ben Murdochc7cc0282012-03-05 14:35:55 +00006797 return SeededNumberDictionary::cast(elements());
Steve Blocka7e24c12009-10-30 11:49:00 +00006798}
6799
6800
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006801bool Name::IsHashFieldComputed(uint32_t field) {
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01006802 return (field & kHashNotComputedMask) == 0;
6803}
6804
6805
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006806bool Name::HasHashCode() {
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01006807 return IsHashFieldComputed(hash_field());
Steve Blocka7e24c12009-10-30 11:49:00 +00006808}
6809
6810
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006811uint32_t Name::Hash() {
Steve Blocka7e24c12009-10-30 11:49:00 +00006812 // Fast case: has hash code already been computed?
Steve Blockd0582a62009-12-15 09:54:21 +00006813 uint32_t field = hash_field();
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01006814 if (IsHashFieldComputed(field)) return field >> kHashShift;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006815 // Slow case: compute hash code and set it. Has to be a string.
6816 return String::cast(this)->ComputeAndSetHash();
6817}
6818
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006819
6820bool Name::IsPrivate() {
6821 return this->IsSymbol() && Symbol::cast(this)->is_private();
Steve Blocka7e24c12009-10-30 11:49:00 +00006822}
6823
6824
Ben Murdochc7cc0282012-03-05 14:35:55 +00006825StringHasher::StringHasher(int length, uint32_t seed)
Steve Blocka7e24c12009-10-30 11:49:00 +00006826 : length_(length),
Ben Murdochc7cc0282012-03-05 14:35:55 +00006827 raw_running_hash_(seed),
Steve Blocka7e24c12009-10-30 11:49:00 +00006828 array_index_(0),
6829 is_array_index_(0 < length_ && length_ <= String::kMaxArrayIndexSize),
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006830 is_first_char_(true) {
6831 DCHECK(FLAG_randomize_hashes || raw_running_hash_ == 0);
Ben Murdochc7cc0282012-03-05 14:35:55 +00006832}
Steve Blocka7e24c12009-10-30 11:49:00 +00006833
6834
6835bool StringHasher::has_trivial_hash() {
Steve Blockd0582a62009-12-15 09:54:21 +00006836 return length_ > String::kMaxHashCalcLength;
Steve Blocka7e24c12009-10-30 11:49:00 +00006837}
6838
6839
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006840uint32_t StringHasher::AddCharacterCore(uint32_t running_hash, uint16_t c) {
6841 running_hash += c;
6842 running_hash += (running_hash << 10);
6843 running_hash ^= (running_hash >> 6);
6844 return running_hash;
6845}
6846
6847
6848uint32_t StringHasher::GetHashCore(uint32_t running_hash) {
6849 running_hash += (running_hash << 3);
6850 running_hash ^= (running_hash >> 11);
6851 running_hash += (running_hash << 15);
6852 if ((running_hash & String::kHashBitMask) == 0) {
6853 return kZeroHash;
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006854 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006855 return running_hash;
6856}
6857
6858
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006859uint32_t StringHasher::ComputeRunningHash(uint32_t running_hash,
6860 const uc16* chars, int length) {
6861 DCHECK_NOT_NULL(chars);
6862 DCHECK(length >= 0);
6863 for (int i = 0; i < length; ++i) {
6864 running_hash = AddCharacterCore(running_hash, *chars++);
6865 }
6866 return running_hash;
6867}
6868
6869
6870uint32_t StringHasher::ComputeRunningHashOneByte(uint32_t running_hash,
6871 const char* chars,
6872 int length) {
6873 DCHECK_NOT_NULL(chars);
6874 DCHECK(length >= 0);
6875 for (int i = 0; i < length; ++i) {
6876 uint16_t c = static_cast<uint16_t>(*chars++);
6877 running_hash = AddCharacterCore(running_hash, c);
6878 }
6879 return running_hash;
6880}
6881
6882
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006883void StringHasher::AddCharacter(uint16_t c) {
Steve Blocka7e24c12009-10-30 11:49:00 +00006884 // Use the Jenkins one-at-a-time hash function to update the hash
6885 // for the given character.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006886 raw_running_hash_ = AddCharacterCore(raw_running_hash_, c);
6887}
6888
6889
6890bool StringHasher::UpdateIndex(uint16_t c) {
6891 DCHECK(is_array_index_);
6892 if (c < '0' || c > '9') {
6893 is_array_index_ = false;
6894 return false;
6895 }
6896 int d = c - '0';
6897 if (is_first_char_) {
6898 is_first_char_ = false;
6899 if (c == '0' && length_ > 1) {
Steve Blocka7e24c12009-10-30 11:49:00 +00006900 is_array_index_ = false;
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006901 return false;
6902 }
6903 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006904 if (array_index_ > 429496729U - ((d + 3) >> 3)) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006905 is_array_index_ = false;
6906 return false;
6907 }
6908 array_index_ = array_index_ * 10 + d;
6909 return true;
6910}
6911
6912
6913template<typename Char>
6914inline void StringHasher::AddCharacters(const Char* chars, int length) {
6915 DCHECK(sizeof(Char) == 1 || sizeof(Char) == 2);
6916 int i = 0;
6917 if (is_array_index_) {
6918 for (; i < length; i++) {
6919 AddCharacter(chars[i]);
6920 if (!UpdateIndex(chars[i])) {
6921 i++;
6922 break;
Steve Blocka7e24c12009-10-30 11:49:00 +00006923 }
6924 }
6925 }
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006926 for (; i < length; i++) {
6927 DCHECK(!is_array_index_);
6928 AddCharacter(chars[i]);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01006929 }
Steve Blocka7e24c12009-10-30 11:49:00 +00006930}
6931
6932
Steve Block44f0eee2011-05-26 01:26:41 +01006933template <typename schar>
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006934uint32_t StringHasher::HashSequentialString(const schar* chars,
6935 int length,
6936 uint32_t seed) {
Ben Murdochc7cc0282012-03-05 14:35:55 +00006937 StringHasher hasher(length, seed);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006938 if (!hasher.has_trivial_hash()) hasher.AddCharacters(chars, length);
6939 return hasher.GetHashField();
6940}
6941
6942
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00006943IteratingStringHasher::IteratingStringHasher(int len, uint32_t seed)
6944 : StringHasher(len, seed) {}
6945
6946
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006947uint32_t IteratingStringHasher::Hash(String* string, uint32_t seed) {
6948 IteratingStringHasher hasher(string->length(), seed);
6949 // Nothing to do.
6950 if (hasher.has_trivial_hash()) return hasher.GetHashField();
6951 ConsString* cons_string = String::VisitFlat(&hasher, string);
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006952 if (cons_string == nullptr) return hasher.GetHashField();
6953 hasher.VisitConsString(cons_string);
Steve Block44f0eee2011-05-26 01:26:41 +01006954 return hasher.GetHashField();
6955}
6956
6957
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006958void IteratingStringHasher::VisitOneByteString(const uint8_t* chars,
6959 int length) {
6960 AddCharacters(chars, length);
6961}
6962
6963
6964void IteratingStringHasher::VisitTwoByteString(const uint16_t* chars,
6965 int length) {
6966 AddCharacters(chars, length);
6967}
6968
6969
6970bool Name::AsArrayIndex(uint32_t* index) {
6971 return IsString() && String::cast(this)->AsArrayIndex(index);
6972}
6973
6974
Steve Blocka7e24c12009-10-30 11:49:00 +00006975bool String::AsArrayIndex(uint32_t* index) {
Steve Blockd0582a62009-12-15 09:54:21 +00006976 uint32_t field = hash_field();
Ben Murdoch7f4d5bd2010-06-15 11:15:29 +01006977 if (IsHashFieldComputed(field) && (field & kIsNotArrayIndexMask)) {
6978 return false;
6979 }
Steve Blocka7e24c12009-10-30 11:49:00 +00006980 return SlowAsArrayIndex(index);
6981}
6982
6983
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006984void String::SetForwardedInternalizedString(String* canonical) {
6985 DCHECK(IsInternalizedString());
6986 DCHECK(HasHashCode());
6987 if (canonical == this) return; // No need to forward.
6988 DCHECK(SlowEquals(canonical));
6989 DCHECK(canonical->IsInternalizedString());
6990 DCHECK(canonical->HasHashCode());
Emily Bernierd0a1eb72015-03-24 16:35:39 -04006991 WRITE_FIELD(this, kHashFieldSlot, canonical);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006992 // Setting the hash field to a tagged value sets the LSB, causing the hash
6993 // code to be interpreted as uninitialized. We use this fact to recognize
6994 // that we have a forwarded string.
6995 DCHECK(!HasHashCode());
Steve Blocka7e24c12009-10-30 11:49:00 +00006996}
6997
6998
Ben Murdochb8a8cc12014-11-26 15:28:44 +00006999String* String::GetForwardedInternalizedString() {
7000 DCHECK(IsInternalizedString());
7001 if (HasHashCode()) return this;
Emily Bernierd0a1eb72015-03-24 16:35:39 -04007002 String* canonical = String::cast(READ_FIELD(this, kHashFieldSlot));
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007003 DCHECK(canonical->IsInternalizedString());
7004 DCHECK(SlowEquals(canonical));
7005 DCHECK(canonical->HasHashCode());
7006 return canonical;
7007}
7008
7009
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007010// static
Ben Murdoch097c5b22016-05-18 11:27:45 +01007011Maybe<bool> Object::GreaterThan(Handle<Object> x, Handle<Object> y) {
7012 Maybe<ComparisonResult> result = Compare(x, y);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007013 if (result.IsJust()) {
7014 switch (result.FromJust()) {
7015 case ComparisonResult::kGreaterThan:
7016 return Just(true);
7017 case ComparisonResult::kLessThan:
7018 case ComparisonResult::kEqual:
7019 case ComparisonResult::kUndefined:
7020 return Just(false);
7021 }
7022 }
7023 return Nothing<bool>();
7024}
7025
7026
7027// static
Ben Murdoch097c5b22016-05-18 11:27:45 +01007028Maybe<bool> Object::GreaterThanOrEqual(Handle<Object> x, Handle<Object> y) {
7029 Maybe<ComparisonResult> result = Compare(x, y);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007030 if (result.IsJust()) {
7031 switch (result.FromJust()) {
7032 case ComparisonResult::kEqual:
7033 case ComparisonResult::kGreaterThan:
7034 return Just(true);
7035 case ComparisonResult::kLessThan:
7036 case ComparisonResult::kUndefined:
7037 return Just(false);
7038 }
7039 }
7040 return Nothing<bool>();
7041}
7042
7043
7044// static
Ben Murdoch097c5b22016-05-18 11:27:45 +01007045Maybe<bool> Object::LessThan(Handle<Object> x, Handle<Object> y) {
7046 Maybe<ComparisonResult> result = Compare(x, y);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007047 if (result.IsJust()) {
7048 switch (result.FromJust()) {
7049 case ComparisonResult::kLessThan:
7050 return Just(true);
7051 case ComparisonResult::kEqual:
7052 case ComparisonResult::kGreaterThan:
7053 case ComparisonResult::kUndefined:
7054 return Just(false);
7055 }
7056 }
7057 return Nothing<bool>();
7058}
7059
7060
7061// static
Ben Murdoch097c5b22016-05-18 11:27:45 +01007062Maybe<bool> Object::LessThanOrEqual(Handle<Object> x, Handle<Object> y) {
7063 Maybe<ComparisonResult> result = Compare(x, y);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007064 if (result.IsJust()) {
7065 switch (result.FromJust()) {
7066 case ComparisonResult::kEqual:
7067 case ComparisonResult::kLessThan:
7068 return Just(true);
7069 case ComparisonResult::kGreaterThan:
7070 case ComparisonResult::kUndefined:
7071 return Just(false);
7072 }
7073 }
7074 return Nothing<bool>();
7075}
7076
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007077MaybeHandle<Object> Object::GetPropertyOrElement(Handle<Object> object,
Ben Murdoch097c5b22016-05-18 11:27:45 +01007078 Handle<Name> name) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007079 LookupIterator it =
7080 LookupIterator::PropertyOrElement(name->GetIsolate(), object, name);
Ben Murdoch097c5b22016-05-18 11:27:45 +01007081 return GetProperty(&it);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007082}
7083
Ben Murdochda12d292016-06-02 14:46:10 +01007084MaybeHandle<Object> Object::SetPropertyOrElement(Handle<Object> object,
7085 Handle<Name> name,
7086 Handle<Object> value,
7087 LanguageMode language_mode,
7088 StoreFromKeyed store_mode) {
7089 LookupIterator it =
7090 LookupIterator::PropertyOrElement(name->GetIsolate(), object, name);
7091 MAYBE_RETURN_NULL(SetProperty(&it, value, language_mode, store_mode));
7092 return value;
7093}
7094
Ben Murdoch097c5b22016-05-18 11:27:45 +01007095MaybeHandle<Object> Object::GetPropertyOrElement(Handle<Object> receiver,
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007096 Handle<Name> name,
Ben Murdoch097c5b22016-05-18 11:27:45 +01007097 Handle<JSReceiver> holder) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007098 LookupIterator it = LookupIterator::PropertyOrElement(
7099 name->GetIsolate(), receiver, name, holder);
Ben Murdoch097c5b22016-05-18 11:27:45 +01007100 return GetProperty(&it);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007101}
7102
7103
7104void JSReceiver::initialize_properties() {
7105 DCHECK(!GetHeap()->InNewSpace(GetHeap()->empty_fixed_array()));
7106 DCHECK(!GetHeap()->InNewSpace(GetHeap()->empty_properties_dictionary()));
7107 if (map()->is_dictionary_map()) {
7108 WRITE_FIELD(this, kPropertiesOffset,
7109 GetHeap()->empty_properties_dictionary());
7110 } else {
7111 WRITE_FIELD(this, kPropertiesOffset, GetHeap()->empty_fixed_array());
7112 }
7113}
7114
7115
7116bool JSReceiver::HasFastProperties() {
7117 DCHECK(properties()->IsDictionary() == map()->is_dictionary_map());
7118 return !properties()->IsDictionary();
7119}
7120
7121
7122NameDictionary* JSReceiver::property_dictionary() {
7123 DCHECK(!HasFastProperties());
7124 DCHECK(!IsJSGlobalObject());
7125 return NameDictionary::cast(properties());
7126}
7127
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007128Maybe<bool> JSReceiver::HasProperty(Handle<JSReceiver> object,
7129 Handle<Name> name) {
Ben Murdochda12d292016-06-02 14:46:10 +01007130 LookupIterator it = LookupIterator::PropertyOrElement(object->GetIsolate(),
7131 object, name, object);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007132 return HasProperty(&it);
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00007133}
7134
7135
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007136Maybe<bool> JSReceiver::HasOwnProperty(Handle<JSReceiver> object,
7137 Handle<Name> name) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007138 if (object->IsJSObject()) { // Shortcut
7139 LookupIterator it = LookupIterator::PropertyOrElement(
Ben Murdochc5610432016-08-08 18:44:38 +01007140 object->GetIsolate(), object, name, object, LookupIterator::OWN);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007141 return HasProperty(&it);
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00007142 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007143
7144 Maybe<PropertyAttributes> attributes =
7145 JSReceiver::GetOwnPropertyAttributes(object, name);
7146 MAYBE_RETURN(attributes, Nothing<bool>());
7147 return Just(attributes.FromJust() != ABSENT);
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00007148}
7149
Ben Murdochc5610432016-08-08 18:44:38 +01007150Maybe<bool> JSReceiver::HasOwnProperty(Handle<JSReceiver> object,
7151 uint32_t index) {
7152 if (object->IsJSObject()) { // Shortcut
7153 LookupIterator it(object->GetIsolate(), object, index, object,
7154 LookupIterator::OWN);
7155 return HasProperty(&it);
7156 }
7157
7158 Maybe<PropertyAttributes> attributes =
7159 JSReceiver::GetOwnPropertyAttributes(object, index);
7160 MAYBE_RETURN(attributes, Nothing<bool>());
7161 return Just(attributes.FromJust() != ABSENT);
7162}
Ben Murdoch3fb3ca82011-12-02 17:19:32 +00007163
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007164Maybe<PropertyAttributes> JSReceiver::GetPropertyAttributes(
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007165 Handle<JSReceiver> object, Handle<Name> name) {
Ben Murdochda12d292016-06-02 14:46:10 +01007166 LookupIterator it = LookupIterator::PropertyOrElement(name->GetIsolate(),
7167 object, name, object);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007168 return GetPropertyAttributes(&it);
Steve Blockd0582a62009-12-15 09:54:21 +00007169}
7170
7171
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007172Maybe<PropertyAttributes> JSReceiver::GetOwnPropertyAttributes(
7173 Handle<JSReceiver> object, Handle<Name> name) {
7174 LookupIterator it = LookupIterator::PropertyOrElement(
Ben Murdochc5610432016-08-08 18:44:38 +01007175 name->GetIsolate(), object, name, object, LookupIterator::OWN);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007176 return GetPropertyAttributes(&it);
7177}
7178
Ben Murdochc5610432016-08-08 18:44:38 +01007179Maybe<PropertyAttributes> JSReceiver::GetOwnPropertyAttributes(
7180 Handle<JSReceiver> object, uint32_t index) {
7181 LookupIterator it(object->GetIsolate(), object, index, object,
7182 LookupIterator::OWN);
7183 return GetPropertyAttributes(&it);
7184}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007185
7186Maybe<bool> JSReceiver::HasElement(Handle<JSReceiver> object, uint32_t index) {
Ben Murdochda12d292016-06-02 14:46:10 +01007187 LookupIterator it(object->GetIsolate(), object, index, object);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007188 return HasProperty(&it);
7189}
7190
7191
7192Maybe<PropertyAttributes> JSReceiver::GetElementAttributes(
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007193 Handle<JSReceiver> object, uint32_t index) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007194 Isolate* isolate = object->GetIsolate();
Ben Murdochda12d292016-06-02 14:46:10 +01007195 LookupIterator it(isolate, object, index, object);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007196 return GetPropertyAttributes(&it);
7197}
7198
7199
7200Maybe<PropertyAttributes> JSReceiver::GetOwnElementAttributes(
7201 Handle<JSReceiver> object, uint32_t index) {
7202 Isolate* isolate = object->GetIsolate();
Ben Murdochc5610432016-08-08 18:44:38 +01007203 LookupIterator it(isolate, object, index, object, LookupIterator::OWN);
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007204 return GetPropertyAttributes(&it);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007205}
7206
7207
7208bool JSGlobalObject::IsDetached() {
7209 return JSGlobalProxy::cast(global_proxy())->IsDetachedFrom(this);
7210}
7211
7212
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007213bool JSGlobalProxy::IsDetachedFrom(JSGlobalObject* global) const {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007214 const PrototypeIterator iter(this->GetIsolate(),
7215 const_cast<JSGlobalProxy*>(this));
7216 return iter.GetCurrent() != global;
7217}
7218
7219
7220Handle<Smi> JSReceiver::GetOrCreateIdentityHash(Handle<JSReceiver> object) {
7221 return object->IsJSProxy()
7222 ? JSProxy::GetOrCreateIdentityHash(Handle<JSProxy>::cast(object))
7223 : JSObject::GetOrCreateIdentityHash(Handle<JSObject>::cast(object));
7224}
7225
Ben Murdochda12d292016-06-02 14:46:10 +01007226Handle<Object> JSReceiver::GetIdentityHash(Isolate* isolate,
7227 Handle<JSReceiver> receiver) {
7228 return receiver->IsJSProxy() ? JSProxy::GetIdentityHash(
7229 isolate, Handle<JSProxy>::cast(receiver))
7230 : JSObject::GetIdentityHash(
7231 isolate, Handle<JSObject>::cast(receiver));
Steve Blockd0582a62009-12-15 09:54:21 +00007232}
7233
7234
Steve Blocka7e24c12009-10-30 11:49:00 +00007235bool AccessorInfo::all_can_read() {
7236 return BooleanBit::get(flag(), kAllCanReadBit);
7237}
7238
7239
7240void AccessorInfo::set_all_can_read(bool value) {
7241 set_flag(BooleanBit::set(flag(), kAllCanReadBit, value));
7242}
7243
7244
7245bool AccessorInfo::all_can_write() {
7246 return BooleanBit::get(flag(), kAllCanWriteBit);
7247}
7248
7249
7250void AccessorInfo::set_all_can_write(bool value) {
7251 set_flag(BooleanBit::set(flag(), kAllCanWriteBit, value));
7252}
7253
7254
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007255bool AccessorInfo::is_special_data_property() {
7256 return BooleanBit::get(flag(), kSpecialDataProperty);
7257}
7258
7259
7260void AccessorInfo::set_is_special_data_property(bool value) {
7261 set_flag(BooleanBit::set(flag(), kSpecialDataProperty, value));
7262}
7263
Ben Murdochda12d292016-06-02 14:46:10 +01007264bool AccessorInfo::is_sloppy() { return BooleanBit::get(flag(), kIsSloppy); }
7265
7266void AccessorInfo::set_is_sloppy(bool value) {
7267 set_flag(BooleanBit::set(flag(), kIsSloppy, value));
7268}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007269
Steve Blocka7e24c12009-10-30 11:49:00 +00007270PropertyAttributes AccessorInfo::property_attributes() {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007271 return AttributesField::decode(static_cast<uint32_t>(flag()));
Steve Blocka7e24c12009-10-30 11:49:00 +00007272}
7273
7274
7275void AccessorInfo::set_property_attributes(PropertyAttributes attributes) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007276 set_flag(AttributesField::update(flag(), attributes));
Steve Blocka7e24c12009-10-30 11:49:00 +00007277}
7278
Ben Murdoch8b112d22011-06-08 16:22:53 +01007279
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007280bool AccessorInfo::IsCompatibleReceiver(Object* receiver) {
7281 if (!HasExpectedReceiverType()) return true;
7282 if (!receiver->IsJSObject()) return false;
7283 return FunctionTemplateInfo::cast(expected_receiver_type())
7284 ->IsTemplateFor(JSObject::cast(receiver)->map());
7285}
7286
7287
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007288bool AccessorInfo::HasExpectedReceiverType() {
7289 return expected_receiver_type()->IsFunctionTemplateInfo();
7290}
7291
7292
7293Object* AccessorPair::get(AccessorComponent component) {
7294 return component == ACCESSOR_GETTER ? getter() : setter();
7295}
7296
7297
7298void AccessorPair::set(AccessorComponent component, Object* value) {
7299 if (component == ACCESSOR_GETTER) {
7300 set_getter(value);
7301 } else {
7302 set_setter(value);
7303 }
7304}
7305
7306
7307void AccessorPair::SetComponents(Object* getter, Object* setter) {
7308 if (!getter->IsNull()) set_getter(getter);
7309 if (!setter->IsNull()) set_setter(setter);
7310}
7311
7312
7313bool AccessorPair::Equals(AccessorPair* pair) {
7314 return (this == pair) || pair->Equals(getter(), setter());
7315}
7316
7317
7318bool AccessorPair::Equals(Object* getter_value, Object* setter_value) {
7319 return (getter() == getter_value) && (setter() == setter_value);
7320}
7321
7322
7323bool AccessorPair::ContainsAccessor() {
7324 return IsJSAccessor(getter()) || IsJSAccessor(setter());
7325}
7326
7327
7328bool AccessorPair::IsJSAccessor(Object* obj) {
7329 return obj->IsCallable() || obj->IsUndefined();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007330}
7331
7332
7333template<typename Derived, typename Shape, typename Key>
7334void Dictionary<Derived, Shape, Key>::SetEntry(int entry,
7335 Handle<Object> key,
7336 Handle<Object> value) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007337 this->SetEntry(entry, key, value, PropertyDetails(Smi::FromInt(0)));
Ben Murdoch8b112d22011-06-08 16:22:53 +01007338}
7339
7340
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007341template<typename Derived, typename Shape, typename Key>
7342void Dictionary<Derived, Shape, Key>::SetEntry(int entry,
7343 Handle<Object> key,
7344 Handle<Object> value,
7345 PropertyDetails details) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007346 Shape::SetEntry(static_cast<Derived*>(this), entry, key, value, details);
7347}
7348
7349
7350template <typename Key>
7351template <typename Dictionary>
7352void BaseDictionaryShape<Key>::SetEntry(Dictionary* dict, int entry,
7353 Handle<Object> key,
7354 Handle<Object> value,
7355 PropertyDetails details) {
7356 STATIC_ASSERT(Dictionary::kEntrySize == 3);
7357 DCHECK(!key->IsName() || details.dictionary_index() > 0);
7358 int index = dict->EntryToIndex(entry);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007359 DisallowHeapAllocation no_gc;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007360 WriteBarrierMode mode = dict->GetWriteBarrierMode(no_gc);
7361 dict->set(index, *key, mode);
7362 dict->set(index + 1, *value, mode);
7363 dict->set(index + 2, details.AsSmi());
7364}
7365
7366
7367template <typename Dictionary>
7368void GlobalDictionaryShape::SetEntry(Dictionary* dict, int entry,
7369 Handle<Object> key, Handle<Object> value,
7370 PropertyDetails details) {
7371 STATIC_ASSERT(Dictionary::kEntrySize == 2);
7372 DCHECK(!key->IsName() || details.dictionary_index() > 0);
7373 DCHECK(value->IsPropertyCell());
7374 int index = dict->EntryToIndex(entry);
7375 DisallowHeapAllocation no_gc;
7376 WriteBarrierMode mode = dict->GetWriteBarrierMode(no_gc);
7377 dict->set(index, *key, mode);
7378 dict->set(index + 1, *value, mode);
7379 PropertyCell::cast(*value)->set_property_details(details);
Steve Blocka7e24c12009-10-30 11:49:00 +00007380}
7381
7382
Steve Block44f0eee2011-05-26 01:26:41 +01007383bool NumberDictionaryShape::IsMatch(uint32_t key, Object* other) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007384 DCHECK(other->IsNumber());
Steve Block44f0eee2011-05-26 01:26:41 +01007385 return key == static_cast<uint32_t>(other->Number());
7386}
7387
7388
Ben Murdochc7cc0282012-03-05 14:35:55 +00007389uint32_t UnseededNumberDictionaryShape::Hash(uint32_t key) {
7390 return ComputeIntegerHash(key, 0);
Steve Block44f0eee2011-05-26 01:26:41 +01007391}
7392
7393
Ben Murdochc7cc0282012-03-05 14:35:55 +00007394uint32_t UnseededNumberDictionaryShape::HashForObject(uint32_t key,
7395 Object* other) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007396 DCHECK(other->IsNumber());
Ben Murdochc7cc0282012-03-05 14:35:55 +00007397 return ComputeIntegerHash(static_cast<uint32_t>(other->Number()), 0);
Steve Block44f0eee2011-05-26 01:26:41 +01007398}
7399
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007400
Ben Murdochc7cc0282012-03-05 14:35:55 +00007401uint32_t SeededNumberDictionaryShape::SeededHash(uint32_t key, uint32_t seed) {
7402 return ComputeIntegerHash(key, seed);
7403}
7404
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007405
Ben Murdochc7cc0282012-03-05 14:35:55 +00007406uint32_t SeededNumberDictionaryShape::SeededHashForObject(uint32_t key,
7407 uint32_t seed,
7408 Object* other) {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007409 DCHECK(other->IsNumber());
Ben Murdochc7cc0282012-03-05 14:35:55 +00007410 return ComputeIntegerHash(static_cast<uint32_t>(other->Number()), seed);
7411}
Steve Block44f0eee2011-05-26 01:26:41 +01007412
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007413
7414Handle<Object> NumberDictionaryShape::AsHandle(Isolate* isolate, uint32_t key) {
7415 return isolate->factory()->NewNumberFromUint(key);
Steve Block44f0eee2011-05-26 01:26:41 +01007416}
7417
7418
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007419bool NameDictionaryShape::IsMatch(Handle<Name> key, Object* other) {
Steve Block44f0eee2011-05-26 01:26:41 +01007420 // We know that all entries in a hash table had their hash keys created.
7421 // Use that knowledge to have fast failure.
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007422 if (key->Hash() != Name::cast(other)->Hash()) return false;
7423 return key->Equals(Name::cast(other));
Steve Block44f0eee2011-05-26 01:26:41 +01007424}
7425
7426
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007427uint32_t NameDictionaryShape::Hash(Handle<Name> key) {
Steve Block44f0eee2011-05-26 01:26:41 +01007428 return key->Hash();
7429}
7430
7431
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007432uint32_t NameDictionaryShape::HashForObject(Handle<Name> key, Object* other) {
7433 return Name::cast(other)->Hash();
Steve Block44f0eee2011-05-26 01:26:41 +01007434}
7435
7436
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007437Handle<Object> NameDictionaryShape::AsHandle(Isolate* isolate,
7438 Handle<Name> key) {
7439 DCHECK(key->IsUniqueName());
Steve Block44f0eee2011-05-26 01:26:41 +01007440 return key;
7441}
7442
7443
Emily Bernierd0a1eb72015-03-24 16:35:39 -04007444Handle<FixedArray> NameDictionary::DoGenerateNewEnumerationIndices(
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007445 Handle<NameDictionary> dictionary) {
Emily Bernierd0a1eb72015-03-24 16:35:39 -04007446 return DerivedDictionary::GenerateNewEnumerationIndices(dictionary);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007447}
7448
7449
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007450template <typename Dictionary>
7451PropertyDetails GlobalDictionaryShape::DetailsAt(Dictionary* dict, int entry) {
7452 DCHECK(entry >= 0); // Not found is -1, which is not caught by get().
7453 Object* raw_value = dict->ValueAt(entry);
7454 DCHECK(raw_value->IsPropertyCell());
7455 PropertyCell* cell = PropertyCell::cast(raw_value);
7456 return cell->property_details();
7457}
7458
7459
7460template <typename Dictionary>
7461void GlobalDictionaryShape::DetailsAtPut(Dictionary* dict, int entry,
7462 PropertyDetails value) {
7463 DCHECK(entry >= 0); // Not found is -1, which is not caught by get().
7464 Object* raw_value = dict->ValueAt(entry);
7465 DCHECK(raw_value->IsPropertyCell());
7466 PropertyCell* cell = PropertyCell::cast(raw_value);
7467 cell->set_property_details(value);
7468}
7469
7470
7471template <typename Dictionary>
7472bool GlobalDictionaryShape::IsDeleted(Dictionary* dict, int entry) {
7473 DCHECK(dict->ValueAt(entry)->IsPropertyCell());
7474 return PropertyCell::cast(dict->ValueAt(entry))->value()->IsTheHole();
7475}
7476
7477
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007478bool ObjectHashTableShape::IsMatch(Handle<Object> key, Object* other) {
7479 return key->SameValue(other);
7480}
7481
7482
7483uint32_t ObjectHashTableShape::Hash(Handle<Object> key) {
7484 return Smi::cast(key->GetHash())->value();
7485}
7486
7487
7488uint32_t ObjectHashTableShape::HashForObject(Handle<Object> key,
7489 Object* other) {
7490 return Smi::cast(other->GetHash())->value();
7491}
7492
7493
7494Handle<Object> ObjectHashTableShape::AsHandle(Isolate* isolate,
7495 Handle<Object> key) {
7496 return key;
7497}
7498
7499
7500Handle<ObjectHashTable> ObjectHashTable::Shrink(
7501 Handle<ObjectHashTable> table, Handle<Object> key) {
7502 return DerivedHashTable::Shrink(table, key);
7503}
7504
7505
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007506Object* OrderedHashMap::ValueAt(int entry) {
7507 return get(EntryToIndex(entry) + kValueOffset);
7508}
7509
7510
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007511template <int entrysize>
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007512bool WeakHashTableShape<entrysize>::IsMatch(Handle<Object> key, Object* other) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007513 if (other->IsWeakCell()) other = WeakCell::cast(other)->value();
7514 return key->IsWeakCell() ? WeakCell::cast(*key)->value() == other
7515 : *key == other;
Ben Murdoch69a99ed2011-11-30 16:03:39 +00007516}
7517
7518
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007519template <int entrysize>
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007520uint32_t WeakHashTableShape<entrysize>::Hash(Handle<Object> key) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007521 intptr_t hash =
7522 key->IsWeakCell()
7523 ? reinterpret_cast<intptr_t>(WeakCell::cast(*key)->value())
7524 : reinterpret_cast<intptr_t>(*key);
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007525 return (uint32_t)(hash & 0xFFFFFFFF);
Ben Murdoch69a99ed2011-11-30 16:03:39 +00007526}
7527
7528
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007529template <int entrysize>
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007530uint32_t WeakHashTableShape<entrysize>::HashForObject(Handle<Object> key,
7531 Object* other) {
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007532 if (other->IsWeakCell()) other = WeakCell::cast(other)->value();
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007533 intptr_t hash = reinterpret_cast<intptr_t>(other);
7534 return (uint32_t)(hash & 0xFFFFFFFF);
Ben Murdoch69a99ed2011-11-30 16:03:39 +00007535}
7536
7537
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007538template <int entrysize>
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007539Handle<Object> WeakHashTableShape<entrysize>::AsHandle(Isolate* isolate,
7540 Handle<Object> key) {
Ben Murdoch69a99ed2011-11-30 16:03:39 +00007541 return key;
7542}
7543
7544
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007545bool ScopeInfo::IsAsmModule() { return AsmModuleField::decode(Flags()); }
7546
7547
7548bool ScopeInfo::IsAsmFunction() { return AsmFunctionField::decode(Flags()); }
7549
7550
7551bool ScopeInfo::HasSimpleParameters() {
7552 return HasSimpleParametersField::decode(Flags());
7553}
7554
7555
7556#define SCOPE_INFO_FIELD_ACCESSORS(name) \
7557 void ScopeInfo::Set##name(int value) { set(k##name, Smi::FromInt(value)); } \
7558 int ScopeInfo::name() { \
7559 if (length() > 0) { \
7560 return Smi::cast(get(k##name))->value(); \
7561 } else { \
7562 return 0; \
7563 } \
7564 }
7565FOR_EACH_SCOPE_INFO_NUMERIC_FIELD(SCOPE_INFO_FIELD_ACCESSORS)
7566#undef SCOPE_INFO_FIELD_ACCESSORS
7567
7568
Steve Block44f0eee2011-05-26 01:26:41 +01007569void Map::ClearCodeCache(Heap* heap) {
Steve Blocka7e24c12009-10-30 11:49:00 +00007570 // No write barrier is needed since empty_fixed_array is not in new space.
7571 // Please note this function is used during marking:
7572 // - MarkCompactCollector::MarkUnmarkedObject
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007573 // - IncrementalMarking::Step
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007574 WRITE_FIELD(this, kCodeCacheOffset, heap->empty_fixed_array());
Steve Blocka7e24c12009-10-30 11:49:00 +00007575}
7576
7577
Emily Bernierd0a1eb72015-03-24 16:35:39 -04007578int Map::SlackForArraySize(int old_size, int size_limit) {
7579 const int max_slack = size_limit - old_size;
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007580 CHECK_LE(0, max_slack);
7581 if (old_size < 4) {
7582 DCHECK_LE(1, max_slack);
7583 return 1;
Steve Blockd0582a62009-12-15 09:54:21 +00007584 }
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007585 return Min(max_slack, old_size / 4);
Steve Blocka7e24c12009-10-30 11:49:00 +00007586}
7587
7588
Leon Clarke4515c472010-02-03 11:58:03 +00007589void JSArray::set_length(Smi* length) {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007590 // Don't need a write barrier for a Smi.
Leon Clarke4515c472010-02-03 11:58:03 +00007591 set_length(static_cast<Object*>(length), SKIP_WRITE_BARRIER);
7592}
7593
7594
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007595bool JSArray::SetLengthWouldNormalize(Heap* heap, uint32_t new_length) {
7596 // If the new array won't fit in a some non-trivial fraction of the max old
7597 // space size, then force it to go dictionary mode.
7598 uint32_t max_fast_array_size =
7599 static_cast<uint32_t>((heap->MaxOldGenerationSize() / kDoubleSize) / 4);
7600 return new_length >= max_fast_array_size;
7601}
7602
7603
7604bool JSArray::AllowsSetLength() {
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007605 bool result = elements()->IsFixedArray() || elements()->IsFixedDoubleArray();
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007606 DCHECK(result == !HasFixedTypedArrayElements());
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007607 return result;
7608}
7609
7610
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007611void JSArray::SetContent(Handle<JSArray> array,
7612 Handle<FixedArrayBase> storage) {
7613 EnsureCanContainElements(array, storage, storage->length(),
7614 ALLOW_COPIED_DOUBLE_ELEMENTS);
7615
7616 DCHECK((storage->map() == array->GetHeap()->fixed_double_array_map() &&
7617 IsFastDoubleElementsKind(array->GetElementsKind())) ||
7618 ((storage->map() != array->GetHeap()->fixed_double_array_map()) &&
7619 (IsFastObjectElementsKind(array->GetElementsKind()) ||
7620 (IsFastSmiElementsKind(array->GetElementsKind()) &&
7621 Handle<FixedArray>::cast(storage)->ContainsOnlySmisOrHoles()))));
7622 array->set_elements(*storage);
7623 array->set_length(Smi::FromInt(storage->length()));
Steve Blocka7e24c12009-10-30 11:49:00 +00007624}
7625
7626
Ben Murdochda12d292016-06-02 14:46:10 +01007627bool JSArray::HasArrayPrototype(Isolate* isolate) {
7628 return map()->prototype() == *isolate->initial_array_prototype();
7629}
7630
7631
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007632int TypeFeedbackInfo::ic_total_count() {
7633 int current = Smi::cast(READ_FIELD(this, kStorage1Offset))->value();
7634 return ICTotalCountField::decode(current);
Steve Block44f0eee2011-05-26 01:26:41 +01007635}
7636
7637
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007638void TypeFeedbackInfo::set_ic_total_count(int count) {
7639 int value = Smi::cast(READ_FIELD(this, kStorage1Offset))->value();
7640 value = ICTotalCountField::update(value,
7641 ICTotalCountField::decode(count));
7642 WRITE_FIELD(this, kStorage1Offset, Smi::FromInt(value));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007643}
7644
7645
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007646int TypeFeedbackInfo::ic_with_type_info_count() {
7647 int current = Smi::cast(READ_FIELD(this, kStorage2Offset))->value();
7648 return ICsWithTypeInfoCountField::decode(current);
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007649}
7650
7651
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007652void TypeFeedbackInfo::change_ic_with_type_info_count(int delta) {
7653 if (delta == 0) return;
7654 int value = Smi::cast(READ_FIELD(this, kStorage2Offset))->value();
7655 int new_count = ICsWithTypeInfoCountField::decode(value) + delta;
7656 // We can get negative count here when the type-feedback info is
7657 // shared between two code objects. The can only happen when
7658 // the debugger made a shallow copy of code object (see Heap::CopyCode).
7659 // Since we do not optimize when the debugger is active, we can skip
7660 // this counter update.
7661 if (new_count >= 0) {
7662 new_count &= ICsWithTypeInfoCountField::kMask;
7663 value = ICsWithTypeInfoCountField::update(value, new_count);
7664 WRITE_FIELD(this, kStorage2Offset, Smi::FromInt(value));
7665 }
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007666}
7667
7668
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007669int TypeFeedbackInfo::ic_generic_count() {
7670 return Smi::cast(READ_FIELD(this, kStorage3Offset))->value();
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007671}
7672
7673
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007674void TypeFeedbackInfo::change_ic_generic_count(int delta) {
7675 if (delta == 0) return;
7676 int new_count = ic_generic_count() + delta;
7677 if (new_count >= 0) {
7678 new_count &= ~Smi::kMinValue;
7679 WRITE_FIELD(this, kStorage3Offset, Smi::FromInt(new_count));
7680 }
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007681}
7682
7683
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007684void TypeFeedbackInfo::initialize_storage() {
7685 WRITE_FIELD(this, kStorage1Offset, Smi::FromInt(0));
7686 WRITE_FIELD(this, kStorage2Offset, Smi::FromInt(0));
7687 WRITE_FIELD(this, kStorage3Offset, Smi::FromInt(0));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007688}
7689
7690
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007691void TypeFeedbackInfo::change_own_type_change_checksum() {
7692 int value = Smi::cast(READ_FIELD(this, kStorage1Offset))->value();
7693 int checksum = OwnTypeChangeChecksum::decode(value);
7694 checksum = (checksum + 1) % (1 << kTypeChangeChecksumBits);
7695 value = OwnTypeChangeChecksum::update(value, checksum);
7696 // Ensure packed bit field is in Smi range.
7697 if (value > Smi::kMaxValue) value |= Smi::kMinValue;
7698 if (value < Smi::kMinValue) value &= ~Smi::kMinValue;
7699 WRITE_FIELD(this, kStorage1Offset, Smi::FromInt(value));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007700}
7701
7702
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007703void TypeFeedbackInfo::set_inlined_type_change_checksum(int checksum) {
7704 int value = Smi::cast(READ_FIELD(this, kStorage2Offset))->value();
7705 int mask = (1 << kTypeChangeChecksumBits) - 1;
7706 value = InlinedTypeChangeChecksum::update(value, checksum & mask);
7707 // Ensure packed bit field is in Smi range.
7708 if (value > Smi::kMaxValue) value |= Smi::kMinValue;
7709 if (value < Smi::kMinValue) value &= ~Smi::kMinValue;
7710 WRITE_FIELD(this, kStorage2Offset, Smi::FromInt(value));
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007711}
7712
7713
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007714int TypeFeedbackInfo::own_type_change_checksum() {
7715 int value = Smi::cast(READ_FIELD(this, kStorage1Offset))->value();
7716 return OwnTypeChangeChecksum::decode(value);
7717}
7718
7719
7720bool TypeFeedbackInfo::matches_inlined_type_change_checksum(int checksum) {
7721 int value = Smi::cast(READ_FIELD(this, kStorage2Offset))->value();
7722 int mask = (1 << kTypeChangeChecksumBits) - 1;
7723 return InlinedTypeChangeChecksum::decode(value) == (checksum & mask);
7724}
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007725
7726
7727SMI_ACCESSORS(AliasedArgumentsEntry, aliased_context_slot, kAliasedContextSlot)
7728
7729
Steve Block44f0eee2011-05-26 01:26:41 +01007730Relocatable::Relocatable(Isolate* isolate) {
Steve Block44f0eee2011-05-26 01:26:41 +01007731 isolate_ = isolate;
7732 prev_ = isolate->relocatable_top();
7733 isolate->set_relocatable_top(this);
7734}
7735
7736
7737Relocatable::~Relocatable() {
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007738 DCHECK_EQ(isolate_->relocatable_top(), this);
Steve Block44f0eee2011-05-26 01:26:41 +01007739 isolate_->set_relocatable_top(prev_);
Steve Blocka7e24c12009-10-30 11:49:00 +00007740}
7741
7742
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007743template<class Derived, class TableType>
7744Object* OrderedHashTableIterator<Derived, TableType>::CurrentKey() {
7745 TableType* table(TableType::cast(this->table()));
7746 int index = Smi::cast(this->index())->value();
7747 Object* key = table->KeyAt(index);
7748 DCHECK(!key->IsTheHole());
7749 return key;
7750}
7751
7752
7753void JSSetIterator::PopulateValueArray(FixedArray* array) {
7754 array->set(0, CurrentKey());
7755}
7756
7757
7758void JSMapIterator::PopulateValueArray(FixedArray* array) {
7759 array->set(0, CurrentKey());
7760 array->set(1, CurrentValue());
7761}
7762
7763
7764Object* JSMapIterator::CurrentValue() {
7765 OrderedHashMap* table(OrderedHashMap::cast(this->table()));
7766 int index = Smi::cast(this->index())->value();
7767 Object* value = table->ValueAt(index);
7768 DCHECK(!value->IsTheHole());
7769 return value;
7770}
7771
Iain Merrick75681382010-08-19 15:07:18 +01007772
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007773String::SubStringRange::SubStringRange(String* string, int first, int length)
7774 : string_(string),
7775 first_(first),
7776 length_(length == -1 ? string->length() : length) {}
7777
7778
7779class String::SubStringRange::iterator final {
7780 public:
7781 typedef std::forward_iterator_tag iterator_category;
7782 typedef int difference_type;
7783 typedef uc16 value_type;
7784 typedef uc16* pointer;
7785 typedef uc16& reference;
7786
7787 iterator(const iterator& other)
7788 : content_(other.content_), offset_(other.offset_) {}
7789
7790 uc16 operator*() { return content_.Get(offset_); }
7791 bool operator==(const iterator& other) const {
7792 return content_.UsesSameString(other.content_) && offset_ == other.offset_;
7793 }
7794 bool operator!=(const iterator& other) const {
7795 return !content_.UsesSameString(other.content_) || offset_ != other.offset_;
7796 }
7797 iterator& operator++() {
7798 ++offset_;
7799 return *this;
7800 }
7801 iterator operator++(int);
7802
7803 private:
7804 friend class String;
7805 iterator(String* from, int offset)
7806 : content_(from->GetFlatContent()), offset_(offset) {}
7807 String::FlatContent content_;
7808 int offset_;
7809};
7810
7811
7812String::SubStringRange::iterator String::SubStringRange::begin() {
7813 return String::SubStringRange::iterator(string_, first_);
7814}
7815
7816
7817String::SubStringRange::iterator String::SubStringRange::end() {
7818 return String::SubStringRange::iterator(string_, first_ + length_);
7819}
7820
7821
7822// Predictably converts HeapObject* or Address to uint32 by calculating
7823// offset of the address in respective MemoryChunk.
7824static inline uint32_t ObjectAddressForHashing(void* object) {
7825 uint32_t value = static_cast<uint32_t>(reinterpret_cast<uintptr_t>(object));
7826 return value & MemoryChunk::kAlignmentMask;
7827}
7828
Ben Murdochda12d292016-06-02 14:46:10 +01007829static inline Handle<Object> MakeEntryPair(Isolate* isolate, uint32_t index,
7830 Handle<Object> value) {
7831 Handle<Object> key = isolate->factory()->Uint32ToString(index);
7832 Handle<FixedArray> entry_storage =
7833 isolate->factory()->NewUninitializedFixedArray(2);
7834 {
7835 entry_storage->set(0, *key, SKIP_WRITE_BARRIER);
7836 entry_storage->set(1, *value, SKIP_WRITE_BARRIER);
7837 }
7838 return isolate->factory()->NewJSArrayWithElements(entry_storage,
7839 FAST_ELEMENTS, 2);
7840}
7841
7842static inline Handle<Object> MakeEntryPair(Isolate* isolate, Handle<Name> key,
7843 Handle<Object> value) {
7844 Handle<FixedArray> entry_storage =
7845 isolate->factory()->NewUninitializedFixedArray(2);
7846 {
7847 entry_storage->set(0, *key, SKIP_WRITE_BARRIER);
7848 entry_storage->set(1, *value, SKIP_WRITE_BARRIER);
7849 }
7850 return isolate->factory()->NewJSArrayWithElements(entry_storage,
7851 FAST_ELEMENTS, 2);
7852}
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007853
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007854#undef TYPE_CHECKER
Steve Blocka7e24c12009-10-30 11:49:00 +00007855#undef CAST_ACCESSOR
7856#undef INT_ACCESSORS
Ben Murdoch85b71792012-04-11 18:30:58 +01007857#undef ACCESSORS
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007858#undef SMI_ACCESSORS
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007859#undef SYNCHRONIZED_SMI_ACCESSORS
7860#undef NOBARRIER_SMI_ACCESSORS
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007861#undef BOOL_GETTER
7862#undef BOOL_ACCESSORS
Steve Blocka7e24c12009-10-30 11:49:00 +00007863#undef FIELD_ADDR
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007864#undef FIELD_ADDR_CONST
Steve Blocka7e24c12009-10-30 11:49:00 +00007865#undef READ_FIELD
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007866#undef NOBARRIER_READ_FIELD
Steve Blocka7e24c12009-10-30 11:49:00 +00007867#undef WRITE_FIELD
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007868#undef NOBARRIER_WRITE_FIELD
Steve Blocka7e24c12009-10-30 11:49:00 +00007869#undef WRITE_BARRIER
7870#undef CONDITIONAL_WRITE_BARRIER
Steve Blocka7e24c12009-10-30 11:49:00 +00007871#undef READ_DOUBLE_FIELD
7872#undef WRITE_DOUBLE_FIELD
7873#undef READ_INT_FIELD
7874#undef WRITE_INT_FIELD
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007875#undef READ_INTPTR_FIELD
7876#undef WRITE_INTPTR_FIELD
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007877#undef READ_UINT8_FIELD
7878#undef WRITE_UINT8_FIELD
7879#undef READ_INT8_FIELD
7880#undef WRITE_INT8_FIELD
7881#undef READ_UINT16_FIELD
7882#undef WRITE_UINT16_FIELD
7883#undef READ_INT16_FIELD
7884#undef WRITE_INT16_FIELD
Ben Murdoch3ef787d2012-04-12 10:51:47 +01007885#undef READ_UINT32_FIELD
7886#undef WRITE_UINT32_FIELD
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007887#undef READ_INT32_FIELD
7888#undef WRITE_INT32_FIELD
7889#undef READ_FLOAT_FIELD
7890#undef WRITE_FLOAT_FIELD
7891#undef READ_UINT64_FIELD
7892#undef WRITE_UINT64_FIELD
7893#undef READ_INT64_FIELD
7894#undef WRITE_INT64_FIELD
Steve Blocka7e24c12009-10-30 11:49:00 +00007895#undef READ_BYTE_FIELD
7896#undef WRITE_BYTE_FIELD
Ben Murdochb8a8cc12014-11-26 15:28:44 +00007897#undef NOBARRIER_READ_BYTE_FIELD
7898#undef NOBARRIER_WRITE_BYTE_FIELD
Steve Blocka7e24c12009-10-30 11:49:00 +00007899
Ben Murdoch4a90d5f2016-03-22 12:00:34 +00007900} // namespace internal
7901} // namespace v8
Steve Blocka7e24c12009-10-30 11:49:00 +00007902
7903#endif // V8_OBJECTS_INL_H_