| /* |
| * Copyright (c) 2011-2018, 2020 The Linux Foundation. All rights reserved. |
| * |
| * Permission to use, copy, modify, and/or distribute this software for |
| * any purpose with or without fee is hereby granted, provided that the |
| * above copyright notice and this permission notice appear in all |
| * copies. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL |
| * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE |
| * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL |
| * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR |
| * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER |
| * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR |
| * PERFORMANCE OF THIS SOFTWARE. |
| */ |
| |
| /* |
| * DOC: contains scan 11d api and functionality |
| */ |
| #include <qdf_status.h> |
| #include <wlan_objmgr_psoc_obj.h> |
| #include <wlan_objmgr_pdev_obj.h> |
| #include <wlan_objmgr_vdev_obj.h> |
| #include <wlan_scan_public_structs.h> |
| #include <wlan_scan_utils_api.h> |
| #include "wlan_scan_main.h" |
| #include "wlan_scan_11d.h" |
| #include "wlan_reg_services_api.h" |
| #include "wlan_reg_ucfg_api.h" |
| |
| /** |
| * wlan_pdevid_get_cc_db() - private API to get cc db from pdev id |
| * @psoc: psoc object |
| * @pdev_id: pdev id |
| * |
| * Return: cc db for the pdev id |
| */ |
| static struct scan_country_code_db * |
| wlan_pdevid_get_cc_db(struct wlan_objmgr_psoc *psoc, uint8_t pdev_id) |
| { |
| struct wlan_scan_obj *scan_obj; |
| |
| if (pdev_id > WLAN_UMAC_MAX_PDEVS) { |
| scm_err("invalid pdev_id %d", pdev_id); |
| return NULL; |
| } |
| |
| scan_obj = wlan_psoc_get_scan_obj(psoc); |
| if (!scan_obj) |
| return NULL; |
| |
| return &scan_obj->cc_db[pdev_id]; |
| } |
| |
| /** |
| * wlan_pdev_get_cc_db() - private API to get cc db from pdev |
| * @psoc: psoc object |
| * @pdev: Pdev object |
| * |
| * Return: cc db for the pdev |
| */ |
| static struct scan_country_code_db * |
| wlan_pdev_get_cc_db(struct wlan_objmgr_psoc *psoc, |
| struct wlan_objmgr_pdev *pdev) |
| { |
| uint8_t pdev_id; |
| |
| if (!pdev) { |
| scm_err("pdev is NULL"); |
| return NULL; |
| } |
| pdev_id = wlan_objmgr_pdev_get_pdev_id(pdev); |
| |
| return wlan_pdevid_get_cc_db(psoc, pdev_id); |
| } |
| |
| /** |
| * scm_11d_elected_country_algo_fcc - private api to get cc per fcc algo |
| * @cc_db: scan country code db |
| * |
| * Return: true or false |
| */ |
| static bool |
| scm_11d_elected_country_algo_fcc(struct scan_country_code_db *cc_db) |
| { |
| uint8_t i; |
| uint8_t country_idx; |
| uint16_t max_votes; |
| bool found = false; |
| |
| if (!cc_db->num_country_codes) { |
| scm_err("No AP with 11d Country code is present in scan list"); |
| return false; |
| } |
| |
| max_votes = cc_db->votes[0].votes; |
| if (wlan_reg_is_us(cc_db->votes[0].cc)) { |
| found = true; |
| country_idx = 0; |
| goto algo_done; |
| } else if (max_votes >= MIN_11D_AP_COUNT) { |
| found = true; |
| country_idx = 0; |
| } |
| |
| for (i = 1; i < cc_db->num_country_codes; i++) { |
| if (wlan_reg_is_us(cc_db->votes[i].cc)) { |
| found = true; |
| country_idx = i; |
| goto algo_done; |
| } |
| |
| if ((max_votes < cc_db->votes[i].votes) && |
| (cc_db->votes[i].votes >= MIN_11D_AP_COUNT)) { |
| scm_debug("Votes for Country %c%c : %d", |
| cc_db->votes[i].cc[0], |
| cc_db->votes[i].cc[1], |
| cc_db->votes[i].votes); |
| max_votes = cc_db->votes[i].votes; |
| country_idx = i; |
| found = true; |
| } |
| } |
| |
| algo_done: |
| if (found) { |
| qdf_mem_copy(cc_db->elected_cc, |
| cc_db->votes[country_idx].cc, |
| REG_ALPHA2_LEN + 1); |
| |
| scm_debug("Selected Country is %c%c With count %d", |
| cc_db->votes[country_idx].cc[0], |
| cc_db->votes[country_idx].cc[1], |
| cc_db->votes[country_idx].votes); |
| } |
| |
| return found; |
| } |
| |
| /** |
| * scm_11d_elected_country_info - private api to get cc |
| * @cc_db: scan country code db |
| * |
| * Return: true or false |
| */ |
| static bool |
| scm_11d_elected_country_info(struct scan_country_code_db *cc_db) |
| { |
| uint8_t i, j = 0; |
| uint8_t max_votes; |
| |
| if (!cc_db->num_country_codes) { |
| scm_err("No AP with 11d Country code is present in scan list"); |
| return false; |
| } |
| |
| max_votes = cc_db->votes[0].votes; |
| |
| for (i = 1; i < cc_db->num_country_codes; i++) { |
| /* |
| * If we have a tie for max votes for 2 different country codes, |
| * pick random. |
| */ |
| if (max_votes < cc_db->votes[i].votes) { |
| scm_debug("Votes for Country %c%c : %d", |
| cc_db->votes[i].cc[0], |
| cc_db->votes[i].cc[1], |
| cc_db->votes[i].votes); |
| |
| max_votes = cc_db->votes[i].votes; |
| j = i; |
| } |
| } |
| |
| qdf_mem_copy(cc_db->elected_cc, cc_db->votes[j].cc, |
| REG_ALPHA2_LEN + 1); |
| |
| scm_debug("Selected Country is %c%c With count %d", |
| cc_db->votes[j].cc[0], |
| cc_db->votes[j].cc[1], |
| cc_db->votes[j].votes); |
| |
| return true; |
| } |
| |
| /** |
| * scm_11d_set_country_code - private api to set cc per 11d learning |
| * @pdev: pdev object |
| * @elected_cc: elected country code |
| * @current_cc: current country code |
| * |
| * Return: true or false |
| */ |
| static bool |
| scm_11d_set_country_code(struct wlan_objmgr_pdev *pdev, |
| uint8_t *elected_cc, uint8_t *current_cc) |
| { |
| scm_debug("elected country %c%c, current country %c%c", |
| elected_cc[0], elected_cc[1], current_cc[0], current_cc[1]); |
| |
| if (!qdf_mem_cmp(elected_cc, current_cc, REG_ALPHA2_LEN + 1)) |
| return true; |
| |
| wlan_reg_set_11d_country(pdev, elected_cc); |
| return true; |
| } |
| |
| /** |
| * scm_11d_reset_cc_db - reset the country code db |
| * @cc_db: the pointer of country code db |
| * |
| * Return: void |
| */ |
| static void scm_11d_reset_cc_db(struct scan_country_code_db *cc_db) |
| { |
| qdf_mem_zero(cc_db->votes, sizeof(cc_db->votes)); |
| qdf_mem_zero(cc_db->elected_cc, sizeof(cc_db->elected_cc)); |
| cc_db->num_country_codes = 0; |
| } |
| |
| QDF_STATUS scm_11d_cc_db_init(struct wlan_objmgr_psoc *psoc) |
| { |
| struct scan_country_code_db *cc_db; |
| struct wlan_scan_obj *scan_obj; |
| |
| if (!psoc) { |
| scm_err("psoc is NULL"); |
| return QDF_STATUS_E_INVAL; |
| } |
| |
| scan_obj = wlan_psoc_get_scan_obj(psoc); |
| if (!scan_obj) { |
| scm_err("scan_obj is NULL"); |
| return QDF_STATUS_E_INVAL; |
| } |
| |
| cc_db = (struct scan_country_code_db *)qdf_mem_malloc_atomic( |
| sizeof(struct scan_country_code_db) * WLAN_UMAC_MAX_PDEVS); |
| if (!cc_db) { |
| scm_err("alloc country code db error"); |
| return QDF_STATUS_E_INVAL; |
| } |
| |
| qdf_mem_zero(cc_db, |
| sizeof(struct scan_country_code_db) * |
| WLAN_UMAC_MAX_PDEVS); |
| |
| scan_obj->cc_db = cc_db; |
| return QDF_STATUS_SUCCESS; |
| } |
| |
| QDF_STATUS scm_11d_cc_db_deinit(struct wlan_objmgr_psoc *psoc) |
| { |
| struct wlan_scan_obj *scan_obj; |
| |
| if (!psoc) { |
| scm_err("psoc is NULL"); |
| return QDF_STATUS_E_INVAL; |
| } |
| |
| scan_obj = wlan_psoc_get_scan_obj(psoc); |
| if (!scan_obj) { |
| scm_err("scan_obj is NULL"); |
| return QDF_STATUS_E_INVAL; |
| } |
| |
| qdf_mem_free(scan_obj->cc_db); |
| return QDF_STATUS_SUCCESS; |
| } |
| |
| /** |
| * scm_11d_handle_country_info() - API to handle 11d country info |
| * @arg: pointer to country code db |
| * @scan_entry: the pointer to scan entry |
| * |
| * Update the country code database per the country code from country IE |
| * |
| * Return: QDF_STATUS |
| */ |
| static QDF_STATUS |
| scm_11d_handle_country_info(void *arg, |
| struct scan_cache_entry *scan_entry) |
| { |
| uint8_t i; |
| bool match = false; |
| uint8_t num_country_codes; |
| struct wlan_country_ie *cc_ie; |
| struct scan_country_code_db *cc_db; |
| |
| cc_ie = util_scan_entry_country(scan_entry); |
| cc_db = (struct scan_country_code_db *)arg; |
| |
| if (!cc_db) |
| return QDF_STATUS_E_INVAL; |
| |
| /* return success to continue iterate */ |
| if (!cc_ie) |
| return QDF_STATUS_SUCCESS; |
| |
| /* just to be sure, convert to UPPER case here */ |
| for (i = 0; i < 3; i++) |
| cc_ie->cc[i] = qdf_toupper(cc_ie->cc[i]); |
| |
| num_country_codes = cc_db->num_country_codes; |
| for (i = 0; i < num_country_codes; i++) { |
| match = !qdf_mem_cmp(cc_db->votes[i].cc, cc_ie->cc, |
| REG_ALPHA2_LEN); |
| if (match) |
| break; |
| } |
| |
| if (match) { |
| cc_db->votes[i].votes++; |
| return QDF_STATUS_SUCCESS; |
| } |
| |
| if (num_country_codes >= SCAN_MAX_NUM_COUNTRY_CODE) { |
| scm_debug("country code db already full: %d", |
| num_country_codes); |
| return QDF_STATUS_SUCCESS; |
| } |
| |
| /* add country code to end of the list */ |
| qdf_mem_copy(cc_db->votes[num_country_codes].cc, cc_ie->cc, |
| REG_ALPHA2_LEN + 1); |
| cc_db->votes[num_country_codes].votes = 1; |
| cc_db->num_country_codes++; |
| return QDF_STATUS_SUCCESS; |
| } |
| |
| #define CC_VOTE_CHAR_LEN 10 |
| |
| /** |
| * scm_11d_dump_cc_db() - Function to dump country db info |
| * @cc_db: pointer to country code db |
| * |
| * Return: None |
| */ |
| static void scm_11d_dump_cc_db(struct scan_country_code_db *cc_db) |
| { |
| uint32_t i, buf_len, num = 0; |
| uint8_t *cc, *cc_buf; |
| |
| if (!cc_db || !cc_db->num_country_codes) |
| return; |
| |
| buf_len = QDF_MIN(cc_db->num_country_codes, SCAN_MAX_NUM_COUNTRY_CODE); |
| buf_len = buf_len * CC_VOTE_CHAR_LEN + 1; |
| cc_buf = qdf_mem_malloc(buf_len); |
| if (!cc_buf) |
| return; |
| |
| for (i = 0; i < cc_db->num_country_codes; i++) { |
| cc = cc_db->votes[i].cc; |
| num += qdf_scnprintf(cc_buf + num, buf_len - num, " %c%c:%d", |
| cc[0], cc[1], cc_db->votes[i].votes); |
| } |
| scm_debug("%s", cc_buf); |
| qdf_mem_free(cc_buf); |
| } |
| |
| void scm_11d_decide_country_code(struct wlan_objmgr_vdev *vdev) |
| { |
| uint8_t current_cc[REG_ALPHA2_LEN + 1]; |
| bool found; |
| struct scan_country_code_db *cc_db; |
| struct wlan_objmgr_pdev *pdev = wlan_vdev_get_pdev(vdev); |
| struct wlan_objmgr_psoc *psoc = wlan_pdev_get_psoc(pdev); |
| |
| if (!wlan_reg_11d_enabled_on_host(psoc)) |
| return; |
| |
| if (SOURCE_UNKNOWN == ucfg_reg_get_cc_and_src(psoc, current_cc)) { |
| scm_err("fail to get current country code"); |
| return; |
| } |
| |
| cc_db = wlan_pdev_get_cc_db(psoc, pdev); |
| if (!cc_db) { |
| scm_err("scan_db is NULL"); |
| return; |
| } |
| |
| scm_iterate_scan_db(pdev, scm_11d_handle_country_info, cc_db); |
| |
| scm_11d_dump_cc_db(cc_db); |
| |
| if (wlan_reg_is_us(current_cc) || wlan_reg_is_world(current_cc)) |
| found = scm_11d_elected_country_algo_fcc(cc_db); |
| else |
| found = scm_11d_elected_country_info(cc_db); |
| |
| if (found) |
| scm_11d_set_country_code(pdev, cc_db->elected_cc, |
| current_cc); |
| scm_11d_reset_cc_db(cc_db); |
| } |