wlan_cfg80211_coex.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. /*
  2. * Copyright (c) 2020, The Linux Foundation. All rights reserved.
  3. * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
  4. *
  5. * Permission to use, copy, modify, and/or distribute this software for any
  6. * purpose with or without fee is hereby granted, provided that the above
  7. * copyright notice and this permission notice appear in all copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  10. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  11. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  12. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  13. * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
  14. * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
  15. * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  16. */
  17. /**
  18. * DOC: defines driver functions interfacing with linux kernel
  19. */
  20. #include <wmi_unified_param.h>
  21. #include <wlan_osif_request_manager.h>
  22. #include <osif_sync.h>
  23. #include <wlan_objmgr_psoc_obj_i.h>
  24. #include <wlan_coex_main.h>
  25. #include <wlan_coex_ucfg_api.h>
  26. #include <wlan_cfg80211_coex.h>
  27. const struct nla_policy
  28. btc_chain_mode_policy[QCA_VENDOR_ATTR_BTC_CHAIN_MODE_MAX + 1] = {
  29. [QCA_VENDOR_ATTR_BTC_CHAIN_MODE] = {.type = NLA_U32},
  30. [QCA_VENDOR_ATTR_BTC_CHAIN_MODE_RESTART] = {.type = NLA_FLAG},
  31. };
  32. static enum coex_btc_chain_mode
  33. __wlan_cfg80211_coex_map_btc_chain_mode(enum qca_btc_chain_mode mode)
  34. {
  35. switch (mode) {
  36. case QCA_BTC_CHAIN_SHARED:
  37. return WLAN_COEX_BTC_CHAIN_MODE_SHARED;
  38. case QCA_BTC_CHAIN_SEPARATED_HYBRID:
  39. return WLAN_COEX_BTC_CHAIN_MODE_HYBRID;
  40. case QCA_BTC_CHAIN_SEPARATED_FDD:
  41. return WLAN_COEX_BTC_CHAIN_MODE_FDD;
  42. default:
  43. return WLAN_COEX_BTC_CHAIN_MODE_UNSETTLED;
  44. }
  45. }
  46. static int
  47. __wlan_cfg80211_coex_set_btc_chain_mode(struct wlan_objmgr_vdev *vdev,
  48. enum coex_btc_chain_mode mode,
  49. bool do_restart)
  50. {
  51. QDF_STATUS status;
  52. enum coex_btc_chain_mode cur_mode;
  53. int err;
  54. struct wlan_objmgr_psoc *psoc;
  55. struct wlan_objmgr_vdev *vdev_tmp;
  56. int vdev_id;
  57. struct coex_psoc_obj *coex_obj;
  58. if (!vdev) {
  59. coex_err("Null vdev");
  60. return -EINVAL;
  61. }
  62. psoc = wlan_vdev_get_psoc(vdev);
  63. if (!psoc) {
  64. coex_err("NULL psoc");
  65. return -EINVAL;
  66. }
  67. coex_obj = wlan_psoc_get_coex_obj(psoc);
  68. if (!coex_obj)
  69. return -EINVAL;
  70. status = ucfg_coex_psoc_get_btc_chain_mode(psoc, &cur_mode);
  71. if (QDF_IS_STATUS_ERROR(status)) {
  72. coex_err("failed to get cur BTC chain mode, status %d", status);
  73. return -EFAULT;
  74. }
  75. if (cur_mode == mode)
  76. return -EALREADY;
  77. status = ucfg_coex_psoc_set_btc_chain_mode(psoc, mode);
  78. if (!QDF_IS_STATUS_SUCCESS(status)) {
  79. coex_err("unable to set BTC chain mode to %d", mode);
  80. return -EFAULT;
  81. }
  82. wlan_objmgr_for_each_psoc_vdev(psoc, vdev_id, vdev_tmp) {
  83. status = ucfg_coex_send_btc_chain_mode(vdev_tmp, mode);
  84. err = qdf_status_to_os_return(status);
  85. if (err) {
  86. coex_err("Failed to set btc chain mode to %d for vdev %d",
  87. mode, vdev_id);
  88. return err;
  89. }
  90. coex_debug("Set btc chain mode to %d for vdev %d",
  91. mode, vdev_id);
  92. if (!do_restart)
  93. continue;
  94. wlan_coex_config_updated(vdev_tmp, COEX_CONFIG_BTC_CHAIN_MODE);
  95. }
  96. return 0;
  97. }
  98. /**
  99. * wlan_cfg80211_coex_set_btc_chain_mode() - set btc chain mode
  100. * @vdev: pointer to vdev structure.
  101. * @data: pointer to btc chain mode command parameters.
  102. * @data_len: the length in byte of btc chain mode command parameters.
  103. *
  104. * Return: An error code or 0 on success.
  105. */
  106. int wlan_cfg80211_coex_set_btc_chain_mode(struct wlan_objmgr_vdev *vdev,
  107. const void *data, int data_len)
  108. {
  109. struct nlattr *tb[QCA_VENDOR_ATTR_BTC_CHAIN_MODE_MAX + 1];
  110. uint32_t mode;
  111. enum coex_btc_chain_mode chain_mode;
  112. bool restart;
  113. if (wlan_cfg80211_nla_parse(tb, QCA_VENDOR_ATTR_BTC_CHAIN_MODE_MAX,
  114. data, data_len, btc_chain_mode_policy)) {
  115. coex_err("Invalid btc chain mode ATTR");
  116. return -EINVAL;
  117. }
  118. if (!tb[QCA_VENDOR_ATTR_BTC_CHAIN_MODE]) {
  119. coex_err("btc chain mode - no attr mode");
  120. return -EINVAL;
  121. }
  122. mode = nla_get_u32(tb[QCA_VENDOR_ATTR_BTC_CHAIN_MODE]);
  123. if (mode > QCA_BTC_CHAIN_SEPARATED_FDD) {
  124. coex_err("Invalid btc chain mode %d", mode);
  125. return -EINVAL;
  126. }
  127. restart = nla_get_flag(tb[QCA_VENDOR_ATTR_BTC_CHAIN_MODE_RESTART]);
  128. /* map to internal mode definitions */
  129. chain_mode = __wlan_cfg80211_coex_map_btc_chain_mode(mode);
  130. if (chain_mode == WLAN_COEX_BTC_CHAIN_MODE_UNSETTLED) {
  131. coex_err("Invalid wlan btc chain mode %d", chain_mode);
  132. return -EINVAL;
  133. }
  134. coex_debug("vdev_id %u mode %u restart %u",
  135. wlan_vdev_get_id(vdev), chain_mode, restart);
  136. return __wlan_cfg80211_coex_set_btc_chain_mode(vdev,
  137. chain_mode,
  138. restart);
  139. }