Browse Source

qcacmn: Update correct PHY mode while processing ch_info_req

Currently PHY mode is not getting updated and it is being set as
0, which is resulting in an invalid channel setting to FW
for LOWI.
To address this issue update the PHY mode correctly.

Change-Id: I77b2a517915a05d6a13dec5cee064d83bfae2992
CRs-Fixed: 2318563
Ashish Kumar Dhanotiya 6 years ago
parent
commit
04c074d219

+ 14 - 1
umac/wifi_pos/inc/wifi_pos_api.h

@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2012-2017 The Linux Foundation. All rights reserved.
+ * Copyright (c) 2012-2018 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
@@ -368,4 +368,17 @@ static inline QDF_STATUS wifi_pos_init_cir_cfr_rings(
 }
 #endif
 
+/**
+ * wifi_pos_register_get_phy_mode_cb: API to register callback to get
+ * current PHY mode
+ * @psoc: pointer to psoc object
+ * @handler: callback to be registered
+ *
+ * Return: QDF_STATUS_SUCCESS in case of success, error codes in
+ * case of failure
+ */
+QDF_STATUS wifi_pos_register_get_phy_mode_cb(
+			struct wlan_objmgr_psoc *psoc,
+			void (*handler)(uint8_t, uint32_t, uint32_t *));
+
 #endif

+ 26 - 0
umac/wifi_pos/src/wifi_pos_api.c

@@ -256,3 +256,29 @@ QDF_STATUS wifi_pos_init_cir_cfr_rings(struct wlan_objmgr_psoc *psoc,
 }
 #endif
 
+QDF_STATUS wifi_pos_register_get_phy_mode_cb(
+				struct wlan_objmgr_psoc *psoc,
+				void (*handler)(uint8_t, uint32_t, uint32_t *))
+{
+	struct wifi_pos_psoc_priv_obj *wifi_pos_psoc;
+
+	if (!psoc) {
+		wifi_pos_err("psoc is null");
+		return QDF_STATUS_E_NULL_VALUE;
+	}
+
+	if (!handler) {
+		wifi_pos_err("Null callback");
+		return QDF_STATUS_E_NULL_VALUE;
+	}
+	wifi_pos_psoc = wifi_pos_get_psoc_priv_obj(psoc);
+	if (!wifi_pos_psoc) {
+		wifi_pos_err("wifi_pos priv obj is null");
+		return QDF_STATUS_E_NULL_VALUE;
+	}
+
+	wifi_pos_psoc->wifi_pos_get_phy_mode = handler;
+
+	return QDF_STATUS_SUCCESS;
+}
+

+ 114 - 9
umac/wifi_pos/src/wifi_pos_main.c

@@ -38,6 +38,24 @@
 /* forward declartion */
 struct regulatory_channel;
 
