| // |
| // 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_android.h" |
| |
| #include <memory> |
| |
| #include <base/bind.h> |
| #include <base/logging.h> |
| #include <binderwrapper/binder_wrapper.h> |
| #include <brillo/errors/error.h> |
| #include <utils/String8.h> |
| |
| using android::binder::Status; |
| using android::os::IUpdateEngineCallback; |
| using android::os::ParcelFileDescriptor; |
| using std::string; |
| using std::vector; |
| using update_engine::UpdateEngineStatus; |
| |
| namespace { |
| Status ErrorPtrToStatus(const brillo::ErrorPtr& error) { |
| return Status::fromServiceSpecificError( |
| 1, android::String8{error->GetMessage().c_str()}); |
| } |
| |
| vector<string> ToVecString(const vector<android::String16>& inp) { |
| vector<string> out; |
| out.reserve(inp.size()); |
| for (const auto& e : inp) { |
| out.emplace_back(android::String8{e}.string()); |
| } |
| return out; |
| } |
| |
| } // namespace |
| |
| namespace chromeos_update_engine { |
| |
| BinderUpdateEngineAndroidService::BinderUpdateEngineAndroidService( |
| ServiceDelegateAndroidInterface* service_delegate) |
| : service_delegate_(service_delegate) {} |
| |
| void BinderUpdateEngineAndroidService::SendStatusUpdate( |
| const UpdateEngineStatus& update_engine_status) { |
| last_status_ = static_cast<int>(update_engine_status.status); |
| last_progress_ = update_engine_status.progress; |
| for (auto& callback : callbacks_) { |
| callback->onStatusUpdate(last_status_, last_progress_); |
| } |
| } |
| |
| void BinderUpdateEngineAndroidService::SendPayloadApplicationComplete( |
| ErrorCode error_code) { |
| for (auto& callback : callbacks_) { |
| callback->onPayloadApplicationComplete(static_cast<int>(error_code)); |
| } |
| } |
| |
| Status BinderUpdateEngineAndroidService::bind( |
| const android::sp<IUpdateEngineCallback>& callback, bool* return_value) { |
| // Send an status update on connection (except when no update sent so far). |
| // Even though the status update is oneway, it still returns an erroneous |
| // status in case of a selinux denial. We should at least check this status |
| // and fails the binding. |
| 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(); |
| } |
| } |
| |
| callbacks_.emplace_back(callback); |
| |
| const android::sp<IBinder>& callback_binder = |
| IUpdateEngineCallback::asBinder(callback); |
| auto binder_wrapper = android::BinderWrapper::Get(); |
| binder_wrapper->RegisterForDeathNotifications( |
| callback_binder, |
| base::Bind( |
| base::IgnoreResult(&BinderUpdateEngineAndroidService::UnbindCallback), |
| base::Unretained(this), |
| base::Unretained(callback_binder.get()))); |
| |
| *return_value = true; |
| return Status::ok(); |
| } |
| |
| Status BinderUpdateEngineAndroidService::unbind( |
| const android::sp<IUpdateEngineCallback>& callback, bool* return_value) { |
| const android::sp<IBinder>& callback_binder = |
| IUpdateEngineCallback::asBinder(callback); |
| auto binder_wrapper = android::BinderWrapper::Get(); |
| binder_wrapper->UnregisterForDeathNotifications(callback_binder); |
| |
| *return_value = UnbindCallback(callback_binder.get()); |
| return Status::ok(); |
| } |
| |
| Status BinderUpdateEngineAndroidService::applyPayload( |
| const android::String16& url, |
| int64_t payload_offset, |
| int64_t payload_size, |
| const vector<android::String16>& header_kv_pairs) { |
| const string payload_url{android::String8{url}.string()}; |
| vector<string> str_headers = ToVecString(header_kv_pairs); |
| |
| brillo::ErrorPtr error; |
| if (!service_delegate_->ApplyPayload( |
| payload_url, payload_offset, payload_size, str_headers, &error)) { |
| return ErrorPtrToStatus(error); |
| } |
| return Status::ok(); |
| } |
| |
| Status BinderUpdateEngineAndroidService::applyPayloadFd( |
| const ParcelFileDescriptor& pfd, |
| int64_t payload_offset, |
| int64_t payload_size, |
| const vector<android::String16>& header_kv_pairs) { |
| vector<string> str_headers = ToVecString(header_kv_pairs); |
| |
| brillo::ErrorPtr error; |
| if (!service_delegate_->ApplyPayload( |
| pfd.get(), payload_offset, payload_size, str_headers, &error)) { |
| return ErrorPtrToStatus(error); |
| } |
| return Status::ok(); |
| } |
| |
| Status BinderUpdateEngineAndroidService::suspend() { |
| brillo::ErrorPtr error; |
| if (!service_delegate_->SuspendUpdate(&error)) |
| return ErrorPtrToStatus(error); |
| return Status::ok(); |
| } |
| |
| Status BinderUpdateEngineAndroidService::resume() { |
| brillo::ErrorPtr error; |
| if (!service_delegate_->ResumeUpdate(&error)) |
| return ErrorPtrToStatus(error); |
| return Status::ok(); |
| } |
| |
| Status BinderUpdateEngineAndroidService::cancel() { |
| brillo::ErrorPtr error; |
| if (!service_delegate_->CancelUpdate(&error)) |
| return ErrorPtrToStatus(error); |
| return Status::ok(); |
| } |
| |
| Status BinderUpdateEngineAndroidService::resetStatus() { |
| brillo::ErrorPtr error; |
| if (!service_delegate_->ResetStatus(&error)) |
| return ErrorPtrToStatus(error); |
| return Status::ok(); |
| } |
| |
| Status BinderUpdateEngineAndroidService::verifyPayloadApplicable( |
| const android::String16& metadata_filename, bool* return_value) { |
| const std::string payload_metadata{ |
| android::String8{metadata_filename}.string()}; |
| LOG(INFO) << "Received a request of verifying payload metadata in " |
| << payload_metadata << "."; |
| brillo::ErrorPtr error; |
| *return_value = |
| service_delegate_->VerifyPayloadApplicable(payload_metadata, &error); |
| if (error != nullptr) |
| return ErrorPtrToStatus(error); |
| return Status::ok(); |
| } |
| |
| bool BinderUpdateEngineAndroidService::UnbindCallback(const IBinder* callback) { |
| auto it = std::find_if( |
| callbacks_.begin(), |
| callbacks_.end(), |
| [&callback](const android::sp<IUpdateEngineCallback>& elem) { |
| return IUpdateEngineCallback::asBinder(elem).get() == callback; |
| }); |
| if (it == callbacks_.end()) { |
| LOG(ERROR) << "Unable to unbind unknown callback."; |
| return false; |
| } |
| callbacks_.erase(it); |
| return true; |
| } |
| |
| Status BinderUpdateEngineAndroidService::allocateSpaceForPayload( |
| const android::String16& metadata_filename, |
| const vector<android::String16>& header_kv_pairs, |
| int64_t* return_value) { |
| const std::string payload_metadata{ |
| android::String8{metadata_filename}.string()}; |
| vector<string> str_headers = ToVecString(header_kv_pairs); |
| LOG(INFO) << "Received a request of allocating space for " << payload_metadata |
| << "."; |
| brillo::ErrorPtr error; |
| *return_value = |
| static_cast<int64_t>(service_delegate_->AllocateSpaceForPayload( |
| payload_metadata, str_headers, &error)); |
| if (error != nullptr) |
| return ErrorPtrToStatus(error); |
| return Status::ok(); |
| } |
| |
| class CleanupSuccessfulUpdateCallback |
| : public CleanupSuccessfulUpdateCallbackInterface { |
| public: |
| CleanupSuccessfulUpdateCallback( |
| const android::sp<IUpdateEngineCallback>& callback) |
| : callback_(callback) {} |
| void OnCleanupComplete(int32_t error_code) { |
| ignore_result(callback_->onPayloadApplicationComplete(error_code)); |
| } |
| void OnCleanupProgressUpdate(double progress) { |
| ignore_result(callback_->onStatusUpdate( |
| static_cast<int32_t>( |
| update_engine::UpdateStatus::CLEANUP_PREVIOUS_UPDATE), |
| progress)); |
| } |
| void RegisterForDeathNotifications(base::Closure unbind) { |
| const android::sp<android::IBinder>& callback_binder = |
| IUpdateEngineCallback::asBinder(callback_); |
| auto binder_wrapper = android::BinderWrapper::Get(); |
| binder_wrapper->RegisterForDeathNotifications(callback_binder, unbind); |
| } |
| |
| private: |
| android::sp<IUpdateEngineCallback> callback_; |
| }; |
| |
| Status BinderUpdateEngineAndroidService::cleanupSuccessfulUpdate( |
| const android::sp<IUpdateEngineCallback>& callback) { |
| brillo::ErrorPtr error; |
| service_delegate_->CleanupSuccessfulUpdate( |
| std::make_unique<CleanupSuccessfulUpdateCallback>(callback), &error); |
| if (error != nullptr) |
| return ErrorPtrToStatus(error); |
| return Status::ok(); |
| } |
| |
| } // namespace chromeos_update_engine |