//
// Copyright (C) 2015 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#include "update_engine/binder_service_brillo.h"
#include <base/bind.h>
#include <binderwrapper/binder_wrapper.h>
#include <utils/String16.h>
#include <utils/StrongPointer.h>
#include "update_engine/update_status_utils.h"
using android::String16;
using android::String8;
using android::binder::Status;
using android::brillo::IUpdateEngineStatusCallback;
using android::brillo::ParcelableUpdateEngineStatus;
using android::sp;
using brillo::ErrorPtr;
using std::string;
namespace chromeos_update_engine {
namespace {
string NormalString(const String16& in) {
return string{String8{in}.string()};
}
Status ToStatus(ErrorPtr* error) {
return Status::fromServiceSpecificError(
1, String8{error->get()->GetMessage().c_str()});
}
} // namespace
template <typename... Parameters, typename... Arguments>
Status BinderUpdateEngineBrilloService::CallCommonHandler(
bool (UpdateEngineService::*Handler)(ErrorPtr*, Parameters...),
Arguments... arguments) {
ErrorPtr error;
if (((common_.get())->*Handler)(&error, arguments...))
return Status::ok();
return ToStatus(&error);
}
Status BinderUpdateEngineBrilloService::AttemptUpdate(
const String16& app_version, const String16& omaha_url, int flags) {
return CallCommonHandler(&UpdateEngineService::AttemptUpdate,
NormalString(app_version),
NormalString(omaha_url),
flags);
}
Status BinderUpdateEngineBrilloService::AttemptRollback(bool powerwash) {
return CallCommonHandler(&UpdateEngineService::AttemptRollback, powerwash);
}
Status BinderUpdateEngineBrilloService::CanRollback(bool* out_can_rollback) {
return CallCommonHandler(&UpdateEngineService::CanRollback, out_can_rollback);
}
Status BinderUpdateEngineBrilloService::ResetStatus() {
return CallCommonHandler(&UpdateEngineService::ResetStatus);
}
Status BinderUpdateEngineBrilloService::GetStatus(
ParcelableUpdateEngineStatus* status) {
string current_op;
string new_version;
auto ret = CallCommonHandler(&UpdateEngineService::GetStatus,
&status->last_checked_time_,
&status->progress_,
¤t_op,
&new_version,
&status->new_size_);
if (ret.isOk()) {
status->current_operation_ = String16{current_op.c_str()};
status->new_version_ = String16{new_version.c_str()};
}
return ret;
}
Status BinderUpdateEngineBrilloService::RebootIfNeeded() {
return CallCommonHandler(&UpdateEngineService::RebootIfNeeded);
}
Status BinderUpdateEngineBrilloService::SetChannel(
const String16& target_channel, bool powerwash) {
return CallCommonHandler(&UpdateEngineService::SetChannel,
NormalString(target_channel),
powerwash);
}
Status BinderUpdateEngineBrilloService::GetChannel(bool get_current_channel,
String16* out_channel) {
string channel_string;
auto ret = CallCommonHandler(
&UpdateEngineService::GetChannel, get_current_channel, &channel_string);
*out_channel = String16(channel_string.c_str());
return ret;
}
Status BinderUpdateEngineBrilloService::SetCohortHint(
const String16& in_cohort_hint) {
return CallCommonHandler(&UpdateEngineService::SetCohortHint,
NormalString(in_cohort_hint));
}
Status BinderUpdateEngineBrilloService::GetCohortHint(
String16* out_cohort_hint) {
string cohort_hint;
auto ret =
CallCommonHandler(&UpdateEngineService::GetCohortHint, &cohort_hint);
*out_cohort_hint = String16(cohort_hint.c_str());
return ret;
}
Status BinderUpdateEngineBrilloService::SetP2PUpdatePermission(bool enabled) {
return CallCommonHandler(&UpdateEngineService::SetP2PUpdatePermission,
enabled);
}
Status BinderUpdateEngineBrilloService::GetP2PUpdatePermission(
bool* out_p2p_permission) {
return CallCommonHandler(&UpdateEngineService::GetP2PUpdatePermission,
out_p2p_permission);
}
Status BinderUpdateEngineBrilloService::SetUpdateOverCellularPermission(
bool enabled) {
return CallCommonHandler(
&UpdateEngineService::SetUpdateOverCellularPermission, enabled);
}
Status BinderUpdateEngineBrilloService::GetUpdateOverCellularPermission(
bool* out_cellular_permission) {
return CallCommonHandler(
&UpdateEngineService::GetUpdateOverCellularPermission,
out_cellular_permission);
}
Status BinderUpdateEngineBrilloService::GetDurationSinceUpdate(
int64_t* out_duration) {
return CallCommonHandler(&UpdateEngineService::GetDurationSinceUpdate,
out_duration);
}
Status BinderUpdateEngineBrilloService::GetPrevVersion(
String16* out_prev_version) {
string version_string;
auto ret =
CallCommonHandler(&UpdateEngineService::GetPrevVersion, &version_string);
*out_prev_version = String16(version_string.c_str());
return ret;
}
Status BinderUpdateEngineBrilloService::GetRollbackPartition(
String16* out_rollback_partition) {
string partition_string;
auto ret = CallCommonHandler(&UpdateEngineService::GetRollbackPartition,
&partition_string);
if (ret.isOk()) {
*out_rollback_partition = String16(partition_string.c_str());
}
return ret;
}
Status BinderUpdateEngineBrilloService::RegisterStatusCallback(
const sp<IUpdateEngineStatusCallback>& callback) {
callbacks_.emplace_back(callback);
auto binder_wrapper = android::BinderWrapper::Get();
binder_wrapper->RegisterForDeathNotifications(
IUpdateEngineStatusCallback::asBinder(callback),
base::Bind(&BinderUpdateEngineBrilloService::UnregisterStatusCallback,
base::Unretained(this),
base::Unretained(callback.get())));
return Status::ok();
}
Status BinderUpdateEngineBrilloService::GetLastAttemptError(
int* out_last_attempt_error) {
return CallCommonHandler(&UpdateEngineService::GetLastAttemptError,
out_last_attempt_error);
}
Status BinderUpdateEngineBrilloService::GetEolStatus(int* out_eol_status) {
return CallCommonHandler(&UpdateEngineService::GetEolStatus, out_eol_status);
}
void BinderUpdateEngineBrilloService::UnregisterStatusCallback(
IUpdateEngineStatusCallback* callback) {
auto it = callbacks_.begin();
while (it != callbacks_.end() && it->get() != callback)
it++;
if (it == callbacks_.end()) {
LOG(ERROR) << "Got death notification for unknown callback.";
return;
}
LOG(INFO) << "Erasing orphan callback";
callbacks_.erase(it);
}
void BinderUpdateEngineBrilloService::SendStatusUpdate(
int64_t last_checked_time,
double progress,
update_engine::UpdateStatus status,
const string& new_version,
int64_t new_size) {
const string str_status = UpdateStatusToString(status);
for (auto& callback : callbacks_) {
callback->HandleStatusUpdate(last_checked_time,
progress,
String16{str_status.c_str()},
String16{new_version.c_str()},
new_size);
}
}
} // namespace chromeos_update_engine