/* * Copyright (C) 2016 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 #include #define LOG_TAG "DevicesFactoryHalHidl" //#define LOG_NDEBUG 0 #include #include #include PATH(android/hardware/audio/FILE_VERSION/IDevice.h) #include #include #include "DeviceHalHidl.h" #include "DevicesFactoryHalHidl.h" using ::android::detail::AudioHalVersionInfo; using ::android::hardware::audio::CPP_VERSION::IDevice; using ::android::hardware::audio::CORE_TYPES_CPP_VERSION::Result; using ::android::hardware::Return; using ::android::hardware::Void; using ::android::hidl::manager::V1_0::IServiceManager; using ::android::hidl::manager::V1_0::IServiceNotification; namespace android { class ServiceNotificationListener : public IServiceNotification { public: explicit ServiceNotificationListener(sp factory) : mFactory(factory) {} Return onRegistration(const hidl_string& /*fully_qualified_name*/, const hidl_string& instance_name, bool /*pre_existing*/) override { if (static_cast(instance_name) == "default") return Void(); sp factory = mFactory.promote(); if (!factory) return Void(); sp halFactory = IDevicesFactory::getService(instance_name); if (halFactory) { factory->addDeviceFactory(halFactory, true /*needToNotify*/); } return Void(); } private: wp mFactory; }; DevicesFactoryHalHidl::DevicesFactoryHalHidl(sp devicesFactory) { ALOG_ASSERT(devicesFactory != nullptr, "Provided default IDevicesFactory service is NULL"); addDeviceFactory(devicesFactory, false /*needToNotify*/); } void DevicesFactoryHalHidl::onFirstRef() { sp sm = IServiceManager::getService(); ALOG_ASSERT(sm != nullptr, "Hardware service manager is not running"); sp listener = new ServiceNotificationListener(this); Return result = sm->registerForNotifications( IDevicesFactory::descriptor, "", listener); if (result.isOk()) { ALOGE_IF(!static_cast(result), "Hardware service manager refused to register listener"); } else { ALOGE("Failed to register for hardware service manager notifications: %s", result.description().c_str()); } } #if MAJOR_VERSION == 2 static IDevicesFactory::Device idFromHal(const char *name, status_t* status) { *status = OK; if (strcmp(name, AUDIO_HARDWARE_MODULE_ID_PRIMARY) == 0) { return IDevicesFactory::Device::PRIMARY; } else if(strcmp(name, AUDIO_HARDWARE_MODULE_ID_A2DP) == 0) { return IDevicesFactory::Device::A2DP; } else if(strcmp(name, AUDIO_HARDWARE_MODULE_ID_USB) == 0) { return IDevicesFactory::Device::USB; } else if(strcmp(name, AUDIO_HARDWARE_MODULE_ID_REMOTE_SUBMIX) == 0) { return IDevicesFactory::Device::R_SUBMIX; } else if(strcmp(name, AUDIO_HARDWARE_MODULE_ID_STUB) == 0) { return IDevicesFactory::Device::STUB; } ALOGE("Invalid device name %s", name); *status = BAD_VALUE; return {}; } #elif MAJOR_VERSION >= 4 static const char* idFromHal(const char *name, status_t* status) { *status = OK; return name; } #endif status_t DevicesFactoryHalHidl::getDeviceNames(std::vector *names __unused) { return INVALID_OPERATION; } status_t DevicesFactoryHalHidl::openDevice(const char *name, sp *device) { auto factories = copyDeviceFactories(); if (factories.empty()) return NO_INIT; status_t status; auto hidlId = idFromHal(name, &status); if (status != OK) return status; Result retval = Result::NOT_INITIALIZED; for (const auto& factory : factories) { Return ret; if (strcmp(name, AUDIO_HARDWARE_MODULE_ID_PRIMARY) == 0) { // In V7.1 it's not possible to cast IDevice back to IPrimaryDevice, // thus openPrimaryDevice must be used. #if MAJOR_VERSION == 7 && MINOR_VERSION == 1 ret = factory->openPrimaryDevice_7_1( #else ret = factory->openPrimaryDevice( #endif [&](Result r, const sp<::android::hardware::audio::CPP_VERSION::IPrimaryDevice>& result) { retval = r; if (retval == Result::OK) { *device = new DeviceHalHidl(result); } }); } else { #if MAJOR_VERSION == 7 && MINOR_VERSION == 1 ret = factory->openDevice_7_1( #else ret = factory->openDevice( #endif hidlId, [&](Result r, const sp<::android::hardware::audio::CPP_VERSION::IDevice>& result) { retval = r; if (retval == Result::OK) { *device = new DeviceHalHidl(result); } }); } if (!ret.isOk()) return FAILED_TRANSACTION; switch (retval) { // Device was found and was initialized successfully. case Result::OK: return OK; // Device was found but failed to initalize. case Result::NOT_INITIALIZED: return NO_INIT; // Otherwise continue iterating. default: ; } } ALOGW("The specified device name is not recognized: \"%s\"", name); return BAD_VALUE; } status_t DevicesFactoryHalHidl::setCallbackOnce(sp callback) { ALOG_ASSERT(callback != nullptr); bool needToCallCallback = false; { std::lock_guard lock(mLock); if (mCallback.unsafe_get()) return INVALID_OPERATION; mCallback = callback; if (mHaveUndeliveredNotifications) { needToCallCallback = true; mHaveUndeliveredNotifications = false; } } if (needToCallCallback) { callback->onNewDevicesAvailable(); } return NO_ERROR; } void DevicesFactoryHalHidl::addDeviceFactory( sp<::android::hardware::audio::CPP_VERSION::IDevicesFactory> factory, bool needToNotify) { // It is assumed that the DevicesFactoryHalInterface instance is owned // by AudioFlinger and thus have the same lifespan. factory->linkToDeath(HalDeathHandler::getInstance(), 0 /*cookie*/); sp callback; { std::lock_guard lock(mLock); mDeviceFactories.push_back(factory); if (needToNotify) { callback = mCallback.promote(); if (!callback) { mHaveUndeliveredNotifications = true; } } } if (callback) { callback->onNewDevicesAvailable(); } } std::vector> DevicesFactoryHalHidl::copyDeviceFactories() { std::lock_guard lock(mLock); return mDeviceFactories; } AudioHalVersionInfo DevicesFactoryHalHidl::getHalVersion() const { return AudioHalVersionInfo(AudioHalVersionInfo::Type::HIDL, MAJOR_VERSION, MINOR_VERSION); } status_t DevicesFactoryHalHidl::getSurroundSoundConfig( media::SurroundSoundConfig *config __unused) { return INVALID_OPERATION; } status_t DevicesFactoryHalHidl::getEngineConfig( media::audio::common::AudioHalEngineConfig *config __unused) { return INVALID_OPERATION; } // Main entry-point to the shared library. extern "C" __attribute__((visibility("default"))) void* createIDevicesFactoryImpl() { auto service = hardware::audio::CPP_VERSION::IDevicesFactory::getService(); return service ? new DevicesFactoryHalHidl(service) : nullptr; } } // namespace android