/* Copyright (c) 2014, The Linux Foundation. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following * disclaimer in the documentation and/or other materials provided * with the distribution. * * Neither the name of The Linux Foundation nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED "AS IS" AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN * IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #define LOG_TAG "WifiHAL" #include <cutils/sched_policy.h> #include <unistd.h> #include <utils/Log.h> #include <time.h> #include "common.h" #include "cpp_bindings.h" #include "rtt.h" #include "wifi_hal.h" #include "wifihal_internal.h" /* Implementation of the API functions exposed in rtt.h */ wifi_error wifi_get_rtt_capabilities(wifi_interface_handle iface, wifi_rtt_capabilities *capabilities) { int ret = WIFI_SUCCESS; lowi_cb_table_t *lowiWifiHalApi = NULL; if (iface == NULL) { ALOGE("wifi_get_rtt_capabilities: NULL iface pointer provided." " Exit."); return WIFI_ERROR_INVALID_ARGS; } interface_info *ifaceInfo = getIfaceInfo(iface); wifi_handle wifiHandle = getWifiHandle(iface); if (capabilities == NULL) { ALOGE("wifi_get_rtt_capabilities: NULL capabilities pointer provided." " Exit."); return WIFI_ERROR_INVALID_ARGS; } /* RTT commands are diverted through LOWI interface. */ /* Open LOWI dynamic library, retrieve handler to LOWI APIs and initialize * LOWI if it isn't up yet. */ lowiWifiHalApi = getLowiCallbackTable( ONE_SIDED_RANGING_SUPPORTED|DUAL_SIDED_RANGING_SUPPORED); if (lowiWifiHalApi == NULL || lowiWifiHalApi->get_rtt_capabilities == NULL) { ALOGE("wifi_get_rtt_capabilities: getLowiCallbackTable returned NULL or " "the function pointer is NULL. Exit."); ret = WIFI_ERROR_NOT_SUPPORTED; goto cleanup; } ret = lowiWifiHalApi->get_rtt_capabilities(iface, capabilities); if (ret != WIFI_SUCCESS) { ALOGE("wifi_get_rtt_capabilities: lowi_wifihal_get_rtt_capabilities " "returned error:%d. Exit.", ret); goto cleanup; } cleanup: return (wifi_error)ret; } /* API to request RTT measurement */ wifi_error wifi_rtt_range_request(wifi_request_id id, wifi_interface_handle iface, unsigned num_rtt_config, wifi_rtt_config rtt_config[], wifi_rtt_event_handler handler) { int ret = WIFI_SUCCESS; lowi_cb_table_t *lowiWifiHalApi = NULL; if (iface == NULL) { ALOGE("wifi_rtt_range_request: NULL iface pointer provided." " Exit."); return WIFI_ERROR_INVALID_ARGS; } interface_info *ifaceInfo = getIfaceInfo(iface); wifi_handle wifiHandle = getWifiHandle(iface); if (rtt_config == NULL) { ALOGE("wifi_rtt_range_request: NULL rtt_config pointer provided." " Exit."); return WIFI_ERROR_INVALID_ARGS; } if (num_rtt_config <= 0) { ALOGE("wifi_rtt_range_request: number of destination BSSIDs to " "measure RTT on = 0. Exit."); return WIFI_ERROR_INVALID_ARGS; } if (handler.on_rtt_results == NULL) { ALOGE("wifi_rtt_range_request: NULL capabilities pointer provided." " Exit."); return WIFI_ERROR_INVALID_ARGS; } /* RTT commands are diverted through LOWI interface. */ /* Open LOWI dynamic library, retrieve handler to LOWI APIs and initialize * LOWI if it isn't up yet. */ lowiWifiHalApi = getLowiCallbackTable( ONE_SIDED_RANGING_SUPPORTED|DUAL_SIDED_RANGING_SUPPORED); if (lowiWifiHalApi == NULL || lowiWifiHalApi->rtt_range_request == NULL) { ALOGE("wifi_rtt_range_request: getLowiCallbackTable returned NULL or " "the function pointer is NULL. Exit."); ret = WIFI_ERROR_NOT_SUPPORTED; goto cleanup; } ret = lowiWifiHalApi->rtt_range_request(id, iface, num_rtt_config, rtt_config, handler); if (ret != WIFI_SUCCESS) { ALOGE("wifi_rtt_range_request: lowi_wifihal_rtt_range_request " "returned error:%d. Exit.", ret); goto cleanup; } cleanup: return (wifi_error)ret; } /* API to cancel RTT measurements */ wifi_error wifi_rtt_range_cancel(wifi_request_id id, wifi_interface_handle iface, unsigned num_devices, mac_addr addr[]) { int ret = WIFI_SUCCESS; lowi_cb_table_t *lowiWifiHalApi = NULL; if (iface == NULL) { ALOGE("wifi_rtt_range_cancel: NULL iface pointer provided." " Exit."); return WIFI_ERROR_INVALID_ARGS; } interface_info *ifaceInfo = getIfaceInfo(iface); wifi_handle wifiHandle = getWifiHandle(iface); if (addr == NULL) { ALOGE("wifi_rtt_range_cancel: NULL addr pointer provided." " Exit."); return WIFI_ERROR_INVALID_ARGS; } if (num_devices <= 0) { ALOGE("wifi_rtt_range_cancel: number of destination BSSIDs to " "measure RTT on = 0. Exit."); return WIFI_ERROR_INVALID_ARGS; } /* RTT commands are diverted through LOWI interface. */ /* Open LOWI dynamic library, retrieve handler to LOWI APIs and initialize * LOWI if it isn't up yet. */ lowiWifiHalApi = getLowiCallbackTable( ONE_SIDED_RANGING_SUPPORTED|DUAL_SIDED_RANGING_SUPPORED); if (lowiWifiHalApi == NULL || lowiWifiHalApi->rtt_range_cancel == NULL) { ALOGE("wifi_rtt_range_cancel: getLowiCallbackTable returned NULL or " "the function pointer is NULL. Exit."); ret = WIFI_ERROR_NOT_SUPPORTED; goto cleanup; } ret = lowiWifiHalApi->rtt_range_cancel(id, num_devices, addr); if (ret != WIFI_SUCCESS) { ALOGE("wifi_rtt_range_cancel: lowi_wifihal_rtt_range_cancel " "returned error:%d. Exit.", ret); goto cleanup; } cleanup: return (wifi_error)ret; } // API to configure the LCI. Used in RTT Responder mode only wifi_error wifi_set_lci(wifi_request_id id, wifi_interface_handle iface, wifi_lci_information *lci) { int ret = WIFI_SUCCESS; lowi_cb_table_t *lowiWifiHalApi = NULL; if (iface == NULL) { ALOGE("%s: NULL iface pointer provided." " Exit.", __FUNCTION__); return WIFI_ERROR_INVALID_ARGS; } interface_info *ifaceInfo = getIfaceInfo(iface); wifi_handle wifiHandle = getWifiHandle(iface); if (lci == NULL) { ALOGE("%s: NULL lci pointer provided." " Exit.", __FUNCTION__); return WIFI_ERROR_INVALID_ARGS; } /* RTT commands are diverted through LOWI interface. */ /* Open LOWI dynamic library, retrieve handler to LOWI APIs and initialize * LOWI if it isn't up yet. */ lowiWifiHalApi = getLowiCallbackTable( ONE_SIDED_RANGING_SUPPORTED|DUAL_SIDED_RANGING_SUPPORED); if (lowiWifiHalApi == NULL || lowiWifiHalApi->rtt_set_lci == NULL) { ALOGE("%s: getLowiCallbackTable returned NULL or " "the function pointer is NULL. Exit.", __FUNCTION__); ret = WIFI_ERROR_NOT_SUPPORTED; goto cleanup; } ret = lowiWifiHalApi->rtt_set_lci(id, iface, lci); if (ret != WIFI_SUCCESS) { ALOGE("%s: returned error:%d. Exit.", __FUNCTION__, ret); goto cleanup; } cleanup: return (wifi_error)ret; } // API to configure the LCR. Used in RTT Responder mode only. wifi_error wifi_set_lcr(wifi_request_id id, wifi_interface_handle iface, wifi_lcr_information *lcr) { int ret = WIFI_SUCCESS; lowi_cb_table_t *lowiWifiHalApi = NULL; if (iface == NULL) { ALOGE("%s: NULL iface pointer provided." " Exit.", __FUNCTION__); return WIFI_ERROR_INVALID_ARGS; } interface_info *ifaceInfo = getIfaceInfo(iface); wifi_handle wifiHandle = getWifiHandle(iface); if (lcr == NULL) { ALOGE("%s: NULL lcr pointer provided." " Exit.", __FUNCTION__); return WIFI_ERROR_INVALID_ARGS; } /* RTT commands are diverted through LOWI interface. */ /* Open LOWI dynamic library, retrieve handler to LOWI APIs and initialize * LOWI if it isn't up yet. */ lowiWifiHalApi = getLowiCallbackTable( ONE_SIDED_RANGING_SUPPORTED|DUAL_SIDED_RANGING_SUPPORED); if (lowiWifiHalApi == NULL || lowiWifiHalApi->rtt_set_lcr == NULL) { ALOGE("%s: getLowiCallbackTable returned NULL or " "the function pointer is NULL. Exit.", __FUNCTION__); ret = WIFI_ERROR_NOT_SUPPORTED; goto cleanup; } ret = lowiWifiHalApi->rtt_set_lcr(id, iface, lcr); if (ret != WIFI_SUCCESS) { ALOGE("%s: returned error:%d. Exit.", __FUNCTION__, ret); goto cleanup; } cleanup: return (wifi_error)ret; } /* * Get RTT responder information e.g. WiFi channel to enable responder on. */ wifi_error wifi_rtt_get_responder_info(wifi_interface_handle iface, wifi_rtt_responder *responder_info) { int ret = WIFI_SUCCESS; lowi_cb_table_t *lowiWifiHalApi = NULL; if (iface == NULL || responder_info == NULL) { ALOGE("%s: iface : %p responder_info : %p", __FUNCTION__, iface, responder_info); return WIFI_ERROR_INVALID_ARGS; } /* Open LOWI dynamic library, retrieve handler to LOWI APIs */ lowiWifiHalApi = getLowiCallbackTable( ONE_SIDED_RANGING_SUPPORTED|DUAL_SIDED_RANGING_SUPPORED); if (lowiWifiHalApi == NULL || lowiWifiHalApi->rtt_get_responder_info == NULL) { ALOGE("%s: getLowiCallbackTable returned NULL or " "the function pointer is NULL. Exit.", __FUNCTION__); ret = WIFI_ERROR_NOT_SUPPORTED; goto cleanup; } ret = lowiWifiHalApi->rtt_get_responder_info(iface, responder_info); if (ret != WIFI_SUCCESS) { ALOGE("%s: returned error:%d. Exit.", __FUNCTION__, ret); goto cleanup; } cleanup: return (wifi_error)ret; } /** * Enable RTT responder mode. * channel_hint - hint of the channel information where RTT responder should * be enabled on. * max_duration_seconds - timeout of responder mode. * responder_info - responder information e.g. channel used for RTT responder, * NULL if responder is not enabled. */ wifi_error wifi_enable_responder(wifi_request_id id, wifi_interface_handle iface, wifi_channel_info channel_hint, unsigned max_duration_seconds, wifi_rtt_responder *responder_info) { int ret = WIFI_SUCCESS; lowi_cb_table_t *lowiWifiHalApi = NULL; if (iface == NULL || responder_info == NULL) { ALOGE("%s: iface : %p responder_info : %p", __FUNCTION__, iface, responder_info); return WIFI_ERROR_INVALID_ARGS; } /* Open LOWI dynamic library, retrieve handler to LOWI APIs */ lowiWifiHalApi = getLowiCallbackTable( ONE_SIDED_RANGING_SUPPORTED|DUAL_SIDED_RANGING_SUPPORED); if (lowiWifiHalApi == NULL || lowiWifiHalApi->enable_responder == NULL) { ALOGE("%s: getLowiCallbackTable returned NULL or " "the function pointer is NULL. Exit.", __FUNCTION__); ret = WIFI_ERROR_NOT_SUPPORTED; goto cleanup; } ret = lowiWifiHalApi->enable_responder(id, iface, channel_hint, max_duration_seconds, responder_info); if (ret != WIFI_SUCCESS) { ALOGE("%s: returned error:%d. Exit.", __FUNCTION__, ret); goto cleanup; } cleanup: return (wifi_error)ret; } /** * Disable RTT responder mode. */ wifi_error wifi_disable_responder(wifi_request_id id, wifi_interface_handle iface) { int ret = WIFI_SUCCESS; lowi_cb_table_t *lowiWifiHalApi = NULL; if (iface == NULL) { ALOGE("%s: iface : %p", __FUNCTION__, iface); return WIFI_ERROR_INVALID_ARGS; } /* Open LOWI dynamic library, retrieve handler to LOWI APIs */ lowiWifiHalApi = getLowiCallbackTable( ONE_SIDED_RANGING_SUPPORTED|DUAL_SIDED_RANGING_SUPPORED); if (lowiWifiHalApi == NULL || lowiWifiHalApi->disable_responder == NULL) { ALOGE("%s: getLowiCallbackTable returned NULL or " "the function pointer is NULL. Exit.", __FUNCTION__); ret = WIFI_ERROR_NOT_SUPPORTED; goto cleanup; } ret = lowiWifiHalApi->disable_responder(id, iface); if (ret != WIFI_SUCCESS) { ALOGE("%s: returned error:%d. Exit.", __FUNCTION__, ret); goto cleanup; } cleanup: return (wifi_error)ret; }