| // |
| // Copyright (C) 2015 Google, Inc. |
| // |
| // 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. |
| // |
| |
| #define LOG_TAG "hal_util" |
| |
| #include <hardware/bluetooth.h> |
| #include <hardware/hardware.h> |
| |
| #include <dlfcn.h> |
| #include <errno.h> |
| #include <string.h> |
| |
| #include "btcore/include/hal_util.h" |
| #include "osi/include/log.h" |
| |
| #if defined(OS_GENERIC) |
| |
| // TODO(armansito): All logging macros should include __func__ by default (see |
| // Bug: 22671731) |
| #define HULOGERR(fmt, args...) \ |
| LOG_ERROR(LOG_TAG, "[%s] failed to load the Bluetooth library: " fmt, \ |
| __func__, ##args) |
| |
| // TODO(armansito): It might be better to pass the library name in a more |
| // generic manner as opposed to hard-coding it here. |
| static const char kBluetoothLibraryName[] = "libbluetooth.default.so"; |
| |
| static int load_bt_library(const struct hw_module_t** module) { |
| const char* id = BT_STACK_MODULE_ID; |
| const char* sym = HAL_MODULE_INFO_SYM_AS_STR; |
| struct hw_module_t* hmi = nullptr; |
| |
| // Always try to load the default Bluetooth stack on GN builds. |
| void* handle = dlopen(kBluetoothLibraryName, RTLD_NOW); |
| if (!handle) { |
| char const* err_str = dlerror(); |
| HULOGERR("%s", err_str ? err_str : "error unknown"); |
| goto error; |
| } |
| |
| // Get the address of the struct hal_module_info. |
| hmi = (struct hw_module_t*)dlsym(handle, sym); |
| if (!hmi) { |
| HULOGERR("%s", sym); |
| goto error; |
| } |
| |
| // Check that the id matches. |
| if (strcmp(id, hmi->id) != 0) { |
| HULOGERR("id=%s does not match HAL module ID: %s", id, hmi->id); |
| goto error; |
| } |
| |
| hmi->dso = handle; |
| |
| // Success. |
| LOG_INFO(LOG_TAG, "[%s] loaded HAL id=%s path=%s hmi=%p handle=%p", __func__, |
| id, kBluetoothLibraryName, hmi, handle); |
| |
| *module = hmi; |
| return 0; |
| |
| error: |
| *module = NULL; |
| if (handle) dlclose(handle); |
| |
| return -EINVAL; |
| } |
| |
| #endif // defined(OS_GENERIC) |
| |
| int hal_util_load_bt_library(const struct hw_module_t** module) { |
| #if defined(OS_GENERIC) |
| return load_bt_library(module); |
| #else // !defined(OS_GENERIC) |
| return hw_get_module(BT_STACK_MODULE_ID, module); |
| #endif // defined(OS_GENERIC) |
| } |