/****************************************************************************** * * Copyright 2016 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 "bt_btif_ble_advertiser" #include <hardware/bluetooth.h> #include <hardware/bt_gatt.h> #include <base/bind.h> #include <vector> #include "ble_advertiser.h" #include "btif_common.h" #include "stack/include/btu.h" using base::Bind; using base::Owned; using std::vector; namespace { template <typename T> class OwnedArrayWrapper { public: explicit OwnedArrayWrapper(T* o) : ptr_(o) {} ~OwnedArrayWrapper() { delete[] ptr_; } T* get() const { return ptr_; } OwnedArrayWrapper(OwnedArrayWrapper&& other) noexcept { ptr_ = other.ptr_; other.ptr_ = NULL; } private: mutable T* ptr_; }; template <typename T> T* Unwrap(const OwnedArrayWrapper<T>& o) { return o.get(); } template <typename T> static inline OwnedArrayWrapper<T> OwnedArray(T* o) { return OwnedArrayWrapper<T>(o); } void parseParams(tBTM_BLE_ADV_PARAMS* p_params, const AdvertiseParameters& params) { p_params->advertising_event_properties = params.advertising_event_properties; p_params->adv_int_min = params.min_interval; p_params->adv_int_max = params.max_interval; p_params->channel_map = params.channel_map; p_params->adv_filter_policy = 0; p_params->tx_power = params.tx_power; p_params->primary_advertising_phy = params.primary_advertising_phy; p_params->secondary_advertising_phy = params.secondary_advertising_phy; p_params->scan_request_notification_enable = params.scan_request_notification_enable; } void parsePeriodicParams(tBLE_PERIODIC_ADV_PARAMS* p_periodic_params, PeriodicAdvertisingParameters periodic_params) { p_periodic_params->enable = periodic_params.enable; p_periodic_params->min_interval = periodic_params.min_interval; p_periodic_params->max_interval = periodic_params.max_interval; p_periodic_params->periodic_advertising_properties = periodic_params.periodic_advertising_properties; } class BleAdvertiserInterfaceImpl : public BleAdvertiserInterface { ~BleAdvertiserInterfaceImpl(){}; void RegisterAdvertiserCb(IdStatusCallback cb, uint8_t advertiser_id, uint8_t status) { LOG(INFO) << __func__ << " status: " << +status << " , adveriser_id: " << +advertiser_id; do_in_jni_thread(Bind(cb, advertiser_id, status)); } void RegisterAdvertiser(IdStatusCallback cb) override { do_in_main_thread( FROM_HERE, Bind(&BleAdvertisingManager::RegisterAdvertiser, BleAdvertisingManager::Get(), Bind(&BleAdvertiserInterfaceImpl::RegisterAdvertiserCb, base::Unretained(this), cb))); } void Unregister(uint8_t advertiser_id) override { do_in_main_thread( FROM_HERE, Bind( [](uint8_t advertiser_id) { if (!BleAdvertisingManager::IsInitialized()) { LOG(WARNING) << "Stack already shutdown"; return; } BleAdvertisingManager::Get()->Unregister(advertiser_id); }, advertiser_id)); } void GetOwnAddress(uint8_t advertiser_id, GetAddressCallback cb) override { if (!BleAdvertisingManager::IsInitialized()) return; do_in_main_thread(FROM_HERE, Bind(&BleAdvertisingManager::GetOwnAddress, BleAdvertisingManager::Get(), advertiser_id, jni_thread_wrapper(FROM_HERE, cb))); } void SetParameters(uint8_t advertiser_id, AdvertiseParameters params, ParametersCallback cb) override { VLOG(1) << __func__; if (!BleAdvertisingManager::IsInitialized()) return; tBTM_BLE_ADV_PARAMS* p_params = new tBTM_BLE_ADV_PARAMS; parseParams(p_params, params); do_in_main_thread(FROM_HERE, Bind(&BleAdvertisingManager::SetParameters, BleAdvertisingManager::Get(), advertiser_id, base::Owned(p_params), jni_thread_wrapper(FROM_HERE, cb))); } void SetData(int advertiser_id, bool set_scan_rsp, vector<uint8_t> data, StatusCallback cb) override { if (!BleAdvertisingManager::IsInitialized()) return; do_in_main_thread( FROM_HERE, Bind(&BleAdvertisingManager::SetData, BleAdvertisingManager::Get(), advertiser_id, set_scan_rsp, std::move(data), jni_thread_wrapper(FROM_HERE, cb))); } void Enable(uint8_t advertiser_id, bool enable, StatusCallback cb, uint16_t duration, uint8_t maxExtAdvEvents, StatusCallback timeout_cb) override { VLOG(1) << __func__ << " advertiser_id: " << +advertiser_id << " ,enable: " << enable; if (!BleAdvertisingManager::IsInitialized()) return; do_in_main_thread( FROM_HERE, Bind(&BleAdvertisingManager::Enable, BleAdvertisingManager::Get(), advertiser_id, enable, jni_thread_wrapper(FROM_HERE, cb), duration, maxExtAdvEvents, jni_thread_wrapper(FROM_HERE, timeout_cb))); } void StartAdvertising(uint8_t advertiser_id, StatusCallback cb, AdvertiseParameters params, std::vector<uint8_t> advertise_data, std::vector<uint8_t> scan_response_data, int timeout_s, MultiAdvCb timeout_cb) override { VLOG(1) << __func__; if (!BleAdvertisingManager::IsInitialized()) return; tBTM_BLE_ADV_PARAMS* p_params = new tBTM_BLE_ADV_PARAMS; parseParams(p_params, params); do_in_main_thread( FROM_HERE, Bind(&BleAdvertisingManager::StartAdvertising, BleAdvertisingManager::Get(), advertiser_id, jni_thread_wrapper(FROM_HERE, cb), base::Owned(p_params), std::move(advertise_data), std::move(scan_response_data), timeout_s * 100, jni_thread_wrapper(FROM_HERE, timeout_cb))); } void StartAdvertisingSet(IdTxPowerStatusCallback cb, AdvertiseParameters params, std::vector<uint8_t> advertise_data, std::vector<uint8_t> scan_response_data, PeriodicAdvertisingParameters periodic_params, std::vector<uint8_t> periodic_data, uint16_t duration, uint8_t maxExtAdvEvents, IdStatusCallback timeout_cb) override { VLOG(1) << __func__; if (!BleAdvertisingManager::IsInitialized()) return; tBTM_BLE_ADV_PARAMS* p_params = new tBTM_BLE_ADV_PARAMS; parseParams(p_params, params); tBLE_PERIODIC_ADV_PARAMS* p_periodic_params = new tBLE_PERIODIC_ADV_PARAMS; parsePeriodicParams(p_periodic_params, periodic_params); do_in_main_thread( FROM_HERE, Bind(&BleAdvertisingManager::StartAdvertisingSet, BleAdvertisingManager::Get(), jni_thread_wrapper(FROM_HERE, cb), base::Owned(p_params), std::move(advertise_data), std::move(scan_response_data), base::Owned(p_periodic_params), std::move(periodic_data), duration, maxExtAdvEvents, jni_thread_wrapper(FROM_HERE, timeout_cb))); } void SetPeriodicAdvertisingParameters( int advertiser_id, PeriodicAdvertisingParameters periodic_params, StatusCallback cb) override { VLOG(1) << __func__ << " advertiser_id: " << +advertiser_id; if (!BleAdvertisingManager::IsInitialized()) return; tBLE_PERIODIC_ADV_PARAMS* p_periodic_params = new tBLE_PERIODIC_ADV_PARAMS; parsePeriodicParams(p_periodic_params, periodic_params); do_in_main_thread( FROM_HERE, Bind(&BleAdvertisingManager::SetPeriodicAdvertisingParameters, BleAdvertisingManager::Get(), advertiser_id, base::Owned(p_periodic_params), jni_thread_wrapper(FROM_HERE, cb))); } void SetPeriodicAdvertisingData(int advertiser_id, std::vector<uint8_t> data, StatusCallback cb) override { VLOG(1) << __func__ << " advertiser_id: " << +advertiser_id; if (!BleAdvertisingManager::IsInitialized()) return; do_in_main_thread(FROM_HERE, Bind(&BleAdvertisingManager::SetPeriodicAdvertisingData, BleAdvertisingManager::Get(), advertiser_id, std::move(data), jni_thread_wrapper(FROM_HERE, cb))); } void SetPeriodicAdvertisingEnable(int advertiser_id, bool enable, StatusCallback cb) override { VLOG(1) << __func__ << " advertiser_id: " << +advertiser_id << " ,enable: " << enable; if (!BleAdvertisingManager::IsInitialized()) return; do_in_main_thread(FROM_HERE, Bind(&BleAdvertisingManager::SetPeriodicAdvertisingEnable, BleAdvertisingManager::Get(), advertiser_id, enable, jni_thread_wrapper(FROM_HERE, cb))); } }; BleAdvertiserInterface* btLeAdvertiserInstance = nullptr; } // namespace BleAdvertiserInterface* get_ble_advertiser_instance() { if (btLeAdvertiserInstance == nullptr) btLeAdvertiserInstance = new BleAdvertiserInterfaceImpl(); return btLeAdvertiserInstance; }