transitions-inl.h 5.52 KB
Newer Older
1
// Copyright 2012 the V8 project authors. All rights reserved.
2 3
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
4 5 6 7

#ifndef V8_TRANSITIONS_INL_H_
#define V8_TRANSITIONS_INL_H_

8
#include "src/transitions.h"
9 10 11 12 13 14

namespace v8 {
namespace internal {


TransitionArray* TransitionArray::cast(Object* object) {
15
  DCHECK(object->IsTransitionArray());
16 17 18 19
  return reinterpret_cast<TransitionArray*>(object);
}


20 21 22 23 24 25 26 27
Object* TransitionArray::next_link() { return get(kNextLinkIndex); }


void TransitionArray::set_next_link(Object* next, WriteBarrierMode mode) {
  return set(kNextLinkIndex, next, mode);
}


28
bool TransitionArray::HasPrototypeTransitions() {
29
  return get(kPrototypeTransitionsIndex) != Smi::FromInt(0);
30 31 32 33
}


FixedArray* TransitionArray::GetPrototypeTransitions() {
34
  DCHECK(HasPrototypeTransitions());  // Callers must check first.
35 36 37 38 39
  Object* prototype_transitions = get(kPrototypeTransitionsIndex);
  return FixedArray::cast(prototype_transitions);
}


40
void TransitionArray::SetPrototypeTransitions(FixedArray* transitions) {
41
  DCHECK(transitions->IsFixedArray());
42
  set(kPrototypeTransitionsIndex, transitions);
43 44 45 46
}


Object** TransitionArray::GetPrototypeTransitionsSlot() {
47
  return RawFieldOfElementAt(kPrototypeTransitionsIndex);
48 49 50
}


51
Object** TransitionArray::GetKeySlot(int transition_number) {
52
  DCHECK(transition_number < number_of_transitions());
53
  return RawFieldOfElementAt(ToKeyIndex(transition_number));
54 55 56
}


57
Name* TransitionArray::GetKey(int transition_number) {
58
  DCHECK(transition_number < number_of_transitions());
59
  return Name::cast(get(ToKeyIndex(transition_number)));
60 61 62
}


63 64 65 66 67 68 69 70 71 72
Name* TransitionArray::GetKey(Object* raw_transitions, int transition_number) {
  if (IsSimpleTransition(raw_transitions)) {
    DCHECK(transition_number == 0);
    return GetSimpleTransitionKey(GetSimpleTransition(raw_transitions));
  }
  DCHECK(IsFullTransitionArray(raw_transitions));
  return TransitionArray::cast(raw_transitions)->GetKey(transition_number);
}


73
void TransitionArray::SetKey(int transition_number, Name* key) {
74
  DCHECK(transition_number < number_of_transitions());
75 76 77 78
  set(ToKeyIndex(transition_number), key);
}


79
Map* TransitionArray::GetTarget(int transition_number) {
80
  DCHECK(transition_number < number_of_transitions());
81
  return Map::cast(get(ToTargetIndex(transition_number)));
82 83 84
}


85 86 87 88 89
Map* TransitionArray::GetTarget(Object* raw_transitions,
                                int transition_number) {
  if (IsSimpleTransition(raw_transitions)) {
    DCHECK(transition_number == 0);
    return GetSimpleTransition(raw_transitions);
90
  }
91 92
  DCHECK(IsFullTransitionArray(raw_transitions));
  return TransitionArray::cast(raw_transitions)->GetTarget(transition_number);
93 94 95
}


96
void TransitionArray::SetTarget(int transition_number, Map* value) {
97 98
  DCHECK(transition_number < number_of_transitions());
  set(ToTargetIndex(transition_number), value);
99 100 101
}


102
int TransitionArray::SearchName(Name* name, int* out_insertion_index) {
103 104 105
  DCHECK(name->IsUniqueName());
  return internal::Search<ALL_ENTRIES>(this, name, number_of_entries(),
                                       out_insertion_index);
106 107 108
}


109 110 111 112
#ifdef DEBUG
bool TransitionArray::IsSpecialTransition(Name* name) {
  if (!name->IsSymbol()) return false;
  Heap* heap = name->GetHeap();
113 114
  return name == heap->nonextensible_symbol() ||
         name == heap->sealed_symbol() || name == heap->frozen_symbol() ||
115
         name == heap->elements_transition_symbol() ||
116
         name == heap->strict_function_transition_symbol();
117 118 119 120
}
#endif


121
int TransitionArray::CompareKeys(Name* key1, uint32_t hash1, PropertyKind kind1,
122
                                 PropertyAttributes attributes1, Name* key2,
123
                                 uint32_t hash2, PropertyKind kind2,
124 125 126 127
                                 PropertyAttributes attributes2) {
  int cmp = CompareNames(key1, hash1, key2, hash2);
  if (cmp != 0) return cmp;

128
  return CompareDetails(kind1, attributes1, kind2, attributes2);
129 130 131 132 133 134 135 136 137 138 139 140 141 142
}


int TransitionArray::CompareNames(Name* key1, uint32_t hash1, Name* key2,
                                  uint32_t hash2) {
  if (key1 != key2) {
    // In case of hash collisions key1 is always "less" than key2.
    return hash1 <= hash2 ? -1 : 1;
  }

  return 0;
}


143
int TransitionArray::CompareDetails(PropertyKind kind1,
144
                                    PropertyAttributes attributes1,
145
                                    PropertyKind kind2,
146
                                    PropertyAttributes attributes2) {
147 148
  if (kind1 != kind2) {
    return static_cast<int>(kind1) < static_cast<int>(kind2) ? -1 : 1;
149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169
  }

  if (attributes1 != attributes2) {
    return static_cast<int>(attributes1) < static_cast<int>(attributes2) ? -1
                                                                         : 1;
  }

  return 0;
}


PropertyDetails TransitionArray::GetTargetDetails(Name* name, Map* target) {
  DCHECK(!IsSpecialTransition(name));
  int descriptor = target->LastAdded();
  DescriptorArray* descriptors = target->instance_descriptors();
  // Transitions are allowed only for the last added property.
  DCHECK(descriptors->GetKey(descriptor)->Equals(name));
  return descriptors->GetDetails(descriptor);
}


170 171 172
void TransitionArray::Set(int transition_number, Name* key, Map* target) {
  set(ToKeyIndex(transition_number), key);
  set(ToTargetIndex(transition_number), target);
173 174 175
}


176
void TransitionArray::SetNumberOfTransitions(int number_of_transitions) {
177 178
  DCHECK(number_of_transitions <= Capacity(this));
  set(kTransitionLengthIndex, Smi::FromInt(number_of_transitions));
179 180
}

181 182
}  // namespace internal
}  // namespace v8
183 184

#endif  // V8_TRANSITIONS_INL_H_