blob: c6c6f80cc93dc63a126fd4525c30debaf6b7c3a5 [file] [log] [blame]
developerb85e8752023-09-25 14:16:05 +08001From 6219efda5b74baf62086236cc7f368f4307b9869 Mon Sep 17 00:00:00 2001
developerbddc9db2023-09-11 13:34:36 +08002From: Michael-CY Lee <michael-cy.lee@mediatek.com>
developerb85e8752023-09-25 14:16:05 +08003Date: Mon, 11 Sep 2023 10:16:35 +0800
developerbddc9db2023-09-11 13:34:36 +08004Subject: [PATCH] hostapd: mtk: synchronize bandwidth in AP/STA support
5
6Signed-off-by: Michael Lee <michael-cy.lee@mediatek.com>
7---
developerb85e8752023-09-25 14:16:05 +08008 src/ap/ucode.c | 39 ++++++++++++++++++--
9 src/utils/ucode.c | 12 +++++--
10 wpa_supplicant/ucode.c | 82 ++++++++++++++++++++++++++++++++++--------
11 3 files changed, 115 insertions(+), 18 deletions(-)
developerbddc9db2023-09-11 13:34:36 +080012
developerb85e8752023-09-25 14:16:05 +080013diff --git a/src/ap/ucode.c b/src/ap/ucode.c
14index 7bc797a..a3716ea 100644
15--- a/src/ap/ucode.c
16+++ b/src/ap/ucode.c
17@@ -382,6 +382,9 @@ uc_hostapd_iface_stop(uc_vm_t *vm, size_t nargs)
18 struct hostapd_iface *iface = uc_fn_thisval("hostapd.iface");
19 int i;
20
21+ wpa_printf(MSG_INFO, "ucode: mtk: stop iface for %s in state %s\n",
22+ iface->phy, hostapd_state_text(iface->state));
23+
24 switch (iface->state) {
25 case HAPD_IFACE_ENABLED:
26 case HAPD_IFACE_DISABLED:
27@@ -420,6 +423,7 @@ uc_hostapd_iface_start(uc_vm_t *vm, size_t nargs)
28 uint64_t intval;
29 int i;
30
31+ wpa_printf(MSG_INFO, "ucode: mtk: start iface for %s\n", iface->phy);
32 if (!iface)
33 return NULL;
34
35@@ -442,7 +446,13 @@ uc_hostapd_iface_start(uc_vm_t *vm, size_t nargs)
36 UPDATE_VAL(op_class, "op_class");
37 UPDATE_VAL(hw_mode, "hw_mode");
38 UPDATE_VAL(channel, "channel");
39- UPDATE_VAL(secondary_channel, "sec_channel");
40+
41+ intval = ucv_int64_get(ucv_object_get(info, "sec_channel", NULL));
42+ if (!errno) {
43+ conf->secondary_channel = intval;
44+ changed = true;
45+ }
46+
47 if (!changed &&
48 (iface->bss[0]->beacon_set_done ||
49 iface->state == HAPD_IFACE_DFS))
50@@ -490,6 +500,18 @@ out:
51 return ucv_boolean_new(true);
52 }
53
54+ wpa_printf(MSG_INFO, "ucode: mtk: updated channel information:\n");
55+ wpa_printf(MSG_INFO, " * channel: %d\n", conf->channel);
56+ wpa_printf(MSG_INFO, " * op_class: %d\n", conf->op_class);
57+ wpa_printf(MSG_INFO, " * secondary channel: %d\n",
58+ conf->secondary_channel);
59+ wpa_printf(MSG_INFO, " * seg0: %d\n",
60+ hostapd_get_oper_centr_freq_seg0_idx(conf));
61+ wpa_printf(MSG_INFO, " * seg1: %d\n",
62+ hostapd_get_oper_centr_freq_seg0_idx(conf));
63+ wpa_printf(MSG_INFO, " * oper_chwidth: %d\n",
64+ hostapd_get_oper_chwidth(conf));
65+
66 for (i = 0; i < iface->num_bss; i++) {
67 struct hostapd_data *hapd = iface->bss[i];
68 int ret;
69@@ -524,6 +546,7 @@ uc_hostapd_iface_switch_channel(uc_vm_t *vm, size_t nargs)
70 uint64_t intval;
71 int i, ret = 0;
72
73+ wpa_printf(MSG_INFO, "ucode: mtk: channel switch for %s\n", iface->phy);
74 if (!iface || ucv_type(info) != UC_OBJECT)
75 return NULL;
76
77@@ -543,7 +566,8 @@ uc_hostapd_iface_switch_channel(uc_vm_t *vm, size_t nargs)
78 if (errno)
79 intval = hostapd_get_oper_chwidth(conf);
80 if (intval)
81- csa.freq_params.bandwidth = 40 << intval;
82+ csa.freq_params.bandwidth = 40 <<
83+ (intval == CONF_OPER_CHWIDTH_320MHZ ? 3 : intval);
84 else
85 csa.freq_params.bandwidth = csa.freq_params.sec_channel_offset ? 40 : 20;
86
87@@ -554,6 +578,17 @@ uc_hostapd_iface_switch_channel(uc_vm_t *vm, size_t nargs)
88 if ((intval = ucv_int64_get(ucv_object_get(info, "center_freq2", NULL))) && !errno)
89 csa.freq_params.center_freq2 = intval;
90
91+ wpa_printf(MSG_INFO, "ucode: mtk: switch channel information:\n");
92+ wpa_printf(MSG_INFO, " * freq is %d\n", csa.freq_params.freq);
93+ wpa_printf(MSG_INFO, " * bandwidth is %d\n",
94+ csa.freq_params.bandwidth);
95+ wpa_printf(MSG_INFO, " * sec_chan_offset is %d\n",
96+ csa.freq_params.sec_channel_offset);
97+ wpa_printf(MSG_INFO, " * center_freq1 is %d\n",
98+ csa.freq_params.center_freq1);
99+ wpa_printf(MSG_INFO, " * center_freq2 is %d\n",
100+ csa.freq_params.center_freq2);
101+
102 for (i = 0; i < iface->num_bss; i++)
103 ret = hostapd_switch_channel(iface->bss[i], &csa);
104
developerbddc9db2023-09-11 13:34:36 +0800105diff --git a/src/utils/ucode.c b/src/utils/ucode.c
developerb85e8752023-09-25 14:16:05 +0800106index 122c619..0990e7b 100644
developerbddc9db2023-09-11 13:34:36 +0800107--- a/src/utils/ucode.c
108+++ b/src/utils/ucode.c
developerb85e8752023-09-25 14:16:05 +0800109@@ -51,6 +51,7 @@ uc_value_t *uc_wpa_freq_info(uc_vm_t *vm, size_t nargs)
110 uc_value_t *freq = uc_fn_arg(0);
111 uc_value_t *sec = uc_fn_arg(1);
112 int width = ucv_uint64_get(uc_fn_arg(2));
113+ int bw320_offset = 1;
114 int freq_val, center_idx, center_ofs;
115 enum oper_chan_width chanwidth;
116 enum hostapd_hw_mode hw_mode;
117@@ -88,6 +89,9 @@ uc_value_t *uc_wpa_freq_info(uc_vm_t *vm, size_t nargs)
118 case 9:
119 width = 3;
120 chanwidth = CONF_OPER_CHWIDTH_320MHZ;
121+
122+ /* bw320_offset is 1 for 320 MHz-1, and 2 for 320 MHz-2 */
123+ bw320_offset = ucv_uint64_get(uc_fn_arg(3));
124 break;
125 default:
126 return NULL;
127@@ -119,12 +123,16 @@ uc_value_t *uc_wpa_freq_info(uc_vm_t *vm, size_t nargs)
developerbddc9db2023-09-11 13:34:36 +0800128 ucv_object_add(ret, "hw_mode_str", ucv_get(ucv_string_new(modestr)));
129 ucv_object_add(ret, "sec_channel", ucv_int64_new(sec_channel));
130 ucv_object_add(ret, "frequency", ucv_int64_new(freq_val));
131+ ucv_object_add(ret, "oper_chwidth", ucv_int64_new(chanwidth));
132
developerb85e8752023-09-25 14:16:05 +0800133- if (!sec_channel)
134+ if (chanwidth == CONF_OPER_CHWIDTH_USE_HT && !sec_channel) {
135+ ucv_object_add(ret, "center_seg0_idx", ucv_int64_new(channel));
136+ ucv_object_add(ret, "center_freq1", ucv_int64_new(freq_val));
developerbddc9db2023-09-11 13:34:36 +0800137 return ret;
developerb85e8752023-09-25 14:16:05 +0800138+ }
139
140 if (freq_val >= 5900)
141- center_ofs = 0;
142+ center_ofs = 32 * (1 - bw320_offset);
143 else if (freq_val >= 5745)
144 center_ofs = 20;
145 else
developerbddc9db2023-09-11 13:34:36 +0800146diff --git a/wpa_supplicant/ucode.c b/wpa_supplicant/ucode.c
developerb85e8752023-09-25 14:16:05 +0800147index 55d2258..d8a19b8 100644
developerbddc9db2023-09-11 13:34:36 +0800148--- a/wpa_supplicant/ucode.c
149+++ b/wpa_supplicant/ucode.c
developerb85e8752023-09-25 14:16:05 +0800150@@ -7,6 +7,8 @@
developerbddc9db2023-09-11 13:34:36 +0800151 #include "wps_supplicant.h"
152 #include "bss.h"
153 #include "ucode.h"
154+#include "driver_i.h"
developerb85e8752023-09-25 14:16:05 +0800155+#include "common/ieee802_11_common.h"
developerbddc9db2023-09-11 13:34:36 +0800156
157 static struct wpa_global *wpa_global;
158 static uc_resource_type_t *global_type, *iface_type;
developerb85e8752023-09-25 14:16:05 +0800159@@ -96,6 +98,8 @@ void wpas_ucode_event(struct wpa_supplicant *wpa_s, int event, union wpa_event_d
160 {
161 const char *state;
162 uc_value_t *val;
163+ enum oper_chan_width ch_width;
164+ int center_freq1, bw320_offset = 1;
165
166 if (event != EVENT_CH_SWITCH_STARTED)
167 return;
168@@ -114,11 +118,42 @@ void wpas_ucode_event(struct wpa_supplicant *wpa_s, int event, union wpa_event_d
169 uc_value_push(ucv_get(val));
170
171 if (event == EVENT_CH_SWITCH_STARTED) {
172+ center_freq1 = data->ch_switch.cf1;
173+
174+ switch (data->ch_switch.ch_width) {
175+ case CHAN_WIDTH_80:
176+ ch_width = CONF_OPER_CHWIDTH_80MHZ;
177+ break;
178+ case CHAN_WIDTH_80P80:
179+ ch_width = CONF_OPER_CHWIDTH_80P80MHZ;
180+ break;
181+ case CHAN_WIDTH_160:
182+ ch_width = CONF_OPER_CHWIDTH_160MHZ;
183+ break;
184+ case CHAN_WIDTH_320:
185+ ch_width = CONF_OPER_CHWIDTH_320MHZ;
186+ break;
187+ case CHAN_WIDTH_20_NOHT:
188+ case CHAN_WIDTH_20:
189+ case CHAN_WIDTH_40:
190+ default:
191+ ch_width = CONF_OPER_CHWIDTH_USE_HT;
192+ break;
193+ }
194+
195+ /* Check bandwidth 320 MHz-2 */
196+ if (ch_width == CONF_OPER_CHWIDTH_320MHZ &&
197+ (center_freq1 == 6265) || center_freq1 == 6585 ||
198+ center_freq1 == 6905)
199+ bw320_offset = 2;
200+
201 ucv_object_add(val, "csa_count", ucv_int64_new(data->ch_switch.count));
202 ucv_object_add(val, "frequency", ucv_int64_new(data->ch_switch.freq));
203 ucv_object_add(val, "sec_chan_offset", ucv_int64_new(data->ch_switch.ch_offset));
204- ucv_object_add(val, "center_freq1", ucv_int64_new(data->ch_switch.cf1));
205+ ucv_object_add(val, "center_freq1", ucv_int64_new(center_freq1));
206 ucv_object_add(val, "center_freq2", ucv_int64_new(data->ch_switch.cf2));
207+ ucv_object_add(val, "ch_width", ucv_int64_new(ch_width));
208+ ucv_object_add(val, "bw320_offset", ucv_int64_new(bw320_offset));
209 }
210
211 ucv_put(wpa_ucode_call(4));
212@@ -195,6 +230,11 @@ uc_wpas_iface_status(uc_vm_t *vm, size_t nargs)
developerbddc9db2023-09-11 13:34:36 +0800213 struct wpa_supplicant *wpa_s = uc_fn_thisval("wpas.iface");
214 struct wpa_bss *bss;
215 uc_value_t *ret, *val;
216+ struct wpa_channel_info ci;
217+ u8 op_class, channel;
218+ enum oper_chan_width ch_width;
developerb85e8752023-09-25 14:16:05 +0800219+ int center_freq1, bw320_offset = 1, is_24ghz;
220+ enum hostapd_hw_mode hw_mode;
developerbddc9db2023-09-11 13:34:36 +0800221
222 if (!wpa_s)
223 return NULL;
developerb85e8752023-09-25 14:16:05 +0800224@@ -207,23 +247,37 @@ uc_wpas_iface_status(uc_vm_t *vm, size_t nargs)
225 bss = wpa_s->current_bss;
226 if (bss) {
227 int sec_chan = 0;
228- const u8 *ie;
229-
230- ie = wpa_bss_get_ie(bss, WLAN_EID_HT_OPERATION);
231- if (ie && ie[1] >= 2) {
232- const struct ieee80211_ht_operation *ht_oper;
233- int sec;
234-
235- ht_oper = (const void *) (ie + 2);
236- sec = ht_oper->ht_param & HT_INFO_HT_PARAM_SECONDARY_CHNL_OFF_MASK;
237- if (sec == HT_INFO_HT_PARAM_SECONDARY_CHNL_ABOVE)
238- sec_chan = 1;
239- else if (sec == HT_INFO_HT_PARAM_SECONDARY_CHNL_BELOW)
240- sec_chan = -1;
241+
242+ hw_mode = ieee80211_freq_to_chan(bss->freq, &channel);
243+ is_24ghz = hw_mode == HOSTAPD_MODE_IEEE80211G ||
244+ hw_mode == HOSTAPD_MODE_IEEE80211B;
developerbddc9db2023-09-11 13:34:36 +0800245+
246+ wpa_drv_channel_info(wpa_s, &ci);
developerb85e8752023-09-25 14:16:05 +0800247+ center_freq1 = ci.center_frq1;
248+
249+ if (bss->freq != center_freq1) {
250+ if (is_24ghz)
251+ sec_chan = (bss->freq < center_freq1) ? 1 : -1;
252+ else
253+ sec_chan = (bss->freq / 20) & 1 ? 1 : -1;
254+ }
255+
256+ if (ieee80211_chaninfo_to_channel(ci.frequency, ci.chanwidth,
257+ sec_chan, &op_class, &channel))
258+ return NULL;
259+
developerbddc9db2023-09-11 13:34:36 +0800260+ ch_width = op_class_to_ch_width(op_class);
developerb85e8752023-09-25 14:16:05 +0800261+ if (ch_width == CONF_OPER_CHWIDTH_320MHZ &&
262+ (center_freq1 == 6265) || center_freq1 == 6585 ||
263+ center_freq1 == 6905) {
264+ /* Bandwidth 320 MHz-2 */
265+ bw320_offset = 2;
266 }
267
268 ucv_object_add(ret, "sec_chan_offset", ucv_int64_new(sec_chan));
269 ucv_object_add(ret, "frequency", ucv_int64_new(bss->freq));
developerbddc9db2023-09-11 13:34:36 +0800270+ ucv_object_add(ret, "ch_width", ucv_int64_new(ch_width));
developerb85e8752023-09-25 14:16:05 +0800271+ ucv_object_add(ret, "bw320_offset", ucv_int64_new(bw320_offset));
developerbddc9db2023-09-11 13:34:36 +0800272 }
273
developerb85e8752023-09-25 14:16:05 +0800274 #ifdef CONFIG_MESH
developerbddc9db2023-09-11 13:34:36 +0800275--
2762.25.1
277