// Copyright 2014 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef V8_FIELD_INDEX_INL_H_ #define V8_FIELD_INDEX_INL_H_ #include "src/field-index.h" #include "src/ic/handler-configuration.h" namespace v8 { namespace internal { inline FieldIndex FieldIndex::ForInObjectOffset(int offset, Map* map) { DCHECK((offset % kPointerSize) == 0); int index = offset / kPointerSize; DCHECK(map == NULL || index < (map->GetInObjectPropertyOffset(0) / kPointerSize + map->GetInObjectProperties())); return FieldIndex(true, index, false, 0, 0, true); } inline FieldIndex FieldIndex::ForPropertyIndex(Map* map, int property_index, bool is_double) { DCHECK(map->instance_type() >= FIRST_NONSTRING_TYPE); int inobject_properties = map->GetInObjectProperties(); bool is_inobject = property_index < inobject_properties; int first_inobject_offset; if (is_inobject) { first_inobject_offset = map->GetInObjectPropertyOffset(0); } else { first_inobject_offset = FixedArray::kHeaderSize; property_index -= inobject_properties; } return FieldIndex(is_inobject, property_index + first_inobject_offset / kPointerSize, is_double, inobject_properties, first_inobject_offset); } // Takes an index as computed by GetLoadByFieldIndex and reconstructs a // FieldIndex object from it. inline FieldIndex FieldIndex::ForLoadByFieldIndex(Map* map, int orig_index) { int field_index = orig_index; int is_inobject = true; bool is_double = field_index & 1; int first_inobject_offset = 0; field_index >>= 1; if (field_index < 0) { field_index = -(field_index + 1); is_inobject = false; first_inobject_offset = FixedArray::kHeaderSize; field_index += FixedArray::kHeaderSize / kPointerSize; } else { first_inobject_offset = map->GetInObjectPropertyOffset(0); field_index += JSObject::kHeaderSize / kPointerSize; } FieldIndex result(is_inobject, field_index, is_double, map->GetInObjectProperties(), first_inobject_offset); DCHECK(result.GetLoadByFieldIndex() == orig_index); return result; } // Returns the index format accepted by the HLoadFieldByIndex instruction. // (In-object: zero-based from (object start + JSObject::kHeaderSize), // out-of-object: zero-based from FixedArray::kHeaderSize.) inline int FieldIndex::GetLoadByFieldIndex() const { // For efficiency, the LoadByFieldIndex instruction takes an index that is // optimized for quick access. If the property is inline, the index is // positive. If it's out-of-line, the encoded index is -raw_index - 1 to // disambiguate the zero out-of-line index from the zero inobject case. // The index itself is shifted up by one bit, the lower-most bit // signifying if the field is a mutable double box (1) or not (0). int result = index(); if (is_inobject()) { result -= JSObject::kHeaderSize / kPointerSize; } else { result -= FixedArray::kHeaderSize / kPointerSize; result = -result - 1; } result <<= 1; return is_double() ? (result | 1) : result; } // Takes an offset as computed by GetLoadByFieldOffset and reconstructs a // FieldIndex object from it. // static inline FieldIndex FieldIndex::ForLoadByFieldOffset(Map* map, int offset) { DCHECK(LoadHandlerTypeBit::decode(offset) == kLoadICHandlerForProperties); bool is_inobject = FieldOffsetIsInobject::decode(offset); bool is_double = FieldOffsetIsDouble::decode(offset); int field_index = FieldOffsetOffset::decode(offset) >> kPointerSizeLog2; int first_inobject_offset = 0; if (is_inobject) { first_inobject_offset = map->IsJSObjectMap() ? map->GetInObjectPropertyOffset(0) : 0; } else { first_inobject_offset = FixedArray::kHeaderSize; } int inobject_properties = map->IsJSObjectMap() ? map->GetInObjectProperties() : 0; FieldIndex result(is_inobject, field_index, is_double, inobject_properties, first_inobject_offset); DCHECK(result.GetLoadByFieldOffset() == offset); return result; } // Returns the offset format consumed by TurboFan stubs: // (offset << 3) | (is_double << 2) | (is_inobject << 1) | is_property // Where |offset| is relative to object start or FixedArray start, respectively. inline int FieldIndex::GetLoadByFieldOffset() const { return FieldOffsetIsInobject::encode(is_inobject()) | FieldOffsetIsDouble::encode(is_double()) | FieldOffsetOffset::encode(index() << kPointerSizeLog2) | LoadHandlerTypeBit::encode(kLoadICHandlerForProperties); } inline FieldIndex FieldIndex::ForDescriptor(Map* map, int descriptor_index) { PropertyDetails details = map->instance_descriptors()->GetDetails(descriptor_index); int field_index = details.field_index(); return ForPropertyIndex(map, field_index, details.representation().IsDouble()); } inline FieldIndex FieldIndex::ForKeyedLookupCacheIndex(Map* map, int index) { if (FLAG_compiled_keyed_generic_loads) { return ForLoadByFieldIndex(map, index); } else { return ForPropertyIndex(map, index); } } inline FieldIndex FieldIndex::FromFieldAccessStubKey(int key) { return FieldIndex(key); } inline int FieldIndex::GetKeyedLookupCacheIndex() const { if (FLAG_compiled_keyed_generic_loads) { return GetLoadByFieldIndex(); } else { return property_index(); } } } // namespace internal } // namespace v8 #endif