OLD | NEW |
1 // Copyright 2013 The Chromium Authors. All rights reserved. | 1 // Copyright 2013 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 "gin/modules/timer.h" | 5 #include "gin/modules/timer.h" |
6 | 6 |
7 #include "base/bind.h" | 7 #include "base/bind.h" |
8 #include "gin/object_template_builder.h" | 8 #include "gin/object_template_builder.h" |
9 #include "gin/per_context_data.h" | 9 #include "gin/per_context_data.h" |
10 | 10 |
11 namespace gin { | 11 namespace gin { |
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, "::gin::Timer"); | 16 return gin::StringToSymbol(isolate, "::gin::Timer"); |
17 } | 17 } |
18 | 18 |
19 } // namespace | 19 } // namespace |
20 | 20 |
21 // Timer | 21 // Timer ======================================================================= |
22 | 22 |
23 gin::WrapperInfo Timer::kWrapperInfo = { gin::kEmbedderNativeGin }; | 23 gin::WrapperInfo Timer::kWrapperInfo = { gin::kEmbedderNativeGin }; |
24 | 24 |
25 // static | 25 // static |
26 Handle<Timer> Timer::Create(TimerType type, v8::Isolate* isolate, int delay_ms, | 26 Handle<Timer> Timer::Create(TimerType type, v8::Isolate* isolate, int delay_ms, |
27 v8::Handle<v8::Function> function) { | 27 v8::Handle<v8::Function> function) { |
28 return CreateHandle(isolate, new Timer(isolate, type == TYPE_REPEATING, | 28 return CreateHandle(isolate, new Timer(isolate, type == TYPE_REPEATING, |
29 delay_ms, function)); | 29 delay_ms, function)); |
30 } | 30 } |
31 | 31 |
32 ObjectTemplateBuilder Timer::GetObjectTemplateBuilder(v8::Isolate* isolate) { | 32 ObjectTemplateBuilder Timer::GetObjectTemplateBuilder(v8::Isolate* isolate) { |
| 33 // We use Unretained() here because we directly own timer_, so we know it will |
| 34 // be alive when these methods are called. |
33 return Wrappable<Timer>::GetObjectTemplateBuilder(isolate) | 35 return Wrappable<Timer>::GetObjectTemplateBuilder(isolate) |
34 .SetMethod("cancel", | 36 .SetMethod("cancel", |
35 base::Bind(&base::Timer::Stop, base::Unretained(&timer_))) | 37 base::Bind(&base::Timer::Stop, base::Unretained(&timer_))) |
36 .SetMethod("reset", | 38 .SetMethod("reset", |
37 base::Bind(&base::Timer::Reset, base::Unretained(&timer_))); | 39 base::Bind(&base::Timer::Reset, base::Unretained(&timer_))); |
38 } | 40 } |
39 | 41 |
40 Timer::Timer(v8::Isolate* isolate, bool repeating, int delay_ms, | 42 Timer::Timer(v8::Isolate* isolate, bool repeating, int delay_ms, |
41 v8::Handle<v8::Function> function) | 43 v8::Handle<v8::Function> function) |
42 : weak_factory_(this), | 44 : weak_factory_(this), |
43 timer_(false, repeating), | 45 timer_(false, repeating), |
44 runner_(PerContextData::From( | 46 runner_(PerContextData::From( |
45 isolate->GetCurrentContext())->runner()->GetWeakPtr()) { | 47 isolate->GetCurrentContext())->runner()->GetWeakPtr()) { |
46 GetWrapper(runner_->isolate())->SetHiddenValue(GetHiddenPropertyName(isolate), | 48 GetWrapper(runner_->isolate())->SetHiddenValue(GetHiddenPropertyName(isolate), |
47 function); | 49 function); |
48 timer_.Start(FROM_HERE, base::TimeDelta::FromMilliseconds(delay_ms), | 50 timer_.Start(FROM_HERE, base::TimeDelta::FromMilliseconds(delay_ms), |
49 base::Bind(&Timer::OnTimerFired, weak_factory_.GetWeakPtr())); | 51 base::Bind(&Timer::OnTimerFired, weak_factory_.GetWeakPtr())); |
50 } | 52 } |
51 | 53 |
52 Timer::~Timer() { | 54 Timer::~Timer() { |
53 } | 55 } |
54 | 56 |
55 void Timer::OnTimerFired() { | 57 void Timer::OnTimerFired() { |
56 if (!runner_) | 58 // This can happen in spite of the weak callback because it is possible for |
| 59 // a gin::Handle<> to keep this object alive past when the isolate it is part |
| 60 // of is destroyed. |
| 61 if (!runner_.get()) { |
57 return; | 62 return; |
| 63 } |
| 64 |
58 Runner::Scope scope(runner_.get()); | 65 Runner::Scope scope(runner_.get()); |
59 v8::Handle<v8::Function> function = v8::Handle<v8::Function>::Cast( | 66 v8::Handle<v8::Function> function = v8::Handle<v8::Function>::Cast( |
60 GetWrapper(runner_->isolate())->GetHiddenValue( | 67 GetWrapper(runner_->isolate())->GetHiddenValue( |
61 GetHiddenPropertyName(runner_->isolate()))); | 68 GetHiddenPropertyName(runner_->isolate()))); |
62 runner_->Call(function, v8::Undefined(runner_->isolate()), 0, NULL); | 69 runner_->Call(function, v8::Undefined(runner_->isolate()), 0, NULL); |
63 } | 70 } |
64 | 71 |
65 | 72 |
66 // TimerModule | 73 // TimerModule ================================================================= |
67 | 74 |
68 WrapperInfo TimerModule::kWrapperInfo = { kEmbedderNativeGin }; | 75 WrapperInfo TimerModule::kWrapperInfo = { kEmbedderNativeGin }; |
69 | 76 |
70 // static | 77 // static |
71 Handle<TimerModule> TimerModule::Create(v8::Isolate* isolate) { | 78 Handle<TimerModule> TimerModule::Create(v8::Isolate* isolate) { |
72 return CreateHandle(isolate, new TimerModule()); | 79 return CreateHandle(isolate, new TimerModule()); |
73 } | 80 } |
74 | 81 |
75 // static | 82 // static |
76 v8::Local<v8::Value> TimerModule::GetModule(v8::Isolate* isolate) { | 83 v8::Local<v8::Value> TimerModule::GetModule(v8::Isolate* isolate) { |
77 return Create(isolate)->GetWrapper(isolate); | 84 return Create(isolate)->GetWrapper(isolate); |
78 } | 85 } |
79 | 86 |
80 TimerModule::TimerModule() { | 87 TimerModule::TimerModule() { |
81 } | 88 } |
82 | 89 |
83 TimerModule::~TimerModule() { | 90 TimerModule::~TimerModule() { |
84 } | 91 } |
85 | 92 |
86 ObjectTemplateBuilder TimerModule::GetObjectTemplateBuilder( | 93 ObjectTemplateBuilder TimerModule::GetObjectTemplateBuilder( |
87 v8::Isolate* isolate) { | 94 v8::Isolate* isolate) { |
88 return Wrappable<TimerModule>::GetObjectTemplateBuilder(isolate) | 95 return Wrappable<TimerModule>::GetObjectTemplateBuilder(isolate) |
89 .SetMethod("createOneShot", | 96 .SetMethod("createOneShot", |
90 base::Bind(&Timer::Create, Timer::TYPE_ONE_SHOT)) | 97 base::Bind(&Timer::Create, Timer::TYPE_ONE_SHOT)) |
91 .SetMethod("createRepeating", | 98 .SetMethod("createRepeating", |
92 base::Bind(&Timer::Create, Timer::TYPE_REPEATING)); | 99 base::Bind(&Timer::Create, Timer::TYPE_REPEATING)); |
93 } | 100 } |
94 | 101 |
95 } // namespace gin | 102 } // namespace gin |
OLD | NEW |