blob: d1d6cc01d299b1f364a70927ffd62c2f0afefec9 [file] [log] [blame]
Casey Dahlina93cd532016-01-14 16:55:11 -08001//
2// Copyright (C) 2015 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
16
17#include "update_engine/client_library/client_dbus.h"
18
19#include <base/message_loop/message_loop.h>
20
Amin Hassanic80d2d82019-06-21 17:43:32 -070021#include <memory>
22
Casey Dahlina93cd532016-01-14 16:55:11 -080023#include <dbus/bus.h>
Amin Hassani9b66aa62019-03-14 14:13:30 -070024#include <dlcservice/proto_bindings/dlcservice.pb.h>
Casey Dahlina93cd532016-01-14 16:55:11 -080025#include <update_engine/dbus-constants.h>
26
27#include "update_engine/update_status_utils.h"
28
Casey Dahlina93cd532016-01-14 16:55:11 -080029using dbus::Bus;
30using org::chromium::UpdateEngineInterfaceProxy;
31using std::string;
Amin Hassanic80d2d82019-06-21 17:43:32 -070032using std::unique_ptr;
Xiaochu Liu88d90382018-08-29 16:09:11 -070033using std::vector;
Casey Dahlina93cd532016-01-14 16:55:11 -080034
35namespace update_engine {
Amin Hassanic80d2d82019-06-21 17:43:32 -070036
37unique_ptr<UpdateEngineClient> UpdateEngineClient::CreateInstance() {
38 auto ret = std::make_unique<internal::DBusUpdateEngineClient>();
39 if (!ret->Init()) {
40 ret.reset();
41 }
42 return ret;
43}
44
Casey Dahlina93cd532016-01-14 16:55:11 -080045namespace internal {
46
Amin Hassanieb463ee2019-06-20 19:23:03 -070047namespace {
48// This converts the status from Protobuf |StatusResult| to The internal
49// |UpdateEngineStatus| struct.
Amin Hassania4417432019-07-08 12:52:40 -070050void ConvertToUpdateEngineStatus(const StatusResult& status,
Amin Hassanieb463ee2019-06-20 19:23:03 -070051 UpdateEngineStatus* out_status) {
52 out_status->last_checked_time = status.last_checked_time();
53 out_status->progress = status.progress();
54 out_status->new_version = status.new_version();
55 out_status->new_size_bytes = status.new_size();
Amin Hassania4417432019-07-08 12:52:40 -070056 out_status->status = static_cast<UpdateStatus>(status.current_operation());
Amin Hassani9be122e2019-08-29 09:20:12 -070057 out_status->is_enterprise_rollback = status.is_enterprise_rollback();
Jae Hoon Kim2f78c1c2019-07-25 13:20:43 -070058 out_status->is_install = status.is_install();
Jae Hoon Kim051627a2019-09-03 12:56:32 -070059 out_status->eol_date = status.eol_date();
Amin Hassanieb463ee2019-06-20 19:23:03 -070060}
61} // namespace
62
Casey Dahlina93cd532016-01-14 16:55:11 -080063bool DBusUpdateEngineClient::Init() {
64 Bus::Options options;
65 options.bus_type = Bus::SYSTEM;
66 scoped_refptr<Bus> bus{new Bus{options}};
67
68 if (!bus->Connect())
69 return false;
70
71 proxy_.reset(new UpdateEngineInterfaceProxy{bus});
72 return true;
73}
74
75bool DBusUpdateEngineClient::AttemptUpdate(const string& in_app_version,
76 const string& in_omaha_url,
77 bool at_user_request) {
78 return proxy_->AttemptUpdateWithFlags(
79 in_app_version,
80 in_omaha_url,
Amin Hassani6bb001f2018-02-26 14:33:02 -080081 (at_user_request)
82 ? 0
83 : update_engine::UpdateAttemptFlags::kFlagNonInteractive,
Casey Dahlina93cd532016-01-14 16:55:11 -080084 nullptr);
85}
86
Amin Hassani9b66aa62019-03-14 14:13:30 -070087bool DBusUpdateEngineClient::AttemptInstall(const string& omaha_url,
88 const vector<string>& dlc_ids) {
Xiaochu Liu88d90382018-08-29 16:09:11 -070089 // Convert parameters into protobuf.
Amin Hassani9b66aa62019-03-14 14:13:30 -070090 dlcservice::DlcModuleList dlc_parameters;
Xiaochu Liu88d90382018-08-29 16:09:11 -070091 dlc_parameters.set_omaha_url(omaha_url);
Amin Hassani9b66aa62019-03-14 14:13:30 -070092 for (const auto& dlc_id : dlc_ids) {
93 dlcservice::DlcModuleInfo* dlc_module_info =
94 dlc_parameters.add_dlc_module_infos();
95 dlc_module_info->set_dlc_id(dlc_id);
Xiaochu Liu88d90382018-08-29 16:09:11 -070096 }
Amin Hassani3bab1772019-06-21 14:58:25 -070097 return proxy_->AttemptInstall(dlc_parameters,
98 nullptr /* brillo::ErrorPtr* */);
Xiaochu Liu88d90382018-08-29 16:09:11 -070099}
100
Casey Dahlina93cd532016-01-14 16:55:11 -0800101bool DBusUpdateEngineClient::GetStatus(int64_t* out_last_checked_time,
102 double* out_progress,
103 UpdateStatus* out_update_status,
104 string* out_new_version,
105 int64_t* out_new_size) const {
Amin Hassanieb463ee2019-06-20 19:23:03 -0700106 StatusResult status;
107 if (!proxy_->GetStatusAdvanced(&status, nullptr)) {
Casey Dahlina93cd532016-01-14 16:55:11 -0800108 return false;
109 }
110
Amin Hassanieb463ee2019-06-20 19:23:03 -0700111 *out_last_checked_time = status.last_checked_time();
112 *out_progress = status.progress();
113 *out_new_version = status.new_version();
114 *out_new_size = status.new_size();
Amin Hassania4417432019-07-08 12:52:40 -0700115 *out_update_status = static_cast<UpdateStatus>(status.current_operation());
116 return true;
Amin Hassanieb463ee2019-06-20 19:23:03 -0700117}
118
119bool DBusUpdateEngineClient::GetStatus(UpdateEngineStatus* out_status) const {
120 StatusResult status;
121 if (!proxy_->GetStatusAdvanced(&status, nullptr)) {
122 return false;
123 }
124
Amin Hassania4417432019-07-08 12:52:40 -0700125 ConvertToUpdateEngineStatus(status, out_status);
126 return true;
Casey Dahlina93cd532016-01-14 16:55:11 -0800127}
128
Alex Deymo5b5fa8b2016-10-06 15:40:49 -0700129bool DBusUpdateEngineClient::SetCohortHint(const string& cohort_hint) {
130 return proxy_->SetCohortHint(cohort_hint, nullptr);
131}
132
133bool DBusUpdateEngineClient::GetCohortHint(string* cohort_hint) const {
134 return proxy_->GetCohortHint(cohort_hint, nullptr);
135}
136
Casey Dahlina93cd532016-01-14 16:55:11 -0800137bool DBusUpdateEngineClient::SetUpdateOverCellularPermission(bool allowed) {
138 return proxy_->SetUpdateOverCellularPermission(allowed, nullptr);
139}
140
141bool DBusUpdateEngineClient::GetUpdateOverCellularPermission(
142 bool* allowed) const {
143 return proxy_->GetUpdateOverCellularPermission(allowed, nullptr);
144}
145
146bool DBusUpdateEngineClient::SetP2PUpdatePermission(bool enabled) {
147 return proxy_->SetP2PUpdatePermission(enabled, nullptr);
148}
149
150bool DBusUpdateEngineClient::GetP2PUpdatePermission(bool* enabled) const {
151 return proxy_->GetP2PUpdatePermission(enabled, nullptr);
152}
153
154bool DBusUpdateEngineClient::Rollback(bool powerwash) {
155 return proxy_->AttemptRollback(powerwash, nullptr);
156}
157
158bool DBusUpdateEngineClient::GetRollbackPartition(
159 string* rollback_partition) const {
160 return proxy_->GetRollbackPartition(rollback_partition, nullptr);
161}
162
163bool DBusUpdateEngineClient::GetPrevVersion(string* prev_version) const {
164 return proxy_->GetPrevVersion(prev_version, nullptr);
165}
166
167void DBusUpdateEngineClient::RebootIfNeeded() {
168 bool ret = proxy_->RebootIfNeeded(nullptr);
169 if (!ret) {
170 // Reboot error code doesn't necessarily mean that a reboot
171 // failed. For example, D-Bus may be shutdown before we receive the
172 // result.
173 LOG(INFO) << "RebootIfNeeded() failure ignored.";
174 }
175}
176
177bool DBusUpdateEngineClient::ResetStatus() {
178 return proxy_->ResetStatus(nullptr);
179}
180
Alex Deymo492eaf52016-02-02 12:05:02 -0800181void DBusUpdateEngineClient::DBusStatusHandlersRegistered(
Amin Hassaniaefaab22019-01-14 16:20:16 -0800182 const string& interface, const string& signal_name, bool success) const {
Casey Dahlina93cd532016-01-14 16:55:11 -0800183 if (!success) {
Casey Dahlina715f7b2016-01-29 16:38:51 -0800184 for (auto handler : handlers_) {
Amin Hassaniaefaab22019-01-14 16:20:16 -0800185 handler->IPCError("Could not connect to" + signal_name + " on " +
186 interface);
Casey Dahlina715f7b2016-01-29 16:38:51 -0800187 }
188 } else {
189 StatusUpdateHandlersRegistered(nullptr);
Casey Dahlina93cd532016-01-14 16:55:11 -0800190 }
Casey Dahlina715f7b2016-01-29 16:38:51 -0800191}
Casey Dahlina93cd532016-01-14 16:55:11 -0800192
Casey Dahlina715f7b2016-01-29 16:38:51 -0800193void DBusUpdateEngineClient::StatusUpdateHandlersRegistered(
194 StatusUpdateHandler* handler) const {
Amin Hassanieb463ee2019-06-20 19:23:03 -0700195 UpdateEngineStatus status;
196 if (!GetStatus(&status)) {
Casey Dahlina715f7b2016-01-29 16:38:51 -0800197 handler->IPCError("Could not query current status");
Casey Dahlina93cd532016-01-14 16:55:11 -0800198 return;
199 }
200
Alex Deymo492eaf52016-02-02 12:05:02 -0800201 std::vector<update_engine::StatusUpdateHandler*> just_handler = {handler};
202 for (auto h : handler ? just_handler : handlers_) {
Amin Hassanieb463ee2019-06-20 19:23:03 -0700203 h->HandleStatusUpdate(status);
Casey Dahlina715f7b2016-01-29 16:38:51 -0800204 }
Casey Dahlina93cd532016-01-14 16:55:11 -0800205}
206
Casey Dahlina715f7b2016-01-29 16:38:51 -0800207void DBusUpdateEngineClient::RunStatusUpdateHandlers(
Amin Hassanieb463ee2019-06-20 19:23:03 -0700208 const StatusResult& status) {
209 UpdateEngineStatus ue_status;
210 ConvertToUpdateEngineStatus(status, &ue_status);
Casey Dahlina93cd532016-01-14 16:55:11 -0800211
Casey Dahlina715f7b2016-01-29 16:38:51 -0800212 for (auto handler : handlers_) {
Amin Hassanieb463ee2019-06-20 19:23:03 -0700213 handler->HandleStatusUpdate(ue_status);
Casey Dahlina715f7b2016-01-29 16:38:51 -0800214 }
215}
216
217bool DBusUpdateEngineClient::UnregisterStatusUpdateHandler(
218 StatusUpdateHandler* handler) {
Alex Deymob3fa53b2016-04-18 19:57:58 -0700219 auto it = std::find(handlers_.begin(), handlers_.end(), handler);
Casey Dahlina715f7b2016-01-29 16:38:51 -0800220 if (it != handlers_.end()) {
221 handlers_.erase(it);
222 return true;
223 }
224
225 return false;
Casey Dahlina93cd532016-01-14 16:55:11 -0800226}
227
Casey Dahlin40892492016-01-25 16:55:28 -0800228bool DBusUpdateEngineClient::RegisterStatusUpdateHandler(
Casey Dahlina93cd532016-01-14 16:55:11 -0800229 StatusUpdateHandler* handler) {
230 if (!base::MessageLoopForIO::current()) {
231 LOG(FATAL) << "Cannot get UpdateEngineClient outside of message loop.";
Casey Dahlin40892492016-01-25 16:55:28 -0800232 return false;
Casey Dahlina93cd532016-01-14 16:55:11 -0800233 }
234
Casey Dahlina715f7b2016-01-29 16:38:51 -0800235 handlers_.push_back(handler);
236
237 if (dbus_handler_registered_) {
238 StatusUpdateHandlersRegistered(handler);
239 return true;
240 }
241
Amin Hassanieb463ee2019-06-20 19:23:03 -0700242 proxy_->RegisterStatusUpdateAdvancedSignalHandler(
Casey Dahlina715f7b2016-01-29 16:38:51 -0800243 base::Bind(&DBusUpdateEngineClient::RunStatusUpdateHandlers,
244 base::Unretained(this)),
Alex Deymo492eaf52016-02-02 12:05:02 -0800245 base::Bind(&DBusUpdateEngineClient::DBusStatusHandlersRegistered,
246 base::Unretained(this)));
Casey Dahlina715f7b2016-01-29 16:38:51 -0800247
248 dbus_handler_registered_ = true;
Casey Dahlin40892492016-01-25 16:55:28 -0800249
250 return true;
Casey Dahlina93cd532016-01-14 16:55:11 -0800251}
252
253bool DBusUpdateEngineClient::SetTargetChannel(const string& in_target_channel,
254 bool allow_powerwash) {
255 return proxy_->SetChannel(in_target_channel, allow_powerwash, nullptr);
256}
257
258bool DBusUpdateEngineClient::GetTargetChannel(string* out_channel) const {
259 return proxy_->GetChannel(false, // Get the target channel.
260 out_channel,
261 nullptr);
262}
263
264bool DBusUpdateEngineClient::GetChannel(string* out_channel) const {
265 return proxy_->GetChannel(true, // Get the current channel.
266 out_channel,
267 nullptr);
268}
269
Shuqian Zhao29971732016-02-05 11:29:32 -0800270bool DBusUpdateEngineClient::GetLastAttemptError(
271 int32_t* last_attempt_error) const {
272 return proxy_->GetLastAttemptError(last_attempt_error, nullptr);
273}
274
Jae Hoon Kima1f4a7d2019-09-03 18:12:33 +0000275bool DBusUpdateEngineClient::GetEolStatus(int32_t* eol_status) const {
276 return proxy_->GetEolStatus(eol_status, nullptr);
Alex Deymob3fa53b2016-04-18 19:57:58 -0700277}
278
Casey Dahlina93cd532016-01-14 16:55:11 -0800279} // namespace internal
280} // namespace update_engine