| OLD | NEW |
| 1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include <stdlib.h> | 5 #include <stdlib.h> |
| 6 | 6 |
| 7 #include "v8.h" | 7 #include "v8.h" |
| 8 | 8 |
| 9 #include "ast.h" | 9 #include "ast.h" |
| 10 #include "bootstrapper.h" | 10 #include "bootstrapper.h" |
| (...skipping 2216 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2227 | 2227 |
| 2228 | 2228 |
| 2229 void Isolate::FireCallCompletedCallback() { | 2229 void Isolate::FireCallCompletedCallback() { |
| 2230 bool has_call_completed_callbacks = !call_completed_callbacks_.is_empty(); | 2230 bool has_call_completed_callbacks = !call_completed_callbacks_.is_empty(); |
| 2231 bool run_microtasks = autorun_microtasks() && pending_microtask_count(); | 2231 bool run_microtasks = autorun_microtasks() && pending_microtask_count(); |
| 2232 if (!has_call_completed_callbacks && !run_microtasks) return; | 2232 if (!has_call_completed_callbacks && !run_microtasks) return; |
| 2233 | 2233 |
| 2234 if (!handle_scope_implementer()->CallDepthIsZero()) return; | 2234 if (!handle_scope_implementer()->CallDepthIsZero()) return; |
| 2235 if (run_microtasks) RunMicrotasks(); | 2235 if (run_microtasks) RunMicrotasks(); |
| 2236 // Fire callbacks. Increase call depth to prevent recursive callbacks. | 2236 // Fire callbacks. Increase call depth to prevent recursive callbacks. |
| 2237 handle_scope_implementer()->IncrementCallDepth(); | 2237 v8::Isolate::SuppressMicrotaskExecutionScope suppress( |
| 2238 reinterpret_cast<v8::Isolate*>(this)); |
| 2238 for (int i = 0; i < call_completed_callbacks_.length(); i++) { | 2239 for (int i = 0; i < call_completed_callbacks_.length(); i++) { |
| 2239 call_completed_callbacks_.at(i)(); | 2240 call_completed_callbacks_.at(i)(); |
| 2240 } | 2241 } |
| 2241 handle_scope_implementer()->DecrementCallDepth(); | |
| 2242 } | 2242 } |
| 2243 | 2243 |
| 2244 | 2244 |
| 2245 void Isolate::EnqueueMicrotask(Handle<JSFunction> microtask) { | 2245 void Isolate::EnqueueMicrotask(Handle<JSFunction> microtask) { |
| 2246 Handle<FixedArray> queue(heap()->microtask_queue(), this); | 2246 Handle<FixedArray> queue(heap()->microtask_queue(), this); |
| 2247 int num_tasks = pending_microtask_count(); | 2247 int num_tasks = pending_microtask_count(); |
| 2248 ASSERT(num_tasks <= queue->length()); | 2248 ASSERT(num_tasks <= queue->length()); |
| 2249 if (num_tasks == 0) { | 2249 if (num_tasks == 0) { |
| 2250 queue = factory()->NewFixedArray(8); | 2250 queue = factory()->NewFixedArray(8); |
| 2251 heap()->set_microtask_queue(*queue); | 2251 heap()->set_microtask_queue(*queue); |
| 2252 } else if (num_tasks == queue->length()) { | 2252 } else if (num_tasks == queue->length()) { |
| 2253 queue = FixedArray::CopySize(queue, num_tasks * 2); | 2253 queue = FixedArray::CopySize(queue, num_tasks * 2); |
| 2254 heap()->set_microtask_queue(*queue); | 2254 heap()->set_microtask_queue(*queue); |
| 2255 } | 2255 } |
| 2256 ASSERT(queue->get(num_tasks)->IsUndefined()); | 2256 ASSERT(queue->get(num_tasks)->IsUndefined()); |
| 2257 queue->set(num_tasks, *microtask); | 2257 queue->set(num_tasks, *microtask); |
| 2258 set_pending_microtask_count(num_tasks + 1); | 2258 set_pending_microtask_count(num_tasks + 1); |
| 2259 } | 2259 } |
| 2260 | 2260 |
| 2261 | 2261 |
| 2262 void Isolate::RunMicrotasks() { | 2262 void Isolate::RunMicrotasks() { |
| 2263 // TODO(adamk): This ASSERT triggers in mjsunit tests which | 2263 // TODO(adamk): This ASSERT triggers in mjsunit tests which |
| 2264 // call the %RunMicrotasks runtime function. But it should | 2264 // call the %RunMicrotasks runtime function. But it should |
| 2265 // never happen outside of tests, so it would be nice to | 2265 // never happen outside of tests, so it would be nice to |
| 2266 // uncomment it. | 2266 // uncomment it. |
| 2267 // | 2267 // |
| 2268 // ASSERT(handle_scope_implementer()->CallDepthIsZero()); | 2268 // ASSERT(handle_scope_implementer()->CallDepthIsZero()); |
| 2269 | 2269 |
| 2270 // Increase call depth to prevent recursive callbacks. | 2270 // Increase call depth to prevent recursive callbacks. |
| 2271 handle_scope_implementer()->IncrementCallDepth(); | 2271 v8::Isolate::SuppressMicrotaskExecutionScope suppress( |
| 2272 reinterpret_cast<v8::Isolate*>(this)); |
| 2272 | 2273 |
| 2273 while (pending_microtask_count() > 0) { | 2274 while (pending_microtask_count() > 0) { |
| 2274 HandleScope scope(this); | 2275 HandleScope scope(this); |
| 2275 int num_tasks = pending_microtask_count(); | 2276 int num_tasks = pending_microtask_count(); |
| 2276 Handle<FixedArray> queue(heap()->microtask_queue(), this); | 2277 Handle<FixedArray> queue(heap()->microtask_queue(), this); |
| 2277 ASSERT(num_tasks <= queue->length()); | 2278 ASSERT(num_tasks <= queue->length()); |
| 2278 set_pending_microtask_count(0); | 2279 set_pending_microtask_count(0); |
| 2279 heap()->set_microtask_queue(heap()->empty_fixed_array()); | 2280 heap()->set_microtask_queue(heap()->empty_fixed_array()); |
| 2280 | 2281 |
| 2281 for (int i = 0; i < num_tasks; i++) { | 2282 for (int i = 0; i < num_tasks; i++) { |
| 2282 HandleScope scope(this); | 2283 HandleScope scope(this); |
| 2283 Handle<JSFunction> microtask(JSFunction::cast(queue->get(i)), this); | 2284 Handle<JSFunction> microtask(JSFunction::cast(queue->get(i)), this); |
| 2284 // TODO(adamk): This should ignore/clear exceptions instead of Checking. | 2285 Handle<Object> exception; |
| 2285 Execution::Call(this, microtask, factory()->undefined_value(), | 2286 MaybeHandle<Object> result = Execution::TryCall( |
| 2286 0, NULL).Check(); | 2287 microtask, factory()->undefined_value(), 0, NULL, &exception); |
| 2288 // If execution is terminating, just bail out. |
| 2289 if (result.is_null() && |
| 2290 !exception.is_null() && |
| 2291 *exception == heap()->termination_exception()) { |
| 2292 // Clear out any remaining callbacks in the queue. |
| 2293 heap()->set_microtask_queue(heap()->empty_fixed_array()); |
| 2294 set_pending_microtask_count(0); |
| 2295 return; |
| 2296 } |
| 2287 } | 2297 } |
| 2288 } | 2298 } |
| 2289 | |
| 2290 handle_scope_implementer()->DecrementCallDepth(); | |
| 2291 } | 2299 } |
| 2292 | 2300 |
| 2293 | 2301 |
| 2294 } } // namespace v8::internal | 2302 } } // namespace v8::internal |
| OLD | NEW |