cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

debugfs.c (10999B)


      1// SPDX-License-Identifier: ISC
      2/* Copyright (C) 2020 MediaTek Inc. */
      3
      4#include "mt7921.h"
      5#include "eeprom.h"
      6
      7static int
      8mt7921_reg_set(void *data, u64 val)
      9{
     10	struct mt7921_dev *dev = data;
     11
     12	mt7921_mutex_acquire(dev);
     13	mt76_wr(dev, dev->mt76.debugfs_reg, val);
     14	mt7921_mutex_release(dev);
     15
     16	return 0;
     17}
     18
     19static int
     20mt7921_reg_get(void *data, u64 *val)
     21{
     22	struct mt7921_dev *dev = data;
     23
     24	mt7921_mutex_acquire(dev);
     25	*val = mt76_rr(dev, dev->mt76.debugfs_reg);
     26	mt7921_mutex_release(dev);
     27
     28	return 0;
     29}
     30
     31DEFINE_DEBUGFS_ATTRIBUTE(fops_regval, mt7921_reg_get, mt7921_reg_set,
     32			 "0x%08llx\n");
     33static int
     34mt7921_fw_debug_set(void *data, u64 val)
     35{
     36	struct mt7921_dev *dev = data;
     37
     38	mt7921_mutex_acquire(dev);
     39
     40	dev->fw_debug = (u8)val;
     41	mt7921_mcu_fw_log_2_host(dev, dev->fw_debug);
     42
     43	mt7921_mutex_release(dev);
     44
     45	return 0;
     46}
     47
     48static int
     49mt7921_fw_debug_get(void *data, u64 *val)
     50{
     51	struct mt7921_dev *dev = data;
     52
     53	*val = dev->fw_debug;
     54
     55	return 0;
     56}
     57
     58DEFINE_DEBUGFS_ATTRIBUTE(fops_fw_debug, mt7921_fw_debug_get,
     59			 mt7921_fw_debug_set, "%lld\n");
     60
     61static void
     62mt7921_ampdu_stat_read_phy(struct mt7921_phy *phy,
     63			   struct seq_file *file)
     64{
     65	struct mt7921_dev *dev = file->private;
     66	int bound[15], range[4], i;
     67
     68	if (!phy)
     69		return;
     70
     71	mt7921_mac_update_mib_stats(phy);
     72
     73	/* Tx ampdu stat */
     74	for (i = 0; i < ARRAY_SIZE(range); i++)
     75		range[i] = mt76_rr(dev, MT_MIB_ARNG(0, i));
     76
     77	for (i = 0; i < ARRAY_SIZE(bound); i++)
     78		bound[i] = MT_MIB_ARNCR_RANGE(range[i / 4], i % 4) + 1;
     79
     80	seq_printf(file, "\nPhy0\n");
     81
     82	seq_printf(file, "Length: %8d | ", bound[0]);
     83	for (i = 0; i < ARRAY_SIZE(bound) - 1; i++)
     84		seq_printf(file, "%3d  %3d | ", bound[i] + 1, bound[i + 1]);
     85
     86	seq_puts(file, "\nCount:  ");
     87	for (i = 0; i < ARRAY_SIZE(bound); i++)
     88		seq_printf(file, "%8d | ", dev->mt76.aggr_stats[i]);
     89	seq_puts(file, "\n");
     90
     91	seq_printf(file, "BA miss count: %d\n", phy->mib.ba_miss_cnt);
     92}
     93
     94static int
     95mt7921_tx_stats_show(struct seq_file *file, void *data)
     96{
     97	struct mt7921_dev *dev = file->private;
     98	struct mt7921_phy *phy = &dev->phy;
     99	struct mib_stats *mib = &phy->mib;
    100	int i;
    101
    102	mt7921_mutex_acquire(dev);
    103
    104	mt7921_ampdu_stat_read_phy(phy, file);
    105
    106	seq_puts(file, "Tx MSDU stat:\n");
    107	for (i = 0; i < ARRAY_SIZE(mib->tx_amsdu); i++) {
    108		seq_printf(file, "AMSDU pack count of %d MSDU in TXD: %8d ",
    109			   i + 1, mib->tx_amsdu[i]);
    110		if (mib->tx_amsdu_cnt)
    111			seq_printf(file, "(%3d%%)\n",
    112				   mib->tx_amsdu[i] * 100 / mib->tx_amsdu_cnt);
    113		else
    114			seq_puts(file, "\n");
    115	}
    116
    117	mt7921_mutex_release(dev);
    118
    119	return 0;
    120}
    121
    122DEFINE_SHOW_ATTRIBUTE(mt7921_tx_stats);
    123
    124static int
    125mt7921_queues_acq(struct seq_file *s, void *data)
    126{
    127	struct mt7921_dev *dev = dev_get_drvdata(s->private);
    128	int i;
    129
    130	mt7921_mutex_acquire(dev);
    131
    132	for (i = 0; i < 4; i++) {
    133		u32 ctrl, val, qlen = 0;
    134		int j;
    135
    136		val = mt76_rr(dev, MT_PLE_AC_QEMPTY(i));
    137		ctrl = BIT(31) | BIT(11) | (i << 24);
    138
    139		for (j = 0; j < 32; j++) {
    140			if (val & BIT(j))
    141				continue;
    142
    143			mt76_wr(dev, MT_PLE_FL_Q0_CTRL, ctrl | j);
    144			qlen += mt76_get_field(dev, MT_PLE_FL_Q3_CTRL,
    145					       GENMASK(11, 0));
    146		}
    147		seq_printf(s, "AC%d: queued=%d\n", i, qlen);
    148	}
    149
    150	mt7921_mutex_release(dev);
    151
    152	return 0;
    153}
    154
    155static int
    156mt7921_queues_read(struct seq_file *s, void *data)
    157{
    158	struct mt7921_dev *dev = dev_get_drvdata(s->private);
    159	struct {
    160		struct mt76_queue *q;
    161		char *queue;
    162	} queue_map[] = {
    163		{ dev->mphy.q_tx[MT_TXQ_BE],	 "WFDMA0" },
    164		{ dev->mt76.q_mcu[MT_MCUQ_WM],	 "MCUWM"  },
    165		{ dev->mt76.q_mcu[MT_MCUQ_FWDL], "MCUFWQ" },
    166	};
    167	int i;
    168
    169	for (i = 0; i < ARRAY_SIZE(queue_map); i++) {
    170		struct mt76_queue *q = queue_map[i].q;
    171
    172		if (!q)
    173			continue;
    174
    175		seq_printf(s,
    176			   "%s:	queued=%d head=%d tail=%d\n",
    177			   queue_map[i].queue, q->queued, q->head,
    178			   q->tail);
    179	}
    180
    181	return 0;
    182}
    183
    184static void
    185mt7921_seq_puts_array(struct seq_file *file, const char *str,
    186		      s8 *val, int len)
    187{
    188	int i;
    189
    190	seq_printf(file, "%-16s:", str);
    191	for (i = 0; i < len; i++)
    192		if (val[i] == 127)
    193			seq_printf(file, " %6s", "N.A");
    194		else
    195			seq_printf(file, " %6d", val[i]);
    196	seq_puts(file, "\n");
    197}
    198
    199#define mt7921_print_txpwr_entry(prefix, rate)				\
    200({									\
    201	mt7921_seq_puts_array(s, #prefix " (user)",			\
    202			      txpwr.data[TXPWR_USER].rate,		\
    203			      ARRAY_SIZE(txpwr.data[TXPWR_USER].rate)); \
    204	mt7921_seq_puts_array(s, #prefix " (eeprom)",			\
    205			      txpwr.data[TXPWR_EEPROM].rate,		\
    206			      ARRAY_SIZE(txpwr.data[TXPWR_EEPROM].rate)); \
    207	mt7921_seq_puts_array(s, #prefix " (tmac)",			\
    208			      txpwr.data[TXPWR_MAC].rate,		\
    209			      ARRAY_SIZE(txpwr.data[TXPWR_MAC].rate));	\
    210})
    211
    212static int
    213mt7921_txpwr(struct seq_file *s, void *data)
    214{
    215	struct mt7921_dev *dev = dev_get_drvdata(s->private);
    216	struct mt7921_txpwr txpwr;
    217	int ret;
    218
    219	mt7921_mutex_acquire(dev);
    220	ret = mt7921_get_txpwr_info(dev, &txpwr);
    221	mt7921_mutex_release(dev);
    222
    223	if (ret)
    224		return ret;
    225
    226	seq_printf(s, "Tx power table (channel %d)\n", txpwr.ch);
    227	seq_printf(s, "%-16s  %6s %6s %6s %6s\n",
    228		   " ", "1m", "2m", "5m", "11m");
    229	mt7921_print_txpwr_entry(CCK, cck);
    230
    231	seq_printf(s, "%-16s  %6s %6s %6s %6s %6s %6s %6s %6s\n",
    232		   " ", "6m", "9m", "12m", "18m", "24m", "36m",
    233		   "48m", "54m");
    234	mt7921_print_txpwr_entry(OFDM, ofdm);
    235
    236	seq_printf(s, "%-16s  %6s %6s %6s %6s %6s %6s %6s %6s\n",
    237		   " ", "mcs0", "mcs1", "mcs2", "mcs3", "mcs4", "mcs5",
    238		   "mcs6", "mcs7");
    239	mt7921_print_txpwr_entry(HT20, ht20);
    240
    241	seq_printf(s, "%-16s  %6s %6s %6s %6s %6s %6s %6s %6s %6s\n",
    242		   " ", "mcs0", "mcs1", "mcs2", "mcs3", "mcs4", "mcs5",
    243		   "mcs6", "mcs7", "mcs32");
    244	mt7921_print_txpwr_entry(HT40, ht40);
    245
    246	seq_printf(s, "%-16s  %6s %6s %6s %6s %6s %6s %6s %6s %6s %6s %6s %6s\n",
    247		   " ", "mcs0", "mcs1", "mcs2", "mcs3", "mcs4", "mcs5",
    248		   "mcs6", "mcs7", "mcs8", "mcs9", "mcs10", "mcs11");
    249	mt7921_print_txpwr_entry(VHT20, vht20);
    250	mt7921_print_txpwr_entry(VHT40, vht40);
    251	mt7921_print_txpwr_entry(VHT80, vht80);
    252	mt7921_print_txpwr_entry(VHT160, vht160);
    253	mt7921_print_txpwr_entry(HE26, he26);
    254	mt7921_print_txpwr_entry(HE52, he52);
    255	mt7921_print_txpwr_entry(HE106, he106);
    256	mt7921_print_txpwr_entry(HE242, he242);
    257	mt7921_print_txpwr_entry(HE484, he484);
    258	mt7921_print_txpwr_entry(HE996, he996);
    259	mt7921_print_txpwr_entry(HE996x2, he996x2);
    260
    261	return 0;
    262}
    263
    264static int
    265mt7921_pm_set(void *data, u64 val)
    266{
    267	struct mt7921_dev *dev = data;
    268	struct mt76_connac_pm *pm = &dev->pm;
    269
    270	if (mt76_is_usb(&dev->mt76))
    271		return -EOPNOTSUPP;
    272
    273	mutex_lock(&dev->mt76.mutex);
    274
    275	if (val == pm->enable_user)
    276		goto out;
    277
    278	if (!pm->enable_user) {
    279		pm->stats.last_wake_event = jiffies;
    280		pm->stats.last_doze_event = jiffies;
    281	}
    282	/* make sure the chip is awake here and ps_work is scheduled
    283	 * just at end of the this routine.
    284	 */
    285	pm->enable = false;
    286	mt76_connac_pm_wake(&dev->mphy, pm);
    287
    288	pm->enable_user = val;
    289	mt7921_set_runtime_pm(dev);
    290	mt76_connac_power_save_sched(&dev->mphy, pm);
    291out:
    292	mutex_unlock(&dev->mt76.mutex);
    293
    294	return 0;
    295}
    296
    297static int
    298mt7921_pm_get(void *data, u64 *val)
    299{
    300	struct mt7921_dev *dev = data;
    301
    302	*val = dev->pm.enable_user;
    303
    304	return 0;
    305}
    306
    307DEFINE_DEBUGFS_ATTRIBUTE(fops_pm, mt7921_pm_get, mt7921_pm_set, "%lld\n");
    308
    309static int
    310mt7921_deep_sleep_set(void *data, u64 val)
    311{
    312	struct mt7921_dev *dev = data;
    313	struct mt76_connac_pm *pm = &dev->pm;
    314	bool monitor = !!(dev->mphy.hw->conf.flags & IEEE80211_CONF_MONITOR);
    315	bool enable = !!val;
    316
    317	if (mt76_is_usb(&dev->mt76))
    318		return -EOPNOTSUPP;
    319
    320	mt7921_mutex_acquire(dev);
    321	if (pm->ds_enable_user == enable)
    322		goto out;
    323
    324	pm->ds_enable_user = enable;
    325	pm->ds_enable = enable && !monitor;
    326	mt76_connac_mcu_set_deep_sleep(&dev->mt76, pm->ds_enable);
    327out:
    328	mt7921_mutex_release(dev);
    329
    330	return 0;
    331}
    332
    333static int
    334mt7921_deep_sleep_get(void *data, u64 *val)
    335{
    336	struct mt7921_dev *dev = data;
    337
    338	*val = dev->pm.ds_enable_user;
    339
    340	return 0;
    341}
    342
    343DEFINE_DEBUGFS_ATTRIBUTE(fops_ds, mt7921_deep_sleep_get,
    344			 mt7921_deep_sleep_set, "%lld\n");
    345
    346static int
    347mt7921_pm_stats(struct seq_file *s, void *data)
    348{
    349	struct mt7921_dev *dev = dev_get_drvdata(s->private);
    350	struct mt76_connac_pm *pm = &dev->pm;
    351
    352	unsigned long awake_time = pm->stats.awake_time;
    353	unsigned long doze_time = pm->stats.doze_time;
    354
    355	if (!test_bit(MT76_STATE_PM, &dev->mphy.state))
    356		awake_time += jiffies - pm->stats.last_wake_event;
    357	else
    358		doze_time += jiffies - pm->stats.last_doze_event;
    359
    360	seq_printf(s, "awake time: %14u\ndoze time: %15u\n",
    361		   jiffies_to_msecs(awake_time),
    362		   jiffies_to_msecs(doze_time));
    363
    364	seq_printf(s, "low power wakes: %9d\n", pm->stats.lp_wake);
    365
    366	return 0;
    367}
    368
    369static int
    370mt7921_pm_idle_timeout_set(void *data, u64 val)
    371{
    372	struct mt7921_dev *dev = data;
    373
    374	dev->pm.idle_timeout = msecs_to_jiffies(val);
    375
    376	return 0;
    377}
    378
    379static int
    380mt7921_pm_idle_timeout_get(void *data, u64 *val)
    381{
    382	struct mt7921_dev *dev = data;
    383
    384	*val = jiffies_to_msecs(dev->pm.idle_timeout);
    385
    386	return 0;
    387}
    388
    389DEFINE_DEBUGFS_ATTRIBUTE(fops_pm_idle_timeout, mt7921_pm_idle_timeout_get,
    390			 mt7921_pm_idle_timeout_set, "%lld\n");
    391
    392static int mt7921_chip_reset(void *data, u64 val)
    393{
    394	struct mt7921_dev *dev = data;
    395	int ret = 0;
    396
    397	switch (val) {
    398	case 1:
    399		/* Reset wifisys directly. */
    400		mt7921_reset(&dev->mt76);
    401		break;
    402	default:
    403		/* Collect the core dump before reset wifisys. */
    404		mt7921_mutex_acquire(dev);
    405		ret = mt76_connac_mcu_chip_config(&dev->mt76);
    406		mt7921_mutex_release(dev);
    407		break;
    408	}
    409
    410	return ret;
    411}
    412
    413DEFINE_DEBUGFS_ATTRIBUTE(fops_reset, NULL, mt7921_chip_reset, "%lld\n");
    414
    415static int
    416mt7921s_sched_quota_read(struct seq_file *s, void *data)
    417{
    418	struct mt7921_dev *dev = dev_get_drvdata(s->private);
    419	struct mt76_sdio *sdio = &dev->mt76.sdio;
    420
    421	seq_printf(s, "pse_data_quota\t%d\n", sdio->sched.pse_data_quota);
    422	seq_printf(s, "ple_data_quota\t%d\n", sdio->sched.ple_data_quota);
    423	seq_printf(s, "pse_mcu_quota\t%d\n", sdio->sched.pse_mcu_quota);
    424	seq_printf(s, "sched_deficit\t%d\n", sdio->sched.deficit);
    425
    426	return 0;
    427}
    428
    429int mt7921_init_debugfs(struct mt7921_dev *dev)
    430{
    431	struct dentry *dir;
    432
    433	dir = mt76_register_debugfs_fops(&dev->mphy, &fops_regval);
    434	if (!dir)
    435		return -ENOMEM;
    436
    437	if (mt76_is_mmio(&dev->mt76))
    438		debugfs_create_devm_seqfile(dev->mt76.dev, "xmit-queues",
    439					    dir, mt7921_queues_read);
    440	else
    441		debugfs_create_devm_seqfile(dev->mt76.dev, "xmit-queues",
    442					    dir, mt76_queues_read);
    443
    444	debugfs_create_devm_seqfile(dev->mt76.dev, "acq", dir,
    445				    mt7921_queues_acq);
    446	debugfs_create_devm_seqfile(dev->mt76.dev, "txpower_sku", dir,
    447				    mt7921_txpwr);
    448	debugfs_create_file("tx_stats", 0400, dir, dev, &mt7921_tx_stats_fops);
    449	debugfs_create_file("fw_debug", 0600, dir, dev, &fops_fw_debug);
    450	debugfs_create_file("runtime-pm", 0600, dir, dev, &fops_pm);
    451	debugfs_create_file("idle-timeout", 0600, dir, dev,
    452			    &fops_pm_idle_timeout);
    453	debugfs_create_file("chip_reset", 0600, dir, dev, &fops_reset);
    454	debugfs_create_devm_seqfile(dev->mt76.dev, "runtime_pm_stats", dir,
    455				    mt7921_pm_stats);
    456	debugfs_create_file("deep-sleep", 0600, dir, dev, &fops_ds);
    457	if (mt76_is_sdio(&dev->mt76))
    458		debugfs_create_devm_seqfile(dev->mt76.dev, "sched-quota", dir,
    459					    mt7921s_sched_quota_read);
    460	return 0;
    461}