162306a36Sopenharmony_ci// SPDX-License-Identifier: ISC
262306a36Sopenharmony_ci/* Copyright (C) 2020 MediaTek Inc. */
362306a36Sopenharmony_ci
462306a36Sopenharmony_ci#include <linux/etherdevice.h>
562306a36Sopenharmony_ci#include <linux/hwmon.h>
662306a36Sopenharmony_ci#include <linux/hwmon-sysfs.h>
762306a36Sopenharmony_ci#include <linux/thermal.h>
862306a36Sopenharmony_ci#include <linux/firmware.h>
962306a36Sopenharmony_ci#include "mt7921.h"
1062306a36Sopenharmony_ci#include "../mt76_connac2_mac.h"
1162306a36Sopenharmony_ci#include "mcu.h"
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_cistatic ssize_t mt7921_thermal_temp_show(struct device *dev,
1462306a36Sopenharmony_ci					struct device_attribute *attr,
1562306a36Sopenharmony_ci					char *buf)
1662306a36Sopenharmony_ci{
1762306a36Sopenharmony_ci	switch (to_sensor_dev_attr(attr)->index) {
1862306a36Sopenharmony_ci	case 0: {
1962306a36Sopenharmony_ci		struct mt792x_phy *phy = dev_get_drvdata(dev);
2062306a36Sopenharmony_ci		struct mt792x_dev *mdev = phy->dev;
2162306a36Sopenharmony_ci		int temperature;
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_ci		mt792x_mutex_acquire(mdev);
2462306a36Sopenharmony_ci		temperature = mt7921_mcu_get_temperature(phy);
2562306a36Sopenharmony_ci		mt792x_mutex_release(mdev);
2662306a36Sopenharmony_ci
2762306a36Sopenharmony_ci		if (temperature < 0)
2862306a36Sopenharmony_ci			return temperature;
2962306a36Sopenharmony_ci		/* display in millidegree Celsius */
3062306a36Sopenharmony_ci		return sprintf(buf, "%u\n", temperature * 1000);
3162306a36Sopenharmony_ci	}
3262306a36Sopenharmony_ci	default:
3362306a36Sopenharmony_ci		return -EINVAL;
3462306a36Sopenharmony_ci	}
3562306a36Sopenharmony_ci}
3662306a36Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RO(temp1_input, mt7921_thermal_temp, 0);
3762306a36Sopenharmony_ci
3862306a36Sopenharmony_cistatic struct attribute *mt7921_hwmon_attrs[] = {
3962306a36Sopenharmony_ci	&sensor_dev_attr_temp1_input.dev_attr.attr,
4062306a36Sopenharmony_ci	NULL,
4162306a36Sopenharmony_ci};
4262306a36Sopenharmony_ciATTRIBUTE_GROUPS(mt7921_hwmon);
4362306a36Sopenharmony_ci
4462306a36Sopenharmony_cistatic int mt7921_thermal_init(struct mt792x_phy *phy)
4562306a36Sopenharmony_ci{
4662306a36Sopenharmony_ci	struct wiphy *wiphy = phy->mt76->hw->wiphy;
4762306a36Sopenharmony_ci	struct device *hwmon;
4862306a36Sopenharmony_ci	const char *name;
4962306a36Sopenharmony_ci
5062306a36Sopenharmony_ci	if (!IS_REACHABLE(CONFIG_HWMON))
5162306a36Sopenharmony_ci		return 0;
5262306a36Sopenharmony_ci
5362306a36Sopenharmony_ci	name = devm_kasprintf(&wiphy->dev, GFP_KERNEL, "mt7921_%s",
5462306a36Sopenharmony_ci			      wiphy_name(wiphy));
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ci	hwmon = devm_hwmon_device_register_with_groups(&wiphy->dev, name, phy,
5762306a36Sopenharmony_ci						       mt7921_hwmon_groups);
5862306a36Sopenharmony_ci	if (IS_ERR(hwmon))
5962306a36Sopenharmony_ci		return PTR_ERR(hwmon);
6062306a36Sopenharmony_ci
6162306a36Sopenharmony_ci	return 0;
6262306a36Sopenharmony_ci}
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_cistatic void
6562306a36Sopenharmony_cimt7921_regd_notifier(struct wiphy *wiphy,
6662306a36Sopenharmony_ci		     struct regulatory_request *request)
6762306a36Sopenharmony_ci{
6862306a36Sopenharmony_ci	struct ieee80211_hw *hw = wiphy_to_ieee80211_hw(wiphy);
6962306a36Sopenharmony_ci	struct mt792x_dev *dev = mt792x_hw_dev(hw);
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	memcpy(dev->mt76.alpha2, request->alpha2, sizeof(dev->mt76.alpha2));
7262306a36Sopenharmony_ci	dev->mt76.region = request->dfs_region;
7362306a36Sopenharmony_ci	dev->country_ie_env = request->country_ie_env;
7462306a36Sopenharmony_ci
7562306a36Sopenharmony_ci	mt792x_mutex_acquire(dev);
7662306a36Sopenharmony_ci	mt7921_mcu_set_clc(dev, request->alpha2, request->country_ie_env);
7762306a36Sopenharmony_ci	mt76_connac_mcu_set_channel_domain(hw->priv);
7862306a36Sopenharmony_ci	mt7921_set_tx_sar_pwr(hw, NULL);
7962306a36Sopenharmony_ci	mt792x_mutex_release(dev);
8062306a36Sopenharmony_ci}
8162306a36Sopenharmony_ci
8262306a36Sopenharmony_ciint mt7921_mac_init(struct mt792x_dev *dev)
8362306a36Sopenharmony_ci{
8462306a36Sopenharmony_ci	int i;
8562306a36Sopenharmony_ci
8662306a36Sopenharmony_ci	mt76_rmw_field(dev, MT_MDP_DCR1, MT_MDP_DCR1_MAX_RX_LEN, 1536);
8762306a36Sopenharmony_ci	/* enable hardware de-agg */
8862306a36Sopenharmony_ci	mt76_set(dev, MT_MDP_DCR0, MT_MDP_DCR0_DAMSDU_EN);
8962306a36Sopenharmony_ci	/* enable hardware rx header translation */
9062306a36Sopenharmony_ci	mt76_set(dev, MT_MDP_DCR0, MT_MDP_DCR0_RX_HDR_TRANS_EN);
9162306a36Sopenharmony_ci
9262306a36Sopenharmony_ci	for (i = 0; i < MT792x_WTBL_SIZE; i++)
9362306a36Sopenharmony_ci		mt7921_mac_wtbl_update(dev, i,
9462306a36Sopenharmony_ci				       MT_WTBL_UPDATE_ADM_COUNT_CLEAR);
9562306a36Sopenharmony_ci	for (i = 0; i < 2; i++)
9662306a36Sopenharmony_ci		mt792x_mac_init_band(dev, i);
9762306a36Sopenharmony_ci
9862306a36Sopenharmony_ci	return mt76_connac_mcu_set_rts_thresh(&dev->mt76, 0x92b, 0);
9962306a36Sopenharmony_ci}
10062306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mt7921_mac_init);
10162306a36Sopenharmony_ci
10262306a36Sopenharmony_cistatic int __mt7921_init_hardware(struct mt792x_dev *dev)
10362306a36Sopenharmony_ci{
10462306a36Sopenharmony_ci	int ret;
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	/* force firmware operation mode into normal state,
10762306a36Sopenharmony_ci	 * which should be set before firmware download stage.
10862306a36Sopenharmony_ci	 */
10962306a36Sopenharmony_ci	mt76_wr(dev, MT_SWDEF_MODE, MT_SWDEF_NORMAL_MODE);
11062306a36Sopenharmony_ci	ret = mt792x_mcu_init(dev);
11162306a36Sopenharmony_ci	if (ret)
11262306a36Sopenharmony_ci		goto out;
11362306a36Sopenharmony_ci
11462306a36Sopenharmony_ci	mt76_eeprom_override(&dev->mphy);
11562306a36Sopenharmony_ci
11662306a36Sopenharmony_ci	ret = mt7921_mcu_set_eeprom(dev);
11762306a36Sopenharmony_ci	if (ret)
11862306a36Sopenharmony_ci		goto out;
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci	ret = mt7921_mac_init(dev);
12162306a36Sopenharmony_ciout:
12262306a36Sopenharmony_ci	return ret;
12362306a36Sopenharmony_ci}
12462306a36Sopenharmony_ci
12562306a36Sopenharmony_cistatic int mt7921_init_hardware(struct mt792x_dev *dev)
12662306a36Sopenharmony_ci{
12762306a36Sopenharmony_ci	int ret, i;
12862306a36Sopenharmony_ci
12962306a36Sopenharmony_ci	set_bit(MT76_STATE_INITIALIZED, &dev->mphy.state);
13062306a36Sopenharmony_ci
13162306a36Sopenharmony_ci	for (i = 0; i < MT792x_MCU_INIT_RETRY_COUNT; i++) {
13262306a36Sopenharmony_ci		ret = __mt7921_init_hardware(dev);
13362306a36Sopenharmony_ci		if (!ret)
13462306a36Sopenharmony_ci			break;
13562306a36Sopenharmony_ci
13662306a36Sopenharmony_ci		mt792x_init_reset(dev);
13762306a36Sopenharmony_ci	}
13862306a36Sopenharmony_ci
13962306a36Sopenharmony_ci	if (i == MT792x_MCU_INIT_RETRY_COUNT) {
14062306a36Sopenharmony_ci		dev_err(dev->mt76.dev, "hardware init failed\n");
14162306a36Sopenharmony_ci		return ret;
14262306a36Sopenharmony_ci	}
14362306a36Sopenharmony_ci
14462306a36Sopenharmony_ci	return 0;
14562306a36Sopenharmony_ci}
14662306a36Sopenharmony_ci
14762306a36Sopenharmony_cistatic void mt7921_init_work(struct work_struct *work)
14862306a36Sopenharmony_ci{
14962306a36Sopenharmony_ci	struct mt792x_dev *dev = container_of(work, struct mt792x_dev,
15062306a36Sopenharmony_ci					      init_work);
15162306a36Sopenharmony_ci	int ret;
15262306a36Sopenharmony_ci
15362306a36Sopenharmony_ci	ret = mt7921_init_hardware(dev);
15462306a36Sopenharmony_ci	if (ret)
15562306a36Sopenharmony_ci		return;
15662306a36Sopenharmony_ci
15762306a36Sopenharmony_ci	mt76_set_stream_caps(&dev->mphy, true);
15862306a36Sopenharmony_ci	mt7921_set_stream_he_caps(&dev->phy);
15962306a36Sopenharmony_ci
16062306a36Sopenharmony_ci	ret = mt76_register_device(&dev->mt76, true, mt76_rates,
16162306a36Sopenharmony_ci				   ARRAY_SIZE(mt76_rates));
16262306a36Sopenharmony_ci	if (ret) {
16362306a36Sopenharmony_ci		dev_err(dev->mt76.dev, "register device failed\n");
16462306a36Sopenharmony_ci		return;
16562306a36Sopenharmony_ci	}
16662306a36Sopenharmony_ci
16762306a36Sopenharmony_ci	ret = mt7921_init_debugfs(dev);
16862306a36Sopenharmony_ci	if (ret) {
16962306a36Sopenharmony_ci		dev_err(dev->mt76.dev, "register debugfs failed\n");
17062306a36Sopenharmony_ci		return;
17162306a36Sopenharmony_ci	}
17262306a36Sopenharmony_ci
17362306a36Sopenharmony_ci	ret = mt7921_thermal_init(&dev->phy);
17462306a36Sopenharmony_ci	if (ret) {
17562306a36Sopenharmony_ci		dev_err(dev->mt76.dev, "thermal init failed\n");
17662306a36Sopenharmony_ci		return;
17762306a36Sopenharmony_ci	}
17862306a36Sopenharmony_ci
17962306a36Sopenharmony_ci	/* we support chip reset now */
18062306a36Sopenharmony_ci	dev->hw_init_done = true;
18162306a36Sopenharmony_ci
18262306a36Sopenharmony_ci	mt76_connac_mcu_set_deep_sleep(&dev->mt76, dev->pm.ds_enable);
18362306a36Sopenharmony_ci}
18462306a36Sopenharmony_ci
18562306a36Sopenharmony_ciint mt7921_register_device(struct mt792x_dev *dev)
18662306a36Sopenharmony_ci{
18762306a36Sopenharmony_ci	struct ieee80211_hw *hw = mt76_hw(dev);
18862306a36Sopenharmony_ci	int ret;
18962306a36Sopenharmony_ci
19062306a36Sopenharmony_ci	dev->phy.dev = dev;
19162306a36Sopenharmony_ci	dev->phy.mt76 = &dev->mt76.phy;
19262306a36Sopenharmony_ci	dev->mt76.phy.priv = &dev->phy;
19362306a36Sopenharmony_ci	dev->mt76.tx_worker.fn = mt792x_tx_worker;
19462306a36Sopenharmony_ci
19562306a36Sopenharmony_ci	INIT_DELAYED_WORK(&dev->pm.ps_work, mt792x_pm_power_save_work);
19662306a36Sopenharmony_ci	INIT_WORK(&dev->pm.wake_work, mt792x_pm_wake_work);
19762306a36Sopenharmony_ci	spin_lock_init(&dev->pm.wake.lock);
19862306a36Sopenharmony_ci	mutex_init(&dev->pm.mutex);
19962306a36Sopenharmony_ci	init_waitqueue_head(&dev->pm.wait);
20062306a36Sopenharmony_ci	if (mt76_is_sdio(&dev->mt76))
20162306a36Sopenharmony_ci		init_waitqueue_head(&dev->mt76.sdio.wait);
20262306a36Sopenharmony_ci	spin_lock_init(&dev->pm.txq_lock);
20362306a36Sopenharmony_ci	INIT_DELAYED_WORK(&dev->mphy.mac_work, mt792x_mac_work);
20462306a36Sopenharmony_ci	INIT_DELAYED_WORK(&dev->phy.scan_work, mt7921_scan_work);
20562306a36Sopenharmony_ci	INIT_DELAYED_WORK(&dev->coredump.work, mt7921_coredump_work);
20662306a36Sopenharmony_ci#if IS_ENABLED(CONFIG_IPV6)
20762306a36Sopenharmony_ci	INIT_WORK(&dev->ipv6_ns_work, mt7921_set_ipv6_ns_work);
20862306a36Sopenharmony_ci	skb_queue_head_init(&dev->ipv6_ns_list);
20962306a36Sopenharmony_ci#endif
21062306a36Sopenharmony_ci	skb_queue_head_init(&dev->phy.scan_event_list);
21162306a36Sopenharmony_ci	skb_queue_head_init(&dev->coredump.msg_list);
21262306a36Sopenharmony_ci
21362306a36Sopenharmony_ci	INIT_WORK(&dev->reset_work, mt7921_mac_reset_work);
21462306a36Sopenharmony_ci	INIT_WORK(&dev->init_work, mt7921_init_work);
21562306a36Sopenharmony_ci
21662306a36Sopenharmony_ci	INIT_WORK(&dev->phy.roc_work, mt7921_roc_work);
21762306a36Sopenharmony_ci	timer_setup(&dev->phy.roc_timer, mt792x_roc_timer, 0);
21862306a36Sopenharmony_ci	init_waitqueue_head(&dev->phy.roc_wait);
21962306a36Sopenharmony_ci
22062306a36Sopenharmony_ci	dev->pm.idle_timeout = MT792x_PM_TIMEOUT;
22162306a36Sopenharmony_ci	dev->pm.stats.last_wake_event = jiffies;
22262306a36Sopenharmony_ci	dev->pm.stats.last_doze_event = jiffies;
22362306a36Sopenharmony_ci	if (!mt76_is_usb(&dev->mt76)) {
22462306a36Sopenharmony_ci		dev->pm.enable_user = true;
22562306a36Sopenharmony_ci		dev->pm.enable = true;
22662306a36Sopenharmony_ci		dev->pm.ds_enable_user = true;
22762306a36Sopenharmony_ci		dev->pm.ds_enable = true;
22862306a36Sopenharmony_ci	}
22962306a36Sopenharmony_ci
23062306a36Sopenharmony_ci	if (!mt76_is_mmio(&dev->mt76))
23162306a36Sopenharmony_ci		hw->extra_tx_headroom += MT_SDIO_TXD_SIZE + MT_SDIO_HDR_SIZE;
23262306a36Sopenharmony_ci
23362306a36Sopenharmony_ci	mt792x_init_acpi_sar(dev);
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci	ret = mt792x_init_wcid(dev);
23662306a36Sopenharmony_ci	if (ret)
23762306a36Sopenharmony_ci		return ret;
23862306a36Sopenharmony_ci
23962306a36Sopenharmony_ci	ret = mt792x_init_wiphy(hw);
24062306a36Sopenharmony_ci	if (ret)
24162306a36Sopenharmony_ci		return ret;
24262306a36Sopenharmony_ci
24362306a36Sopenharmony_ci	hw->wiphy->reg_notifier = mt7921_regd_notifier;
24462306a36Sopenharmony_ci	dev->mphy.sband_2g.sband.ht_cap.cap |=
24562306a36Sopenharmony_ci			IEEE80211_HT_CAP_LDPC_CODING |
24662306a36Sopenharmony_ci			IEEE80211_HT_CAP_MAX_AMSDU;
24762306a36Sopenharmony_ci	dev->mphy.sband_5g.sband.ht_cap.cap |=
24862306a36Sopenharmony_ci			IEEE80211_HT_CAP_LDPC_CODING |
24962306a36Sopenharmony_ci			IEEE80211_HT_CAP_MAX_AMSDU;
25062306a36Sopenharmony_ci	dev->mphy.sband_5g.sband.vht_cap.cap |=
25162306a36Sopenharmony_ci			IEEE80211_VHT_CAP_MAX_MPDU_LENGTH_11454 |
25262306a36Sopenharmony_ci			IEEE80211_VHT_CAP_MAX_A_MPDU_LENGTH_EXPONENT_MASK |
25362306a36Sopenharmony_ci			IEEE80211_VHT_CAP_SU_BEAMFORMEE_CAPABLE |
25462306a36Sopenharmony_ci			IEEE80211_VHT_CAP_MU_BEAMFORMEE_CAPABLE |
25562306a36Sopenharmony_ci			(3 << IEEE80211_VHT_CAP_BEAMFORMEE_STS_SHIFT);
25662306a36Sopenharmony_ci	if (is_mt7922(&dev->mt76))
25762306a36Sopenharmony_ci		dev->mphy.sband_5g.sband.vht_cap.cap |=
25862306a36Sopenharmony_ci			IEEE80211_VHT_CAP_SUPP_CHAN_WIDTH_160MHZ |
25962306a36Sopenharmony_ci			IEEE80211_VHT_CAP_SHORT_GI_160;
26062306a36Sopenharmony_ci
26162306a36Sopenharmony_ci	dev->mphy.hw->wiphy->available_antennas_rx = dev->mphy.chainmask;
26262306a36Sopenharmony_ci	dev->mphy.hw->wiphy->available_antennas_tx = dev->mphy.chainmask;
26362306a36Sopenharmony_ci
26462306a36Sopenharmony_ci	queue_work(system_wq, &dev->init_work);
26562306a36Sopenharmony_ci
26662306a36Sopenharmony_ci	return 0;
26762306a36Sopenharmony_ci}
26862306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mt7921_register_device);
269