Path: blob/main/sys/contrib/dev/mediatek/mt76/mt76x2/pci.c
48525 views
// SPDX-License-Identifier: ISC1/*2* Copyright (C) 2016 Felix Fietkau <[email protected]>3*/45#include <linux/kernel.h>6#include <linux/module.h>7#include <linux/pci.h>89#include "mt76x2.h"1011static const struct pci_device_id mt76x2e_device_table[] = {12{ PCI_DEVICE(PCI_VENDOR_ID_MEDIATEK, 0x7662) },13{ PCI_DEVICE(PCI_VENDOR_ID_MEDIATEK, 0x7612) },14{ PCI_DEVICE(PCI_VENDOR_ID_MEDIATEK, 0x7602) },15{ },16};1718static int19mt76x2e_probe(struct pci_dev *pdev, const struct pci_device_id *id)20{21static const struct mt76_driver_ops drv_ops = {22.txwi_size = sizeof(struct mt76x02_txwi),23.drv_flags = MT_DRV_TX_ALIGNED4_SKBS |24MT_DRV_SW_RX_AIRTIME |25MT_DRV_IGNORE_TXS_FAILED,26.survey_flags = SURVEY_INFO_TIME_TX,27.update_survey = mt76x02_update_channel,28.set_channel = mt76x2e_set_channel,29.tx_prepare_skb = mt76x02_tx_prepare_skb,30.tx_complete_skb = mt76x02_tx_complete_skb,31.rx_skb = mt76x02_queue_rx_skb,32.rx_poll_complete = mt76x02_rx_poll_complete,33.sta_ps = mt76x02_sta_ps,34.sta_add = mt76x02_sta_add,35.sta_remove = mt76x02_sta_remove,36};37struct mt76x02_dev *dev;38struct mt76_dev *mdev;39int ret;4041ret = pcim_enable_device(pdev);42if (ret)43return ret;4445ret = pcim_iomap_regions(pdev, BIT(0), pci_name(pdev));46if (ret)47return ret;4849pci_set_master(pdev);5051ret = dma_set_mask(&pdev->dev, DMA_BIT_MASK(32));52if (ret)53return ret;5455mdev = mt76_alloc_device(&pdev->dev, sizeof(*dev), &mt76x2_ops,56&drv_ops);57if (!mdev)58return -ENOMEM;5960dev = container_of(mdev, struct mt76x02_dev, mt76);61mt76_mmio_init(mdev, pcim_iomap_table(pdev)[0]);62mt76x2_reset_wlan(dev, false);6364mdev->rev = mt76_rr(dev, MT_ASIC_VERSION);65dev_info(mdev->dev, "ASIC revision: %08x\n", mdev->rev);6667mt76_wr(dev, MT_INT_MASK_CSR, 0);6869ret = devm_request_irq(mdev->dev, pdev->irq, mt76x02_irq_handler,70IRQF_SHARED, KBUILD_MODNAME, dev);71if (ret)72goto error;7374ret = mt76x2_register_device(dev);75if (ret)76goto error;7778/* Fix up ASPM configuration */7980/* RG_SSUSB_G1_CDR_BIR_LTR = 0x9 */81mt76_rmw_field(dev, 0x15a10, 0x1f << 16, 0x9);8283/* RG_SSUSB_G1_CDR_BIC_LTR = 0xf */84mt76_rmw_field(dev, 0x15a0c, 0xfU << 28, 0xf);8586/* RG_SSUSB_CDR_BR_PE1D = 0x3 */87mt76_rmw_field(dev, 0x15c58, 0x3 << 6, 0x3);8889mt76_pci_disable_aspm(pdev);9091return 0;9293error:94mt76_free_device(&dev->mt76);9596return ret;97}9899static void100mt76x2e_remove(struct pci_dev *pdev)101{102struct mt76_dev *mdev = pci_get_drvdata(pdev);103struct mt76x02_dev *dev = container_of(mdev, struct mt76x02_dev, mt76);104105mt76_unregister_device(mdev);106mt76x2_cleanup(dev);107mt76_free_device(mdev);108}109110static int __maybe_unused111mt76x2e_suspend(struct pci_dev *pdev, pm_message_t state)112{113struct mt76_dev *mdev = pci_get_drvdata(pdev);114int i, err;115116napi_disable(&mdev->tx_napi);117tasklet_kill(&mdev->pre_tbtt_tasklet);118mt76_worker_disable(&mdev->tx_worker);119120mt76_for_each_q_rx(mdev, i)121napi_disable(&mdev->napi[i]);122123pci_enable_wake(pdev, pci_choose_state(pdev, state), true);124pci_save_state(pdev);125err = pci_set_power_state(pdev, pci_choose_state(pdev, state));126if (err)127goto restore;128129return 0;130131restore:132mt76_for_each_q_rx(mdev, i)133napi_enable(&mdev->napi[i]);134napi_enable(&mdev->tx_napi);135136return err;137}138139static int __maybe_unused140mt76x2e_resume(struct pci_dev *pdev)141{142struct mt76_dev *mdev = pci_get_drvdata(pdev);143struct mt76x02_dev *dev = container_of(mdev, struct mt76x02_dev, mt76);144int i, err;145146err = pci_set_power_state(pdev, PCI_D0);147if (err)148return err;149150pci_restore_state(pdev);151152mt76_worker_enable(&mdev->tx_worker);153154mt76_for_each_q_rx(mdev, i) {155napi_enable(&mdev->napi[i]);156}157napi_enable(&mdev->tx_napi);158159local_bh_disable();160mt76_for_each_q_rx(mdev, i) {161napi_schedule(&mdev->napi[i]);162}163napi_schedule(&mdev->tx_napi);164local_bh_enable();165166return mt76x2_resume_device(dev);167}168169MODULE_DEVICE_TABLE(pci, mt76x2e_device_table);170MODULE_FIRMWARE(MT7662_FIRMWARE);171MODULE_FIRMWARE(MT7662_ROM_PATCH);172MODULE_DESCRIPTION("MediaTek MT76x2E (PCIe) wireless driver");173MODULE_LICENSE("Dual BSD/GPL");174175static struct pci_driver mt76pci_driver = {176.name = KBUILD_MODNAME,177.id_table = mt76x2e_device_table,178.probe = mt76x2e_probe,179.remove = mt76x2e_remove,180#ifdef CONFIG_PM181.suspend = mt76x2e_suspend,182.resume = mt76x2e_resume,183#endif /* CONFIG_PM */184};185186module_pci_driver(mt76pci_driver);187188189