diff options
Diffstat (limited to 'drivers/net/wireless/ath/wil6210/wmi.c')
| -rw-r--r-- | drivers/net/wireless/ath/wil6210/wmi.c | 88 | 
1 files changed, 82 insertions, 6 deletions
| diff --git a/drivers/net/wireless/ath/wil6210/wmi.c b/drivers/net/wireless/ath/wil6210/wmi.c index 7a0d934eb271..23e1ed6a9d6d 100644 --- a/drivers/net/wireless/ath/wil6210/wmi.c +++ b/drivers/net/wireless/ath/wil6210/wmi.c @@ -196,8 +196,8 @@ const struct fw_map talyn_mb_fw_mapping[] = {  	{0x8c0000, 0x8c0210, 0x8c0000, "dum_user_rgf", true, true},  	/* DMA OFU 296b */  	{0x8c2000, 0x8c2128, 0x8c2000, "dma_ofu", true, true}, -	/* ucode debug 4k */ -	{0x8c3000, 0x8c4000, 0x8c3000, "ucode_debug", true, true}, +	/* ucode debug 256b */ +	{0x8c3000, 0x8c3100, 0x8c3000, "ucode_debug", true, true},  	/* upper area 1536k */  	{0x900000, 0xa80000, 0x900000, "upper", true, true},  	/* UCODE areas - accessible by debugfs blobs but not by @@ -476,6 +476,8 @@ static const char *cmdid2name(u16 cmdid)  		return "WMI_RBUFCAP_CFG_CMD";  	case WMI_TEMP_SENSE_ALL_CMDID:  		return "WMI_TEMP_SENSE_ALL_CMDID"; +	case WMI_SET_LINK_MONITOR_CMDID: +		return "WMI_SET_LINK_MONITOR_CMD";  	default:  		return "Untracked CMD";  	} @@ -624,6 +626,10 @@ static const char *eventid2name(u16 eventid)  		return "WMI_RBUFCAP_CFG_EVENT";  	case WMI_TEMP_SENSE_ALL_DONE_EVENTID:  		return "WMI_TEMP_SENSE_ALL_DONE_EVENTID"; +	case WMI_SET_LINK_MONITOR_EVENTID: +		return "WMI_SET_LINK_MONITOR_EVENT"; +	case WMI_LINK_MONITOR_EVENTID: +		return "WMI_LINK_MONITOR_EVENT";  	default:  		return "Untracked EVENT";  	} @@ -1507,14 +1513,14 @@ static void wmi_link_stats_parse(struct wil6210_vif *vif, u64 tsf,  			if (vif->fw_stats_ready) {  				/* clean old statistics */  				vif->fw_stats_tsf = 0; -				vif->fw_stats_ready = 0; +				vif->fw_stats_ready = false;  			}  			wil_link_stats_store_basic(vif, payload + hdr_size);  			if (!has_next) {  				vif->fw_stats_tsf = tsf; -				vif->fw_stats_ready = 1; +				vif->fw_stats_ready = true;  			}  			break; @@ -1529,14 +1535,14 @@ static void wmi_link_stats_parse(struct wil6210_vif *vif, u64 tsf,  			if (wil->fw_stats_global.ready) {  				/* clean old statistics */  				wil->fw_stats_global.tsf = 0; -				wil->fw_stats_global.ready = 0; +				wil->fw_stats_global.ready = false;  			}  			wil_link_stats_store_global(vif, payload + hdr_size);  			if (!has_next) {  				wil->fw_stats_global.tsf = tsf; -				wil->fw_stats_global.ready = 1; +				wil->fw_stats_global.ready = true;  			}  			break; @@ -1836,6 +1842,32 @@ fail:  	wil6210_disconnect(vif, NULL, WLAN_REASON_PREV_AUTH_NOT_VALID);  } +static void +wmi_evt_link_monitor(struct wil6210_vif *vif, int id, void *d, int len) +{ +	struct wil6210_priv *wil = vif_to_wil(vif); +	struct net_device *ndev = vif_to_ndev(vif); +	struct wmi_link_monitor_event *evt = d; +	enum nl80211_cqm_rssi_threshold_event event_type; + +	if (len < sizeof(*evt)) { +		wil_err(wil, "link monitor event too short %d\n", len); +		return; +	} + +	wil_dbg_wmi(wil, "link monitor event, type %d rssi %d (stored %d)\n", +		    evt->type, evt->rssi_level, wil->cqm_rssi_thold); + +	if (evt->type != WMI_LINK_MONITOR_NOTIF_RSSI_THRESHOLD_EVT) +		/* ignore */ +		return; + +	event_type = (evt->rssi_level > wil->cqm_rssi_thold ? +		      NL80211_CQM_RSSI_THRESHOLD_EVENT_HIGH : +		      NL80211_CQM_RSSI_THRESHOLD_EVENT_LOW); +	cfg80211_cqm_rssi_notify(ndev, event_type, evt->rssi_level, GFP_KERNEL); +} +  /**   * Some events are ignored for purpose; and need not be interpreted as   * "unhandled events" @@ -1869,6 +1901,7 @@ static const struct {  	{WMI_LINK_STATS_EVENTID,		wmi_evt_link_stats},  	{WMI_FT_AUTH_STATUS_EVENTID,		wmi_evt_auth_status},  	{WMI_FT_REASSOC_STATUS_EVENTID,		wmi_evt_reassoc_status}, +	{WMI_LINK_MONITOR_EVENTID,		wmi_evt_link_monitor},  };  /* @@ -3981,3 +4014,46 @@ int wmi_link_stats_cfg(struct wil6210_vif *vif, u32 type, u8 cid, u32 interval)  	return 0;  } + +int wmi_set_cqm_rssi_config(struct wil6210_priv *wil, +			    s32 rssi_thold, u32 rssi_hyst) +{ +	struct net_device *ndev = wil->main_ndev; +	struct wil6210_vif *vif = ndev_to_vif(ndev); +	int rc; +	struct { +		struct wmi_set_link_monitor_cmd cmd; +		s8 rssi_thold; +	} __packed cmd = { +		.cmd = { +			.rssi_hyst = rssi_hyst, +			.rssi_thresholds_list_size = 1, +		}, +		.rssi_thold = rssi_thold, +	}; +	struct { +		struct wmi_cmd_hdr hdr; +		struct wmi_set_link_monitor_event evt; +	} __packed reply = { +		.evt = {.status = WMI_FW_STATUS_FAILURE}, +	}; + +	if (rssi_thold > S8_MAX || rssi_thold < S8_MIN || rssi_hyst > U8_MAX) +		return -EINVAL; + +	rc = wmi_call(wil, WMI_SET_LINK_MONITOR_CMDID, vif->mid, &cmd, +		      sizeof(cmd), WMI_SET_LINK_MONITOR_EVENTID, +		      &reply, sizeof(reply), WIL_WMI_CALL_GENERAL_TO_MS); +	if (rc) { +		wil_err(wil, "WMI_SET_LINK_MONITOR_CMDID failed, rc %d\n", rc); +		return rc; +	} + +	if (reply.evt.status != WMI_FW_STATUS_SUCCESS) { +		wil_err(wil, "WMI_SET_LINK_MONITOR_CMDID failed, status %d\n", +			reply.evt.status); +		return -EINVAL; +	} + +	return 0; +} | 
