|  | // SPDX-License-Identifier: GPL-2.0-only | 
|  | /* Copyright (C) 2021 Felix Fietkau <nbd@nbd.name> */ | 
|  |  | 
|  | #ifndef __MTK_WED_PRIV_H | 
|  | #define __MTK_WED_PRIV_H | 
|  |  | 
|  | #include <linux/soc/mediatek/mtk_wed.h> | 
|  | #include <linux/debugfs.h> | 
|  | #include <linux/regmap.h> | 
|  | #include <linux/netdevice.h> | 
|  |  | 
|  | #include "mtk_wed_regs.h" | 
|  |  | 
|  | struct mtk_eth; | 
|  | struct mtk_wed_wo; | 
|  |  | 
|  | struct mtk_wed_soc_data { | 
|  | struct { | 
|  | u32 tx_bm_tkid; | 
|  | u32 wpdma_rx_ring0; | 
|  | u32 reset_idx_tx_mask; | 
|  | u32 reset_idx_rx_mask; | 
|  | } regmap; | 
|  | u32 tx_ring_desc_size; | 
|  | u32 wdma_desc_size; | 
|  | }; | 
|  |  | 
|  | struct mtk_wed_amsdu { | 
|  | void *txd; | 
|  | dma_addr_t txd_phy; | 
|  | }; | 
|  |  | 
|  | struct mtk_wed_hw { | 
|  | const struct mtk_wed_soc_data *soc; | 
|  | struct device_node *node; | 
|  | struct mtk_eth *eth; | 
|  | struct regmap *regs; | 
|  | struct regmap *hifsys; | 
|  | struct device *dev; | 
|  | void __iomem *wdma; | 
|  | phys_addr_t wdma_phy; | 
|  | struct regmap *mirror; | 
|  | struct dentry *debugfs_dir; | 
|  | struct mtk_wed_device *wed_dev; | 
|  | struct mtk_wed_wo *wed_wo; | 
|  | struct mtk_wed_amsdu *wed_amsdu; | 
|  | u32 pcie_base; | 
|  | u32 debugfs_reg; | 
|  | u32 num_flows; | 
|  | u8 version; | 
|  | char dirname[5]; | 
|  | int irq; | 
|  | int index; | 
|  | }; | 
|  |  | 
|  | struct mtk_wdma_info { | 
|  | u8 wdma_idx; | 
|  | u8 queue; | 
|  | u16 wcid; | 
|  | u8 bss; | 
|  | u8 amsdu; | 
|  | }; | 
|  |  | 
|  | #ifdef CONFIG_NET_MEDIATEK_SOC_WED | 
|  | static inline bool mtk_wed_is_v1(struct mtk_wed_hw *hw) | 
|  | { | 
|  | return hw->version == 1; | 
|  | } | 
|  |  | 
|  | static inline bool mtk_wed_is_v2(struct mtk_wed_hw *hw) | 
|  | { | 
|  | return hw->version == 2; | 
|  | } | 
|  |  | 
|  | static inline bool mtk_wed_is_v3(struct mtk_wed_hw *hw) | 
|  | { | 
|  | return hw->version == 3; | 
|  | } | 
|  |  | 
|  | static inline bool mtk_wed_is_v3_or_greater(struct mtk_wed_hw *hw) | 
|  | { | 
|  | return hw->version > 2; | 
|  | } | 
|  |  | 
|  | static inline void | 
|  | wed_w32(struct mtk_wed_device *dev, u32 reg, u32 val) | 
|  | { | 
|  | regmap_write(dev->hw->regs, reg, val); | 
|  | } | 
|  |  | 
|  | static inline u32 | 
|  | wed_r32(struct mtk_wed_device *dev, u32 reg) | 
|  | { | 
|  | unsigned int val; | 
|  |  | 
|  | regmap_read(dev->hw->regs, reg, &val); | 
|  |  | 
|  | return val; | 
|  | } | 
|  |  | 
|  | static inline void | 
|  | wdma_w32(struct mtk_wed_device *dev, u32 reg, u32 val) | 
|  | { | 
|  | writel(val, dev->hw->wdma + reg); | 
|  | } | 
|  |  | 
|  | static inline u32 | 
|  | wdma_r32(struct mtk_wed_device *dev, u32 reg) | 
|  | { | 
|  | return readl(dev->hw->wdma + reg); | 
|  | } | 
|  |  | 
|  | static inline u32 | 
|  | wpdma_tx_r32(struct mtk_wed_device *dev, int ring, u32 reg) | 
|  | { | 
|  | if (!dev->tx_ring[ring].wpdma) | 
|  | return 0; | 
|  |  | 
|  | return readl(dev->tx_ring[ring].wpdma + reg); | 
|  | } | 
|  |  | 
|  | static inline void | 
|  | wpdma_tx_w32(struct mtk_wed_device *dev, int ring, u32 reg, u32 val) | 
|  | { | 
|  | if (!dev->tx_ring[ring].wpdma) | 
|  | return; | 
|  |  | 
|  | writel(val, dev->tx_ring[ring].wpdma + reg); | 
|  | } | 
|  |  | 
|  | static inline u32 | 
|  | wpdma_rx_r32(struct mtk_wed_device *dev, int ring, u32 reg) | 
|  | { | 
|  | if (!dev->rx_ring[ring].wpdma) | 
|  | return 0; | 
|  |  | 
|  | return readl(dev->rx_ring[ring].wpdma + reg); | 
|  | } | 
|  |  | 
|  | static inline void | 
|  | wpdma_rx_w32(struct mtk_wed_device *dev, int ring, u32 reg, u32 val) | 
|  | { | 
|  | if (!dev->rx_ring[ring].wpdma) | 
|  | return; | 
|  |  | 
|  | writel(val, dev->rx_ring[ring].wpdma + reg); | 
|  | } | 
|  |  | 
|  | static inline u32 | 
|  | wpdma_txfree_r32(struct mtk_wed_device *dev, u32 reg) | 
|  | { | 
|  | if (!dev->txfree_ring.wpdma) | 
|  | return 0; | 
|  |  | 
|  | return readl(dev->txfree_ring.wpdma + reg); | 
|  | } | 
|  |  | 
|  | static inline void | 
|  | wpdma_txfree_w32(struct mtk_wed_device *dev, u32 reg, u32 val) | 
|  | { | 
|  | if (!dev->txfree_ring.wpdma) | 
|  | return; | 
|  |  | 
|  | writel(val, dev->txfree_ring.wpdma + reg); | 
|  | } | 
|  |  | 
|  | static inline u32 mtk_wed_get_pcie_base(struct mtk_wed_device *dev) | 
|  | { | 
|  | if (!mtk_wed_is_v3_or_greater(dev->hw)) | 
|  | return MTK_WED_PCIE_BASE; | 
|  |  | 
|  | switch (dev->hw->index) { | 
|  | case 1: | 
|  | return MTK_WED_PCIE_BASE1; | 
|  | case 2: | 
|  | return MTK_WED_PCIE_BASE2; | 
|  | default: | 
|  | return MTK_WED_PCIE_BASE0; | 
|  | } | 
|  | } | 
|  |  | 
|  | void mtk_wed_add_hw(struct device_node *np, struct mtk_eth *eth, | 
|  | void __iomem *wdma, phys_addr_t wdma_phy, | 
|  | int index); | 
|  | void mtk_wed_exit(void); | 
|  | int mtk_wed_flow_add(int index); | 
|  | void mtk_wed_flow_remove(int index); | 
|  | void mtk_wed_fe_reset(void); | 
|  | void mtk_wed_fe_reset_complete(void); | 
|  | #else | 
|  | static inline void | 
|  | mtk_wed_add_hw(struct device_node *np, struct mtk_eth *eth, | 
|  | void __iomem *wdma, phys_addr_t wdma_phy, | 
|  | int index) | 
|  | { | 
|  | } | 
|  | static inline void | 
|  | mtk_wed_exit(void) | 
|  | { | 
|  | } | 
|  | static inline int mtk_wed_flow_add(int index) | 
|  | { | 
|  | return -EINVAL; | 
|  | } | 
|  | static inline void mtk_wed_flow_remove(int index) | 
|  | { | 
|  | } | 
|  |  | 
|  | static inline void mtk_wed_fe_reset(void) | 
|  | { | 
|  | } | 
|  |  | 
|  | static inline void mtk_wed_fe_reset_complete(void) | 
|  | { | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #ifdef CONFIG_DEBUG_FS | 
|  | void mtk_wed_hw_add_debugfs(struct mtk_wed_hw *hw); | 
|  | #else | 
|  | static inline void mtk_wed_hw_add_debugfs(struct mtk_wed_hw *hw) | 
|  | { | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #endif |