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 2214 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2225 for (int i = 0; i < call_completed_callbacks_.length(); i++) { | 2225 for (int i = 0; i < call_completed_callbacks_.length(); i++) { |
2226 if (callback == call_completed_callbacks_.at(i)) { | 2226 if (callback == call_completed_callbacks_.at(i)) { |
2227 call_completed_callbacks_.Remove(i); | 2227 call_completed_callbacks_.Remove(i); |
2228 } | 2228 } |
2229 } | 2229 } |
2230 } | 2230 } |
2231 | 2231 |
2232 | 2232 |
2233 void Isolate::FireCallCompletedCallback() { | 2233 void Isolate::FireCallCompletedCallback() { |
2234 bool has_call_completed_callbacks = !call_completed_callbacks_.is_empty(); | 2234 bool has_call_completed_callbacks = !call_completed_callbacks_.is_empty(); |
2235 bool run_microtasks = autorun_microtasks() && microtask_pending(); | 2235 bool run_microtasks = autorun_microtasks() && pending_microtask_count(); |
2236 if (!has_call_completed_callbacks && !run_microtasks) return; | 2236 if (!has_call_completed_callbacks && !run_microtasks) return; |
2237 | 2237 |
2238 if (!handle_scope_implementer()->CallDepthIsZero()) return; | 2238 if (!handle_scope_implementer()->CallDepthIsZero()) return; |
| 2239 if (run_microtasks) RunMicrotasks(); |
2239 // Fire callbacks. Increase call depth to prevent recursive callbacks. | 2240 // Fire callbacks. Increase call depth to prevent recursive callbacks. |
2240 handle_scope_implementer()->IncrementCallDepth(); | 2241 handle_scope_implementer()->IncrementCallDepth(); |
2241 if (run_microtasks) Execution::RunMicrotasks(this); | |
2242 for (int i = 0; i < call_completed_callbacks_.length(); i++) { | 2242 for (int i = 0; i < call_completed_callbacks_.length(); i++) { |
2243 call_completed_callbacks_.at(i)(); | 2243 call_completed_callbacks_.at(i)(); |
2244 } | 2244 } |
2245 handle_scope_implementer()->DecrementCallDepth(); | 2245 handle_scope_implementer()->DecrementCallDepth(); |
2246 } | 2246 } |
2247 | 2247 |
2248 | 2248 |
| 2249 void Isolate::EnqueueMicrotask(Handle<JSFunction> microtask) { |
| 2250 Handle<FixedArray> queue(heap()->microtask_queue(), this); |
| 2251 int num_tasks = pending_microtask_count(); |
| 2252 ASSERT(num_tasks <= queue->length()); |
| 2253 if (num_tasks == 0) { |
| 2254 queue = factory()->NewFixedArray(8); |
| 2255 heap()->set_microtask_queue(*queue); |
| 2256 } else if (num_tasks == queue->length()) { |
| 2257 queue = FixedArray::CopySize(queue, num_tasks * 2); |
| 2258 heap()->set_microtask_queue(*queue); |
| 2259 } |
| 2260 ASSERT(queue->get(num_tasks)->IsUndefined()); |
| 2261 queue->set(num_tasks, *microtask); |
| 2262 set_pending_microtask_count(num_tasks + 1); |
| 2263 } |
| 2264 |
| 2265 |
2249 void Isolate::RunMicrotasks() { | 2266 void Isolate::RunMicrotasks() { |
2250 if (!microtask_pending()) | |
2251 return; | |
2252 | |
2253 ASSERT(handle_scope_implementer()->CallDepthIsZero()); | 2267 ASSERT(handle_scope_implementer()->CallDepthIsZero()); |
2254 | 2268 |
2255 // Increase call depth to prevent recursive callbacks. | 2269 // Increase call depth to prevent recursive callbacks. |
2256 handle_scope_implementer()->IncrementCallDepth(); | 2270 handle_scope_implementer()->IncrementCallDepth(); |
2257 Execution::RunMicrotasks(this); | 2271 |
| 2272 while (pending_microtask_count() > 0) { |
| 2273 HandleScope scope(this); |
| 2274 int num_tasks = pending_microtask_count(); |
| 2275 Handle<FixedArray> queue(heap()->microtask_queue(), this); |
| 2276 ASSERT(num_tasks <= queue->length()); |
| 2277 set_pending_microtask_count(0); |
| 2278 heap()->set_microtask_queue(heap()->empty_fixed_array()); |
| 2279 |
| 2280 for (int i = 0; i < num_tasks; i++) { |
| 2281 HandleScope scope(this); |
| 2282 Handle<JSFunction> microtask(JSFunction::cast(queue->get(i)), this); |
| 2283 // TODO(adamk): This should ignore/clear exceptions instead of Checking. |
| 2284 Execution::Call(this, microtask, factory()->undefined_value(), |
| 2285 0, NULL).Check(); |
| 2286 } |
| 2287 } |
| 2288 |
2258 handle_scope_implementer()->DecrementCallDepth(); | 2289 handle_scope_implementer()->DecrementCallDepth(); |
2259 } | 2290 } |
2260 | 2291 |
2261 | 2292 |
2262 } } // namespace v8::internal | 2293 } } // namespace v8::internal |
OLD | NEW |