/*
* 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 "real_binder_wrapper.h"
#include <base/logging.h>
#include <binder/Binder.h>
#include <binder/IBinder.h>
#include <binder/IPCThreadState.h>
#include <binder/IServiceManager.h>
namespace android {
// Class that handles binder death notifications. libbinder wants the recipient
// to be wrapped in sp<>, so registering RealBinderWrapper as a recipient would
// be awkward.
class RealBinderWrapper::DeathRecipient : public IBinder::DeathRecipient {
public:
explicit DeathRecipient(const base::Closure& callback)
: callback_(callback) {}
~DeathRecipient() = default;
// IBinder::DeathRecipient:
void binderDied(const wp<IBinder>& who) override {
callback_.Run();
}
private:
// Callback to run in response to binder death.
base::Closure callback_;
DISALLOW_COPY_AND_ASSIGN(DeathRecipient);
};
RealBinderWrapper::RealBinderWrapper() = default;
RealBinderWrapper::~RealBinderWrapper() = default;
sp<IBinder> RealBinderWrapper::GetService(const std::string& service_name) {
sp<IServiceManager> service_manager = defaultServiceManager();
if (!service_manager.get()) {
LOG(ERROR) << "Unable to get service manager";
return sp<IBinder>();
}
sp<IBinder> binder =
service_manager->checkService(String16(service_name.c_str()));
if (!binder.get())
LOG(ERROR) << "Unable to get \"" << service_name << "\" service";
return binder;
}
bool RealBinderWrapper::RegisterService(const std::string& service_name,
const sp<IBinder>& binder) {
sp<IServiceManager> service_manager = defaultServiceManager();
if (!service_manager.get()) {
LOG(ERROR) << "Unable to get service manager";
return false;
}
status_t status = defaultServiceManager()->addService(
String16(service_name.c_str()), binder);
if (status != OK) {
LOG(ERROR) << "Failed to register \"" << service_name << "\" with service "
<< "manager";
return false;
}
return true;
}
sp<BBinder> RealBinderWrapper::CreateLocalBinder() {
return sp<BBinder>(new BBinder());
}
bool RealBinderWrapper::RegisterForDeathNotifications(
const sp<IBinder>& binder,
const base::Closure& callback) {
sp<DeathRecipient> recipient(new DeathRecipient(callback));
if (binder->linkToDeath(recipient) != OK) {
LOG(ERROR) << "Failed to register for death notifications on "
<< binder.get();
return false;
}
death_recipients_[binder] = recipient;
return true;
}
bool RealBinderWrapper::UnregisterForDeathNotifications(
const sp<IBinder>& binder) {
auto it = death_recipients_.find(binder);
if (it == death_recipients_.end()) {
LOG(ERROR) << "Not registered for death notifications on " << binder.get();
return false;
}
if (binder->unlinkToDeath(it->second) != OK) {
LOG(ERROR) << "Failed to unregister for death notifications on "
<< binder.get();
return false;
}
death_recipients_.erase(it);
return true;
}
uid_t RealBinderWrapper::GetCallingUid() {
return IPCThreadState::self()->getCallingUid();
}
pid_t RealBinderWrapper::GetCallingPid() {
return IPCThreadState::self()->getCallingPid();
}
} // namespace android