// // Copyright (C) 2020 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/aosp/binder_service_stable_android.h" #include #include #include #include #include "update_engine/aosp/binder_service_android_common.h" using android::binder::Status; using android::os::IUpdateEngineStableCallback; using android::os::ParcelFileDescriptor; using std::string; using std::vector; using update_engine::UpdateEngineStatus; namespace chromeos_update_engine { BinderUpdateEngineAndroidStableService::BinderUpdateEngineAndroidStableService( ServiceDelegateAndroidInterface* service_delegate) : service_delegate_(service_delegate) {} void BinderUpdateEngineAndroidStableService::SendStatusUpdate( const UpdateEngineStatus& update_engine_status) { last_status_ = static_cast(update_engine_status.status); last_progress_ = update_engine_status.progress; if (callback_) { callback_->onStatusUpdate(last_status_, last_progress_); } } void BinderUpdateEngineAndroidStableService::SendPayloadApplicationComplete( ErrorCode error_code) { if (callback_) { callback_->onPayloadApplicationComplete(static_cast(error_code)); } } Status BinderUpdateEngineAndroidStableService::bind( const android::sp& callback, bool* return_value) { // Reject binding if another callback is already bound. if (callback_ != nullptr) { LOG(ERROR) << "Another callback is already bound. Can't bind new callback."; *return_value = false; return Status::ok(); } // See BinderUpdateEngineAndroidService::bind. if (last_status_ != -1) { auto status = callback->onStatusUpdate(last_status_, last_progress_); if (!status.isOk()) { LOG(ERROR) << "Failed to call onStatusUpdate() from callback: " << status.toString8(); *return_value = false; return Status::ok(); } } callback_ = callback; const android::sp& callback_binder = IUpdateEngineStableCallback::asBinder(callback); auto binder_wrapper = android::BinderWrapper::Get(); binder_wrapper->RegisterForDeathNotifications( callback_binder, [this, callback = callback_binder.get()]() { UnbindCallback(callback); }); *return_value = true; return Status::ok(); } Status BinderUpdateEngineAndroidStableService::unbind( const android::sp& callback, bool* return_value) { const android::sp& callback_binder = IUpdateEngineStableCallback::asBinder(callback); auto binder_wrapper = android::BinderWrapper::Get(); binder_wrapper->UnregisterForDeathNotifications(callback_binder); *return_value = UnbindCallback(callback_binder.get()); return Status::ok(); } Status BinderUpdateEngineAndroidStableService::applyPayloadFd( const ParcelFileDescriptor& pfd, int64_t payload_offset, int64_t payload_size, const vector& header_kv_pairs) { vector str_headers = ToVecString(header_kv_pairs); Error error; if (!service_delegate_->ApplyPayload( pfd.get(), payload_offset, payload_size, str_headers, &error)) { return ErrorPtrToStatus(error); } return Status::ok(); } bool BinderUpdateEngineAndroidStableService::UnbindCallback( const IBinder* callback) { if (IUpdateEngineStableCallback::asBinder(callback_).get() != callback) { LOG(ERROR) << "Unable to unbind unknown callback."; return false; } callback_ = nullptr; return true; } android::binder::Status BinderUpdateEngineAndroidStableService::triggerPostinstall( const ::android::String16& partition) { Error error; if (!service_delegate_->TriggerPostinstall( android::String8{partition}.c_str(), &error)) { return ErrorPtrToStatus(error); } return Status::ok(); } } // namespace chromeos_update_engine