Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(8)

Side by Side Diff: src/objects.cc

Issue 10455005: TraversableMap only dependent on DescriptorArray (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Removed assert since it increased brittleness. It assumed that the map transition iterator was neveā€¦ Created 8 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « src/objects.h ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without 2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are 3 // modification, are permitted provided that the following conditions are
4 // met: 4 // met:
5 // 5 //
6 // * Redistributions of source code must retain the above copyright 6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer. 7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above 8 // * Redistributions in binary form must reproduce the above
9 // copyright notice, this list of conditions and the following 9 // copyright notice, this list of conditions and the following
10 // disclaimer in the documentation and/or other materials provided 10 // disclaimer in the documentation and/or other materials provided
(...skipping 5070 matching lines...) Expand 10 before | Expand all | Expand 10 after
5081 5081
5082 // An iterator over all map transitions in an descriptor array, reusing the map 5082 // An iterator over all map transitions in an descriptor array, reusing the map
5083 // field of the contens array while it is running. 5083 // field of the contens array while it is running.
5084 class IntrusiveMapTransitionIterator { 5084 class IntrusiveMapTransitionIterator {
5085 public: 5085 public:
5086 explicit IntrusiveMapTransitionIterator(DescriptorArray* descriptor_array) 5086 explicit IntrusiveMapTransitionIterator(DescriptorArray* descriptor_array)
5087 : descriptor_array_(descriptor_array) { } 5087 : descriptor_array_(descriptor_array) { }
5088 5088
5089 void Start() { 5089 void Start() {
5090 ASSERT(!IsIterating()); 5090 ASSERT(!IsIterating());
5091 if (HasContentArray()) *ContentHeader() = Smi::FromInt(0); 5091 if (HasDescriptors()) *DescriptorArrayHeader() = Smi::FromInt(0);
5092 } 5092 }
5093 5093
5094 bool IsIterating() { 5094 bool IsIterating() {
5095 return HasContentArray() && (*ContentHeader())->IsSmi(); 5095 return HasDescriptors() && (*DescriptorArrayHeader())->IsSmi();
5096 } 5096 }
5097 5097
5098 Map* Next() { 5098 Map* Next() {
5099 ASSERT(IsIterating()); 5099 ASSERT(IsIterating());
5100 FixedArray* contents = ContentArray(); 5100 // Attention, tricky index manipulation ahead: Two consecutive indices are
5101 // Attention, tricky index manipulation ahead: Every entry in the contents 5101 // assigned to each descriptor. Most descriptors directly advance to the
5102 // array consists of a value/details pair, so the index is typically even. 5102 // next descriptor by adding 2 to the index. The exceptions are the
5103 // An exception is made for CALLBACKS entries: An even index means we look 5103 // CALLBACKS entries: An even index means we look at its getter, and an odd
5104 // at its getter, and an odd index means we look at its setter. 5104 // index means we look at its setter.
5105 int index = Smi::cast(*ContentHeader())->value(); 5105 int raw_index = Smi::cast(*DescriptorArrayHeader())->value();
5106 while (index < contents->length()) { 5106 int index = raw_index / 2;
5107 PropertyDetails details(Smi::cast(contents->get(index | 1))); 5107 int number_of_descriptors = descriptor_array_->number_of_descriptors();
5108 while (index < number_of_descriptors) {
5109 PropertyDetails details(RawGetDetails(index));
5108 switch (details.type()) { 5110 switch (details.type()) {
5109 case MAP_TRANSITION: 5111 case MAP_TRANSITION:
5110 case CONSTANT_TRANSITION: 5112 case CONSTANT_TRANSITION:
5111 case ELEMENTS_TRANSITION: 5113 case ELEMENTS_TRANSITION:
5112 // We definitely have a map transition. 5114 // We definitely have a map transition.
5113 *ContentHeader() = Smi::FromInt(index + 2); 5115 *DescriptorArrayHeader() = Smi::FromInt(raw_index + 2);
5114 return static_cast<Map*>(contents->get(index)); 5116 return static_cast<Map*>(RawGetValue(index));
5115 case CALLBACKS: { 5117 case CALLBACKS: {
5116 // We might have a map transition in a getter or in a setter. 5118 // We might have a map transition in a getter or in a setter.
5117 AccessorPair* accessors = 5119 AccessorPair* accessors =
5118 static_cast<AccessorPair*>(contents->get(index & ~1)); 5120 static_cast<AccessorPair*>(RawGetValue(index));
5119 Object* accessor = 5121 Object* accessor;
5120 ((index & 1) == 0) ? accessors->getter() : accessors->setter(); 5122 if ((raw_index & 1) == 0) {
5121 index++; 5123 accessor = accessors->setter();
5124 } else {
5125 ++index;
5126 accessor = accessors->getter();
5127 }
5128 ++raw_index;
5122 if (accessor->IsMap()) { 5129 if (accessor->IsMap()) {
5123 *ContentHeader() = Smi::FromInt(index); 5130 *DescriptorArrayHeader() = Smi::FromInt(raw_index);
Michael Starzinger 2012/05/30 13:28:56 Indentation seems off.
5124 return static_cast<Map*>(accessor); 5131 return static_cast<Map*>(accessor);
5125 } 5132 }
5126 break; 5133 break;
5127 } 5134 }
5128 case NORMAL: 5135 case NORMAL:
5129 case FIELD: 5136 case FIELD:
5130 case CONSTANT_FUNCTION: 5137 case CONSTANT_FUNCTION:
5131 case HANDLER: 5138 case HANDLER:
5132 case INTERCEPTOR: 5139 case INTERCEPTOR:
5133 case NULL_DESCRIPTOR: 5140 case NULL_DESCRIPTOR:
5134 // We definitely have no map transition. 5141 // We definitely have no map transition.
5135 index += 2; 5142 raw_index += 2;
5143 ++index;
5136 break; 5144 break;
5137 } 5145 }
5138 } 5146 }
5139 *ContentHeader() = descriptor_array_->GetHeap()->fixed_array_map(); 5147 *DescriptorArrayHeader() = descriptor_array_->GetHeap()->fixed_array_map();
5140 return NULL; 5148 return NULL;
5141 } 5149 }
5142 5150
5143 private: 5151 private:
5144 bool HasContentArray() { 5152 bool HasDescriptors() {
5145 return descriptor_array_-> length() > DescriptorArray::kContentArrayIndex; 5153 return descriptor_array_->length() > DescriptorArray::kFirstIndex;
5146 } 5154 }
5147 5155
5148 FixedArray* ContentArray() { 5156 Object** DescriptorArrayHeader() {
5149 Object* array = descriptor_array_->get(DescriptorArray::kContentArrayIndex); 5157 return HeapObject::RawField(descriptor_array_, DescriptorArray::kMapOffset);
5150 return static_cast<FixedArray*>(array);
5151 } 5158 }
5152 5159
5153 Object** ContentHeader() { 5160 FixedArray* RawGetContentArray() {
5154 return HeapObject::RawField(ContentArray(), DescriptorArray::kMapOffset); 5161 Object* array =
5162 descriptor_array_->get(DescriptorArray::kContentArrayIndex);
5163 return static_cast<FixedArray*>(array);
5155 } 5164 }
5156 5165
5166 Object* RawGetValue(int descriptor_number) {
5167 return RawGetContentArray()->get(
5168 DescriptorArray::ToValueIndex(descriptor_number));
5169 }
5170
5171 PropertyDetails RawGetDetails(int descriptor_number) {
5172 Object* details = RawGetContentArray()->get(
5173 DescriptorArray::ToDetailsIndex(descriptor_number));
5174 return PropertyDetails(Smi::cast(details));
5175 }
5176
5177
5157 DescriptorArray* descriptor_array_; 5178 DescriptorArray* descriptor_array_;
5158 }; 5179 };
5159 5180
5160 5181
5161 // An iterator over all prototype transitions, reusing the map field of the 5182 // An iterator over all prototype transitions, reusing the map field of the
5162 // underlying array while it is running. 5183 // underlying array while it is running.
5163 class IntrusivePrototypeTransitionIterator { 5184 class IntrusivePrototypeTransitionIterator {
5164 public: 5185 public:
5165 explicit IntrusivePrototypeTransitionIterator(HeapObject* proto_trans) 5186 explicit IntrusivePrototypeTransitionIterator(HeapObject* proto_trans)
5166 : proto_trans_(proto_trans) { } 5187 : proto_trans_(proto_trans) { }
(...skipping 78 matching lines...) Expand 10 before | Expand all | Expand 10 after
5245 // this map's map! 5266 // this map's map!
5246 void SetParent(TraversableMap* parent) { set_map_no_write_barrier(parent); } 5267 void SetParent(TraversableMap* parent) { set_map_no_write_barrier(parent); }
5247 5268
5248 // Reset the current map's map, returning the parent previously stored in it. 5269 // Reset the current map's map, returning the parent previously stored in it.
5249 TraversableMap* GetAndResetParent() { 5270 TraversableMap* GetAndResetParent() {
5250 TraversableMap* old_parent = static_cast<TraversableMap*>(map()); 5271 TraversableMap* old_parent = static_cast<TraversableMap*>(map());
5251 set_map_no_write_barrier(GetHeap()->meta_map()); 5272 set_map_no_write_barrier(GetHeap()->meta_map());
5252 return old_parent; 5273 return old_parent;
5253 } 5274 }
5254 5275
5276 // Can either be Smi (no instance descriptors), or a descriptor array with the
5277 // header overwritten as a Smi (thus iterating).
5278 DescriptorArray* MutatedInstanceDescriptors() {
5279 Object* object =
5280 *HeapObject::RawField(this, kInstanceDescriptorsOrBitField3Offset);
5281 if (object->IsSmi()) {
5282 return GetHeap()->empty_descriptor_array();
5283 } else {
5284 DescriptorArray* descriptor_array =
5285 static_cast<DescriptorArray*>(object);
5286 return descriptor_array;
5287 }
5288 }
5289
5255 // Start iterating over this map's children, possibly destroying a FixedArray 5290 // Start iterating over this map's children, possibly destroying a FixedArray
5256 // map (see explanation above). 5291 // map (see explanation above).
5257 void ChildIteratorStart() { 5292 void ChildIteratorStart() {
5258 IntrusiveMapTransitionIterator(instance_descriptors()).Start(); 5293 IntrusiveMapTransitionIterator(instance_descriptors()).Start();
5259 IntrusivePrototypeTransitionIterator( 5294 IntrusivePrototypeTransitionIterator(
5260 unchecked_prototype_transitions()).Start(); 5295 unchecked_prototype_transitions()).Start();
5261 } 5296 }
5262 5297
5263 // If we have an unvisited child map, return that one and advance. If we have 5298 // If we have an unvisited child map, return that one and advance. If we have
5264 // none, return NULL and reset any destroyed FixedArray maps. 5299 // none, return NULL and reset any destroyed FixedArray maps.
5265 TraversableMap* ChildIteratorNext() { 5300 TraversableMap* ChildIteratorNext() {
5266 IntrusiveMapTransitionIterator descriptor_iterator(instance_descriptors());
5267 if (descriptor_iterator.IsIterating()) {
5268 Map* next = descriptor_iterator.Next();
5269 if (next != NULL) return static_cast<TraversableMap*>(next);
5270 }
5271 IntrusivePrototypeTransitionIterator 5301 IntrusivePrototypeTransitionIterator
5272 proto_iterator(unchecked_prototype_transitions()); 5302 proto_iterator(unchecked_prototype_transitions());
5273 if (proto_iterator.IsIterating()) { 5303 if (proto_iterator.IsIterating()) {
5274 Map* next = proto_iterator.Next(); 5304 Map* next = proto_iterator.Next();
5275 if (next != NULL) return static_cast<TraversableMap*>(next); 5305 if (next != NULL) return static_cast<TraversableMap*>(next);
5276 } 5306 }
Michael Starzinger 2012/05/30 13:28:56 Just out of curiosity: What is the difference betw
5307 IntrusiveMapTransitionIterator
5308 descriptor_iterator(MutatedInstanceDescriptors());
5309 if (descriptor_iterator.IsIterating()) {
5310 Map* next = descriptor_iterator.Next();
5311 if (next != NULL) return static_cast<TraversableMap*>(next);
5312 }
5277 return NULL; 5313 return NULL;
5278 } 5314 }
5279 }; 5315 };
5280 5316
5281 5317
5282 // Traverse the transition tree in postorder without using the C++ stack by 5318 // Traverse the transition tree in postorder without using the C++ stack by
5283 // doing pointer reversal. 5319 // doing pointer reversal.
5284 void Map::TraverseTransitionTree(TraverseCallback callback, void* data) { 5320 void Map::TraverseTransitionTree(TraverseCallback callback, void* data) {
5285 TraversableMap* current = static_cast<TraversableMap*>(this); 5321 TraversableMap* current = static_cast<TraversableMap*>(this);
5286 current->ChildIteratorStart(); 5322 current->ChildIteratorStart();
(...skipping 7908 matching lines...) Expand 10 before | Expand all | Expand 10 after
13195 set_year(Smi::FromInt(year), SKIP_WRITE_BARRIER); 13231 set_year(Smi::FromInt(year), SKIP_WRITE_BARRIER);
13196 set_month(Smi::FromInt(month), SKIP_WRITE_BARRIER); 13232 set_month(Smi::FromInt(month), SKIP_WRITE_BARRIER);
13197 set_day(Smi::FromInt(day), SKIP_WRITE_BARRIER); 13233 set_day(Smi::FromInt(day), SKIP_WRITE_BARRIER);
13198 set_weekday(Smi::FromInt(weekday), SKIP_WRITE_BARRIER); 13234 set_weekday(Smi::FromInt(weekday), SKIP_WRITE_BARRIER);
13199 set_hour(Smi::FromInt(hour), SKIP_WRITE_BARRIER); 13235 set_hour(Smi::FromInt(hour), SKIP_WRITE_BARRIER);
13200 set_min(Smi::FromInt(min), SKIP_WRITE_BARRIER); 13236 set_min(Smi::FromInt(min), SKIP_WRITE_BARRIER);
13201 set_sec(Smi::FromInt(sec), SKIP_WRITE_BARRIER); 13237 set_sec(Smi::FromInt(sec), SKIP_WRITE_BARRIER);
13202 } 13238 }
13203 13239
13204 } } // namespace v8::internal 13240 } } // namespace v8::internal
OLDNEW
« no previous file with comments | « src/objects.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698