/* * Copyright (c) 2020, The Linux Foundation. All rights reserved. * Copyright (c) 2022 Qualcomm Innovation Center, Inc. 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: defines driver functions interfacing with linux kernel */ #include #include #include #include #include #include #include const struct nla_policy btc_chain_mode_policy[QCA_VENDOR_ATTR_BTC_CHAIN_MODE_MAX + 1] = { [QCA_VENDOR_ATTR_BTC_CHAIN_MODE] = {.type = NLA_U32}, [QCA_VENDOR_ATTR_BTC_CHAIN_MODE_RESTART] = {.type = NLA_FLAG}, }; static enum coex_btc_chain_mode __wlan_cfg80211_coex_map_btc_chain_mode(enum qca_btc_chain_mode mode) { switch (mode) { case QCA_BTC_CHAIN_SHARED: return WLAN_COEX_BTC_CHAIN_MODE_SHARED; case QCA_BTC_CHAIN_SEPARATED_HYBRID: return WLAN_COEX_BTC_CHAIN_MODE_HYBRID; case QCA_BTC_CHAIN_SEPARATED_FDD: return WLAN_COEX_BTC_CHAIN_MODE_FDD; default: return WLAN_COEX_BTC_CHAIN_MODE_UNSETTLED; } } static int __wlan_cfg80211_coex_set_btc_chain_mode(struct wlan_objmgr_vdev *vdev, enum coex_btc_chain_mode mode, bool do_restart) { QDF_STATUS status; enum coex_btc_chain_mode cur_mode; int err; struct wlan_objmgr_psoc *psoc; struct wlan_objmgr_vdev *vdev_tmp; int vdev_id; struct coex_psoc_obj *coex_obj; if (!vdev) { coex_err("Null vdev"); return -EINVAL; } psoc = wlan_vdev_get_psoc(vdev); if (!psoc) { coex_err("NULL psoc"); return -EINVAL; } coex_obj = wlan_psoc_get_coex_obj(psoc); if (!coex_obj) return -EINVAL; status = ucfg_coex_psoc_get_btc_chain_mode(psoc, &cur_mode); if (QDF_IS_STATUS_ERROR(status)) { coex_err("failed to get cur BTC chain mode, status %d", status); return -EFAULT; } if (cur_mode == mode) return -EALREADY; status = ucfg_coex_psoc_set_btc_chain_mode(psoc, mode); if (!QDF_IS_STATUS_SUCCESS(status)) { coex_err("unable to set BTC chain mode to %d", mode); return -EFAULT; } wlan_objmgr_for_each_psoc_vdev(psoc, vdev_id, vdev_tmp) { status = ucfg_coex_send_btc_chain_mode(vdev_tmp, mode); err = qdf_status_to_os_return(status); if (err) { coex_err("Failed to set btc chain mode to %d for vdev %d", mode, vdev_id); return err; } coex_debug("Set btc chain mode to %d for vdev %d", mode, vdev_id); if (!do_restart) continue; wlan_coex_config_updated(vdev_tmp, COEX_CONFIG_BTC_CHAIN_MODE); } return 0; } /** * wlan_cfg80211_coex_set_btc_chain_mode() - set btc chain mode * @vdev: pointer to vdev structure. * @data: pointer to btc chain mode command parameters. * @data_len: the length in byte of btc chain mode command parameters. * * Return: An error code or 0 on success. */ int wlan_cfg80211_coex_set_btc_chain_mode(struct wlan_objmgr_vdev *vdev, const void *data, int data_len) { struct nlattr *tb[QCA_VENDOR_ATTR_BTC_CHAIN_MODE_MAX + 1]; uint32_t mode; enum coex_btc_chain_mode chain_mode; bool restart; if (wlan_cfg80211_nla_parse(tb, QCA_VENDOR_ATTR_BTC_CHAIN_MODE_MAX, data, data_len, btc_chain_mode_policy)) { coex_err("Invalid btc chain mode ATTR"); return -EINVAL; } if (!tb[QCA_VENDOR_ATTR_BTC_CHAIN_MODE]) { coex_err("btc chain mode - no attr mode"); return -EINVAL; } mode = nla_get_u32(tb[QCA_VENDOR_ATTR_BTC_CHAIN_MODE]); if (mode > QCA_BTC_CHAIN_SEPARATED_FDD) { coex_err("Invalid btc chain mode %d", mode); return -EINVAL; } restart = nla_get_flag(tb[QCA_VENDOR_ATTR_BTC_CHAIN_MODE_RESTART]); /* map to internal mode definitions */ chain_mode = __wlan_cfg80211_coex_map_btc_chain_mode(mode); if (chain_mode == WLAN_COEX_BTC_CHAIN_MODE_UNSETTLED) { coex_err("Invalid wlan btc chain mode %d", chain_mode); return -EINVAL; } coex_debug("vdev_id %u mode %u restart %u", wlan_vdev_get_id(vdev), chain_mode, restart); return __wlan_cfg80211_coex_set_btc_chain_mode(vdev, chain_mode, restart); }