OLD | NEW |
1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium 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 "mojo/bindings/js/waiting_callback.h" | 5 #include "mojo/bindings/js/waiting_callback.h" |
6 | 6 |
7 #include "gin/per_context_data.h" | 7 #include "gin/per_context_data.h" |
8 #include "mojo/public/cpp/environment/environment.h" | 8 #include "mojo/public/cpp/environment/environment.h" |
9 | 9 |
10 namespace mojo { | 10 namespace mojo { |
11 namespace js { | 11 namespace js { |
12 | 12 |
13 namespace { | 13 namespace { |
14 | 14 |
15 v8::Handle<v8::String> GetHiddenPropertyName(v8::Isolate* isolate) { | 15 v8::Handle<v8::String> GetHiddenPropertyName(v8::Isolate* isolate) { |
16 return gin::StringToSymbol(isolate, "::mojo::js::WaitingCallback"); | 16 return gin::StringToSymbol(isolate, "::mojo::js::WaitingCallback"); |
17 } | 17 } |
18 | 18 |
19 } // namespace | 19 } // namespace |
20 | 20 |
21 gin::WrapperInfo WaitingCallback::kWrapperInfo = { gin::kEmbedderNativeGin }; | 21 gin::WrapperInfo WaitingCallback::kWrapperInfo = { gin::kEmbedderNativeGin }; |
22 | 22 |
23 // static | 23 // static |
24 gin::Handle<WaitingCallback> WaitingCallback::Create( | 24 gin::Handle<WaitingCallback> WaitingCallback::Create( |
25 v8::Isolate* isolate, | 25 v8::Isolate* isolate, |
26 v8::Handle<v8::Function> callback, | 26 v8::Handle<v8::Function> callback, |
27 mojo::Handle handle, | 27 gin::Handle<gin::HandleWrapper> handle_wrapper, |
28 MojoHandleSignals signals) { | 28 MojoHandleSignals signals) { |
29 gin::Handle<WaitingCallback> waiting_callback = | 29 gin::Handle<WaitingCallback> waiting_callback = gin::CreateHandle( |
30 gin::CreateHandle(isolate, new WaitingCallback(isolate, callback)); | 30 isolate, new WaitingCallback(isolate, callback, handle_wrapper)); |
31 waiting_callback->wait_id_ = Environment::GetDefaultAsyncWaiter()->AsyncWait( | 31 waiting_callback->wait_id_ = Environment::GetDefaultAsyncWaiter()->AsyncWait( |
32 handle.value(), | 32 handle_wrapper->get().value(), |
33 signals, | 33 signals, |
34 MOJO_DEADLINE_INDEFINITE, | 34 MOJO_DEADLINE_INDEFINITE, |
35 &WaitingCallback::CallOnHandleReady, | 35 &WaitingCallback::CallOnHandleReady, |
36 waiting_callback.get()); | 36 waiting_callback.get()); |
37 return waiting_callback; | 37 return waiting_callback; |
38 } | 38 } |
39 | 39 |
40 void WaitingCallback::Cancel() { | 40 void WaitingCallback::Cancel() { |
41 if (!wait_id_) | 41 if (!wait_id_) |
42 return; | 42 return; |
43 | 43 |
| 44 handle_wrapper_->RemoveCloseObserver(this); |
| 45 handle_wrapper_ = NULL; |
44 Environment::GetDefaultAsyncWaiter()->CancelWait(wait_id_); | 46 Environment::GetDefaultAsyncWaiter()->CancelWait(wait_id_); |
45 wait_id_ = 0; | 47 wait_id_ = 0; |
46 } | 48 } |
47 | 49 |
48 WaitingCallback::WaitingCallback(v8::Isolate* isolate, | 50 WaitingCallback::WaitingCallback(v8::Isolate* isolate, |
49 v8::Handle<v8::Function> callback) | 51 v8::Handle<v8::Function> callback, |
50 : wait_id_() { | 52 gin::Handle<gin::HandleWrapper> handle_wrapper) |
| 53 : wait_id_(0), handle_wrapper_(handle_wrapper.get()) { |
| 54 handle_wrapper_->AddCloseObserver(this); |
51 v8::Handle<v8::Context> context = isolate->GetCurrentContext(); | 55 v8::Handle<v8::Context> context = isolate->GetCurrentContext(); |
52 runner_ = gin::PerContextData::From(context)->runner()->GetWeakPtr(); | 56 runner_ = gin::PerContextData::From(context)->runner()->GetWeakPtr(); |
53 GetWrapper(isolate)->SetHiddenValue(GetHiddenPropertyName(isolate), callback); | 57 GetWrapper(isolate)->SetHiddenValue(GetHiddenPropertyName(isolate), callback); |
54 } | 58 } |
55 | 59 |
56 WaitingCallback::~WaitingCallback() { | 60 WaitingCallback::~WaitingCallback() { |
57 Cancel(); | 61 Cancel(); |
58 } | 62 } |
59 | 63 |
60 // static | 64 // static |
61 void WaitingCallback::CallOnHandleReady(void* closure, MojoResult result) { | 65 void WaitingCallback::CallOnHandleReady(void* closure, MojoResult result) { |
62 static_cast<WaitingCallback*>(closure)->OnHandleReady(result); | 66 static_cast<WaitingCallback*>(closure)->OnHandleReady(result); |
63 } | 67 } |
64 | 68 |
65 void WaitingCallback::OnHandleReady(MojoResult result) { | 69 void WaitingCallback::OnHandleReady(MojoResult result) { |
66 wait_id_ = 0; | 70 wait_id_ = 0; |
| 71 handle_wrapper_->RemoveCloseObserver(this); |
| 72 handle_wrapper_ = NULL; |
67 | 73 |
68 if (!runner_) | 74 if (!runner_) |
69 return; | 75 return; |
70 | 76 |
71 gin::Runner::Scope scope(runner_.get()); | 77 gin::Runner::Scope scope(runner_.get()); |
72 v8::Isolate* isolate = runner_->GetContextHolder()->isolate(); | 78 v8::Isolate* isolate = runner_->GetContextHolder()->isolate(); |
73 | 79 |
74 v8::Handle<v8::Value> hidden_value = | 80 v8::Handle<v8::Value> hidden_value = |
75 GetWrapper(isolate)->GetHiddenValue(GetHiddenPropertyName(isolate)); | 81 GetWrapper(isolate)->GetHiddenValue(GetHiddenPropertyName(isolate)); |
76 v8::Handle<v8::Function> callback; | 82 v8::Handle<v8::Function> callback; |
77 CHECK(gin::ConvertFromV8(isolate, hidden_value, &callback)); | 83 CHECK(gin::ConvertFromV8(isolate, hidden_value, &callback)); |
78 | 84 |
79 v8::Handle<v8::Value> args[] = { gin::ConvertToV8(isolate, result) }; | 85 v8::Handle<v8::Value> args[] = { gin::ConvertToV8(isolate, result) }; |
80 runner_->Call(callback, runner_->global(), 1, args); | 86 runner_->Call(callback, runner_->global(), 1, args); |
81 } | 87 } |
82 | 88 |
| 89 void WaitingCallback::OnWillCloseHandle() { |
| 90 Environment::GetDefaultAsyncWaiter()->CancelWait(wait_id_); |
| 91 OnHandleReady(MOJO_RESULT_INVALID_ARGUMENT); |
| 92 } |
| 93 |
83 } // namespace js | 94 } // namespace js |
84 } // namespace mojo | 95 } // namespace mojo |
OLD | NEW |