// // 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