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