blob: e100eacdd5904bba81b795421b2561a6d8b8342d [file] [log] [blame]
/*
* Copyright (C) 2019 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.
*/
#pragma once
#include <future>
#include <aidl/android/hardware/vibrator/BnVibratorCallback.h>
#include <aidl/android/hardware/vibrator/IVibrator.h>
#include <aidl/android/hardware/vibrator/IVibratorManager.h>
#include <android/binder_manager.h>
#include <android/binder_process.h>
#include <android/hardware/vibrator/1.3/IVibrator.h>
#include "IdlCli.h"
#include "utils.h"
namespace android {
using hardware::Return;
using idlcli::IdlCli;
static constexpr int NUM_TRIES = 2;
// Creates a Return<R> with STATUS::EX_NULL_POINTER.
template <class R>
inline R NullptrStatus() {
using ::android::hardware::Status;
return Status::fromExceptionCode(Status::EX_NULL_POINTER);
}
template <>
inline ndk::ScopedAStatus NullptrStatus() {
return ndk::ScopedAStatus(AStatus_fromExceptionCode(EX_NULL_POINTER));
}
template <typename I>
inline auto getService(std::string name) {
const auto instance = std::string() + I::descriptor + "/" + name;
auto vibBinder = ndk::SpAIBinder(AServiceManager_getService(instance.c_str()));
return I::fromBinder(vibBinder);
}
template <>
inline auto getService<android::hardware::vibrator::V1_0::IVibrator>(std::string name) {
return android::hardware::vibrator::V1_0::IVibrator::getService(name);
}
template <>
inline auto getService<android::hardware::vibrator::V1_1::IVibrator>(std::string name) {
return android::hardware::vibrator::V1_1::IVibrator::getService(name);
}
template <>
inline auto getService<android::hardware::vibrator::V1_2::IVibrator>(std::string name) {
return android::hardware::vibrator::V1_2::IVibrator::getService(name);
}
template <>
inline auto getService<android::hardware::vibrator::V1_3::IVibrator>(std::string name) {
return android::hardware::vibrator::V1_3::IVibrator::getService(name);
}
template <typename I>
using shared_ptr = std::invoke_result_t<decltype(getService<I>)&, std::string>;
template <typename I>
class HalWrapper {
public:
static std::unique_ptr<HalWrapper> Create() {
// Assume that if getService returns a nullptr, HAL is not available on the
// device.
const auto name = IdlCli::Get().getName();
auto hal = getService<I>(name.empty() ? "default" : name);
return hal ? std::unique_ptr<HalWrapper>(new HalWrapper(std::move(hal))) : nullptr;
}
template <class R, class... Args0, class... Args1>
R call(R (I::*fn)(Args0...), Args1&&... args1) {
return (*mHal.*fn)(std::forward<Args1>(args1)...);
}
private:
HalWrapper(shared_ptr<I>&& hal) : mHal(std::move(hal)) {}
private:
shared_ptr<I> mHal;
};
template <typename I>
static auto getHal() {
static auto sHalWrapper = HalWrapper<I>::Create();
return sHalWrapper.get();
}
template <class R, class I, class... Args0, class... Args1>
R halCall(R (I::*fn)(Args0...), Args1&&... args1) {
auto hal = getHal<I>();
return hal ? hal->call(fn, std::forward<Args1>(args1)...) : NullptrStatus<R>();
}
namespace idlcli {
namespace vibrator {
namespace V1_0 = ::android::hardware::vibrator::V1_0;
namespace V1_1 = ::android::hardware::vibrator::V1_1;
namespace V1_2 = ::android::hardware::vibrator::V1_2;
namespace V1_3 = ::android::hardware::vibrator::V1_3;
namespace aidl = ::aidl::android::hardware::vibrator;
class VibratorCallback : public aidl::BnVibratorCallback {
public:
ndk::ScopedAStatus onComplete() override {
mPromise.set_value();
return ndk::ScopedAStatus::ok();
}
void waitForComplete() { mPromise.get_future().wait(); }
private:
std::promise<void> mPromise;
};
} // namespace vibrator
} // namespace idlcli
} // namespace android