blob: 9d85c7cc4457612cd49230e2a05ac80cab46cafa [file] [log] [blame]
developerbd9fa1e2023-10-16 11:04:00 +08001From c4d8ad5eb5f4d9929e923fa2c0683a2e24d8ca0c Mon Sep 17 00:00:00 2001
developerbb6ddff2023-03-08 17:22:32 +08002From: Howard Hsu <howard-yh.hsu@mediatek.com>
3Date: Thu, 2 Feb 2023 21:20:31 +0800
developerbd9fa1e2023-10-16 11:04:00 +08004Subject: [PATCH 05/98] wifi: mt76: mt7996: add thermal protection support
developerbb6ddff2023-03-08 17:22:32 +08005
6This commit includes the following changes:
71. implement MTK thermal protection driver API
82. support Linux cooling device control
9
10Change-Id: I8fecc28f5b17ee50ae4644d1dd17d188dd694731
11---
12 mt76_connac_mcu.h | 1 +
developerbd9fa1e2023-10-16 11:04:00 +080013 mt7996/init.c | 103 +++++++++++++++++++++++++++++++++++++++++++++
developerbb6ddff2023-03-08 17:22:32 +080014 mt7996/main.c | 8 ++++
developerbd9fa1e2023-10-16 11:04:00 +080015 mt7996/mcu.c | 105 ++++++++++++++++++++++++++++++++++++++++++++++
developerbb6ddff2023-03-08 17:22:32 +080016 mt7996/mcu.h | 44 +++++++++++++++++++
17 mt7996/mt7996.h | 15 +++++++
developerbd9fa1e2023-10-16 11:04:00 +080018 6 files changed, 276 insertions(+)
developerbb6ddff2023-03-08 17:22:32 +080019
20diff --git a/mt76_connac_mcu.h b/mt76_connac_mcu.h
developerbd9fa1e2023-10-16 11:04:00 +080021index 6064973..99077f1 100644
developerbb6ddff2023-03-08 17:22:32 +080022--- a/mt76_connac_mcu.h
23+++ b/mt76_connac_mcu.h
developerbd9fa1e2023-10-16 11:04:00 +080024@@ -1021,6 +1021,7 @@ enum {
developerbb6ddff2023-03-08 17:22:32 +080025 MCU_UNI_EVENT_RDD_REPORT = 0x11,
developerbd9fa1e2023-10-16 11:04:00 +080026 MCU_UNI_EVENT_ROC = 0x27,
27 MCU_UNI_EVENT_TX_DONE = 0x2d,
developerbb6ddff2023-03-08 17:22:32 +080028+ MCU_UNI_EVENT_THERMAL = 0x35,
developerbd9fa1e2023-10-16 11:04:00 +080029 MCU_UNI_EVENT_NIC_CAPAB = 0x43,
developerbb6ddff2023-03-08 17:22:32 +080030 };
31
developerbb6ddff2023-03-08 17:22:32 +080032diff --git a/mt7996/init.c b/mt7996/init.c
developerbd9fa1e2023-10-16 11:04:00 +080033index d335b58..610d80e 100644
developerbb6ddff2023-03-08 17:22:32 +080034--- a/mt7996/init.c
35+++ b/mt7996/init.c
developer7af0f762023-05-22 15:16:16 +080036@@ -42,6 +42,98 @@ static const struct ieee80211_iface_combination if_comb[] = {
developerbb6ddff2023-03-08 17:22:32 +080037 }
38 };
39
40+static int
41+mt7996_thermal_get_max_throttle_state(struct thermal_cooling_device *cdev,
42+ unsigned long *state)
43+{
44+ *state = MT7996_CDEV_THROTTLE_MAX;
45+
46+ return 0;
47+}
48+
49+static int
50+mt7996_thermal_get_cur_throttle_state(struct thermal_cooling_device *cdev,
51+ unsigned long *state)
52+{
53+ struct mt7996_phy *phy = cdev->devdata;
54+
55+ *state = phy->cdev_state;
56+
57+ return 0;
58+}
59+
60+static int
61+mt7996_thermal_set_cur_throttle_state(struct thermal_cooling_device *cdev,
62+ unsigned long state)
63+{
64+ struct mt7996_phy *phy = cdev->devdata;
65+ u8 throttling = MT7996_THERMAL_THROTTLE_MAX - state;
66+ int ret;
67+
68+ if (state > MT7996_CDEV_THROTTLE_MAX) {
69+ dev_err(phy->dev->mt76.dev,
70+ "please specify a valid throttling state\n");
71+ return -EINVAL;
72+ }
73+
74+ if (state == phy->cdev_state)
75+ return 0;
76+
77+ /*
78+ * cooling_device convention: 0 = no cooling, more = more cooling
79+ * mcu convention: 1 = max cooling, more = less cooling
80+ */
81+ ret = mt7996_mcu_set_thermal_throttling(phy, throttling);
82+ if (ret)
83+ return ret;
84+
85+ phy->cdev_state = state;
86+
87+ return 0;
88+}
89+
90+static const struct thermal_cooling_device_ops mt7996_thermal_ops = {
91+ .get_max_state = mt7996_thermal_get_max_throttle_state,
92+ .get_cur_state = mt7996_thermal_get_cur_throttle_state,
93+ .set_cur_state = mt7996_thermal_set_cur_throttle_state,
94+};
95+
96+static void mt7996_unregister_thermal(struct mt7996_phy *phy)
97+{
98+ struct wiphy *wiphy = phy->mt76->hw->wiphy;
99+
100+ if (!phy->cdev)
101+ return;
102+
103+ sysfs_remove_link(&wiphy->dev.kobj, "cooling_device");
104+ thermal_cooling_device_unregister(phy->cdev);
105+}
106+
107+static int mt7996_thermal_init(struct mt7996_phy *phy)
108+{
109+ struct wiphy *wiphy = phy->mt76->hw->wiphy;
110+ struct thermal_cooling_device *cdev;
111+ const char *name;
112+
113+ name = devm_kasprintf(&wiphy->dev, GFP_KERNEL, "mt7996_%s",
114+ wiphy_name(wiphy));
115+
116+ cdev = thermal_cooling_device_register(name, phy, &mt7996_thermal_ops);
117+ if (!IS_ERR(cdev)) {
118+ if (sysfs_create_link(&wiphy->dev.kobj, &cdev->device.kobj,
119+ "cooling_device") < 0)
120+ thermal_cooling_device_unregister(cdev);
121+ else
122+ phy->cdev = cdev;
123+ }
124+
125+ /* initialize critical/maximum high temperature */
126+ phy->throttle_temp[MT7996_CRIT_TEMP_IDX] = MT7996_CRIT_TEMP;
127+ phy->throttle_temp[MT7996_MAX_TEMP_IDX] = MT7996_MAX_TEMP;
128+
129+ return 0;
130+}
131+
132 static void mt7996_led_set_config(struct led_classdev *led_cdev,
133 u8 delay_on, u8 delay_off)
134 {
developerbd9fa1e2023-10-16 11:04:00 +0800135@@ -419,6 +511,10 @@ static int mt7996_register_phy(struct mt7996_dev *dev, struct mt7996_phy *phy,
developerbb6ddff2023-03-08 17:22:32 +0800136 if (ret)
137 goto error;
138
139+ ret = mt7996_thermal_init(phy);
140+ if (ret)
141+ goto error;
142+
143 ret = mt7996_init_debugfs(phy);
144 if (ret)
145 goto error;
developerbd9fa1e2023-10-16 11:04:00 +0800146@@ -446,6 +542,8 @@ mt7996_unregister_phy(struct mt7996_phy *phy, enum mt76_band_id band)
developerbb6ddff2023-03-08 17:22:32 +0800147 if (!phy)
148 return;
149
150+ mt7996_unregister_thermal(phy);
151+
152 mphy = phy->dev->mt76.phys[band];
153 mt76_unregister_phy(mphy);
154 ieee80211_free_hw(mphy->hw);
developerbd9fa1e2023-10-16 11:04:00 +0800155@@ -1027,6 +1125,10 @@ int mt7996_register_device(struct mt7996_dev *dev)
developerbb6ddff2023-03-08 17:22:32 +0800156 if (ret)
157 return ret;
158
159+ ret = mt7996_thermal_init(&dev->phy);
160+ if (ret)
161+ return ret;
162+
163 ieee80211_queue_work(mt76_hw(dev), &dev->init_work);
164
165 ret = mt7996_register_phy(dev, mt7996_phy2(dev), MT_BAND1);
developerbd9fa1e2023-10-16 11:04:00 +0800166@@ -1050,6 +1152,7 @@ void mt7996_unregister_device(struct mt7996_dev *dev)
developerbb6ddff2023-03-08 17:22:32 +0800167 {
168 mt7996_unregister_phy(mt7996_phy3(dev), MT_BAND2);
169 mt7996_unregister_phy(mt7996_phy2(dev), MT_BAND1);
developerbb6ddff2023-03-08 17:22:32 +0800170+ mt7996_unregister_thermal(&dev->phy);
developer7af0f762023-05-22 15:16:16 +0800171 mt7996_coredump_unregister(dev);
developerbb6ddff2023-03-08 17:22:32 +0800172 mt76_unregister_device(&dev->mt76);
173 mt7996_mcu_exit(dev);
developerbb6ddff2023-03-08 17:22:32 +0800174diff --git a/mt7996/main.c b/mt7996/main.c
developerbd9fa1e2023-10-16 11:04:00 +0800175index ae4f0ce..280120b 100644
developerbb6ddff2023-03-08 17:22:32 +0800176--- a/mt7996/main.c
177+++ b/mt7996/main.c
developer7af0f762023-05-22 15:16:16 +0800178@@ -51,6 +51,14 @@ int mt7996_run(struct ieee80211_hw *hw)
developerbb6ddff2023-03-08 17:22:32 +0800179 if (ret)
180 goto out;
181
182+ ret = mt7996_mcu_set_thermal_throttling(phy, MT7996_THERMAL_THROTTLE_MAX);
183+ if (ret)
184+ goto out;
185+
developerbd9fa1e2023-10-16 11:04:00 +0800186+ ret = mt7996_mcu_set_thermal_protect(phy, true);
developerbb6ddff2023-03-08 17:22:32 +0800187+ if (ret)
188+ goto out;
189+
190 set_bit(MT76_STATE_RUNNING, &phy->mt76->state);
191
192 ieee80211_queue_delayed_work(hw, &phy->mt76->mac_work,
193diff --git a/mt7996/mcu.c b/mt7996/mcu.c
developerbd9fa1e2023-10-16 11:04:00 +0800194index 328bea9..5310f9b 100644
developerbb6ddff2023-03-08 17:22:32 +0800195--- a/mt7996/mcu.c
196+++ b/mt7996/mcu.c
developerbd9fa1e2023-10-16 11:04:00 +0800197@@ -449,6 +449,34 @@ mt7996_mcu_ie_countdown(struct mt7996_dev *dev, struct sk_buff *skb)
developerbb6ddff2023-03-08 17:22:32 +0800198 }
199 }
200
201+static void
202+mt7996_mcu_rx_thermal_notify(struct mt7996_dev *dev, struct sk_buff *skb)
203+{
204+#define THERMAL_NOTIFY_TAG 0x4
205+#define THERMAL_NOTIFY 0x2
206+ struct mt76_phy *mphy = &dev->mt76.phy;
207+ struct mt7996_mcu_thermal_notify *n;
208+ struct mt7996_phy *phy;
209+
210+ n = (struct mt7996_mcu_thermal_notify *)skb->data;
211+
212+ if (n->tag != THERMAL_NOTIFY_TAG)
213+ return;
214+
215+ if (n->event_id != THERMAL_NOTIFY)
216+ return;
217+
218+ if (n->band_idx > MT_BAND2)
219+ return;
220+
221+ mphy = dev->mt76.phys[n->band_idx];
222+ if (!mphy)
223+ return;
224+
225+ phy = (struct mt7996_phy *)mphy->priv;
226+ phy->throttle_state = n->duty_percent;
227+}
228+
229 static void
230 mt7996_mcu_rx_ext_event(struct mt7996_dev *dev, struct sk_buff *skb)
231 {
developerbd9fa1e2023-10-16 11:04:00 +0800232@@ -493,6 +521,9 @@ mt7996_mcu_uni_rx_unsolicited_event(struct mt7996_dev *dev, struct sk_buff *skb)
developerbb6ddff2023-03-08 17:22:32 +0800233 case MCU_UNI_EVENT_RDD_REPORT:
234 mt7996_mcu_rx_radar_detected(dev, skb);
235 break;
236+ case MCU_UNI_EVENT_THERMAL:
237+ mt7996_mcu_rx_thermal_notify(dev, skb);
238+ break;
239 default:
240 break;
241 }
developerbd9fa1e2023-10-16 11:04:00 +0800242@@ -3450,6 +3481,80 @@ out:
developerbb6ddff2023-03-08 17:22:32 +0800243 return 0;
244 }
245
246+
247+int mt7996_mcu_set_thermal_throttling(struct mt7996_phy *phy, u8 state)
248+{
249+ struct {
250+ u8 _rsv[4];
251+
252+ __le16 tag;
253+ __le16 len;
254+
255+ struct mt7996_mcu_thermal_ctrl ctrl;
256+ } __packed req = {
257+ .tag = cpu_to_le16(UNI_CMD_THERMAL_PROTECT_DUTY_CONFIG),
258+ .len = cpu_to_le16(sizeof(req) - 4),
259+ .ctrl = {
260+ .band_idx = phy->mt76->band_idx,
261+ },
262+ };
263+ int level, ret;
264+
265+ /* set duty cycle and level */
266+ for (level = 0; level < 4; level++) {
267+ req.ctrl.duty.duty_level = level;
268+ req.ctrl.duty.duty_cycle = state;
269+ state /= 2;
270+
271+ ret = mt76_mcu_send_msg(&phy->dev->mt76, MCU_WM_UNI_CMD(THERMAL),
272+ &req, sizeof(req), false);
273+ if (ret)
274+ return ret;
275+ }
276+
277+ return 0;
278+}
279+
developerbd9fa1e2023-10-16 11:04:00 +0800280+int mt7996_mcu_set_thermal_protect(struct mt7996_phy *phy, bool enable)
developerbb6ddff2023-03-08 17:22:32 +0800281+{
282+#define SUSTAIN_PERIOD 10
283+ struct {
284+ u8 _rsv[4];
285+
286+ __le16 tag;
287+ __le16 len;
288+
289+ struct mt7996_mcu_thermal_ctrl ctrl;
290+ struct mt7996_mcu_thermal_enable enable;
291+ } __packed req = {
292+ .len = cpu_to_le16(sizeof(req) - 4 - sizeof(req.enable)),
293+ .ctrl = {
294+ .band_idx = phy->mt76->band_idx,
295+ .type.protect_type = 1,
296+ .type.trigger_type = 1,
297+ },
298+ };
299+ int ret;
300+
301+ req.tag = cpu_to_le16(UNI_CMD_THERMAL_PROTECT_DISABLE);
302+
303+ ret = mt76_mcu_send_msg(&phy->dev->mt76, MCU_WM_UNI_CMD(THERMAL),
304+ &req, sizeof(req) - sizeof(req.enable), false);
developerbd9fa1e2023-10-16 11:04:00 +0800305+ if (ret || !enable)
developerbb6ddff2023-03-08 17:22:32 +0800306+ return ret;
307+
308+ /* set high-temperature trigger threshold */
309+ req.tag = cpu_to_le16(UNI_CMD_THERMAL_PROTECT_ENABLE);
developerbd9fa1e2023-10-16 11:04:00 +0800310+ req.enable.restore_temp = cpu_to_le32(phy->throttle_temp[0]);
developerbb6ddff2023-03-08 17:22:32 +0800311+ req.enable.trigger_temp = cpu_to_le32(phy->throttle_temp[1]);
312+ req.enable.sustain_time = cpu_to_le16(SUSTAIN_PERIOD);
313+
314+ req.len = cpu_to_le16(sizeof(req) - 4);
315+
316+ return mt76_mcu_send_msg(&phy->dev->mt76, MCU_WM_UNI_CMD(THERMAL),
317+ &req, sizeof(req), false);
318+}
319+
320 int mt7996_mcu_set_ser(struct mt7996_dev *dev, u8 action, u8 val, u8 band)
321 {
322 struct {
323diff --git a/mt7996/mcu.h b/mt7996/mcu.h
developerbd9fa1e2023-10-16 11:04:00 +0800324index e4b3122..f1528df 100644
developerbb6ddff2023-03-08 17:22:32 +0800325--- a/mt7996/mcu.h
326+++ b/mt7996/mcu.h
327@@ -30,6 +30,28 @@ struct mt7996_mcu_uni_event {
328 __le32 status; /* 0: success, others: fail */
329 } __packed;
330
331+struct mt7996_mcu_thermal_ctrl {
332+ u8 ctrl_id;
333+ u8 band_idx;
334+ union {
335+ struct {
336+ u8 protect_type; /* 1: duty admit, 2: radio off */
337+ u8 trigger_type; /* 0: low, 1: high */
338+ } __packed type;
339+ struct {
340+ u8 duty_level; /* level 0~3 */
341+ u8 duty_cycle;
342+ } __packed duty;
343+ };
344+} __packed;
345+
346+struct mt7996_mcu_thermal_enable {
347+ __le32 trigger_temp;
348+ __le32 restore_temp;
349+ __le16 sustain_time;
350+ u8 rsv[2];
351+} __packed;
352+
353 struct mt7996_mcu_csa_notify {
354 struct mt7996_mcu_rxd rxd;
355
356@@ -153,6 +175,22 @@ struct mt7996_mcu_mib {
357 __le64 data;
358 } __packed;
359
360+struct mt7996_mcu_thermal_notify {
361+ struct mt7996_mcu_rxd rxd;
362+
363+ u8 __rsv1[4];
364+
365+ __le16 tag;
366+ __le16 len;
367+
368+ u8 event_id;
369+ u8 band_idx;
370+ u8 level_idx;
371+ u8 duty_percent;
372+ __le32 restore_temp;
373+ u8 __rsv2[4];
374+} __packed;
375+
376 enum mt7996_chan_mib_offs {
377 UNI_MIB_OBSS_AIRTIME = 26,
378 UNI_MIB_NON_WIFI_TIME = 27,
developerbd9fa1e2023-10-16 11:04:00 +0800379@@ -656,6 +694,12 @@ enum{
developerbb6ddff2023-03-08 17:22:32 +0800380 UNI_CMD_SR_SET_SIGA = 0xd0,
381 };
382
383+enum {
384+ UNI_CMD_THERMAL_PROTECT_ENABLE = 0x6,
385+ UNI_CMD_THERMAL_PROTECT_DISABLE,
386+ UNI_CMD_THERMAL_PROTECT_DUTY_CONFIG,
387+};
388+
389 enum {
390 UNI_CMD_ACCESS_REG_BASIC = 0x0,
391 UNI_CMD_ACCESS_RF_REG_BASIC,
392diff --git a/mt7996/mt7996.h b/mt7996/mt7996.h
developerbd9fa1e2023-10-16 11:04:00 +0800393index ef84173..e1972e9 100644
developerbb6ddff2023-03-08 17:22:32 +0800394--- a/mt7996/mt7996.h
395+++ b/mt7996/mt7996.h
developerbd9fa1e2023-10-16 11:04:00 +0800396@@ -50,6 +50,13 @@
developer7af0f762023-05-22 15:16:16 +0800397 #define MT7996_BASIC_RATES_TBL 11
398 #define MT7996_BEACON_RATES_TBL 25
developerbb6ddff2023-03-08 17:22:32 +0800399
400+#define MT7996_THERMAL_THROTTLE_MAX 100
401+#define MT7996_CDEV_THROTTLE_MAX 99
402+#define MT7996_CRIT_TEMP_IDX 0
403+#define MT7996_MAX_TEMP_IDX 1
404+#define MT7996_CRIT_TEMP 110
405+#define MT7996_MAX_TEMP 120
406+
developerbd9fa1e2023-10-16 11:04:00 +0800407 #define MT7996_RRO_MAX_SESSION 1024
408 #define MT7996_RRO_WINDOW_MAX_LEN 1024
409 #define MT7996_RRO_ADDR_ELEM_LEN 128
410@@ -191,6 +198,11 @@ struct mt7996_phy {
developerbb6ddff2023-03-08 17:22:32 +0800411
412 struct ieee80211_vif *monitor_vif;
413
414+ struct thermal_cooling_device *cdev;
415+ u8 cdev_state;
416+ u8 throttle_state;
417+ u32 throttle_temp[2]; /* 0: critical high, 1: maximum */
418+
419 u32 rxfilter;
420 u64 omac_mask;
421
developerbd9fa1e2023-10-16 11:04:00 +0800422@@ -445,6 +457,9 @@ int mt7996_mcu_set_radio_en(struct mt7996_phy *phy, bool enable);
developerbb6ddff2023-03-08 17:22:32 +0800423 int mt7996_mcu_set_rts_thresh(struct mt7996_phy *phy, u32 val);
developer692ed9b2023-06-19 12:03:50 +0800424 int mt7996_mcu_set_timing(struct mt7996_phy *phy, struct ieee80211_vif *vif);
developerbb6ddff2023-03-08 17:22:32 +0800425 int mt7996_mcu_get_chan_mib_info(struct mt7996_phy *phy, bool chan_switch);
426+int mt7996_mcu_get_temperature(struct mt7996_phy *phy);
427+int mt7996_mcu_set_thermal_throttling(struct mt7996_phy *phy, u8 state);
developerbd9fa1e2023-10-16 11:04:00 +0800428+int mt7996_mcu_set_thermal_protect(struct mt7996_phy *phy, bool enable);
developerbb6ddff2023-03-08 17:22:32 +0800429 int mt7996_mcu_rdd_cmd(struct mt7996_dev *dev, int cmd, u8 index,
430 u8 rx_sel, u8 val);
431 int mt7996_mcu_rdd_background_enable(struct mt7996_phy *phy,
432--
developer3e11ee32023-09-27 12:24:47 +08004332.18.0
developerbb6ddff2023-03-08 17:22:32 +0800434