blob: 48a563b9c5ba0723245ab488cfc6c7d9408388ae [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
29using chromeos_update_engine::StringToUpdateStatus;
30using dbus::Bus;
31using org::chromium::UpdateEngineInterfaceProxy;
32using std::string;
Amin Hassanic80d2d82019-06-21 17:43:32 -070033using std::unique_ptr;
Xiaochu Liu88d90382018-08-29 16:09:11 -070034using std::vector;
Casey Dahlina93cd532016-01-14 16:55:11 -080035
36namespace update_engine {
Amin Hassanic80d2d82019-06-21 17:43:32 -070037
38unique_ptr<UpdateEngineClient> UpdateEngineClient::CreateInstance() {
39 auto ret = std::make_unique<internal::DBusUpdateEngineClient>();
40 if (!ret->Init()) {
41 ret.reset();
42 }
43 return ret;
44}
45
Casey Dahlina93cd532016-01-14 16:55:11 -080046namespace internal {
47
Amin Hassanieb463ee2019-06-20 19:23:03 -070048namespace {
49// This converts the status from Protobuf |StatusResult| to The internal
50// |UpdateEngineStatus| struct.
51bool ConvertToUpdateEngineStatus(const StatusResult& status,
52 UpdateEngineStatus* out_status) {
53 out_status->last_checked_time = status.last_checked_time();
54 out_status->progress = status.progress();
55 out_status->new_version = status.new_version();
56 out_status->new_size_bytes = status.new_size();
57 return StringToUpdateStatus(status.current_operation(), &out_status->status);
58}
59} // namespace
60
Casey Dahlina93cd532016-01-14 16:55:11 -080061bool DBusUpdateEngineClient::Init() {
62 Bus::Options options;
63 options.bus_type = Bus::SYSTEM;
64 scoped_refptr<Bus> bus{new Bus{options}};
65
66 if (!bus->Connect())
67 return false;
68
69 proxy_.reset(new UpdateEngineInterfaceProxy{bus});
70 return true;
71}
72
73bool DBusUpdateEngineClient::AttemptUpdate(const string& in_app_version,
74 const string& in_omaha_url,
75 bool at_user_request) {
76 return proxy_->AttemptUpdateWithFlags(
77 in_app_version,
78 in_omaha_url,
Amin Hassani6bb001f2018-02-26 14:33:02 -080079 (at_user_request)
80 ? 0
81 : update_engine::UpdateAttemptFlags::kFlagNonInteractive,
Casey Dahlina93cd532016-01-14 16:55:11 -080082 nullptr);
83}
84
Amin Hassani9b66aa62019-03-14 14:13:30 -070085bool DBusUpdateEngineClient::AttemptInstall(const string& omaha_url,
86 const vector<string>& dlc_ids) {
Xiaochu Liu88d90382018-08-29 16:09:11 -070087 // Convert parameters into protobuf.
Amin Hassani9b66aa62019-03-14 14:13:30 -070088 dlcservice::DlcModuleList dlc_parameters;
Xiaochu Liu88d90382018-08-29 16:09:11 -070089 dlc_parameters.set_omaha_url(omaha_url);
Amin Hassani9b66aa62019-03-14 14:13:30 -070090 for (const auto& dlc_id : dlc_ids) {
91 dlcservice::DlcModuleInfo* dlc_module_info =
92 dlc_parameters.add_dlc_module_infos();
93 dlc_module_info->set_dlc_id(dlc_id);
Xiaochu Liu88d90382018-08-29 16:09:11 -070094 }
95 string dlc_request;
96 if (dlc_parameters.SerializeToString(&dlc_request)) {
97 return proxy_->AttemptInstall(dlc_request, nullptr /* brillo::ErrorPtr* */);
98 } else {
99 LOG(ERROR) << "Fail to serialize a protobuf to a string.";
100 return false;
101 }
102}
103
Casey Dahlina93cd532016-01-14 16:55:11 -0800104bool DBusUpdateEngineClient::GetStatus(int64_t* out_last_checked_time,
105 double* out_progress,
106 UpdateStatus* out_update_status,
107 string* out_new_version,
108 int64_t* out_new_size) const {
Amin Hassanieb463ee2019-06-20 19:23:03 -0700109 StatusResult status;
110 if (!proxy_->GetStatusAdvanced(&status, nullptr)) {
Casey Dahlina93cd532016-01-14 16:55:11 -0800111 return false;
112 }
113
Amin Hassanieb463ee2019-06-20 19:23:03 -0700114 *out_last_checked_time = status.last_checked_time();
115 *out_progress = status.progress();
116 *out_new_version = status.new_version();
117 *out_new_size = status.new_size();
118 return StringToUpdateStatus(status.current_operation(), out_update_status);
119}
120
121bool DBusUpdateEngineClient::GetStatus(UpdateEngineStatus* out_status) const {
122 StatusResult status;
123 if (!proxy_->GetStatusAdvanced(&status, nullptr)) {
124 return false;
125 }
126
127 return ConvertToUpdateEngineStatus(status, out_status);
Casey Dahlina93cd532016-01-14 16:55:11 -0800128}
129
Alex Deymo5b5fa8b2016-10-06 15:40:49 -0700130bool DBusUpdateEngineClient::SetCohortHint(const string& cohort_hint) {
131 return proxy_->SetCohortHint(cohort_hint, nullptr);
132}
133
134bool DBusUpdateEngineClient::GetCohortHint(string* cohort_hint) const {
135 return proxy_->GetCohortHint(cohort_hint, nullptr);
136}
137
Casey Dahlina93cd532016-01-14 16:55:11 -0800138bool DBusUpdateEngineClient::SetUpdateOverCellularPermission(bool allowed) {
139 return proxy_->SetUpdateOverCellularPermission(allowed, nullptr);
140}
141
142bool DBusUpdateEngineClient::GetUpdateOverCellularPermission(
143 bool* allowed) const {
144 return proxy_->GetUpdateOverCellularPermission(allowed, nullptr);
145}
146
147bool DBusUpdateEngineClient::SetP2PUpdatePermission(bool enabled) {
148 return proxy_->SetP2PUpdatePermission(enabled, nullptr);
149}
150
151bool DBusUpdateEngineClient::GetP2PUpdatePermission(bool* enabled) const {
152 return proxy_->GetP2PUpdatePermission(enabled, nullptr);
153}
154
155bool DBusUpdateEngineClient::Rollback(bool powerwash) {
156 return proxy_->AttemptRollback(powerwash, nullptr);
157}
158
159bool DBusUpdateEngineClient::GetRollbackPartition(
160 string* rollback_partition) const {
161 return proxy_->GetRollbackPartition(rollback_partition, nullptr);
162}
163
164bool DBusUpdateEngineClient::GetPrevVersion(string* prev_version) const {
165 return proxy_->GetPrevVersion(prev_version, nullptr);
166}
167
168void DBusUpdateEngineClient::RebootIfNeeded() {
169 bool ret = proxy_->RebootIfNeeded(nullptr);
170 if (!ret) {
171 // Reboot error code doesn't necessarily mean that a reboot
172 // failed. For example, D-Bus may be shutdown before we receive the
173 // result.
174 LOG(INFO) << "RebootIfNeeded() failure ignored.";
175 }
176}
177
178bool DBusUpdateEngineClient::ResetStatus() {
179 return proxy_->ResetStatus(nullptr);
180}
181
Alex Deymo492eaf52016-02-02 12:05:02 -0800182void DBusUpdateEngineClient::DBusStatusHandlersRegistered(
Amin Hassaniaefaab22019-01-14 16:20:16 -0800183 const string& interface, const string& signal_name, bool success) const {
Casey Dahlina93cd532016-01-14 16:55:11 -0800184 if (!success) {
Casey Dahlina715f7b2016-01-29 16:38:51 -0800185 for (auto handler : handlers_) {
Amin Hassaniaefaab22019-01-14 16:20:16 -0800186 handler->IPCError("Could not connect to" + signal_name + " on " +
187 interface);
Casey Dahlina715f7b2016-01-29 16:38:51 -0800188 }
189 } else {
190 StatusUpdateHandlersRegistered(nullptr);
Casey Dahlina93cd532016-01-14 16:55:11 -0800191 }
Casey Dahlina715f7b2016-01-29 16:38:51 -0800192}
Casey Dahlina93cd532016-01-14 16:55:11 -0800193
Casey Dahlina715f7b2016-01-29 16:38:51 -0800194void DBusUpdateEngineClient::StatusUpdateHandlersRegistered(
195 StatusUpdateHandler* handler) const {
Amin Hassanieb463ee2019-06-20 19:23:03 -0700196 UpdateEngineStatus status;
197 if (!GetStatus(&status)) {
Casey Dahlina715f7b2016-01-29 16:38:51 -0800198 handler->IPCError("Could not query current status");
Casey Dahlina93cd532016-01-14 16:55:11 -0800199 return;
200 }
201
Alex Deymo492eaf52016-02-02 12:05:02 -0800202 std::vector<update_engine::StatusUpdateHandler*> just_handler = {handler};
203 for (auto h : handler ? just_handler : handlers_) {
Amin Hassanieb463ee2019-06-20 19:23:03 -0700204 h->HandleStatusUpdate(status);
Casey Dahlina715f7b2016-01-29 16:38:51 -0800205 }
Casey Dahlina93cd532016-01-14 16:55:11 -0800206}
207
Casey Dahlina715f7b2016-01-29 16:38:51 -0800208void DBusUpdateEngineClient::RunStatusUpdateHandlers(
Amin Hassanieb463ee2019-06-20 19:23:03 -0700209 const StatusResult& status) {
210 UpdateEngineStatus ue_status;
211 ConvertToUpdateEngineStatus(status, &ue_status);
Casey Dahlina93cd532016-01-14 16:55:11 -0800212
Casey Dahlina715f7b2016-01-29 16:38:51 -0800213 for (auto handler : handlers_) {
Amin Hassanieb463ee2019-06-20 19:23:03 -0700214 handler->HandleStatusUpdate(ue_status);
Casey Dahlina715f7b2016-01-29 16:38:51 -0800215 }
216}
217
218bool DBusUpdateEngineClient::UnregisterStatusUpdateHandler(
219 StatusUpdateHandler* handler) {
Alex Deymob3fa53b2016-04-18 19:57:58 -0700220 auto it = std::find(handlers_.begin(), handlers_.end(), handler);
Casey Dahlina715f7b2016-01-29 16:38:51 -0800221 if (it != handlers_.end()) {
222 handlers_.erase(it);
223 return true;
224 }
225
226 return false;
Casey Dahlina93cd532016-01-14 16:55:11 -0800227}
228
Casey Dahlin40892492016-01-25 16:55:28 -0800229bool DBusUpdateEngineClient::RegisterStatusUpdateHandler(
Casey Dahlina93cd532016-01-14 16:55:11 -0800230 StatusUpdateHandler* handler) {
231 if (!base::MessageLoopForIO::current()) {
232 LOG(FATAL) << "Cannot get UpdateEngineClient outside of message loop.";
Casey Dahlin40892492016-01-25 16:55:28 -0800233 return false;
Casey Dahlina93cd532016-01-14 16:55:11 -0800234 }
235
Casey Dahlina715f7b2016-01-29 16:38:51 -0800236 handlers_.push_back(handler);
237
238 if (dbus_handler_registered_) {
239 StatusUpdateHandlersRegistered(handler);
240 return true;
241 }
242
Amin Hassanieb463ee2019-06-20 19:23:03 -0700243 proxy_->RegisterStatusUpdateAdvancedSignalHandler(
Casey Dahlina715f7b2016-01-29 16:38:51 -0800244 base::Bind(&DBusUpdateEngineClient::RunStatusUpdateHandlers,
245 base::Unretained(this)),
Alex Deymo492eaf52016-02-02 12:05:02 -0800246 base::Bind(&DBusUpdateEngineClient::DBusStatusHandlersRegistered,
247 base::Unretained(this)));
Casey Dahlina715f7b2016-01-29 16:38:51 -0800248
249 dbus_handler_registered_ = true;
Casey Dahlin40892492016-01-25 16:55:28 -0800250
251 return true;
Casey Dahlina93cd532016-01-14 16:55:11 -0800252}
253
254bool DBusUpdateEngineClient::SetTargetChannel(const string& in_target_channel,
255 bool allow_powerwash) {
256 return proxy_->SetChannel(in_target_channel, allow_powerwash, nullptr);
257}
258
259bool DBusUpdateEngineClient::GetTargetChannel(string* out_channel) const {
260 return proxy_->GetChannel(false, // Get the target channel.
261 out_channel,
262 nullptr);
263}
264
265bool DBusUpdateEngineClient::GetChannel(string* out_channel) const {
266 return proxy_->GetChannel(true, // Get the current channel.
267 out_channel,
268 nullptr);
269}
270
Shuqian Zhao29971732016-02-05 11:29:32 -0800271bool DBusUpdateEngineClient::GetLastAttemptError(
272 int32_t* last_attempt_error) const {
273 return proxy_->GetLastAttemptError(last_attempt_error, nullptr);
274}
275
Alex Deymob3fa53b2016-04-18 19:57:58 -0700276bool DBusUpdateEngineClient::GetEolStatus(int32_t* eol_status) const {
277 return proxy_->GetEolStatus(eol_status, nullptr);
278}
279
Casey Dahlina93cd532016-01-14 16:55:11 -0800280} // namespace internal
281} // namespace update_engine