/* * 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 #include #include #include #include #include #include #include "IdlCli.h" #include "utils.h" namespace android { using hardware::Return; using idlcli::IdlCli; static constexpr int NUM_TRIES = 2; // Creates a Return with STATUS::EX_NULL_POINTER. template 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 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(std::string name) { return android::hardware::vibrator::V1_0::IVibrator::getService(name); } template <> inline auto getService(std::string name) { return android::hardware::vibrator::V1_1::IVibrator::getService(name); } template <> inline auto getService(std::string name) { return android::hardware::vibrator::V1_2::IVibrator::getService(name); } template <> inline auto getService(std::string name) { return android::hardware::vibrator::V1_3::IVibrator::getService(name); } template using shared_ptr = std::invoke_result_t)&, std::string>; template class HalWrapper { public: static std::unique_ptr Create() { // Assume that if getService returns a nullptr, HAL is not available on the // device. const auto name = IdlCli::Get().getName(); auto hal = getService(name.empty() ? "default" : name); return hal ? std::unique_ptr(new HalWrapper(std::move(hal))) : nullptr; } template R call(R (I::*fn)(Args0...), Args1&&... args1) { return (*mHal.*fn)(std::forward(args1)...); } private: HalWrapper(shared_ptr&& hal) : mHal(std::move(hal)) {} private: shared_ptr mHal; }; template static auto getHal() { static auto sHalWrapper = HalWrapper::Create(); return sHalWrapper.get(); } template R halCall(R (I::*fn)(Args0...), Args1&&... args1) { auto hal = getHal(); return hal ? hal->call(fn, std::forward(args1)...) : NullptrStatus(); } 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 mPromise; }; } // namespace vibrator } // namespace idlcli } // namespace android