danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 1 | // Copyright 2012 the V8 project authors. All rights reserved. |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 2 | // Redistribution and use in source and binary forms, with or without |
| 3 | // modification, are permitted provided that the following conditions are |
| 4 | // met: |
| 5 | // |
| 6 | // * Redistributions of source code must retain the above copyright |
| 7 | // notice, this list of conditions and the following disclaimer. |
| 8 | // * Redistributions in binary form must reproduce the above |
| 9 | // copyright notice, this list of conditions and the following |
| 10 | // disclaimer in the documentation and/or other materials provided |
| 11 | // with the distribution. |
| 12 | // * Neither the name of Google Inc. nor the names of its |
| 13 | // contributors may be used to endorse or promote products derived |
| 14 | // from this software without specific prior written permission. |
| 15 | // |
| 16 | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| 17 | // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| 18 | // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| 19 | // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| 20 | // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| 21 | // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| 22 | // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| 23 | // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| 24 | // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| 25 | // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| 26 | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| 27 | |
| 28 | #ifndef V8_HEAP_INL_H_ |
| 29 | #define V8_HEAP_INL_H_ |
| 30 | |
machenbach@chromium.org | e31286d | 2014-01-15 10:29:52 +0000 | [diff] [blame] | 31 | #include <cmath> |
| 32 | |
ricow@chromium.org | d236f4d | 2010-09-01 06:52:08 +0000 | [diff] [blame] | 33 | #include "heap.h" |
machenbach@chromium.org | 5c88bc3 | 2014-01-17 08:10:36 +0000 | [diff] [blame] | 34 | #include "heap-profiler.h" |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 35 | #include "isolate.h" |
fschneider@chromium.org | fb144a0 | 2011-05-04 12:43:48 +0000 | [diff] [blame] | 36 | #include "list-inl.h" |
| 37 | #include "objects.h" |
ulan@chromium.org | 9a21ec4 | 2012-03-06 08:42:24 +0000 | [diff] [blame] | 38 | #include "platform.h" |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 39 | #include "v8-counters.h" |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 40 | #include "store-buffer.h" |
| 41 | #include "store-buffer-inl.h" |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 42 | |
kasperl@chromium.org | 71affb5 | 2009-05-26 05:44:31 +0000 | [diff] [blame] | 43 | namespace v8 { |
| 44 | namespace internal { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 45 | |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 46 | void PromotionQueue::insert(HeapObject* target, int size) { |
danno@chromium.org | c612e02 | 2011-11-10 11:38:15 +0000 | [diff] [blame] | 47 | if (emergency_stack_ != NULL) { |
| 48 | emergency_stack_->Add(Entry(target, size)); |
| 49 | return; |
| 50 | } |
| 51 | |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 52 | if (NewSpacePage::IsAtStart(reinterpret_cast<Address>(rear_))) { |
| 53 | NewSpacePage* rear_page = |
| 54 | NewSpacePage::FromAddress(reinterpret_cast<Address>(rear_)); |
| 55 | ASSERT(!rear_page->prev_page()->is_anchor()); |
yangguo@chromium.org | ab30bb8 | 2012-02-24 14:41:46 +0000 | [diff] [blame] | 56 | rear_ = reinterpret_cast<intptr_t*>(rear_page->prev_page()->area_end()); |
danno@chromium.org | c612e02 | 2011-11-10 11:38:15 +0000 | [diff] [blame] | 57 | ActivateGuardIfOnTheSamePage(); |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 58 | } |
danno@chromium.org | c612e02 | 2011-11-10 11:38:15 +0000 | [diff] [blame] | 59 | |
| 60 | if (guard_) { |
| 61 | ASSERT(GetHeadPage() == |
| 62 | Page::FromAllocationTop(reinterpret_cast<Address>(limit_))); |
| 63 | |
| 64 | if ((rear_ - 2) < limit_) { |
| 65 | RelocateQueueHead(); |
| 66 | emergency_stack_->Add(Entry(target, size)); |
| 67 | return; |
| 68 | } |
| 69 | } |
| 70 | |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 71 | *(--rear_) = reinterpret_cast<intptr_t>(target); |
| 72 | *(--rear_) = size; |
| 73 | // Assert no overflow into live objects. |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 74 | #ifdef DEBUG |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 75 | SemiSpace::AssertValidRange(target->GetIsolate()->heap()->new_space()->top(), |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 76 | reinterpret_cast<Address>(rear_)); |
| 77 | #endif |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 78 | } |
| 79 | |
| 80 | |
danno@chromium.org | c612e02 | 2011-11-10 11:38:15 +0000 | [diff] [blame] | 81 | void PromotionQueue::ActivateGuardIfOnTheSamePage() { |
| 82 | guard_ = guard_ || |
| 83 | heap_->new_space()->active_space()->current_page()->address() == |
| 84 | GetHeadPage()->address(); |
| 85 | } |
| 86 | |
| 87 | |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 88 | MaybeObject* Heap::AllocateStringFromUtf8(Vector<const char> str, |
| 89 | PretenureFlag pretenure) { |
| 90 | // Check for ASCII first since this is the common case. |
rossberg@chromium.org | 89e18f5 | 2012-10-22 13:09:53 +0000 | [diff] [blame] | 91 | const char* start = str.start(); |
| 92 | int length = str.length(); |
| 93 | int non_ascii_start = String::NonAsciiStart(start, length); |
| 94 | if (non_ascii_start >= length) { |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 95 | // If the string is ASCII, we do not need to convert the characters |
| 96 | // since UTF8 is backwards compatible with ASCII. |
ulan@chromium.org | 8e8d882 | 2012-11-23 14:36:46 +0000 | [diff] [blame] | 97 | return AllocateStringFromOneByte(str, pretenure); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 98 | } |
| 99 | // Non-ASCII and we need to decode. |
rossberg@chromium.org | 89e18f5 | 2012-10-22 13:09:53 +0000 | [diff] [blame] | 100 | return AllocateStringFromUtf8Slow(str, non_ascii_start, pretenure); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 101 | } |
| 102 | |
| 103 | |
yangguo@chromium.org | a6bbcc8 | 2012-12-21 12:35:02 +0000 | [diff] [blame] | 104 | template<> |
| 105 | bool inline Heap::IsOneByte(Vector<const char> str, int chars) { |
| 106 | // TODO(dcarney): incorporate Latin-1 check when Latin-1 is supported? |
| 107 | // ASCII only check. |
| 108 | return chars == str.length(); |
| 109 | } |
| 110 | |
| 111 | |
| 112 | template<> |
| 113 | bool inline Heap::IsOneByte(String* str, int chars) { |
| 114 | return str->IsOneByteRepresentation(); |
| 115 | } |
| 116 | |
| 117 | |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 118 | MaybeObject* Heap::AllocateInternalizedStringFromUtf8( |
| 119 | Vector<const char> str, int chars, uint32_t hash_field) { |
jkummerow@chromium.org | 59297c7 | 2013-01-09 16:32:23 +0000 | [diff] [blame] | 120 | if (IsOneByte(str, chars)) { |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 121 | return AllocateOneByteInternalizedString( |
| 122 | Vector<const uint8_t>::cast(str), hash_field); |
jkummerow@chromium.org | 59297c7 | 2013-01-09 16:32:23 +0000 | [diff] [blame] | 123 | } |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 124 | return AllocateInternalizedStringImpl<false>(str, chars, hash_field); |
yangguo@chromium.org | a6bbcc8 | 2012-12-21 12:35:02 +0000 | [diff] [blame] | 125 | } |
| 126 | |
| 127 | |
| 128 | template<typename T> |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 129 | MaybeObject* Heap::AllocateInternalizedStringImpl( |
| 130 | T t, int chars, uint32_t hash_field) { |
yangguo@chromium.org | a6bbcc8 | 2012-12-21 12:35:02 +0000 | [diff] [blame] | 131 | if (IsOneByte(t, chars)) { |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 132 | return AllocateInternalizedStringImpl<true>(t, chars, hash_field); |
yangguo@chromium.org | a6bbcc8 | 2012-12-21 12:35:02 +0000 | [diff] [blame] | 133 | } |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 134 | return AllocateInternalizedStringImpl<false>(t, chars, hash_field); |
ager@chromium.org | a74f0da | 2008-12-03 16:05:52 +0000 | [diff] [blame] | 135 | } |
| 136 | |
| 137 | |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 138 | MaybeObject* Heap::AllocateOneByteInternalizedString(Vector<const uint8_t> str, |
| 139 | uint32_t hash_field) { |
yangguo@chromium.org | fb37721 | 2012-11-16 14:43:43 +0000 | [diff] [blame] | 140 | if (str.length() > SeqOneByteString::kMaxLength) { |
jkummerow@chromium.org | 59297c7 | 2013-01-09 16:32:23 +0000 | [diff] [blame] | 141 | return Failure::OutOfMemoryException(0x2); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 142 | } |
| 143 | // Compute map and object size. |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 144 | Map* map = ascii_internalized_string_map(); |
yangguo@chromium.org | fb37721 | 2012-11-16 14:43:43 +0000 | [diff] [blame] | 145 | int size = SeqOneByteString::SizeFor(str.length()); |
machenbach@chromium.org | 3d079fe | 2013-09-25 08:19:55 +0000 | [diff] [blame] | 146 | AllocationSpace space = SelectSpace(size, OLD_DATA_SPACE, TENURED); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 147 | |
| 148 | // Allocate string. |
| 149 | Object* result; |
machenbach@chromium.org | 3d079fe | 2013-09-25 08:19:55 +0000 | [diff] [blame] | 150 | { MaybeObject* maybe_result = AllocateRaw(size, space, OLD_DATA_SPACE); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 151 | if (!maybe_result->ToObject(&result)) return maybe_result; |
| 152 | } |
| 153 | |
ricow@chromium.org | 64e3a4b | 2011-12-13 08:07:27 +0000 | [diff] [blame] | 154 | // String maps are all immortal immovable objects. |
| 155 | reinterpret_cast<HeapObject*>(result)->set_map_no_write_barrier(map); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 156 | // Set length and hash fields of the allocated string. |
| 157 | String* answer = String::cast(result); |
| 158 | answer->set_length(str.length()); |
| 159 | answer->set_hash_field(hash_field); |
| 160 | |
| 161 | ASSERT_EQ(size, answer->Size()); |
| 162 | |
| 163 | // Fill in the characters. |
mstarzinger@chromium.org | e27d617 | 2013-04-17 11:51:44 +0000 | [diff] [blame] | 164 | OS::MemCopy(answer->address() + SeqOneByteString::kHeaderSize, |
| 165 | str.start(), str.length()); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 166 | |
| 167 | return answer; |
| 168 | } |
| 169 | |
| 170 | |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 171 | MaybeObject* Heap::AllocateTwoByteInternalizedString(Vector<const uc16> str, |
| 172 | uint32_t hash_field) { |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 173 | if (str.length() > SeqTwoByteString::kMaxLength) { |
jkummerow@chromium.org | 59297c7 | 2013-01-09 16:32:23 +0000 | [diff] [blame] | 174 | return Failure::OutOfMemoryException(0x3); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 175 | } |
| 176 | // Compute map and object size. |
yangguo@chromium.org | 4a9f655 | 2013-03-04 14:46:33 +0000 | [diff] [blame] | 177 | Map* map = internalized_string_map(); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 178 | int size = SeqTwoByteString::SizeFor(str.length()); |
machenbach@chromium.org | 3d079fe | 2013-09-25 08:19:55 +0000 | [diff] [blame] | 179 | AllocationSpace space = SelectSpace(size, OLD_DATA_SPACE, TENURED); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 180 | |
| 181 | // Allocate string. |
| 182 | Object* result; |
machenbach@chromium.org | 3d079fe | 2013-09-25 08:19:55 +0000 | [diff] [blame] | 183 | { MaybeObject* maybe_result = AllocateRaw(size, space, OLD_DATA_SPACE); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 184 | if (!maybe_result->ToObject(&result)) return maybe_result; |
| 185 | } |
| 186 | |
| 187 | reinterpret_cast<HeapObject*>(result)->set_map(map); |
| 188 | // Set length and hash fields of the allocated string. |
| 189 | String* answer = String::cast(result); |
| 190 | answer->set_length(str.length()); |
| 191 | answer->set_hash_field(hash_field); |
| 192 | |
| 193 | ASSERT_EQ(size, answer->Size()); |
| 194 | |
| 195 | // Fill in the characters. |
mstarzinger@chromium.org | e27d617 | 2013-04-17 11:51:44 +0000 | [diff] [blame] | 196 | OS::MemCopy(answer->address() + SeqTwoByteString::kHeaderSize, |
| 197 | str.start(), str.length() * kUC16Size); |
fschneider@chromium.org | 9e3e0b6 | 2011-01-03 10:16:46 +0000 | [diff] [blame] | 198 | |
| 199 | return answer; |
| 200 | } |
| 201 | |
lrn@chromium.org | 303ada7 | 2010-10-27 09:33:13 +0000 | [diff] [blame] | 202 | MaybeObject* Heap::CopyFixedArray(FixedArray* src) { |
whesse@chromium.org | 4a1fe7d | 2010-09-27 12:32:04 +0000 | [diff] [blame] | 203 | return CopyFixedArrayWithMap(src, src->map()); |
| 204 | } |
| 205 | |
| 206 | |
ricow@chromium.org | 2c99e28 | 2011-07-28 09:15:17 +0000 | [diff] [blame] | 207 | MaybeObject* Heap::CopyFixedDoubleArray(FixedDoubleArray* src) { |
| 208 | return CopyFixedDoubleArrayWithMap(src, src->map()); |
| 209 | } |
| 210 | |
| 211 | |
mstarzinger@chromium.org | a2e1a40 | 2013-10-15 08:25:05 +0000 | [diff] [blame] | 212 | MaybeObject* Heap::CopyConstantPoolArray(ConstantPoolArray* src) { |
| 213 | return CopyConstantPoolArrayWithMap(src, src->map()); |
| 214 | } |
| 215 | |
| 216 | |
lrn@chromium.org | 303ada7 | 2010-10-27 09:33:13 +0000 | [diff] [blame] | 217 | MaybeObject* Heap::AllocateRaw(int size_in_bytes, |
| 218 | AllocationSpace space, |
| 219 | AllocationSpace retry_space) { |
machenbach@chromium.org | 3d079fe | 2013-09-25 08:19:55 +0000 | [diff] [blame] | 220 | ASSERT(AllowHandleAllocation::IsAllowed()); |
| 221 | ASSERT(AllowHeapAllocation::IsAllowed()); |
| 222 | ASSERT(gc_state_ == NOT_IN_GC); |
machenbach@chromium.org | 935a779 | 2013-11-12 09:05:18 +0000 | [diff] [blame] | 223 | HeapProfiler* profiler = isolate_->heap_profiler(); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 224 | #ifdef DEBUG |
| 225 | if (FLAG_gc_interval >= 0 && |
| 226 | !disallow_allocation_failure_ && |
| 227 | Heap::allocation_timeout_-- <= 0) { |
whesse@chromium.org | 4a5224e | 2010-10-20 12:37:07 +0000 | [diff] [blame] | 228 | return Failure::RetryAfterGC(space); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 229 | } |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 230 | isolate_->counters()->objs_since_last_full()->Increment(); |
| 231 | isolate_->counters()->objs_since_last_young()->Increment(); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 232 | #endif |
machenbach@chromium.org | 935a779 | 2013-11-12 09:05:18 +0000 | [diff] [blame] | 233 | |
| 234 | HeapObject* object; |
lrn@chromium.org | 303ada7 | 2010-10-27 09:33:13 +0000 | [diff] [blame] | 235 | MaybeObject* result; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 236 | if (NEW_SPACE == space) { |
kasperl@chromium.org | 9bbf968 | 2008-10-30 11:53:07 +0000 | [diff] [blame] | 237 | result = new_space_.AllocateRaw(size_in_bytes); |
machenbach@chromium.org | 935a779 | 2013-11-12 09:05:18 +0000 | [diff] [blame] | 238 | if (always_allocate() && result->IsFailure() && retry_space != NEW_SPACE) { |
kasperl@chromium.org | 9bbf968 | 2008-10-30 11:53:07 +0000 | [diff] [blame] | 239 | space = retry_space; |
| 240 | } else { |
machenbach@chromium.org | 935a779 | 2013-11-12 09:05:18 +0000 | [diff] [blame] | 241 | if (profiler->is_tracking_allocations() && result->To(&object)) { |
yangguo@chromium.org | cc53605 | 2013-11-29 11:43:20 +0000 | [diff] [blame] | 242 | profiler->AllocationEvent(object->address(), size_in_bytes); |
machenbach@chromium.org | 935a779 | 2013-11-12 09:05:18 +0000 | [diff] [blame] | 243 | } |
kasperl@chromium.org | 9bbf968 | 2008-10-30 11:53:07 +0000 | [diff] [blame] | 244 | return result; |
| 245 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 246 | } |
| 247 | |
ager@chromium.org | 9258b6b | 2008-09-11 09:11:10 +0000 | [diff] [blame] | 248 | if (OLD_POINTER_SPACE == space) { |
| 249 | result = old_pointer_space_->AllocateRaw(size_in_bytes); |
| 250 | } else if (OLD_DATA_SPACE == space) { |
| 251 | result = old_data_space_->AllocateRaw(size_in_bytes); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 252 | } else if (CODE_SPACE == space) { |
| 253 | result = code_space_->AllocateRaw(size_in_bytes); |
| 254 | } else if (LO_SPACE == space) { |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 255 | result = lo_space_->AllocateRaw(size_in_bytes, NOT_EXECUTABLE); |
kasperl@chromium.org | defbd10 | 2009-07-13 14:04:26 +0000 | [diff] [blame] | 256 | } else if (CELL_SPACE == space) { |
| 257 | result = cell_space_->AllocateRaw(size_in_bytes); |
danno@chromium.org | 4172848 | 2013-06-12 22:31:22 +0000 | [diff] [blame] | 258 | } else if (PROPERTY_CELL_SPACE == space) { |
| 259 | result = property_cell_space_->AllocateRaw(size_in_bytes); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 260 | } else { |
| 261 | ASSERT(MAP_SPACE == space); |
| 262 | result = map_space_->AllocateRaw(size_in_bytes); |
| 263 | } |
| 264 | if (result->IsFailure()) old_gen_exhausted_ = true; |
machenbach@chromium.org | 935a779 | 2013-11-12 09:05:18 +0000 | [diff] [blame] | 265 | if (profiler->is_tracking_allocations() && result->To(&object)) { |
yangguo@chromium.org | cc53605 | 2013-11-29 11:43:20 +0000 | [diff] [blame] | 266 | profiler->AllocationEvent(object->address(), size_in_bytes); |
machenbach@chromium.org | 935a779 | 2013-11-12 09:05:18 +0000 | [diff] [blame] | 267 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 268 | return result; |
| 269 | } |
| 270 | |
| 271 | |
erikcorry | 0ad885c | 2011-11-21 13:51:57 +0000 | [diff] [blame] | 272 | MaybeObject* Heap::NumberFromInt32( |
| 273 | int32_t value, PretenureFlag pretenure) { |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 274 | if (Smi::IsValid(value)) return Smi::FromInt(value); |
| 275 | // Bypass NumberFromDouble to avoid various redundant checks. |
erikcorry | 0ad885c | 2011-11-21 13:51:57 +0000 | [diff] [blame] | 276 | return AllocateHeapNumber(FastI2D(value), pretenure); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 277 | } |
| 278 | |
| 279 | |
erikcorry | 0ad885c | 2011-11-21 13:51:57 +0000 | [diff] [blame] | 280 | MaybeObject* Heap::NumberFromUint32( |
| 281 | uint32_t value, PretenureFlag pretenure) { |
jkummerow@chromium.org | 59297c7 | 2013-01-09 16:32:23 +0000 | [diff] [blame] | 282 | if (static_cast<int32_t>(value) >= 0 && |
| 283 | Smi::IsValid(static_cast<int32_t>(value))) { |
| 284 | return Smi::FromInt(static_cast<int32_t>(value)); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 285 | } |
| 286 | // Bypass NumberFromDouble to avoid various redundant checks. |
erikcorry | 0ad885c | 2011-11-21 13:51:57 +0000 | [diff] [blame] | 287 | return AllocateHeapNumber(FastUI2D(value), pretenure); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 288 | } |
| 289 | |
| 290 | |
kmillikin@chromium.org | 13bd294 | 2009-12-16 15:36:05 +0000 | [diff] [blame] | 291 | void Heap::FinalizeExternalString(String* string) { |
| 292 | ASSERT(string->IsExternalString()); |
| 293 | v8::String::ExternalStringResourceBase** resource_addr = |
| 294 | reinterpret_cast<v8::String::ExternalStringResourceBase**>( |
| 295 | reinterpret_cast<byte*>(string) + |
| 296 | ExternalString::kResourceOffset - |
| 297 | kHeapObjectTag); |
lrn@chromium.org | 1af7e1b | 2010-06-07 11:12:01 +0000 | [diff] [blame] | 298 | |
vegorov@chromium.org | 2356e6f | 2010-06-09 09:38:56 +0000 | [diff] [blame] | 299 | // Dispose of the C++ object if it has not already been disposed. |
| 300 | if (*resource_addr != NULL) { |
| 301 | (*resource_addr)->Dispose(); |
mstarzinger@chromium.org | 1b3afd1 | 2011-11-29 14:28:56 +0000 | [diff] [blame] | 302 | *resource_addr = NULL; |
lrn@chromium.org | 1af7e1b | 2010-06-07 11:12:01 +0000 | [diff] [blame] | 303 | } |
kmillikin@chromium.org | 13bd294 | 2009-12-16 15:36:05 +0000 | [diff] [blame] | 304 | } |
| 305 | |
| 306 | |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 307 | bool Heap::InNewSpace(Object* object) { |
sgjesse@chromium.org | b302e56 | 2010-02-03 11:26:59 +0000 | [diff] [blame] | 308 | bool result = new_space_.Contains(object); |
| 309 | ASSERT(!result || // Either not in new space |
| 310 | gc_state_ != NOT_IN_GC || // ... or in the middle of GC |
| 311 | InToSpace(object)); // ... or in to-space (where we allocate). |
| 312 | return result; |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 313 | } |
| 314 | |
| 315 | |
ulan@chromium.org | 750145a | 2013-03-07 15:14:13 +0000 | [diff] [blame] | 316 | bool Heap::InNewSpace(Address address) { |
| 317 | return new_space_.Contains(address); |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 318 | } |
| 319 | |
| 320 | |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 321 | bool Heap::InFromSpace(Object* object) { |
kasperl@chromium.org | 5a8ca6c | 2008-10-23 13:57:19 +0000 | [diff] [blame] | 322 | return new_space_.FromSpaceContains(object); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 323 | } |
| 324 | |
| 325 | |
| 326 | bool Heap::InToSpace(Object* object) { |
kasperl@chromium.org | 5a8ca6c | 2008-10-23 13:57:19 +0000 | [diff] [blame] | 327 | return new_space_.ToSpaceContains(object); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 328 | } |
| 329 | |
| 330 | |
ulan@chromium.org | 750145a | 2013-03-07 15:14:13 +0000 | [diff] [blame] | 331 | bool Heap::InOldPointerSpace(Address address) { |
| 332 | return old_pointer_space_->Contains(address); |
| 333 | } |
| 334 | |
| 335 | |
| 336 | bool Heap::InOldPointerSpace(Object* object) { |
| 337 | return InOldPointerSpace(reinterpret_cast<Address>(object)); |
| 338 | } |
| 339 | |
| 340 | |
mstarzinger@chromium.org | e27d617 | 2013-04-17 11:51:44 +0000 | [diff] [blame] | 341 | bool Heap::InOldDataSpace(Address address) { |
| 342 | return old_data_space_->Contains(address); |
| 343 | } |
| 344 | |
| 345 | |
| 346 | bool Heap::InOldDataSpace(Object* object) { |
| 347 | return InOldDataSpace(reinterpret_cast<Address>(object)); |
| 348 | } |
| 349 | |
| 350 | |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 351 | bool Heap::OldGenerationAllocationLimitReached() { |
| 352 | if (!incremental_marking()->IsStopped()) return false; |
| 353 | return OldGenerationSpaceAvailable() < 0; |
| 354 | } |
| 355 | |
| 356 | |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 357 | bool Heap::ShouldBePromoted(Address old_address, int object_size) { |
| 358 | // An object should be promoted if: |
| 359 | // - the object has survived a scavenge operation or |
| 360 | // - to space is already 25% full. |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 361 | NewSpacePage* page = NewSpacePage::FromAddress(old_address); |
| 362 | Address age_mark = new_space_.age_mark(); |
| 363 | bool below_mark = page->IsFlagSet(MemoryChunk::NEW_SPACE_BELOW_AGE_MARK) && |
| 364 | (!page->ContainsLimit(age_mark) || old_address < age_mark); |
| 365 | return below_mark || (new_space_.Size() + object_size) >= |
| 366 | (new_space_.EffectiveCapacity() >> 2); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 367 | } |
| 368 | |
| 369 | |
| 370 | void Heap::RecordWrite(Address address, int offset) { |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 371 | if (!InNewSpace(address)) store_buffer_.Mark(address + offset); |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 372 | } |
| 373 | |
| 374 | |
ager@chromium.org | ce5e87b | 2010-03-10 10:24:18 +0000 | [diff] [blame] | 375 | void Heap::RecordWrites(Address address, int start, int len) { |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 376 | if (!InNewSpace(address)) { |
| 377 | for (int i = 0; i < len; i++) { |
| 378 | store_buffer_.Mark(address + start + i * kPointerSize); |
| 379 | } |
| 380 | } |
ager@chromium.org | ce5e87b | 2010-03-10 10:24:18 +0000 | [diff] [blame] | 381 | } |
| 382 | |
| 383 | |
ager@chromium.org | 9258b6b | 2008-09-11 09:11:10 +0000 | [diff] [blame] | 384 | OldSpace* Heap::TargetSpace(HeapObject* object) { |
kasperl@chromium.org | 9bbf968 | 2008-10-30 11:53:07 +0000 | [diff] [blame] | 385 | InstanceType type = object->map()->instance_type(); |
| 386 | AllocationSpace space = TargetSpaceId(type); |
| 387 | return (space == OLD_POINTER_SPACE) |
| 388 | ? old_pointer_space_ |
| 389 | : old_data_space_; |
| 390 | } |
| 391 | |
| 392 | |
| 393 | AllocationSpace Heap::TargetSpaceId(InstanceType type) { |
ager@chromium.org | 9258b6b | 2008-09-11 09:11:10 +0000 | [diff] [blame] | 394 | // Heap numbers and sequential strings are promoted to old data space, all |
| 395 | // other object types are promoted to old pointer space. We do not use |
kasper.lund | 7276f14 | 2008-07-30 08:49:36 +0000 | [diff] [blame] | 396 | // object->IsHeapNumber() and object->IsSeqString() because we already |
| 397 | // know that object has the heap object tag. |
fschneider@chromium.org | 0c20e67 | 2010-01-14 15:28:53 +0000 | [diff] [blame] | 398 | |
| 399 | // These objects are never allocated in new space. |
| 400 | ASSERT(type != MAP_TYPE); |
| 401 | ASSERT(type != CODE_TYPE); |
| 402 | ASSERT(type != ODDBALL_TYPE); |
danno@chromium.org | 4172848 | 2013-06-12 22:31:22 +0000 | [diff] [blame] | 403 | ASSERT(type != CELL_TYPE); |
| 404 | ASSERT(type != PROPERTY_CELL_TYPE); |
fschneider@chromium.org | 0c20e67 | 2010-01-14 15:28:53 +0000 | [diff] [blame] | 405 | |
mstarzinger@chromium.org | f705b50 | 2013-04-04 11:38:09 +0000 | [diff] [blame] | 406 | if (type <= LAST_NAME_TYPE) { |
| 407 | if (type == SYMBOL_TYPE) return OLD_POINTER_SPACE; |
| 408 | ASSERT(type < FIRST_NONSTRING_TYPE); |
ricow@chromium.org | 4668a2c | 2011-08-29 10:41:00 +0000 | [diff] [blame] | 409 | // There are four string representations: sequential strings, external |
| 410 | // strings, cons strings, and sliced strings. |
| 411 | // Only the latter two contain non-map-word pointers to heap objects. |
| 412 | return ((type & kIsIndirectStringMask) == kIsIndirectStringTag) |
fschneider@chromium.org | 0c20e67 | 2010-01-14 15:28:53 +0000 | [diff] [blame] | 413 | ? OLD_POINTER_SPACE |
| 414 | : OLD_DATA_SPACE; |
| 415 | } else { |
| 416 | return (type <= LAST_DATA_TYPE) ? OLD_DATA_SPACE : OLD_POINTER_SPACE; |
| 417 | } |
kasper.lund | 7276f14 | 2008-07-30 08:49:36 +0000 | [diff] [blame] | 418 | } |
| 419 | |
| 420 | |
verwaest@chromium.org | ec6855e | 2013-08-22 12:26:58 +0000 | [diff] [blame] | 421 | bool Heap::AllowedToBeMigrated(HeapObject* object, AllocationSpace dst) { |
| 422 | // Object migration is governed by the following rules: |
| 423 | // |
| 424 | // 1) Objects in new-space can be migrated to one of the old spaces |
| 425 | // that matches their target space or they stay in new-space. |
| 426 | // 2) Objects in old-space stay in the same space when migrating. |
| 427 | // 3) Fillers (two or more words) can migrate due to left-trimming of |
| 428 | // fixed arrays in new-space, old-data-space and old-pointer-space. |
| 429 | // 4) Fillers (one word) can never migrate, they are skipped by |
| 430 | // incremental marking explicitly to prevent invalid pattern. |
| 431 | // |
| 432 | // Since this function is used for debugging only, we do not place |
| 433 | // asserts here, but check everything explicitly. |
| 434 | if (object->map() == one_pointer_filler_map()) return false; |
| 435 | InstanceType type = object->map()->instance_type(); |
| 436 | MemoryChunk* chunk = MemoryChunk::FromAddress(object->address()); |
| 437 | AllocationSpace src = chunk->owner()->identity(); |
| 438 | switch (src) { |
| 439 | case NEW_SPACE: |
| 440 | return dst == src || dst == TargetSpaceId(type); |
| 441 | case OLD_POINTER_SPACE: |
| 442 | return dst == src && (dst == TargetSpaceId(type) || object->IsFiller()); |
| 443 | case OLD_DATA_SPACE: |
| 444 | return dst == src && dst == TargetSpaceId(type); |
| 445 | case CODE_SPACE: |
| 446 | return dst == src && type == CODE_TYPE; |
| 447 | case MAP_SPACE: |
| 448 | case CELL_SPACE: |
| 449 | case PROPERTY_CELL_SPACE: |
| 450 | case LO_SPACE: |
| 451 | return false; |
| 452 | } |
| 453 | UNREACHABLE(); |
| 454 | return false; |
| 455 | } |
| 456 | |
| 457 | |
ricow@chromium.org | 30ce411 | 2010-05-31 10:38:25 +0000 | [diff] [blame] | 458 | void Heap::CopyBlock(Address dst, Address src, int byte_size) { |
ricow@chromium.org | 30ce411 | 2010-05-31 10:38:25 +0000 | [diff] [blame] | 459 | CopyWords(reinterpret_cast<Object**>(dst), |
| 460 | reinterpret_cast<Object**>(src), |
mstarzinger@chromium.org | e27d617 | 2013-04-17 11:51:44 +0000 | [diff] [blame] | 461 | static_cast<size_t>(byte_size / kPointerSize)); |
lrn@chromium.org | 25156de | 2010-04-06 13:10:27 +0000 | [diff] [blame] | 462 | } |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 463 | |
kasperl@chromium.org | 5a8ca6c | 2008-10-23 13:57:19 +0000 | [diff] [blame] | 464 | |
ricow@chromium.org | 30ce411 | 2010-05-31 10:38:25 +0000 | [diff] [blame] | 465 | void Heap::MoveBlock(Address dst, Address src, int byte_size) { |
ricow@chromium.org | c9c8082 | 2010-04-21 08:22:37 +0000 | [diff] [blame] | 466 | ASSERT(IsAligned(byte_size, kPointerSize)); |
whesse@chromium.org | b6e43bb | 2010-04-14 09:36:28 +0000 | [diff] [blame] | 467 | |
| 468 | int size_in_words = byte_size / kPointerSize; |
| 469 | |
ricow@chromium.org | 9fa0967 | 2011-07-25 11:05:35 +0000 | [diff] [blame] | 470 | if ((dst < src) || (dst >= (src + byte_size))) { |
ricow@chromium.org | 30ce411 | 2010-05-31 10:38:25 +0000 | [diff] [blame] | 471 | Object** src_slot = reinterpret_cast<Object**>(src); |
| 472 | Object** dst_slot = reinterpret_cast<Object**>(dst); |
| 473 | Object** end_slot = src_slot + size_in_words; |
whesse@chromium.org | b6e43bb | 2010-04-14 09:36:28 +0000 | [diff] [blame] | 474 | |
ricow@chromium.org | 30ce411 | 2010-05-31 10:38:25 +0000 | [diff] [blame] | 475 | while (src_slot != end_slot) { |
| 476 | *dst_slot++ = *src_slot++; |
whesse@chromium.org | b6e43bb | 2010-04-14 09:36:28 +0000 | [diff] [blame] | 477 | } |
| 478 | } else { |
mstarzinger@chromium.org | e27d617 | 2013-04-17 11:51:44 +0000 | [diff] [blame] | 479 | OS::MemMove(dst, src, static_cast<size_t>(byte_size)); |
whesse@chromium.org | b6e43bb | 2010-04-14 09:36:28 +0000 | [diff] [blame] | 480 | } |
| 481 | } |
| 482 | |
| 483 | |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 484 | void Heap::ScavengePointer(HeapObject** p) { |
| 485 | ScavengeObject(p, *p); |
| 486 | } |
| 487 | |
| 488 | |
machenbach@chromium.org | c86e8c2 | 2013-11-27 15:11:04 +0000 | [diff] [blame] | 489 | void Heap::UpdateAllocationSiteFeedback(HeapObject* object) { |
machenbach@chromium.org | 0345396 | 2014-01-10 14:16:31 +0000 | [diff] [blame] | 490 | Heap* heap = object->GetHeap(); |
machenbach@chromium.org | 4ddd2f1 | 2014-01-14 08:13:44 +0000 | [diff] [blame] | 491 | ASSERT(heap->InNewSpace(object)); |
| 492 | |
| 493 | if (!FLAG_allocation_site_pretenuring || |
machenbach@chromium.org | 4ddd2f1 | 2014-01-14 08:13:44 +0000 | [diff] [blame] | 494 | !AllocationSite::CanTrack(object->map()->instance_type())) return; |
| 495 | |
| 496 | // Either object is the last object in the from space, or there is another |
| 497 | // object of at least word size (the header map word) following it, so |
| 498 | // suffices to compare ptr and top here. |
| 499 | Address ptr = object->address() + object->Size(); |
| 500 | Address top = heap->new_space()->FromSpacePageHigh(); |
| 501 | ASSERT(ptr == top || ptr + HeapObject::kHeaderSize <= top); |
| 502 | if (ptr == top) return; |
| 503 | |
| 504 | HeapObject* candidate = HeapObject::FromAddress(ptr); |
| 505 | if (candidate->map() != heap->allocation_memento_map()) return; |
| 506 | |
| 507 | AllocationMemento* memento = AllocationMemento::cast(candidate); |
| 508 | if (!memento->IsValid()) return; |
| 509 | |
| 510 | if (memento->GetAllocationSite()->IncrementMementoFoundCount() && |
| 511 | heap->allocation_sites_scratchpad_length < |
| 512 | kAllocationSiteScratchpadSize) { |
| 513 | heap->allocation_sites_scratchpad[ |
| 514 | heap->allocation_sites_scratchpad_length++] = |
| 515 | memento->GetAllocationSite(); |
machenbach@chromium.org | c86e8c2 | 2013-11-27 15:11:04 +0000 | [diff] [blame] | 516 | } |
| 517 | } |
| 518 | |
| 519 | |
kasperl@chromium.org | b3284ad | 2009-05-18 06:12:45 +0000 | [diff] [blame] | 520 | void Heap::ScavengeObject(HeapObject** p, HeapObject* object) { |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 521 | ASSERT(object->GetIsolate()->heap()->InFromSpace(object)); |
kasperl@chromium.org | b3284ad | 2009-05-18 06:12:45 +0000 | [diff] [blame] | 522 | |
| 523 | // We use the first word (where the map pointer usually is) of a heap |
| 524 | // object to record the forwarding pointer. A forwarding pointer can |
| 525 | // point to an old space, the code space, or the to space of the new |
| 526 | // generation. |
| 527 | MapWord first_word = object->map_word(); |
| 528 | |
| 529 | // If the first word is a forwarding address, the object has already been |
| 530 | // copied. |
| 531 | if (first_word.IsForwardingAddress()) { |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 532 | HeapObject* dest = first_word.ToForwardingAddress(); |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 533 | ASSERT(object->GetIsolate()->heap()->InFromSpace(*p)); |
erik.corry@gmail.com | c3b670f | 2011-10-05 21:44:48 +0000 | [diff] [blame] | 534 | *p = dest; |
kasperl@chromium.org | b3284ad | 2009-05-18 06:12:45 +0000 | [diff] [blame] | 535 | return; |
| 536 | } |
| 537 | |
machenbach@chromium.org | c86e8c2 | 2013-11-27 15:11:04 +0000 | [diff] [blame] | 538 | UpdateAllocationSiteFeedback(object); |
jkummerow@chromium.org | fb7a7c4 | 2013-10-02 11:41:02 +0000 | [diff] [blame] | 539 | |
hpayer@chromium.org | 27ce874 | 2013-09-19 09:59:01 +0000 | [diff] [blame] | 540 | // AllocationMementos are unrooted and shouldn't survive a scavenge |
| 541 | ASSERT(object->map() != object->GetHeap()->allocation_memento_map()); |
kasperl@chromium.org | b3284ad | 2009-05-18 06:12:45 +0000 | [diff] [blame] | 542 | // Call the slow part of scavenge object. |
| 543 | return ScavengeObjectSlow(p, object); |
| 544 | } |
| 545 | |
| 546 | |
machenbach@chromium.org | 4ddd2f1 | 2014-01-14 08:13:44 +0000 | [diff] [blame] | 547 | bool Heap::CollectGarbage(AllocationSpace space, |
| 548 | const char* gc_reason, |
| 549 | const v8::GCCallbackFlags callbackFlags) { |
rossberg@chromium.org | 994edf6 | 2012-02-06 10:12:55 +0000 | [diff] [blame] | 550 | const char* collector_reason = NULL; |
| 551 | GarbageCollector collector = SelectGarbageCollector(space, &collector_reason); |
machenbach@chromium.org | 4ddd2f1 | 2014-01-14 08:13:44 +0000 | [diff] [blame] | 552 | return CollectGarbage( |
| 553 | space, collector, gc_reason, collector_reason, callbackFlags); |
whesse@chromium.org | f0ac72d | 2010-11-08 12:47:26 +0000 | [diff] [blame] | 554 | } |
| 555 | |
| 556 | |
lrn@chromium.org | 303ada7 | 2010-10-27 09:33:13 +0000 | [diff] [blame] | 557 | MaybeObject* Heap::PrepareForCompare(String* str) { |
fschneider@chromium.org | 086aac6 | 2010-03-17 13:18:24 +0000 | [diff] [blame] | 558 | // Always flatten small strings and force flattening of long strings |
| 559 | // after we have accumulated a certain amount we failed to flatten. |
| 560 | static const int kMaxAlwaysFlattenLength = 32; |
| 561 | static const int kFlattenLongThreshold = 16*KB; |
| 562 | |
| 563 | const int length = str->length(); |
lrn@chromium.org | 303ada7 | 2010-10-27 09:33:13 +0000 | [diff] [blame] | 564 | MaybeObject* obj = str->TryFlatten(); |
fschneider@chromium.org | 086aac6 | 2010-03-17 13:18:24 +0000 | [diff] [blame] | 565 | if (length <= kMaxAlwaysFlattenLength || |
kmillikin@chromium.org | 5d8f0e6 | 2010-03-24 08:21:20 +0000 | [diff] [blame] | 566 | unflattened_strings_length_ >= kFlattenLongThreshold) { |
fschneider@chromium.org | 086aac6 | 2010-03-17 13:18:24 +0000 | [diff] [blame] | 567 | return obj; |
| 568 | } |
| 569 | if (obj->IsFailure()) { |
kmillikin@chromium.org | 5d8f0e6 | 2010-03-24 08:21:20 +0000 | [diff] [blame] | 570 | unflattened_strings_length_ += length; |
fschneider@chromium.org | 086aac6 | 2010-03-17 13:18:24 +0000 | [diff] [blame] | 571 | } |
| 572 | return str; |
| 573 | } |
| 574 | |
| 575 | |
machenbach@chromium.org | 7ff7607 | 2013-11-21 09:47:43 +0000 | [diff] [blame] | 576 | int64_t Heap::AdjustAmountOfExternalAllocatedMemory( |
| 577 | int64_t change_in_bytes) { |
erik.corry@gmail.com | f2038fb | 2012-01-16 11:42:08 +0000 | [diff] [blame] | 578 | ASSERT(HasBeenSetUp()); |
machenbach@chromium.org | 7ff7607 | 2013-11-21 09:47:43 +0000 | [diff] [blame] | 579 | int64_t amount = amount_of_external_allocated_memory_ + change_in_bytes; |
mmassi@chromium.org | 49a4467 | 2012-12-04 13:52:03 +0000 | [diff] [blame] | 580 | if (change_in_bytes > 0) { |
kasperl@chromium.org | e959c18 | 2009-07-27 08:59:04 +0000 | [diff] [blame] | 581 | // Avoid overflow. |
| 582 | if (amount > amount_of_external_allocated_memory_) { |
| 583 | amount_of_external_allocated_memory_ = amount; |
mstarzinger@chromium.org | 471f2f1 | 2012-08-10 14:46:33 +0000 | [diff] [blame] | 584 | } else { |
| 585 | // Give up and reset the counters in case of an overflow. |
| 586 | amount_of_external_allocated_memory_ = 0; |
| 587 | amount_of_external_allocated_memory_at_last_global_gc_ = 0; |
kasperl@chromium.org | e959c18 | 2009-07-27 08:59:04 +0000 | [diff] [blame] | 588 | } |
machenbach@chromium.org | 7ff7607 | 2013-11-21 09:47:43 +0000 | [diff] [blame] | 589 | int64_t amount_since_last_global_gc = PromotedExternalMemorySize(); |
kasperl@chromium.org | e959c18 | 2009-07-27 08:59:04 +0000 | [diff] [blame] | 590 | if (amount_since_last_global_gc > external_allocation_limit_) { |
rossberg@chromium.org | 994edf6 | 2012-02-06 10:12:55 +0000 | [diff] [blame] | 591 | CollectAllGarbage(kNoGCFlags, "external memory allocation limit reached"); |
kasperl@chromium.org | e959c18 | 2009-07-27 08:59:04 +0000 | [diff] [blame] | 592 | } |
| 593 | } else { |
| 594 | // Avoid underflow. |
| 595 | if (amount >= 0) { |
| 596 | amount_of_external_allocated_memory_ = amount; |
mstarzinger@chromium.org | 471f2f1 | 2012-08-10 14:46:33 +0000 | [diff] [blame] | 597 | } else { |
mstarzinger@chromium.org | 1510d58 | 2013-06-28 14:00:48 +0000 | [diff] [blame] | 598 | // Give up and reset the counters in case of an underflow. |
mstarzinger@chromium.org | 471f2f1 | 2012-08-10 14:46:33 +0000 | [diff] [blame] | 599 | amount_of_external_allocated_memory_ = 0; |
| 600 | amount_of_external_allocated_memory_at_last_global_gc_ = 0; |
kasperl@chromium.org | e959c18 | 2009-07-27 08:59:04 +0000 | [diff] [blame] | 601 | } |
| 602 | } |
mstarzinger@chromium.org | 471f2f1 | 2012-08-10 14:46:33 +0000 | [diff] [blame] | 603 | if (FLAG_trace_external_memory) { |
| 604 | PrintPID("%8.0f ms: ", isolate()->time_millis_since_init()); |
| 605 | PrintF("Adjust amount of external memory: delta=%6" V8_PTR_PREFIX "d KB, " |
mstarzinger@chromium.org | 1510d58 | 2013-06-28 14:00:48 +0000 | [diff] [blame] | 606 | "amount=%6" V8_PTR_PREFIX "d KB, since_gc=%6" V8_PTR_PREFIX "d KB, " |
| 607 | "isolate=0x%08" V8PRIxPTR ".\n", |
machenbach@chromium.org | 7ff7607 | 2013-11-21 09:47:43 +0000 | [diff] [blame] | 608 | static_cast<intptr_t>(change_in_bytes / KB), |
| 609 | static_cast<intptr_t>(amount_of_external_allocated_memory_ / KB), |
| 610 | static_cast<intptr_t>(PromotedExternalMemorySize() / KB), |
mstarzinger@chromium.org | 471f2f1 | 2012-08-10 14:46:33 +0000 | [diff] [blame] | 611 | reinterpret_cast<intptr_t>(isolate())); |
| 612 | } |
kasperl@chromium.org | e959c18 | 2009-07-27 08:59:04 +0000 | [diff] [blame] | 613 | ASSERT(amount_of_external_allocated_memory_ >= 0); |
| 614 | return amount_of_external_allocated_memory_; |
| 615 | } |
| 616 | |
| 617 | |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 618 | Isolate* Heap::isolate() { |
| 619 | return reinterpret_cast<Isolate*>(reinterpret_cast<intptr_t>(this) - |
| 620 | reinterpret_cast<size_t>(reinterpret_cast<Isolate*>(4)->heap()) + 4); |
| 621 | } |
| 622 | |
kasperl@chromium.org | 7be3c99 | 2009-03-12 07:19:55 +0000 | [diff] [blame] | 623 | |
whesse@chromium.org | 4a5224e | 2010-10-20 12:37:07 +0000 | [diff] [blame] | 624 | #ifdef DEBUG |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 625 | #define GC_GREEDY_CHECK(ISOLATE) \ |
| 626 | if (FLAG_gc_greedy) (ISOLATE)->heap()->GarbageCollectionGreedyCheck() |
whesse@chromium.org | 4a5224e | 2010-10-20 12:37:07 +0000 | [diff] [blame] | 627 | #else |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 628 | #define GC_GREEDY_CHECK(ISOLATE) { } |
whesse@chromium.org | 4a5224e | 2010-10-20 12:37:07 +0000 | [diff] [blame] | 629 | #endif |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 630 | |
kasperl@chromium.org | 9bbf968 | 2008-10-30 11:53:07 +0000 | [diff] [blame] | 631 | // Calls the FUNCTION_CALL function and retries it up to three times |
| 632 | // to guarantee that any allocations performed during the call will |
| 633 | // succeed if there's enough memory. |
| 634 | |
lrn@chromium.org | 303ada7 | 2010-10-27 09:33:13 +0000 | [diff] [blame] | 635 | // Warning: Do not use the identifiers __object__, __maybe_object__ or |
| 636 | // __scope__ in a call to this macro. |
kasperl@chromium.org | 9bbf968 | 2008-10-30 11:53:07 +0000 | [diff] [blame] | 637 | |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 638 | #define CALL_AND_RETRY(ISOLATE, FUNCTION_CALL, RETURN_VALUE, RETURN_EMPTY, OOM)\ |
| 639 | do { \ |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 640 | GC_GREEDY_CHECK(ISOLATE); \ |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 641 | MaybeObject* __maybe_object__ = FUNCTION_CALL; \ |
| 642 | Object* __object__ = NULL; \ |
| 643 | if (__maybe_object__->ToObject(&__object__)) RETURN_VALUE; \ |
| 644 | if (__maybe_object__->IsOutOfMemory()) { \ |
| 645 | OOM; \ |
| 646 | } \ |
| 647 | if (!__maybe_object__->IsRetryAfterGC()) RETURN_EMPTY; \ |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 648 | (ISOLATE)->heap()->CollectGarbage(Failure::cast(__maybe_object__)-> \ |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 649 | allocation_space(), \ |
| 650 | "allocation failure"); \ |
| 651 | __maybe_object__ = FUNCTION_CALL; \ |
| 652 | if (__maybe_object__->ToObject(&__object__)) RETURN_VALUE; \ |
| 653 | if (__maybe_object__->IsOutOfMemory()) { \ |
| 654 | OOM; \ |
| 655 | } \ |
| 656 | if (!__maybe_object__->IsRetryAfterGC()) RETURN_EMPTY; \ |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 657 | (ISOLATE)->counters()->gc_last_resort_from_handles()->Increment(); \ |
| 658 | (ISOLATE)->heap()->CollectAllAvailableGarbage("last resort gc"); \ |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 659 | { \ |
| 660 | AlwaysAllocateScope __scope__; \ |
| 661 | __maybe_object__ = FUNCTION_CALL; \ |
| 662 | } \ |
| 663 | if (__maybe_object__->ToObject(&__object__)) RETURN_VALUE; \ |
| 664 | if (__maybe_object__->IsOutOfMemory()) { \ |
| 665 | OOM; \ |
| 666 | } \ |
| 667 | if (__maybe_object__->IsRetryAfterGC()) { \ |
| 668 | /* TODO(1181417): Fix this. */ \ |
machenbach@chromium.org | 5c88bc3 | 2014-01-17 08:10:36 +0000 | [diff] [blame] | 669 | v8::internal::Heap::FatalProcessOutOfMemory("CALL_AND_RETRY_LAST", true);\ |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 670 | } \ |
| 671 | RETURN_EMPTY; \ |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 672 | } while (false) |
| 673 | |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 674 | #define CALL_AND_RETRY_OR_DIE( \ |
| 675 | ISOLATE, FUNCTION_CALL, RETURN_VALUE, RETURN_EMPTY) \ |
| 676 | CALL_AND_RETRY( \ |
| 677 | ISOLATE, \ |
| 678 | FUNCTION_CALL, \ |
| 679 | RETURN_VALUE, \ |
| 680 | RETURN_EMPTY, \ |
machenbach@chromium.org | 5c88bc3 | 2014-01-17 08:10:36 +0000 | [diff] [blame] | 681 | v8::internal::Heap::FatalProcessOutOfMemory("CALL_AND_RETRY", true)) |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 682 | |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 683 | #define CALL_HEAP_FUNCTION(ISOLATE, FUNCTION_CALL, TYPE) \ |
| 684 | CALL_AND_RETRY_OR_DIE(ISOLATE, \ |
| 685 | FUNCTION_CALL, \ |
| 686 | return Handle<TYPE>(TYPE::cast(__object__), ISOLATE), \ |
| 687 | return Handle<TYPE>()) \ |
kasperl@chromium.org | 9bbf968 | 2008-10-30 11:53:07 +0000 | [diff] [blame] | 688 | |
| 689 | |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 690 | #define CALL_HEAP_FUNCTION_VOID(ISOLATE, FUNCTION_CALL) \ |
| 691 | CALL_AND_RETRY_OR_DIE(ISOLATE, FUNCTION_CALL, return, return) |
kasperl@chromium.org | b912362 | 2008-09-17 14:05:56 +0000 | [diff] [blame] | 692 | |
| 693 | |
mstarzinger@chromium.org | b228be0 | 2013-04-18 14:56:59 +0000 | [diff] [blame] | 694 | #define CALL_HEAP_FUNCTION_PASS_EXCEPTION(ISOLATE, FUNCTION_CALL) \ |
| 695 | CALL_AND_RETRY(ISOLATE, \ |
| 696 | FUNCTION_CALL, \ |
| 697 | return __object__, \ |
ulan@chromium.org | 77ca49a | 2013-04-22 09:43:56 +0000 | [diff] [blame] | 698 | return __maybe_object__, \ |
mstarzinger@chromium.org | b228be0 | 2013-04-18 14:56:59 +0000 | [diff] [blame] | 699 | return __maybe_object__) |
| 700 | |
| 701 | |
kmillikin@chromium.org | 13bd294 | 2009-12-16 15:36:05 +0000 | [diff] [blame] | 702 | void ExternalStringTable::AddString(String* string) { |
| 703 | ASSERT(string->IsExternalString()); |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 704 | if (heap_->InNewSpace(string)) { |
kmillikin@chromium.org | 13bd294 | 2009-12-16 15:36:05 +0000 | [diff] [blame] | 705 | new_space_strings_.Add(string); |
| 706 | } else { |
| 707 | old_space_strings_.Add(string); |
| 708 | } |
| 709 | } |
| 710 | |
| 711 | |
| 712 | void ExternalStringTable::Iterate(ObjectVisitor* v) { |
| 713 | if (!new_space_strings_.is_empty()) { |
| 714 | Object** start = &new_space_strings_[0]; |
| 715 | v->VisitPointers(start, start + new_space_strings_.length()); |
| 716 | } |
| 717 | if (!old_space_strings_.is_empty()) { |
| 718 | Object** start = &old_space_strings_[0]; |
| 719 | v->VisitPointers(start, start + old_space_strings_.length()); |
| 720 | } |
| 721 | } |
| 722 | |
| 723 | |
| 724 | // Verify() is inline to avoid ifdef-s around its calls in release |
| 725 | // mode. |
| 726 | void ExternalStringTable::Verify() { |
| 727 | #ifdef DEBUG |
| 728 | for (int i = 0; i < new_space_strings_.length(); ++i) { |
mstarzinger@chromium.org | 15613d0 | 2012-05-23 12:04:37 +0000 | [diff] [blame] | 729 | Object* obj = Object::cast(new_space_strings_[i]); |
mstarzinger@chromium.org | 15613d0 | 2012-05-23 12:04:37 +0000 | [diff] [blame] | 730 | ASSERT(heap_->InNewSpace(obj)); |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 731 | ASSERT(obj != heap_->the_hole_value()); |
kmillikin@chromium.org | 13bd294 | 2009-12-16 15:36:05 +0000 | [diff] [blame] | 732 | } |
| 733 | for (int i = 0; i < old_space_strings_.length(); ++i) { |
mstarzinger@chromium.org | 15613d0 | 2012-05-23 12:04:37 +0000 | [diff] [blame] | 734 | Object* obj = Object::cast(old_space_strings_[i]); |
mstarzinger@chromium.org | 15613d0 | 2012-05-23 12:04:37 +0000 | [diff] [blame] | 735 | ASSERT(!heap_->InNewSpace(obj)); |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 736 | ASSERT(obj != heap_->the_hole_value()); |
kmillikin@chromium.org | 13bd294 | 2009-12-16 15:36:05 +0000 | [diff] [blame] | 737 | } |
| 738 | #endif |
| 739 | } |
| 740 | |
| 741 | |
| 742 | void ExternalStringTable::AddOldString(String* string) { |
| 743 | ASSERT(string->IsExternalString()); |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 744 | ASSERT(!heap_->InNewSpace(string)); |
kmillikin@chromium.org | 13bd294 | 2009-12-16 15:36:05 +0000 | [diff] [blame] | 745 | old_space_strings_.Add(string); |
| 746 | } |
| 747 | |
| 748 | |
| 749 | void ExternalStringTable::ShrinkNewStrings(int position) { |
| 750 | new_space_strings_.Rewind(position); |
svenpanne@chromium.org | c859c4f | 2012-10-15 11:51:39 +0000 | [diff] [blame] | 751 | #ifdef VERIFY_HEAP |
erik.corry@gmail.com | 394dbcf | 2011-10-27 07:38:48 +0000 | [diff] [blame] | 752 | if (FLAG_verify_heap) { |
| 753 | Verify(); |
| 754 | } |
svenpanne@chromium.org | c859c4f | 2012-10-15 11:51:39 +0000 | [diff] [blame] | 755 | #endif |
kmillikin@chromium.org | 13bd294 | 2009-12-16 15:36:05 +0000 | [diff] [blame] | 756 | } |
| 757 | |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 758 | |
| 759 | void Heap::ClearInstanceofCache() { |
| 760 | set_instanceof_cache_function(the_hole_value()); |
| 761 | } |
| 762 | |
| 763 | |
| 764 | Object* Heap::ToBoolean(bool condition) { |
| 765 | return condition ? true_value() : false_value(); |
| 766 | } |
| 767 | |
| 768 | |
| 769 | void Heap::CompletelyClearInstanceofCache() { |
| 770 | set_instanceof_cache_map(the_hole_value()); |
| 771 | set_instanceof_cache_function(the_hole_value()); |
| 772 | } |
| 773 | |
| 774 | |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 775 | AlwaysAllocateScope::AlwaysAllocateScope() { |
| 776 | // We shouldn't hit any nested scopes, because that requires |
| 777 | // non-handle code to call handle code. The code still works but |
| 778 | // performance will degrade, so we want to catch this situation |
| 779 | // in debug mode. |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 780 | Isolate* isolate = Isolate::Current(); |
| 781 | ASSERT(isolate->heap()->always_allocate_scope_depth_ == 0); |
| 782 | isolate->heap()->always_allocate_scope_depth_++; |
sgjesse@chromium.org | ea88ce9 | 2011-03-23 11:19:56 +0000 | [diff] [blame] | 783 | } |
| 784 | |
| 785 | |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 786 | AlwaysAllocateScope::~AlwaysAllocateScope() { |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 787 | Isolate* isolate = Isolate::Current(); |
| 788 | isolate->heap()->always_allocate_scope_depth_--; |
| 789 | ASSERT(isolate->heap()->always_allocate_scope_depth_ == 0); |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 790 | } |
| 791 | |
| 792 | |
danno@chromium.org | 94b0d6f | 2013-02-04 13:33:20 +0000 | [diff] [blame] | 793 | #ifdef VERIFY_HEAP |
jkummerow@chromium.org | 25b0e21 | 2013-10-04 15:38:52 +0000 | [diff] [blame] | 794 | NoWeakObjectVerificationScope::NoWeakObjectVerificationScope() { |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 795 | Isolate* isolate = Isolate::Current(); |
jkummerow@chromium.org | 25b0e21 | 2013-10-04 15:38:52 +0000 | [diff] [blame] | 796 | isolate->heap()->no_weak_object_verification_scope_depth_++; |
danno@chromium.org | 94b0d6f | 2013-02-04 13:33:20 +0000 | [diff] [blame] | 797 | } |
| 798 | |
| 799 | |
jkummerow@chromium.org | 25b0e21 | 2013-10-04 15:38:52 +0000 | [diff] [blame] | 800 | NoWeakObjectVerificationScope::~NoWeakObjectVerificationScope() { |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 801 | Isolate* isolate = Isolate::Current(); |
jkummerow@chromium.org | 25b0e21 | 2013-10-04 15:38:52 +0000 | [diff] [blame] | 802 | isolate->heap()->no_weak_object_verification_scope_depth_--; |
danno@chromium.org | 94b0d6f | 2013-02-04 13:33:20 +0000 | [diff] [blame] | 803 | } |
| 804 | #endif |
| 805 | |
| 806 | |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 807 | void VerifyPointersVisitor::VisitPointers(Object** start, Object** end) { |
| 808 | for (Object** current = start; current < end; current++) { |
| 809 | if ((*current)->IsHeapObject()) { |
| 810 | HeapObject* object = HeapObject::cast(*current); |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 811 | CHECK(object->GetIsolate()->heap()->Contains(object)); |
svenpanne@chromium.org | c859c4f | 2012-10-15 11:51:39 +0000 | [diff] [blame] | 812 | CHECK(object->map()->IsMap()); |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 813 | } |
| 814 | } |
| 815 | } |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 816 | |
| 817 | |
| 818 | double GCTracer::SizeOfHeapObjects() { |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 819 | return (static_cast<double>(heap_->SizeOfObjects())) / MB; |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 820 | } |
| 821 | |
| 822 | |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 823 | DisallowAllocationFailure::DisallowAllocationFailure() { |
jkummerow@chromium.org | 000f7fb | 2012-08-01 11:14:42 +0000 | [diff] [blame] | 824 | #ifdef DEBUG |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 825 | Isolate* isolate = Isolate::Current(); |
| 826 | old_state_ = isolate->heap()->disallow_allocation_failure_; |
| 827 | isolate->heap()->disallow_allocation_failure_ = true; |
jkummerow@chromium.org | 000f7fb | 2012-08-01 11:14:42 +0000 | [diff] [blame] | 828 | #endif |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 829 | } |
| 830 | |
| 831 | |
| 832 | DisallowAllocationFailure::~DisallowAllocationFailure() { |
jkummerow@chromium.org | 000f7fb | 2012-08-01 11:14:42 +0000 | [diff] [blame] | 833 | #ifdef DEBUG |
hpayer@chromium.org | c5d4971 | 2013-09-11 08:25:48 +0000 | [diff] [blame] | 834 | Isolate* isolate = Isolate::Current(); |
| 835 | isolate->heap()->disallow_allocation_failure_ = old_state_; |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 836 | #endif |
jkummerow@chromium.org | 000f7fb | 2012-08-01 11:14:42 +0000 | [diff] [blame] | 837 | } |
danno@chromium.org | fa458e4 | 2012-02-01 10:48:36 +0000 | [diff] [blame] | 838 | |
| 839 | |
christian.plesner.hansen | 43d26ec | 2008-07-03 15:10:15 +0000 | [diff] [blame] | 840 | } } // namespace v8::internal |
| 841 | |
| 842 | #endif // V8_HEAP_INL_H_ |