+#define REG_SET_CHANNEL_REG_POWER(reg_info_1, val) do { \
+	reg_info_1 &= 0xff00ffff;           \
+	reg_info_1 |= ((val & 0xff) << 16); \
+} while (0)
+
+/* max tx power is in 1 dBm units */
+#define REG_SET_CHANNEL_MAX_TX_POWER(reg_info_2, val) do { \
+	reg_info_2 &= 0xffff00ff;              \
+	reg_info_2 |= ((val & 0xff) << 8);     \
+} while (0)
+
+/* channel info consists of 6 bits of channel mode */
+
+#define REG_SET_CHANNEL_MODE(reg_channel, val) do { \
+	(reg_channel)->info &= 0xffffffc0;            \
+	(reg_channel)->info |= (val);                 \
+} while (0)
+
 /*
  * obj mgr api to iterate over vdevs does not provide a direct array or vdevs,
  * rather takes a callback that is called for every vdev. wifi pos needs to
@@ -183,15 +201,85 @@ static QDF_STATUS wifi_pos_process_get_cap_req(struct wlan_objmgr_psoc *psoc,
 	return QDF_STATUS_SUCCESS;
 }
 
+static void wifi_update_channel_bw_info(struct wlan_objmgr_psoc *psoc,
+					struct wlan_objmgr_pdev *pdev,
+					uint16_t chan,
+					struct wifi_pos_ch_info_rsp *chan_info)
+{
+	struct ch_params ch_params = {0};
+	uint16_t sec_ch_2g = 0;
+	struct wifi_pos_psoc_priv_obj *wifi_pos_psoc =
+		wifi_pos_get_psoc_priv_obj(psoc);
+	uint32_t phy_mode;
+
+	if (!wifi_pos_psoc) {
+		wifi_pos_err("wifi_pos priv obj is null");
+		return;
+	}
+
+	/* Passing CH_WIDTH_MAX will give the max bandwidth supported */
+	ch_params.ch_width = CH_WIDTH_MAX;
+
+	wlan_reg_set_channel_params(pdev, chan, sec_ch_2g, &ch_params);
+	if (ch_params.center_freq_seg0)
+		chan_info->band_center_freq1 =
+			wlan_reg_get_channel_freq(pdev,
+						  ch_params.center_freq_seg0);
+
+	wifi_pos_psoc->wifi_pos_get_phy_mode(chan, ch_params.ch_width,
+					     &phy_mode);
+
+	REG_SET_CHANNEL_MODE(chan_info, phy_mode);
+}
+
+static void wifi_pos_get_reg_info(struct wlan_objmgr_pdev *pdev,
+				  uint32_t chan_num, uint32_t *reg_info_1,
+				  uint32_t *reg_info_2)
+{
+	uint32_t reg_power = wlan_reg_get_channel_reg_power(pdev, chan_num);
+
+	*reg_info_1 = 0;
+	*reg_info_2 = 0;
+
+	REG_SET_CHANNEL_REG_POWER(*reg_info_1, reg_power);
+	REG_SET_CHANNEL_MAX_TX_POWER(*reg_info_2, reg_power);
+}
+
+/**
+ * wifi_pos_get_valid_channels: Get the list of valid channels from the
+ * given channel list
+ * @channels: Channel list to be validated
+ * @num_ch: NUmber of channels in the channel list to be validated
+ * @valid_channel_list: Pointer to valid channel list
+ *
+ * Return: Number of valid channels in the given list
+ */
+
+static uint32_t wifi_pos_get_valid_channels(uint8_t *channels, uint32_t num_ch,
+					    uint8_t *valid_channel_list) {
+	uint32_t i, num_valid_channels = 0;
+
+	for (i = 0; i < num_ch; i++) {
+		if (INVALID_CHANNEL == reg_get_chan_enum(channels[i]))
+			continue;
+		valid_channel_list[num_valid_channels++] = channels[i];
+	}
+	return num_valid_channels;
+}
+
 static QDF_STATUS wifi_pos_process_ch_info_req(struct wlan_objmgr_psoc *psoc,
 					struct wifi_pos_req_msg *req)
 {
 	uint8_t idx;
 	uint8_t *buf;
 	uint32_t len;
+	uint32_t reg_info_1;
+	uint32_t reg_info_2;
 	uint8_t *channels = req->buf;
 	struct wlan_objmgr_pdev *pdev;
 	uint32_t num_ch = req->buf_len;
+	uint8_t valid_channel_list[NUM_CHANNELS];
+	uint32_t num_valid_channels;
 	struct wifi_pos_ch_info_rsp *ch_info;
 	struct wifi_pos_psoc_priv_obj *wifi_pos_obj =
 					wifi_pos_get_psoc_priv_obj(psoc);
@@ -205,8 +293,15 @@ static QDF_STATUS wifi_pos_process_ch_info_req(struct wlan_objmgr_psoc *psoc,
 		wifi_pos_err("pdev get API failed");
 		return QDF_STATUS_E_INVAL;
 	}
+	if (num_ch > NUM_CHANNELS) {
+		wifi_pos_err("Invalid number of channels");
+		return QDF_STATUS_E_INVAL;
+	}
+	num_valid_channels = wifi_pos_get_valid_channels(channels, num_ch,
+							 valid_channel_list);
 
-	len = sizeof(uint8_t) + sizeof(struct wifi_pos_ch_info_rsp) * num_ch;
+	len = sizeof(uint8_t) + sizeof(struct wifi_pos_ch_info_rsp) *
+			num_valid_channels;
 	buf = qdf_mem_malloc(len);
 	if (!buf) {
 		wifi_pos_alert("malloc failed");
@@ -214,21 +309,31 @@ static QDF_STATUS wifi_pos_process_ch_info_req(struct wlan_objmgr_psoc *psoc,
 		return QDF_STATUS_E_NOMEM;
 	}
 
+	qdf_mem_zero(buf, len);
+
 	/* First byte of message body will have num of channels */
-	buf[0] = num_ch;
+	buf[0] = num_valid_channels;
 	ch_info = (struct wifi_pos_ch_info_rsp *)&buf[1];
-	for (idx = 0; idx < num_ch; idx++) {
-		ch_info[idx].chan_id = channels[idx];
+	for (idx = 0; idx < num_valid_channels; idx++) {
+		ch_info[idx].chan_id = valid_channel_list[idx];
+		wifi_pos_get_reg_info(pdev, ch_info[idx].chan_id,
+				      &reg_info_1, &reg_info_2);
 		ch_info[idx].reserved0 = 0;
-		ch_info[idx].mhz = wlan_reg_get_channel_freq(pdev,
-							channels[idx]);
+		ch_info[idx].mhz = wlan_reg_get_channel_freq(
+						pdev,
+						valid_channel_list[idx]);
 		ch_info[idx].band_center_freq1 = ch_info[idx].mhz;
 		ch_info[idx].band_center_freq2 = 0;
 		ch_info[idx].info = 0;
-		if (wlan_reg_is_dfs_ch(pdev, channels[idx]))
+		if (wlan_reg_is_dfs_ch(pdev, valid_channel_list[idx]))
 			WIFI_POS_SET_DFS(ch_info[idx].info);
-		ch_info[idx].reg_info_1 = 0;
-		ch_info[idx].reg_info_2 = 0;
+
+		wifi_update_channel_bw_info(psoc, pdev,
+					    ch_info[idx].chan_id,
+					    &ch_info[idx]);
+
+		ch_info[idx].reg_info_1 = reg_info_1;
+		ch_info[idx].reg_info_2 = reg_info_2;
 	}
 
 	wifi_pos_obj->wifi_pos_send_rsp(wifi_pos_obj->app_pid,

+ 1 - 0
umac/wifi_pos/src/wifi_pos_utils_i.h

@@ -304,6 +304,7 @@ struct wifi_pos_psoc_priv_obj {
 	QDF_STATUS (*wifi_pos_req_handler)(struct wlan_objmgr_psoc *psoc,
 				    struct wifi_pos_req_msg *req);
 	void (*wifi_pos_send_rsp)(uint32_t, uint32_t, uint32_t, uint8_t *);
+	void (*wifi_pos_get_phy_mode)(uint8_t, uint32_t, uint32_t *);
 };
 
 /**