wlan_cfg80211_tdls.c 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998
  1. /*
  2. * Copyright (c) 2017-2019 The Linux Foundation. All rights reserved.
  3. *
  4. * Permission to use, copy, modify, and/or distribute this software for
  5. * any purpose with or without fee is hereby granted, provided that the
  6. * above copyright notice and this permission notice appear in all
  7. * copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  10. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  11. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  12. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  13. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  14. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  15. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  16. * PERFORMANCE OF THIS SOFTWARE.
  17. */
  18. /**
  19. * DOC: defines driver functions interfacing with linux kernel
  20. */
  21. #include <qdf_list.h>
  22. #include <qdf_status.h>
  23. #include <linux/wireless.h>
  24. #include <linux/netdevice.h>
  25. #include <net/cfg80211.h>
  26. #include <wlan_cfg80211.h>
  27. #include <wlan_cfg80211_tdls.h>
  28. #include <wlan_osif_priv.h>
  29. #include <wlan_tdls_public_structs.h>
  30. #include <wlan_tdls_ucfg_api.h>
  31. #include <qdf_mem.h>
  32. #include <wlan_utility.h>
  33. #include <wlan_reg_services_api.h>
  34. #include "wlan_cfg80211_mc_cp_stats.h"
  35. #define TDLS_MAX_NO_OF_2_4_CHANNELS 14
  36. static int wlan_cfg80211_tdls_validate_mac_addr(const uint8_t *mac)
  37. {
  38. static const uint8_t temp_mac[QDF_MAC_ADDR_SIZE] = {0};
  39. if (!qdf_mem_cmp(mac, temp_mac, QDF_MAC_ADDR_SIZE)) {
  40. osif_debug("Invalid Mac address " QDF_MAC_ADDR_STR
  41. " cmd declined.",
  42. QDF_MAC_ADDR_ARRAY(mac));
  43. return -EINVAL;
  44. }
  45. return 0;
  46. }
  47. QDF_STATUS wlan_cfg80211_tdls_osif_priv_init(struct wlan_objmgr_vdev *vdev)
  48. {
  49. struct osif_tdls_vdev *tdls_priv;
  50. struct vdev_osif_priv *osif_priv;
  51. osif_priv = wlan_vdev_get_ospriv(vdev);
  52. if (!osif_priv) {
  53. osif_err("osif_priv is NULL!");
  54. return QDF_STATUS_E_FAULT;
  55. }
  56. osif_debug("initialize tdls os if layer private structure");
  57. tdls_priv = qdf_mem_malloc(sizeof(*tdls_priv));
  58. if (!tdls_priv) {
  59. osif_err("failed to allocate memory for tdls_priv");
  60. return QDF_STATUS_E_NOMEM;
  61. }
  62. init_completion(&tdls_priv->tdls_add_peer_comp);
  63. init_completion(&tdls_priv->tdls_del_peer_comp);
  64. init_completion(&tdls_priv->tdls_mgmt_comp);
  65. init_completion(&tdls_priv->tdls_link_establish_req_comp);
  66. init_completion(&tdls_priv->tdls_teardown_comp);
  67. init_completion(&tdls_priv->tdls_user_cmd_comp);
  68. init_completion(&tdls_priv->tdls_antenna_switch_comp);
  69. osif_priv->osif_tdls = tdls_priv;
  70. return QDF_STATUS_SUCCESS;
  71. }
  72. void wlan_cfg80211_tdls_osif_priv_deinit(struct wlan_objmgr_vdev *vdev)
  73. {
  74. struct vdev_osif_priv *osif_priv;
  75. osif_priv = wlan_vdev_get_ospriv(vdev);
  76. if (!osif_priv) {
  77. osif_err("osif_priv is NULL!");
  78. return;
  79. }
  80. osif_debug("deinitialize tdls os if layer private structure");
  81. if (osif_priv->osif_tdls)
  82. qdf_mem_free(osif_priv->osif_tdls);
  83. osif_priv->osif_tdls = NULL;
  84. }
  85. void hdd_notify_teardown_tdls_links(struct wlan_objmgr_psoc *psoc)
  86. {
  87. struct vdev_osif_priv *osif_priv;
  88. struct osif_tdls_vdev *tdls_priv;
  89. QDF_STATUS status;
  90. unsigned long rc;
  91. struct wlan_objmgr_vdev *vdev;
  92. vdev = ucfg_get_tdls_vdev(psoc, WLAN_OSIF_ID);
  93. if (!vdev) {
  94. osif_err("Unable to get the vdev");
  95. return;
  96. }
  97. osif_priv = wlan_vdev_get_ospriv(vdev);
  98. if (!osif_priv || !osif_priv->osif_tdls) {
  99. osif_err("osif priv or tdls priv is NULL");
  100. goto release_ref;
  101. }
  102. tdls_priv = osif_priv->osif_tdls;
  103. reinit_completion(&tdls_priv->tdls_teardown_comp);
  104. status = ucfg_tdls_teardown_links(psoc);
  105. if (QDF_IS_STATUS_ERROR(status)) {
  106. osif_err("ucfg_tdls_teardown_links failed err %d", status);
  107. goto release_ref;
  108. }
  109. osif_debug("Wait for tdls teardown completion. Timeout %u ms",
  110. WAIT_TIME_FOR_TDLS_TEARDOWN_LINKS);
  111. rc = wait_for_completion_timeout(
  112. &tdls_priv->tdls_teardown_comp,
  113. msecs_to_jiffies(WAIT_TIME_FOR_TDLS_TEARDOWN_LINKS));
  114. if (0 == rc) {
  115. osif_err(" Teardown Completion timed out rc: %ld", rc);
  116. goto release_ref;
  117. }
  118. osif_debug("TDLS teardown completion status %ld ", rc);
  119. release_ref:
  120. wlan_objmgr_vdev_release_ref(vdev,
  121. WLAN_OSIF_ID);
  122. }
  123. void hdd_notify_tdls_reset_adapter(struct wlan_objmgr_vdev *vdev)
  124. {
  125. ucfg_tdls_notify_reset_adapter(vdev);
  126. }
  127. void
  128. hdd_notify_sta_connect(uint8_t session_id,
  129. bool tdls_chan_swit_prohibited,
  130. bool tdls_prohibited,
  131. struct wlan_objmgr_vdev *vdev)
  132. {
  133. struct tdls_sta_notify_params notify_info = {0};
  134. QDF_STATUS status;
  135. if (!vdev) {
  136. osif_err("vdev is NULL");
  137. return;
  138. }
  139. status = wlan_objmgr_vdev_try_get_ref(vdev, WLAN_TDLS_NB_ID);
  140. if (QDF_IS_STATUS_ERROR(status)) {
  141. osif_err("can't get vdev");
  142. return;
  143. }
  144. notify_info.session_id = session_id;
  145. notify_info.vdev = vdev;
  146. notify_info.tdls_chan_swit_prohibited = tdls_chan_swit_prohibited;
  147. notify_info.tdls_prohibited = tdls_prohibited;
  148. ucfg_tdls_notify_sta_connect(&notify_info);
  149. }
  150. void hdd_notify_sta_disconnect(uint8_t session_id,
  151. bool lfr_roam,
  152. bool user_disconnect,
  153. struct wlan_objmgr_vdev *vdev)
  154. {
  155. struct tdls_sta_notify_params notify_info = {0};
  156. QDF_STATUS status;
  157. if (!vdev) {
  158. osif_err("vdev is NULL");
  159. return;
  160. }
  161. status = wlan_objmgr_vdev_try_get_ref(vdev, WLAN_TDLS_NB_ID);
  162. if (QDF_IS_STATUS_ERROR(status)) {
  163. osif_err("can't get vdev");
  164. return;
  165. }
  166. notify_info.session_id = session_id;
  167. notify_info.lfr_roam = lfr_roam;
  168. notify_info.tdls_chan_swit_prohibited = false;
  169. notify_info.tdls_prohibited = false;
  170. notify_info.vdev = vdev;
  171. notify_info.user_disconnect = user_disconnect;
  172. ucfg_tdls_notify_sta_disconnect(&notify_info);
  173. }
  174. int wlan_cfg80211_tdls_add_peer(struct wlan_objmgr_vdev *vdev,
  175. const uint8_t *mac)
  176. {
  177. struct tdls_add_peer_params *add_peer_req;
  178. int status;
  179. struct vdev_osif_priv *osif_priv;
  180. struct osif_tdls_vdev *tdls_priv;
  181. unsigned long rc;
  182. status = wlan_cfg80211_tdls_validate_mac_addr(mac);
  183. if (status)
  184. return status;
  185. osif_debug("Add TDLS peer " QDF_MAC_ADDR_STR,
  186. QDF_MAC_ADDR_ARRAY(mac));
  187. add_peer_req = qdf_mem_malloc(sizeof(*add_peer_req));
  188. if (!add_peer_req) {
  189. osif_err("Failed to allocate tdls add peer request mem");
  190. return -EINVAL;
  191. }
  192. osif_priv = wlan_vdev_get_ospriv(vdev);
  193. if (!osif_priv || !osif_priv->osif_tdls) {
  194. osif_err("osif_tdls_vdev or osif_priv is NULL for the current vdev");
  195. status = -EINVAL;
  196. goto error;
  197. }
  198. tdls_priv = osif_priv->osif_tdls;
  199. add_peer_req->vdev_id = wlan_vdev_get_id(vdev);
  200. qdf_mem_copy(add_peer_req->peer_addr, mac, QDF_MAC_ADDR_SIZE);
  201. reinit_completion(&tdls_priv->tdls_add_peer_comp);
  202. status = ucfg_tdls_add_peer(vdev, add_peer_req);
  203. if (QDF_IS_STATUS_ERROR(status)) {
  204. osif_err("ucfg_tdls_add_peer returned err %d", status);
  205. status = -EIO;
  206. goto error;
  207. }
  208. rc = wait_for_completion_timeout(
  209. &tdls_priv->tdls_add_peer_comp,
  210. msecs_to_jiffies(WAIT_TIME_TDLS_ADD_STA));
  211. if (!rc) {
  212. osif_err("timeout for tdls add peer indication %ld", rc);
  213. status = -EPERM;
  214. goto error;
  215. }
  216. if (QDF_IS_STATUS_ERROR(tdls_priv->tdls_add_peer_status)) {
  217. osif_err("tdls add peer failed, status:%d",
  218. tdls_priv->tdls_add_peer_status);
  219. status = -EPERM;
  220. }
  221. error:
  222. qdf_mem_free(add_peer_req);
  223. return status;
  224. }
  225. static bool
  226. is_duplicate_channel(uint8_t *arr, int index, uint8_t match)
  227. {
  228. int i;
  229. for (i = 0; i < index; i++) {
  230. if (arr[i] == match)
  231. return true;
  232. }
  233. return false;
  234. }
  235. static void
  236. tdls_calc_channels_from_staparams(struct tdls_update_peer_params *req_info,
  237. struct station_parameters *params)
  238. {
  239. int i = 0, j = 0, k = 0, no_of_channels = 0;
  240. int num_unique_channels;
  241. int next;
  242. uint8_t *dest_chans;
  243. const uint8_t *src_chans;
  244. dest_chans = req_info->supported_channels;
  245. src_chans = params->supported_channels;
  246. /* Convert (first channel , number of channels) tuple to
  247. * the total list of channels. This goes with the assumption
  248. * that if the first channel is < 14, then the next channels
  249. * are an incremental of 1 else an incremental of 4 till the number
  250. * of channels.
  251. */
  252. for (i = 0; i < params->supported_channels_len &&
  253. j < WLAN_MAC_MAX_SUPP_CHANNELS; i += 2) {
  254. int wifi_chan_index;
  255. if (!is_duplicate_channel(dest_chans, j, src_chans[i]))
  256. dest_chans[j] = src_chans[i];
  257. else
  258. continue;
  259. wifi_chan_index = ((dest_chans[j] <= WLAN_CHANNEL_14) ? 1 : 4);
  260. no_of_channels = src_chans[i + 1];
  261. osif_debug("i:%d,j:%d,k:%d,[%d]:%d,index:%d,chans_num: %d",
  262. i, j, k, j,
  263. dest_chans[j],
  264. wifi_chan_index,
  265. no_of_channels);
  266. for (k = 1; k <= no_of_channels &&
  267. j < WLAN_MAC_MAX_SUPP_CHANNELS - 1; k++) {
  268. next = dest_chans[j] + wifi_chan_index;
  269. if (!is_duplicate_channel(dest_chans, j + 1, next))
  270. dest_chans[j + 1] = next;
  271. else
  272. continue;
  273. osif_debug("i: %d, j: %d, k: %d, [%d]: %d",
  274. i, j, k, j + 1, dest_chans[j + 1]);
  275. j += 1;
  276. }
  277. }
  278. num_unique_channels = j + 1;
  279. osif_debug("Unique Channel List: supported_channels ");
  280. for (i = 0; i < num_unique_channels; i++)
  281. osif_debug("[%d]: %d,", i, dest_chans[i]);
  282. if (num_unique_channels > NUM_CHANNELS)
  283. num_unique_channels = NUM_CHANNELS;
  284. req_info->supported_channels_len = num_unique_channels;
  285. osif_debug("After removing duplcates supported_channels_len: %d",
  286. req_info->supported_channels_len);
  287. }
  288. static void
  289. wlan_cfg80211_tdls_extract_params(struct tdls_update_peer_params *req_info,
  290. struct station_parameters *params)
  291. {
  292. int i;
  293. osif_debug("sta cap %d, uapsd_queue %d, max_sp %d",
  294. params->capability,
  295. params->uapsd_queues, params->max_sp);
  296. if (!req_info) {
  297. osif_err("reg_info is NULL");
  298. return;
  299. }
  300. req_info->capability = params->capability;
  301. req_info->uapsd_queues = params->uapsd_queues;
  302. req_info->max_sp = params->max_sp;
  303. if (params->supported_channels_len)
  304. tdls_calc_channels_from_staparams(req_info, params);
  305. if (params->supported_oper_classes_len > WLAN_MAX_SUPP_OPER_CLASSES) {
  306. osif_debug("received oper classes:%d, resetting it to max supported: %d",
  307. params->supported_oper_classes_len,
  308. WLAN_MAX_SUPP_OPER_CLASSES);
  309. params->supported_oper_classes_len = WLAN_MAX_SUPP_OPER_CLASSES;
  310. }
  311. qdf_mem_copy(req_info->supported_oper_classes,
  312. params->supported_oper_classes,
  313. params->supported_oper_classes_len);
  314. req_info->supported_oper_classes_len =
  315. params->supported_oper_classes_len;
  316. if (params->ext_capab_len)
  317. qdf_mem_copy(req_info->extn_capability, params->ext_capab,
  318. sizeof(req_info->extn_capability));
  319. if (params->ht_capa) {
  320. req_info->htcap_present = 1;
  321. qdf_mem_copy(&req_info->ht_cap, params->ht_capa,
  322. sizeof(struct htcap_cmn_ie));
  323. }
  324. req_info->supported_rates_len = params->supported_rates_len;
  325. /* Note : The Maximum sizeof supported_rates sent by the Supplicant is
  326. * 32. The supported_rates array , for all the structures propogating
  327. * till Add Sta to the firmware has to be modified , if the supplicant
  328. * (ieee80211) is modified to send more rates.
  329. */
  330. /* To avoid Data Currption , set to max length to SIR_MAC_MAX_SUPP_RATES
  331. */
  332. if (req_info->supported_rates_len > WLAN_MAC_MAX_SUPP_RATES)
  333. req_info->supported_rates_len = WLAN_MAC_MAX_SUPP_RATES;
  334. if (req_info->supported_rates_len) {
  335. qdf_mem_copy(req_info->supported_rates,
  336. params->supported_rates,
  337. req_info->supported_rates_len);
  338. osif_debug("Supported Rates with Length %d",
  339. req_info->supported_rates_len);
  340. for (i = 0; i < req_info->supported_rates_len; i++)
  341. osif_debug("[%d]: %0x", i,
  342. req_info->supported_rates[i]);
  343. }
  344. if (params->vht_capa) {
  345. req_info->vhtcap_present = 1;
  346. qdf_mem_copy(&req_info->vht_cap, params->vht_capa,
  347. sizeof(struct vhtcap));
  348. }
  349. if (params->ht_capa || params->vht_capa ||
  350. (params->sta_flags_set & BIT(NL80211_STA_FLAG_WME)))
  351. req_info->is_qos_wmm_sta = true;
  352. if (params->sta_flags_set & BIT(NL80211_STA_FLAG_MFP)) {
  353. osif_debug("TDLS peer pmf capable");
  354. req_info->is_pmf = 1;
  355. }
  356. }
  357. int wlan_cfg80211_tdls_update_peer(struct wlan_objmgr_vdev *vdev,
  358. const uint8_t *mac,
  359. struct station_parameters *params)
  360. {
  361. struct tdls_update_peer_params *req_info;
  362. int status;
  363. struct vdev_osif_priv *osif_priv;
  364. struct osif_tdls_vdev *tdls_priv;
  365. unsigned long rc;
  366. status = wlan_cfg80211_tdls_validate_mac_addr(mac);
  367. if (status)
  368. return status;
  369. osif_debug("Update TDLS peer " QDF_MAC_ADDR_STR,
  370. QDF_MAC_ADDR_ARRAY(mac));
  371. req_info = qdf_mem_malloc(sizeof(*req_info));
  372. if (!req_info) {
  373. osif_err("Failed to allocate tdls add peer request mem");
  374. return -EINVAL;
  375. }
  376. wlan_cfg80211_tdls_extract_params(req_info, params);
  377. osif_priv = wlan_vdev_get_ospriv(vdev);
  378. if (!osif_priv || !osif_priv->osif_tdls) {
  379. osif_err("osif priv or tdls priv is NULL");
  380. status = -EINVAL;
  381. goto error;
  382. }
  383. tdls_priv = osif_priv->osif_tdls;
  384. req_info->vdev_id = wlan_vdev_get_id(vdev);
  385. qdf_mem_copy(req_info->peer_addr, mac, QDF_MAC_ADDR_SIZE);
  386. reinit_completion(&tdls_priv->tdls_add_peer_comp);
  387. status = ucfg_tdls_update_peer(vdev, req_info);
  388. if (QDF_IS_STATUS_ERROR(status)) {
  389. osif_err("ucfg_tdls_update_peer returned err %d", status);
  390. status = -EIO;
  391. goto error;
  392. }
  393. rc = wait_for_completion_timeout(
  394. &tdls_priv->tdls_add_peer_comp,
  395. msecs_to_jiffies(WAIT_TIME_TDLS_ADD_STA));
  396. if (!rc) {
  397. osif_err("timeout for tdls update peer indication %ld", rc);
  398. status = -EPERM;
  399. goto error;
  400. }
  401. if (QDF_IS_STATUS_ERROR(tdls_priv->tdls_add_peer_status)) {
  402. osif_err("tdls update peer failed, status:%d",
  403. tdls_priv->tdls_add_peer_status);
  404. status = -EPERM;
  405. }
  406. error:
  407. qdf_mem_free(req_info);
  408. return status;
  409. }
  410. static char *tdls_oper_to_str(enum nl80211_tdls_operation oper)
  411. {
  412. switch (oper) {
  413. case NL80211_TDLS_ENABLE_LINK:
  414. return "TDLS_ENABLE_LINK";
  415. case NL80211_TDLS_DISABLE_LINK:
  416. return "TDLS_DISABLE_LINK";
  417. case NL80211_TDLS_TEARDOWN:
  418. return "TDLS_TEARDOWN";
  419. case NL80211_TDLS_SETUP:
  420. return "TDLS_SETUP";
  421. default:
  422. return "UNKNOWN:ERR";
  423. }
  424. }
  425. static enum tdls_command_type tdls_oper_to_cmd(enum nl80211_tdls_operation oper)
  426. {
  427. if (oper == NL80211_TDLS_ENABLE_LINK)
  428. return TDLS_CMD_ENABLE_LINK;
  429. else if (oper == NL80211_TDLS_DISABLE_LINK)
  430. return TDLS_CMD_DISABLE_LINK;
  431. else if (oper == NL80211_TDLS_TEARDOWN)
  432. return TDLS_CMD_REMOVE_FORCE_PEER;
  433. else if (oper == NL80211_TDLS_SETUP)
  434. return TDLS_CMD_CONFIG_FORCE_PEER;
  435. else
  436. return 0;
  437. }
  438. int wlan_cfg80211_tdls_configure_mode(struct wlan_objmgr_vdev *vdev,
  439. uint32_t trigger_mode)
  440. {
  441. enum tdls_feature_mode tdls_mode;
  442. struct tdls_set_mode_params set_mode_params;
  443. int status;
  444. if (!vdev)
  445. return -EINVAL;
  446. switch (trigger_mode) {
  447. case WLAN_VENDOR_TDLS_TRIGGER_MODE_EXPLICIT:
  448. tdls_mode = TDLS_SUPPORT_EXP_TRIG_ONLY;
  449. return 0;
  450. case WLAN_VENDOR_TDLS_TRIGGER_MODE_EXTERNAL:
  451. tdls_mode = TDLS_SUPPORT_EXT_CONTROL;
  452. break;
  453. case WLAN_VENDOR_TDLS_TRIGGER_MODE_IMPLICIT:
  454. tdls_mode = TDLS_SUPPORT_IMP_MODE;
  455. return 0;
  456. default:
  457. osif_err("Invalid TDLS trigger mode");
  458. return -EINVAL;
  459. }
  460. osif_notice("cfg80211 tdls trigger mode %d", trigger_mode);
  461. set_mode_params.source = TDLS_SET_MODE_SOURCE_USER;
  462. set_mode_params.tdls_mode = tdls_mode;
  463. set_mode_params.update_last = false;
  464. set_mode_params.vdev = vdev;
  465. status = ucfg_tdls_set_operating_mode(&set_mode_params);
  466. return status;
  467. }
  468. int wlan_cfg80211_tdls_oper(struct wlan_objmgr_vdev *vdev,
  469. const uint8_t *peer,
  470. enum nl80211_tdls_operation oper)
  471. {
  472. struct vdev_osif_priv *osif_priv;
  473. struct osif_tdls_vdev *tdls_priv;
  474. int status;
  475. unsigned long rc;
  476. enum tdls_command_type cmd;
  477. status = wlan_cfg80211_tdls_validate_mac_addr(peer);
  478. if (status)
  479. return status;
  480. if (NL80211_TDLS_DISCOVERY_REQ == oper) {
  481. osif_warn(
  482. "We don't support in-driver setup/teardown/discovery");
  483. return -ENOTSUPP;
  484. }
  485. osif_debug("%s start", tdls_oper_to_str(oper));
  486. cmd = tdls_oper_to_cmd(oper);
  487. switch (oper) {
  488. case NL80211_TDLS_ENABLE_LINK:
  489. case NL80211_TDLS_TEARDOWN:
  490. case NL80211_TDLS_SETUP:
  491. status = ucfg_tdls_oper(vdev, peer, cmd);
  492. if (QDF_IS_STATUS_ERROR(status)) {
  493. osif_err("%s fail %d",
  494. tdls_oper_to_str(oper), status);
  495. status = -EIO;
  496. goto error;
  497. }
  498. break;
  499. case NL80211_TDLS_DISABLE_LINK:
  500. osif_priv = wlan_vdev_get_ospriv(vdev);
  501. if (!osif_priv || !osif_priv->osif_tdls) {
  502. osif_err("osif priv or tdls priv is NULL");
  503. status = -EINVAL;
  504. goto error;
  505. }
  506. tdls_priv = osif_priv->osif_tdls;
  507. reinit_completion(&tdls_priv->tdls_del_peer_comp);
  508. status = ucfg_tdls_oper(vdev, peer, cmd);
  509. if (QDF_IS_STATUS_ERROR(status)) {
  510. osif_err("ucfg_tdls_disable_link fail %d", status);
  511. status = -EIO;
  512. goto error;
  513. }
  514. rc = wait_for_completion_timeout(
  515. &tdls_priv->tdls_del_peer_comp,
  516. msecs_to_jiffies(WAIT_TIME_TDLS_DEL_STA));
  517. if (!rc) {
  518. osif_err("timeout for tdls disable link %ld", rc);
  519. status = -EPERM;
  520. }
  521. break;
  522. default:
  523. osif_err("unsupported event %d", oper);
  524. status = -ENOTSUPP;
  525. }
  526. error:
  527. return status;
  528. }
  529. void wlan_cfg80211_tdls_rx_callback(void *user_data,
  530. struct tdls_rx_mgmt_frame *rx_frame)
  531. {
  532. struct wlan_objmgr_psoc *psoc;
  533. struct wlan_objmgr_vdev *vdev;
  534. struct vdev_osif_priv *osif_priv;
  535. struct wireless_dev *wdev;
  536. osif_debug("user data:%pK, vdev id:%d, rssi:%d, buf:%pK, len:%d",
  537. user_data, rx_frame->vdev_id, rx_frame->rx_rssi,
  538. rx_frame->buf, rx_frame->frame_len);
  539. psoc = user_data;
  540. if (!psoc) {
  541. osif_err("psoc is null");
  542. return;
  543. }
  544. vdev = wlan_objmgr_get_vdev_by_id_from_psoc(psoc,
  545. rx_frame->vdev_id, WLAN_TDLS_NB_ID);
  546. if (!vdev) {
  547. osif_err("vdev is null");
  548. return;
  549. }
  550. osif_priv = wlan_vdev_get_ospriv(vdev);
  551. if (!osif_priv) {
  552. osif_err("osif_priv is null");
  553. goto fail;
  554. }
  555. wdev = osif_priv->wdev;
  556. if (!wdev) {
  557. osif_err("wdev is null");
  558. goto fail;
  559. }
  560. osif_notice("Indicate frame over nl80211, vdev id:%d, idx:%d",
  561. rx_frame->vdev_id, wdev->netdev->ifindex);
  562. #if (LINUX_VERSION_CODE >= KERNEL_VERSION(3, 18, 0))
  563. cfg80211_rx_mgmt(wdev, rx_frame->rx_freq, rx_frame->rx_rssi * 100,
  564. rx_frame->buf, rx_frame->frame_len,
  565. NL80211_RXMGMT_FLAG_ANSWERED);
  566. #elif (LINUX_VERSION_CODE >= KERNEL_VERSION(3, 12, 0))
  567. cfg80211_rx_mgmt(wdev, rx_frame->rx_freq, rx_frame->rx_rssi * 100,
  568. rx_frame->buf, rx_frame->frame_len,
  569. NL80211_RXMGMT_FLAG_ANSWERED, GFP_ATOMIC);
  570. #else
  571. cfg80211_rx_mgmt(wdev, rx_frame->rx_freq, rx_frame->rx_rssi * 100,
  572. rx_frame->buf, rx_frame->frame_len, GFP_ATOMIC);
  573. #endif /* LINUX_VERSION_CODE */
  574. fail:
  575. wlan_objmgr_vdev_release_ref(vdev, WLAN_TDLS_NB_ID);
  576. }
  577. static void wlan_cfg80211_update_tdls_peers_rssi(struct wlan_objmgr_vdev *vdev)
  578. {
  579. int ret = 0, i;
  580. struct stats_event *rssi_info;
  581. struct qdf_mac_addr bcast_mac = QDF_MAC_ADDR_BCAST_INIT;
  582. rssi_info = wlan_cfg80211_mc_cp_stats_get_peer_rssi(
  583. vdev, bcast_mac.bytes,
  584. &ret);
  585. if (ret || !rssi_info) {
  586. osif_err("get peer rssi fail");
  587. wlan_cfg80211_mc_cp_stats_free_stats_event(rssi_info);
  588. return;
  589. }
  590. for (i = 0; i < rssi_info->num_peer_stats; i++)
  591. ucfg_tdls_set_rssi(vdev, rssi_info->peer_stats[i].peer_macaddr,
  592. rssi_info->peer_stats[i].peer_rssi);
  593. wlan_cfg80211_mc_cp_stats_free_stats_event(rssi_info);
  594. }
  595. int wlan_cfg80211_tdls_get_all_peers(struct wlan_objmgr_vdev *vdev,
  596. char *buf, int buflen)
  597. {
  598. struct vdev_osif_priv *osif_priv;
  599. struct osif_tdls_vdev *tdls_priv;
  600. int32_t len;
  601. QDF_STATUS status;
  602. unsigned long rc;
  603. osif_priv = wlan_vdev_get_ospriv(vdev);
  604. if (!osif_priv || !osif_priv->osif_tdls) {
  605. osif_err("osif_tdls_vdev or osif_priv is NULL for the current vdev");
  606. return -EINVAL;
  607. }
  608. tdls_priv = osif_priv->osif_tdls;
  609. wlan_cfg80211_update_tdls_peers_rssi(vdev);
  610. reinit_completion(&tdls_priv->tdls_user_cmd_comp);
  611. status = ucfg_tdls_get_all_peers(vdev, buf, buflen);
  612. if (QDF_IS_STATUS_ERROR(status)) {
  613. osif_err("ucfg_tdls_get_all_peers failed err %d", status);
  614. len = scnprintf(buf, buflen,
  615. "\nucfg_tdls_send_mgmt failed\n");
  616. goto error_get_tdls_peers;
  617. }
  618. osif_debug("Wait for tdls_user_cmd_comp. Timeout %u ms",
  619. WAIT_TIME_FOR_TDLS_USER_CMD);
  620. rc = wait_for_completion_timeout(
  621. &tdls_priv->tdls_user_cmd_comp,
  622. msecs_to_jiffies(WAIT_TIME_FOR_TDLS_USER_CMD));
  623. if (0 == rc) {
  624. osif_err("TDLS user cmd get all peers timed out rc %ld",
  625. rc);
  626. len = scnprintf(buf, buflen,
  627. "\nTDLS user cmd get all peers timed out\n");
  628. goto error_get_tdls_peers;
  629. }
  630. len = tdls_priv->tdls_user_cmd_len;
  631. error_get_tdls_peers:
  632. return len;
  633. }
  634. int wlan_cfg80211_tdls_mgmt(struct wlan_objmgr_vdev *vdev,
  635. const uint8_t *peer_mac,
  636. uint8_t action_code, uint8_t dialog_token,
  637. uint16_t status_code, uint32_t peer_capability,
  638. const uint8_t *buf, size_t len)
  639. {
  640. struct tdls_action_frame_request mgmt_req;
  641. struct vdev_osif_priv *osif_priv;
  642. struct osif_tdls_vdev *tdls_priv;
  643. int status;
  644. unsigned long rc;
  645. struct tdls_set_responder_req set_responder;
  646. status = wlan_cfg80211_tdls_validate_mac_addr(peer_mac);
  647. if (status)
  648. return status;
  649. osif_priv = wlan_vdev_get_ospriv(vdev);
  650. if (!osif_priv || !osif_priv->osif_tdls) {
  651. osif_err("osif priv or tdls priv is NULL");
  652. return -EINVAL;
  653. }
  654. tdls_priv = osif_priv->osif_tdls;
  655. /* make sure doesn't call send_mgmt() while it is pending */
  656. if (TDLS_VDEV_MAGIC == tdls_priv->mgmt_tx_completion_status) {
  657. osif_err(QDF_MAC_ADDR_STR " action %d couldn't sent, as one is pending. return EBUSY",
  658. QDF_MAC_ADDR_ARRAY(peer_mac), action_code);
  659. return -EBUSY;
  660. }
  661. /* Reset TDLS VDEV magic */
  662. tdls_priv->mgmt_tx_completion_status = TDLS_VDEV_MAGIC;
  663. /*prepare the request */
  664. /* Validate the management Request */
  665. mgmt_req.chk_frame.action_code = action_code;
  666. qdf_mem_copy(mgmt_req.chk_frame.peer_mac, peer_mac, QDF_MAC_ADDR_SIZE);
  667. mgmt_req.chk_frame.dialog_token = dialog_token;
  668. mgmt_req.chk_frame.action_code = action_code;
  669. mgmt_req.chk_frame.status_code = status_code;
  670. mgmt_req.chk_frame.len = len;
  671. mgmt_req.vdev = vdev;
  672. mgmt_req.vdev_id = wlan_vdev_get_id(vdev);
  673. mgmt_req.session_id = mgmt_req.vdev_id;
  674. /* populate management req params */
  675. qdf_mem_copy(mgmt_req.tdls_mgmt.peer_mac.bytes,
  676. peer_mac, QDF_MAC_ADDR_SIZE);
  677. mgmt_req.tdls_mgmt.dialog = dialog_token;
  678. mgmt_req.tdls_mgmt.frame_type = action_code;
  679. mgmt_req.tdls_mgmt.len = len;
  680. mgmt_req.tdls_mgmt.peer_capability = peer_capability;
  681. mgmt_req.tdls_mgmt.status_code = mgmt_req.chk_frame.status_code;
  682. /*populate the additional IE's */
  683. mgmt_req.cmd_buf = buf;
  684. mgmt_req.len = len;
  685. reinit_completion(&tdls_priv->tdls_mgmt_comp);
  686. status = ucfg_tdls_send_mgmt_frame(&mgmt_req);
  687. if (QDF_IS_STATUS_ERROR(status)) {
  688. osif_err("ucfg_tdls_send_mgmt failed err %d", status);
  689. status = -EIO;
  690. tdls_priv->mgmt_tx_completion_status = false;
  691. goto error_mgmt_req;
  692. }
  693. osif_debug("Wait for tdls_mgmt_comp. Timeout %u ms",
  694. WAIT_TIME_FOR_TDLS_MGMT);
  695. rc = wait_for_completion_timeout(
  696. &tdls_priv->tdls_mgmt_comp,
  697. msecs_to_jiffies(WAIT_TIME_FOR_TDLS_MGMT));
  698. if ((0 == rc) || (QDF_STATUS_SUCCESS !=
  699. tdls_priv->mgmt_tx_completion_status)) {
  700. osif_err("%s rc %ld mgmtTxCompletionStatus %u",
  701. !rc ? "Mgmt Tx Completion timed out" :
  702. "Mgmt Tx Completion failed",
  703. rc, tdls_priv->mgmt_tx_completion_status);
  704. tdls_priv->mgmt_tx_completion_status = false;
  705. status = -EINVAL;
  706. goto error_mgmt_req;
  707. }
  708. osif_debug("Mgmt Tx Completion status %ld TxCompletion %u",
  709. rc, tdls_priv->mgmt_tx_completion_status);
  710. if (TDLS_SETUP_RESPONSE == action_code ||
  711. TDLS_SETUP_CONFIRM == action_code) {
  712. qdf_mem_copy(set_responder.peer_mac, peer_mac,
  713. QDF_MAC_ADDR_SIZE);
  714. set_responder.vdev = vdev;
  715. if (TDLS_SETUP_RESPONSE == action_code)
  716. set_responder.responder = false;
  717. if (TDLS_SETUP_CONFIRM == action_code)
  718. set_responder.responder = true;
  719. ucfg_tdls_responder(&set_responder);
  720. }
  721. error_mgmt_req:
  722. return status;
  723. }
  724. int wlan_tdls_antenna_switch(struct wlan_objmgr_vdev *vdev, uint32_t mode)
  725. {
  726. struct vdev_osif_priv *osif_priv;
  727. struct osif_tdls_vdev *tdls_priv;
  728. int ret;
  729. unsigned long rc;
  730. if (!vdev) {
  731. osif_err("vdev is NULL");
  732. return -EAGAIN;
  733. }
  734. osif_priv = wlan_vdev_get_ospriv(vdev);
  735. if (!osif_priv || !osif_priv->osif_tdls) {
  736. osif_err("osif priv or tdls priv is NULL");
  737. ret = -EINVAL;
  738. goto error;
  739. }
  740. tdls_priv = osif_priv->osif_tdls;
  741. reinit_completion(&tdls_priv->tdls_antenna_switch_comp);
  742. ret = ucfg_tdls_antenna_switch(vdev, mode);
  743. if (QDF_IS_STATUS_ERROR(ret)) {
  744. osif_err("ucfg_tdls_antenna_switch failed err %d", ret);
  745. ret = -EAGAIN;
  746. goto error;
  747. }
  748. rc = wait_for_completion_timeout(
  749. &tdls_priv->tdls_antenna_switch_comp,
  750. msecs_to_jiffies(WAIT_TIME_FOR_TDLS_ANTENNA_SWITCH));
  751. if (!rc) {
  752. osif_err("timeout for tdls antenna switch %ld", rc);
  753. ret = -EAGAIN;
  754. goto error;
  755. }
  756. ret = tdls_priv->tdls_antenna_switch_status;
  757. osif_debug("tdls antenna switch status:%d", ret);
  758. error:
  759. return ret;
  760. }
  761. static void
  762. wlan_cfg80211_tdls_indicate_discovery(struct tdls_osif_indication *ind)
  763. {
  764. struct vdev_osif_priv *osif_vdev;
  765. osif_vdev = wlan_vdev_get_ospriv(ind->vdev);
  766. osif_debug("Implicit TDLS, request Send Discovery request");
  767. cfg80211_tdls_oper_request(osif_vdev->wdev->netdev,
  768. ind->peer_mac, NL80211_TDLS_DISCOVERY_REQ,
  769. false, GFP_KERNEL);
  770. }
  771. static void
  772. wlan_cfg80211_tdls_indicate_setup(struct tdls_osif_indication *ind)
  773. {
  774. struct vdev_osif_priv *osif_vdev;
  775. osif_vdev = wlan_vdev_get_ospriv(ind->vdev);
  776. osif_debug("Indication to request TDLS setup");
  777. cfg80211_tdls_oper_request(osif_vdev->wdev->netdev,
  778. ind->peer_mac, NL80211_TDLS_SETUP, false,
  779. GFP_KERNEL);
  780. }
  781. static void
  782. wlan_cfg80211_tdls_indicate_teardown(struct tdls_osif_indication *ind)
  783. {
  784. struct vdev_osif_priv *osif_vdev;
  785. osif_vdev = wlan_vdev_get_ospriv(ind->vdev);
  786. osif_debug("Teardown reason %d", ind->reason);
  787. cfg80211_tdls_oper_request(osif_vdev->wdev->netdev,
  788. ind->peer_mac, NL80211_TDLS_TEARDOWN,
  789. ind->reason, GFP_KERNEL);
  790. }
  791. void wlan_cfg80211_tdls_event_callback(void *user_data,
  792. enum tdls_event_type type,
  793. struct tdls_osif_indication *ind)
  794. {
  795. struct vdev_osif_priv *osif_vdev;
  796. struct osif_tdls_vdev *tdls_priv;
  797. if (!ind || !ind->vdev) {
  798. osif_err("ind: %pK", ind);
  799. return;
  800. }
  801. osif_vdev = wlan_vdev_get_ospriv(ind->vdev);
  802. if (!osif_vdev || !osif_vdev->osif_tdls) {
  803. osif_err("osif priv or tdls priv is NULL");
  804. return;
  805. }
  806. tdls_priv = osif_vdev->osif_tdls;
  807. switch (type) {
  808. case TDLS_EVENT_MGMT_TX_ACK_CNF:
  809. tdls_priv->mgmt_tx_completion_status = ind->status;
  810. complete(&tdls_priv->tdls_mgmt_comp);
  811. break;
  812. case TDLS_EVENT_ADD_PEER:
  813. tdls_priv->tdls_add_peer_status = ind->status;
  814. complete(&tdls_priv->tdls_add_peer_comp);
  815. break;
  816. case TDLS_EVENT_DEL_PEER:
  817. complete(&tdls_priv->tdls_del_peer_comp);
  818. break;
  819. case TDLS_EVENT_DISCOVERY_REQ:
  820. wlan_cfg80211_tdls_indicate_discovery(ind);
  821. break;
  822. case TDLS_EVENT_TEARDOWN_REQ:
  823. wlan_cfg80211_tdls_indicate_teardown(ind);
  824. break;
  825. case TDLS_EVENT_SETUP_REQ:
  826. wlan_cfg80211_tdls_indicate_setup(ind);
  827. break;
  828. case TDLS_EVENT_TEARDOWN_LINKS_DONE:
  829. complete(&tdls_priv->tdls_teardown_comp);
  830. break;
  831. case TDLS_EVENT_USER_CMD:
  832. tdls_priv->tdls_user_cmd_len = ind->status;
  833. complete(&tdls_priv->tdls_user_cmd_comp);
  834. break;
  835. case TDLS_EVENT_ANTENNA_SWITCH:
  836. tdls_priv->tdls_antenna_switch_status = ind->status;
  837. complete(&tdls_priv->tdls_antenna_switch_comp);
  838. default:
  839. break;
  840. }
  841. }