Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 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 "components/arc/arc_bridge_service_impl.h" | 5 #include "components/arc/arc_bridge_service_impl.h" |
| 6 | 6 |
| 7 #include <string> | 7 #include "base/logging.h" |
| 8 #include <utility> | 8 #include "base/memory/ref_counted.h" |
| 9 | 9 #include "base/task_runner.h" |
| 10 #include "base/command_line.h" | |
| 11 #include "base/json/json_writer.h" | |
| 12 #include "base/sequenced_task_runner.h" | |
| 13 #include "base/sys_info.h" | |
| 14 #include "base/task_runner_util.h" | |
| 15 #include "base/threading/thread_task_runner_handle.h" | |
| 16 #include "base/time/time.h" | |
| 17 #include "chromeos/chromeos_switches.h" | |
| 18 #include "chromeos/dbus/dbus_method_call_status.h" | |
| 19 #include "chromeos/dbus/dbus_thread_manager.h" | |
| 20 #include "components/arc/arc_session.h" | 10 #include "components/arc/arc_session.h" |
| 21 #include "components/prefs/pref_registry_simple.h" | |
| 22 #include "components/prefs/pref_service.h" | |
| 23 | 11 |
| 24 namespace arc { | 12 namespace arc { |
| 25 namespace { | 13 namespace { |
| 26 | 14 |
| 27 constexpr base::TimeDelta kDefaultRestartDelay = | 15 constexpr base::TimeDelta kDefaultRestartDelay = |
| 28 base::TimeDelta::FromSeconds(5); | 16 base::TimeDelta::FromSeconds(5); |
| 29 | 17 |
| 30 } // namespace | 18 } // namespace |
| 31 | 19 |
| 32 ArcBridgeServiceImpl::ArcBridgeServiceImpl( | 20 ArcBridgeServiceImpl::ArcBridgeServiceImpl( |
| 33 const scoped_refptr<base::TaskRunner>& blocking_task_runner) | 21 const scoped_refptr<base::TaskRunner>& blocking_task_runner) |
| 34 : session_started_(false), | 22 : restart_delay_(kDefaultRestartDelay), |
| 35 restart_delay_(kDefaultRestartDelay), | |
| 36 factory_(base::Bind(ArcSession::Create, this, blocking_task_runner)), | 23 factory_(base::Bind(ArcSession::Create, this, blocking_task_runner)), |
| 37 weak_factory_(this) {} | 24 weak_ptr_factory_(this) {} |
| 38 | 25 |
| 39 ArcBridgeServiceImpl::~ArcBridgeServiceImpl() { | 26 ArcBridgeServiceImpl::~ArcBridgeServiceImpl() { |
| 27 DCHECK(CalledOnValidThread()); | |
| 40 if (arc_session_) | 28 if (arc_session_) |
| 41 arc_session_->RemoveObserver(this); | 29 arc_session_->RemoveObserver(this); |
| 42 } | 30 } |
| 43 | 31 |
| 44 void ArcBridgeServiceImpl::RequestStart() { | 32 void ArcBridgeServiceImpl::RequestStart() { |
| 45 DCHECK(CalledOnValidThread()); | 33 DCHECK(CalledOnValidThread()); |
| 46 if (session_started_) | 34 |
| 35 // Consecutive RequestStart() call. Do nothing. | |
| 36 if (running_) | |
| 47 return; | 37 return; |
| 38 | |
| 48 VLOG(1) << "Session started"; | 39 VLOG(1) << "Session started"; |
| 49 session_started_ = true; | 40 running_ = true; |
| 50 PrerequisitesChanged(); | 41 // Here the |running_| becomes false to true. So, |restart_timer_| must be |
|
Luis Héctor Chávez
2016/12/16 23:20:12
nit: "Here |running_| transitions from false to tr
hidehiko
2016/12/19 08:05:11
Done.
| |
| 42 // stopped (either not even started, or has been cancelled in previous | |
| 43 // RequestStop() call). | |
| 44 DCHECK(!restart_timer_.IsRunning()); | |
| 45 | |
| 46 if (arc_session_) { | |
| 47 // This case, a client calls RequestStop() in advance, then soon | |
|
Luis Héctor Chávez
2016/12/16 23:20:12
nit: "In this case, RequestStop() was called, and
hidehiko
2016/12/19 08:05:11
Done.
| |
| 48 // RequestStart(). In such a case, |arc_session_| must be requested | |
| 49 // to stop already. Then, do nothing now, and later in OnSessionStopped(), | |
| 50 // restarting should be handled. | |
| 51 DCHECK_EQ(state(), State::STOPPING); | |
| 52 } else { | |
| 53 DCHECK_EQ(state(), State::STOPPED); | |
| 54 StartArcSession(); | |
| 55 } | |
| 51 } | 56 } |
| 52 | 57 |
| 53 void ArcBridgeServiceImpl::RequestStop() { | 58 void ArcBridgeServiceImpl::RequestStop() { |
| 54 DCHECK(CalledOnValidThread()); | 59 DCHECK(CalledOnValidThread()); |
| 55 if (!session_started_) | 60 |
| 61 // Consecutive RequestStop() call. Do nothing. | |
| 62 if (!running_) | |
| 56 return; | 63 return; |
| 64 | |
| 57 VLOG(1) << "Session ended"; | 65 VLOG(1) << "Session ended"; |
| 58 session_started_ = false; | 66 running_ = false; |
| 59 PrerequisitesChanged(); | 67 |
| 68 if (arc_session_) { | |
| 69 // The |state_| could be either STARTING, RUNNING or STOPPING. | |
| 70 DCHECK_NE(state(), State::STOPPED); | |
| 71 | |
| 72 // STOPPING is found in the senario of "RequestStart() -> RequestStop() | |
| 73 // -> RequestStart() -> RequestStop()" case. | |
| 74 // In the first RequestStop() call, |state_| is set to STOPPING, | |
| 75 // and in the second RequestStop() finds it (so this is the second call). | |
| 76 // In that case, ArcSession::Stop() is already called, so do nothing. | |
| 77 if (state() != State::STOPPING) { | |
|
Luis Héctor Chávez
2016/12/16 23:20:12
nit:
if (state() == State::STOPPING) {
// STOPP
hidehiko
2016/12/19 08:05:11
Done.
| |
| 78 SetState(State::STOPPING); | |
| 79 arc_session_->Stop(); | |
| 80 } | |
| 81 } else { | |
| 82 DCHECK_EQ(state(), State::STOPPED); | |
| 83 // In case restarting is in progress, cancel it. | |
| 84 restart_timer_.Stop(); | |
| 85 } | |
| 60 } | 86 } |
| 61 | 87 |
| 62 void ArcBridgeServiceImpl::OnShutdown() { | 88 void ArcBridgeServiceImpl::OnShutdown() { |
| 63 DCHECK(CalledOnValidThread()); | 89 DCHECK(CalledOnValidThread()); |
| 64 | 90 |
| 65 VLOG(1) << "OnShutdown"; | 91 VLOG(1) << "OnShutdown"; |
| 66 session_started_ = false; | 92 running_ = false; |
| 67 restart_timer_.Stop(); | 93 restart_timer_.Stop(); |
| 68 if (arc_session_) | 94 if (arc_session_) |
| 69 arc_session_->OnShutdown(); | 95 arc_session_->OnShutdown(); |
| 70 // ArcSession::OnShutdown() invokes OnSessionStopped() synchronously. | 96 // ArcSession::OnShutdown() invokes OnSessionStopped() synchronously. |
| 71 // In the observer method, |arc_session_| should be destroyed. | 97 // In the observer method, |arc_session_| should be destroyed. |
| 72 DCHECK(!arc_session_); | 98 DCHECK(!arc_session_); |
| 73 } | 99 } |
| 74 | 100 |
| 75 void ArcBridgeServiceImpl::SetArcSessionFactoryForTesting( | 101 void ArcBridgeServiceImpl::SetArcSessionFactoryForTesting( |
| 76 const ArcSessionFactory& factory) { | 102 const ArcSessionFactory& factory) { |
| 77 DCHECK(!factory.is_null()); | 103 DCHECK(!factory.is_null()); |
| 104 DCHECK_EQ(state(), State::STOPPED); | |
| 105 DCHECK(!arc_session_); | |
| 106 DCHECK(!restart_timer_.IsRunning()); | |
| 78 factory_ = factory; | 107 factory_ = factory; |
| 79 } | 108 } |
| 80 | 109 |
| 81 void ArcBridgeServiceImpl::SetRestartDelayForTesting( | 110 void ArcBridgeServiceImpl::SetRestartDelayForTesting( |
| 82 const base::TimeDelta& restart_delay) { | 111 const base::TimeDelta& restart_delay) { |
| 83 DCHECK_EQ(state(), State::STOPPED); | 112 DCHECK_EQ(state(), State::STOPPED); |
| 84 DCHECK(!arc_session_); | 113 DCHECK(!arc_session_); |
| 85 DCHECK(!restart_timer_.IsRunning()); | 114 DCHECK(!restart_timer_.IsRunning()); |
| 86 restart_delay_ = restart_delay; | 115 restart_delay_ = restart_delay; |
| 87 } | 116 } |
| 88 | 117 |
| 89 void ArcBridgeServiceImpl::PrerequisitesChanged() { | |
| 90 DCHECK(CalledOnValidThread()); | |
| 91 VLOG(1) << "Prerequisites changed. " | |
| 92 << "state=" << static_cast<uint32_t>(state()) | |
| 93 << ", session_started=" << session_started_; | |
| 94 if (state() == State::STOPPED) { | |
| 95 if (!session_started_) { | |
| 96 // We were explicitly asked to stop, so do not restart. | |
| 97 restart_timer_.Stop(); | |
| 98 return; | |
| 99 } | |
| 100 DCHECK(!restart_timer_.IsRunning()); | |
| 101 VLOG(0) << "Prerequisites met, starting ARC"; | |
| 102 StartArcSession(); | |
| 103 } else { | |
| 104 DCHECK(!restart_timer_.IsRunning()); | |
| 105 if (session_started_) | |
| 106 return; | |
| 107 VLOG(0) << "Prerequisites stopped being met, stopping ARC"; | |
| 108 StopInstance(); | |
| 109 } | |
| 110 } | |
| 111 | |
| 112 void ArcBridgeServiceImpl::StartArcSession() { | 118 void ArcBridgeServiceImpl::StartArcSession() { |
| 113 DCHECK(CalledOnValidThread()); | 119 DCHECK(CalledOnValidThread()); |
| 114 DCHECK_EQ(state(), State::STOPPED); | 120 DCHECK_EQ(state(), State::STOPPED); |
| 115 DCHECK(!arc_session_); | 121 DCHECK(!arc_session_); |
| 116 DCHECK(!restart_timer_.IsRunning()); | 122 DCHECK(!restart_timer_.IsRunning()); |
| 117 | 123 |
| 118 VLOG(1) << "Starting ARC instance"; | 124 VLOG(1) << "Starting ARC instance"; |
| 119 SetStopReason(StopReason::SHUTDOWN); | 125 SetStopReason(StopReason::SHUTDOWN); |
| 120 arc_session_ = factory_.Run(); | 126 arc_session_ = factory_.Run(); |
| 121 arc_session_->AddObserver(this); | 127 arc_session_->AddObserver(this); |
| 122 SetState(State::CONNECTING); | 128 SetState(State::STARTING); |
| 123 arc_session_->Start(); | 129 arc_session_->Start(); |
| 124 } | 130 } |
| 125 | 131 |
| 126 void ArcBridgeServiceImpl::StopInstance() { | |
| 127 DCHECK(CalledOnValidThread()); | |
| 128 DCHECK_NE(state(), State::STOPPED); | |
| 129 DCHECK(!restart_timer_.IsRunning()); | |
| 130 | |
| 131 if (state() == State::STOPPING) { | |
| 132 VLOG(1) << "StopInstance() called when ARC is not running"; | |
| 133 return; | |
| 134 } | |
| 135 | |
| 136 VLOG(1) << "Stopping ARC"; | |
| 137 DCHECK(arc_session_.get()); | |
| 138 SetState(State::STOPPING); | |
| 139 | |
| 140 // Note: this can call OnStopped() internally as a callback. | |
| 141 arc_session_->Stop(); | |
| 142 } | |
| 143 | |
| 144 void ArcBridgeServiceImpl::OnSessionReady() { | 132 void ArcBridgeServiceImpl::OnSessionReady() { |
| 145 DCHECK(CalledOnValidThread()); | 133 DCHECK(CalledOnValidThread()); |
| 146 if (state() != State::CONNECTING) { | 134 DCHECK_EQ(state(), State::STARTING); |
| 147 VLOG(1) << "StopInstance() called while connecting"; | 135 DCHECK(arc_session_); |
| 148 return; | 136 DCHECK(!restart_timer_.IsRunning()); |
| 149 } | |
| 150 | 137 |
| 151 VLOG(0) << "ARC ready"; | 138 VLOG(0) << "ARC ready"; |
| 152 SetState(State::READY); | 139 SetState(State::RUNNING); |
| 153 } | 140 } |
| 154 | 141 |
| 155 void ArcBridgeServiceImpl::OnSessionStopped(StopReason stop_reason) { | 142 void ArcBridgeServiceImpl::OnSessionStopped(StopReason stop_reason) { |
| 156 DCHECK(CalledOnValidThread()); | 143 DCHECK(CalledOnValidThread()); |
| 144 DCHECK_NE(state(), State::STOPPED); | |
| 145 DCHECK(arc_session_); | |
| 157 DCHECK(!restart_timer_.IsRunning()); | 146 DCHECK(!restart_timer_.IsRunning()); |
| 158 | 147 |
| 159 VLOG(0) << "ARC stopped: " << stop_reason; | 148 VLOG(0) << "ARC stopped: " << stop_reason; |
| 160 arc_session_->RemoveObserver(this); | 149 arc_session_->RemoveObserver(this); |
| 161 arc_session_.reset(); | 150 arc_session_.reset(); |
| 162 | 151 |
| 163 // If READY, ARC instance is unexpectedly crashed so we need to restart it | 152 // If READY, ARC instance is unexpectedly crashed so we need to restart it |
| 164 // automatically. If STOPPING, it is the result of consecutive RequestStop() | 153 // automatically. If STOPPING, it is the result of consecutive RequestStop() |
| 165 // followed by RequestStart() invocation. | 154 // followed by RequestStart() invocation. |
| 166 // If CONNECTING, ARC instance has not been booted properly, so do not | 155 // If CONNECTING, ARC instance has not been booted properly, so do not |
|
Luis Héctor Chávez
2016/12/16 23:20:12
STARTING?
hidehiko
2016/12/19 08:05:11
Done.
| |
| 167 // restart it automatically. | 156 // restart it automatically. |
| 168 if (session_started_ && | 157 if (running_ && (state() == State::RUNNING || state() == State::STOPPING)) { |
| 169 (state() == State::READY || state() == State::STOPPING)) { | |
| 170 // There was a previous invocation and it crashed for some reason. Try | 158 // There was a previous invocation and it crashed for some reason. Try |
| 171 // starting ARC instance later again. | 159 // starting ARC instance later again. |
| 172 // Note that even |restart_delay_| is 0 (for testing), it needs to | 160 // Note that even |restart_delay_| is 0 (for testing), it needs to |
| 173 // PostTask, because observer callback may call RequestStart()/Stop(), | 161 // PostTask, because observer callback may call RequestStart()/Stop(), |
| 174 // which can change restarting. | 162 // which can change restarting. |
| 175 VLOG(0) << "ARC restarting"; | 163 VLOG(0) << "ARC restarting"; |
| 176 restart_timer_.Start(FROM_HERE, restart_delay_, | 164 restart_timer_.Start(FROM_HERE, restart_delay_, |
| 177 base::Bind(&ArcBridgeServiceImpl::StartArcSession, | 165 base::Bind(&ArcBridgeServiceImpl::StartArcSession, |
| 178 weak_factory_.GetWeakPtr())); | 166 weak_ptr_factory_.GetWeakPtr())); |
| 179 } | 167 } |
| 180 | 168 |
| 181 SetStopReason(stop_reason); | 169 SetStopReason(stop_reason); |
| 182 SetState(State::STOPPED); | 170 SetState(State::STOPPED); |
| 183 } | 171 } |
| 184 | 172 |
| 185 } // namespace arc | 173 } // namespace arc |
| OLD | NEW |