transitions-inl.h 6.54 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
#include "src/ic/handler-configuration-inl.h"

12 13 14
namespace v8 {
namespace internal {

15 16 17 18 19 20
template <TransitionsAccessor::Encoding enc>
WeakCell* TransitionsAccessor::GetTargetCell() {
  DCHECK(!needs_reload_);
  if (target_cell_ != nullptr) return target_cell_;
  if (enc == kWeakCell) {
    target_cell_ = WeakCell::cast(raw_transitions_);
21 22
  } else if (enc == kHandler) {
    target_cell_ = StoreHandler::GetTransitionCell(raw_transitions_);
23 24 25 26 27 28 29 30 31 32
  } else {
    UNREACHABLE();
  }
  return target_cell_;
}

TransitionArray* TransitionsAccessor::transitions() {
  DCHECK_EQ(kFullTransitionArray, encoding());
  return TransitionArray::cast(raw_transitions_);
}
33

34
// static
35
TransitionArray* TransitionArray::cast(Object* object) {
36
  DCHECK(object->IsTransitionArray());
37 38 39 40
  return reinterpret_cast<TransitionArray*>(object);
}


41
bool TransitionArray::HasPrototypeTransitions() {
42
  return get(kPrototypeTransitionsIndex) != Smi::kZero;
43 44 45 46
}


FixedArray* TransitionArray::GetPrototypeTransitions() {
47
  DCHECK(HasPrototypeTransitions());  // Callers must check first.
48 49 50 51 52
  Object* prototype_transitions = get(kPrototypeTransitionsIndex);
  return FixedArray::cast(prototype_transitions);
}


53
void TransitionArray::SetPrototypeTransitions(FixedArray* transitions) {
54
  DCHECK(transitions->IsFixedArray());
55
  set(kPrototypeTransitionsIndex, transitions);
56 57 58 59
}


Object** TransitionArray::GetPrototypeTransitionsSlot() {
60
  return RawFieldOfElementAt(kPrototypeTransitionsIndex);
61 62 63
}


64
Object** TransitionArray::GetKeySlot(int transition_number) {
65
  DCHECK(transition_number < number_of_transitions());
66
  return RawFieldOfElementAt(ToKeyIndex(transition_number));
67 68 69
}


70
Name* TransitionArray::GetKey(int transition_number) {
71
  DCHECK(transition_number < number_of_transitions());
72
  return Name::cast(get(ToKeyIndex(transition_number)));
73 74
}

75 76 77 78 79 80 81 82 83 84
Name* TransitionsAccessor::GetKey(int transition_number) {
  WeakCell* cell = nullptr;
  switch (encoding()) {
    case kPrototypeInfo:
    case kUninitialized:
      UNREACHABLE();
      return nullptr;
    case kWeakCell:
      cell = GetTargetCell<kWeakCell>();
      break;
85 86
    case kHandler:
      cell = GetTargetCell<kHandler>();
87 88 89
      break;
    case kFullTransitionArray:
      return transitions()->GetKey(transition_number);
90
  }
91 92
  DCHECK(!cell->cleared());
  return GetSimpleTransitionKey(Map::cast(cell->value()));
93 94
}

95
void TransitionArray::SetKey(int transition_number, Name* key) {
96
  DCHECK(transition_number < number_of_transitions());
97 98 99
  set(ToKeyIndex(transition_number), key);
}

100
Object** TransitionArray::GetTargetSlot(int transition_number) {
101
  DCHECK(transition_number < number_of_transitions());
102
  return RawFieldOfElementAt(ToTargetIndex(transition_number));
103 104
}

105 106 107 108 109 110 111 112 113
// static
PropertyDetails TransitionsAccessor::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);
}
114

115 116
// static
Map* TransitionsAccessor::GetTargetFromRaw(Object* raw) {
117 118
  if (raw->IsWeakCell()) return Map::cast(WeakCell::cast(raw)->value());
  return Map::cast(StoreHandler::GetTransitionCell(raw)->value());
119 120
}

121 122 123 124
Object* TransitionArray::GetRawTarget(int transition_number) {
  DCHECK(transition_number < number_of_transitions());
  return get(ToTargetIndex(transition_number));
}
125

126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
Map* TransitionArray::GetTarget(int transition_number) {
  Object* raw = GetRawTarget(transition_number);
  return TransitionsAccessor::GetTargetFromRaw(raw);
}

Map* TransitionsAccessor::GetTarget(int transition_number) {
  WeakCell* cell = nullptr;
  switch (encoding()) {
    case kPrototypeInfo:
    case kUninitialized:
      UNREACHABLE();
      return nullptr;
    case kWeakCell:
      cell = GetTargetCell<kWeakCell>();
      break;
141 142
    case kHandler:
      cell = GetTargetCell<kHandler>();
143 144 145 146 147 148 149 150 151
      break;
    case kFullTransitionArray:
      return transitions()->GetTarget(transition_number);
  }
  DCHECK(!cell->cleared());
  return Map::cast(cell->value());
}

void TransitionArray::SetTarget(int transition_number, Object* value) {
152
  DCHECK(!value->IsMap());
153 154
  DCHECK(transition_number < number_of_transitions());
  set(ToTargetIndex(transition_number), value);
155 156 157
}


158
int TransitionArray::SearchName(Name* name, int* out_insertion_index) {
159 160 161
  DCHECK(name->IsUniqueName());
  return internal::Search<ALL_ENTRIES>(this, name, number_of_entries(),
                                       out_insertion_index);
162 163
}

164
int TransitionArray::CompareKeys(Name* key1, uint32_t hash1, PropertyKind kind1,
165
                                 PropertyAttributes attributes1, Name* key2,
166
                                 uint32_t hash2, PropertyKind kind2,
167 168 169 170
                                 PropertyAttributes attributes2) {
  int cmp = CompareNames(key1, hash1, key2, hash2);
  if (cmp != 0) return cmp;

171
  return CompareDetails(kind1, attributes1, kind2, attributes2);
172 173 174 175 176 177 178 179 180 181 182 183
}

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;
}

184
int TransitionArray::CompareDetails(PropertyKind kind1,
185
                                    PropertyAttributes attributes1,
186
                                    PropertyKind kind2,
187
                                    PropertyAttributes attributes2) {
188 189
  if (kind1 != kind2) {
    return static_cast<int>(kind1) < static_cast<int>(kind2) ? -1 : 1;
190 191 192 193 194 195 196 197 198 199
  }

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

  return 0;
}

200
void TransitionArray::Set(int transition_number, Name* key, Object* target) {
201 202
  set(ToKeyIndex(transition_number), key);
  set(ToTargetIndex(transition_number), target);
203 204
}

205 206 207 208
int TransitionArray::Capacity() {
  if (length() <= kFirstIndex) return 0;
  return (length() - kFirstIndex) / kTransitionSize;
}
209

210
void TransitionArray::SetNumberOfTransitions(int number_of_transitions) {
211
  DCHECK(number_of_transitions <= Capacity());
212
  set(kTransitionLengthIndex, Smi::FromInt(number_of_transitions));
213 214
}

215 216
}  // namespace internal
}  // namespace v8
217 218

#endif  // V8_TRANSITIONS_INL_H_