From: Oleksij Rempel <linux@rempel-privat.de>
To: barebox@lists.infradead.org
Cc: Oleksij Rempel <linux@rempel-privat.de>
Subject: [PATCH v1 01/10] net: ath79: add ar9344 support
Date: Sun, 27 Aug 2017 09:02:21 +0200 [thread overview]
Message-ID: <20170827070230.13382-2-linux@rempel-privat.de> (raw)
In-Reply-To: <20170827070230.13382-1-linux@rempel-privat.de>
Signed-off-by: Oleksij Rempel <linux@rempel-privat.de>
---
drivers/net/ag71xx.c | 192 +++++++++++++++++++++++++++++++++++++++++++--------
1 file changed, 165 insertions(+), 27 deletions(-)
diff --git a/drivers/net/ag71xx.c b/drivers/net/ag71xx.c
index d13bf9171..b8bd12bc3 100644
--- a/drivers/net/ag71xx.c
+++ b/drivers/net/ag71xx.c
@@ -12,6 +12,7 @@
*/
#include <common.h>
+#include <driver.h>
#include <net.h>
#include <dma.h>
#include <init.h>
@@ -185,6 +186,11 @@
#define AG71XX_ETH_CFG_RGMII_GE0 (1<<0)
#define AG71XX_ETH_CFG_MII_GE0_SLAVE (1<<4)
+enum ag71xx_type {
+ AG71XX_TYPE_AR9331_GE0,
+ AG71XX_TYPE_AR9344_GMAC0,
+};
+
/*
* h/w descriptor
*/
@@ -213,23 +219,31 @@ struct ag71xx {
void __iomem *regs;
void __iomem *regs_gmac;
struct mii_bus miibus;
+ const struct ag71xx_cfg *cfg;
void *rx_buffer;
unsigned char *rx_pkt[NO_OF_RX_FIFOS];
ag7240_desc_t *fifo_tx;
ag7240_desc_t *fifo_rx;
+ dma_addr_t addr_tx;
+ dma_addr_t addr_rx;
int next_tx;
int next_rx;
};
+struct ag71xx_cfg {
+ enum ag71xx_type type;
+ void (*init_mii)(struct ag71xx *priv);
+};
+
static inline void ag71xx_check_reg_offset(struct ag71xx *priv, int reg)
{
switch (reg) {
case AG71XX_REG_MAC_CFG1 ... AG71XX_REG_MAC_MFL:
case AG71XX_REG_MAC_IFCTL ... AG71XX_REG_TX_SM:
- case AG71XX_REG_MII_CFG:
+ case AG71XX_REG_MII_CFG ... AG71XX_REG_MII_IND:
break;
default:
@@ -237,6 +251,16 @@ static inline void ag71xx_check_reg_offset(struct ag71xx *priv, int reg)
}
}
+static inline u32 ar7240_reg_rd(u32 reg)
+{
+ return __raw_readl(KSEG1ADDR(reg));
+}
+
+static inline void ar7240_reg_wr(u32 reg, u32 val)
+{
+ __raw_writel(val, KSEG1ADDR(reg));
+}
+
static inline u32 ag71xx_gmac_rr(struct ag71xx *dev, int reg)
{
return __raw_readl(dev->regs_gmac + reg);
@@ -263,13 +287,81 @@ static inline void ag71xx_wr(struct ag71xx *priv, int reg, u32 val)
(void)__raw_readl(priv->regs + reg);
}
-static int ag71xx_ether_mii_read(struct mii_bus *miidev, int addr, int reg)
+static int ag71xx_ether_mii_read(struct mii_bus *miidev, int phy_addr, int reg)
{
- return 0xffff;
+ struct ag71xx *priv = miidev->priv;
+ const struct ag71xx_cfg *cfg = priv->cfg;
+ volatile int rddata;
+ u16 addr = (phy_addr << MII_ADDR_SHIFT) | reg, val;
+ u16 ii = 0xFFFF;
+
+ if (AG71XX_TYPE_AR9331_GE0 == cfg->type)
+ return 0xffff;
+ /*
+ * Check for previous transactions are complete. Added to avoid
+ * race condition while running at higher frequencies.
+ */
+ do {
+ udelay(5);
+ rddata = ag71xx_rr(priv, AG71XX_REG_MII_IND) & MII_IND_BUSY;
+ } while (rddata && --ii);
+
+ if (ii == 0)
+ printk("ERROR:%s:%d transaction failed\n",__func__,__LINE__);
+
+
+ ag71xx_wr(priv, AG71XX_REG_MII_CMD, MII_CMD_WRITE);
+ ag71xx_wr(priv, AG71XX_REG_MII_ADDR, addr);
+ ag71xx_wr(priv, AG71XX_REG_MII_CMD, MII_CMD_READ);
+
+ do {
+ udelay(5);
+ rddata = ag71xx_rr(priv, AG71XX_REG_MII_IND) & MII_IND_BUSY;
+ } while (rddata && --ii);
+
+ if (ii == 0)
+ printk("Error!!! Leave ag7240_miiphy_read without polling correct status!\n");
+
+ val = ag71xx_rr(priv, AG71XX_REG_MII_STATUS);
+ ag71xx_wr(priv, AG71XX_REG_MII_CMD, MII_CMD_WRITE);
+
+ return val;
}
-static int ag71xx_ether_mii_write(struct mii_bus *miidev, int addr, int reg, u16 val)
+static int ag71xx_ether_mii_write(struct mii_bus *miidev, int phy_addr,
+ int reg, u16 val)
{
+ struct ag71xx *priv = miidev->priv;
+ const struct ag71xx_cfg *cfg = priv->cfg;
+ u16 addr = (phy_addr << MII_ADDR_SHIFT) | reg;
+ u16 ii = 0xFFFF;
+ volatile int rddata;
+
+ if (AG71XX_TYPE_AR9331_GE0 == cfg->type)
+ return 0;
+
+ /*
+ * Check for previous transactions are complete. Added to avoid
+ * race condition while running at higher frequencies.
+ */
+ do {
+ udelay(5);
+ rddata = ag71xx_rr(priv, AG71XX_REG_MII_IND) & MII_IND_BUSY;
+ } while (rddata && --ii);
+
+ if (ii == 0)
+ printk("ERROR:%s:%d transaction failed\n", __func__, __LINE__);
+
+ ag71xx_wr(priv, AG71XX_REG_MII_ADDR, addr);
+ ag71xx_wr(priv, AG71XX_REG_MII_CTRL, val);
+
+ do {
+ rddata = ag71xx_rr(priv, AG71XX_REG_MII_IND) & MII_IND_BUSY;
+ } while (rddata && --ii);
+
+ if (ii == 0)
+ printk("Error!!! Leave ag7240_miiphy_write without polling correct status!\n");
+
return 0;
}
@@ -372,6 +464,13 @@ static int ag71xx_ether_send(struct eth_device *edev, void *packet, int length)
static int ag71xx_ether_open(struct eth_device *edev)
{
+ struct ag71xx *priv = edev->priv;
+ const struct ag71xx_cfg *cfg = priv->cfg;
+
+ if (AG71XX_TYPE_AR9344_GMAC0 == cfg->type)
+ return phy_device_connect(edev, &priv->miibus, 0,
+ NULL, 0, PHY_INTERFACE_MODE_RGMII_TXID);
+
return 0;
}
@@ -408,25 +507,63 @@ static int ag71xx_ether_init(struct eth_device *edev)
return 1;
}
-static int ag71xx_mii_setup(struct ag71xx *priv)
+static void ag71xx_ar9331_ge0_mii_init(struct ag71xx *priv)
{
u32 rd;
- rd = ag71xx_gmac_rr(priv, 0);
+ rd = ag71xx_rr(priv, AG71XX_REG_MAC_CFG2);
+ rd |= (MAC_CFG2_PAD_CRC_EN | MAC_CFG2_LEN_CHECK | MAC_CFG2_IF_10_100);
+ ag71xx_wr(priv, AG71XX_REG_MAC_CFG2, rd);
+
+ /* config FIFOs */
+ ag71xx_wr(priv, AG71XX_REG_FIFO_CFG0, 0x1f00);
+
+ rd = ag71xx_gmac_rr(priv, AG71XX_REG_MAC_CFG1);
rd |= AG71XX_ETH_CFG_MII_GE0_SLAVE;
ag71xx_gmac_wr(priv, 0, rd);
+}
- return 0;
+static void ag71xx_ar9344_gmac0_mii_init(struct ag71xx *priv)
+{
+ u32 rd;
+
+ rd = ag71xx_rr(priv, AG71XX_REG_MAC_CFG2);
+ rd |= (MAC_CFG2_PAD_CRC_EN | MAC_CFG2_LEN_CHECK | MAC_CFG2_IF_1000);
+ ag71xx_wr(priv, AG71XX_REG_MAC_CFG2, rd);
+
+ /* config FIFOs */
+ ag71xx_wr(priv, AG71XX_REG_FIFO_CFG0, 0x1f00);
+
+ ag71xx_gmac_wr(priv, AG71XX_REG_MAC_CFG1, 1);
+ udelay(1000);
+ ag71xx_wr(priv, AG71XX_REG_MII_CFG, 4 | (1 << 31));
+ ag71xx_wr(priv, AG71XX_REG_MII_CFG, 4);
}
+static struct ag71xx_cfg ag71xx_cfg_ar9331_ge0 = {
+ .type = AG71XX_TYPE_AR9331_GE0,
+ .init_mii = ag71xx_ar9331_ge0_mii_init,
+};
+
+static struct ag71xx_cfg ag71xx_cfg_ar9344_gmac0 = {
+ .type = AG71XX_TYPE_AR9344_GMAC0,
+ .init_mii = ag71xx_ar9344_gmac0_mii_init,
+};
+
static int ag71xx_probe(struct device_d *dev)
{
void __iomem *regs, *regs_gmac;
struct mii_bus *miibus;
struct eth_device *edev;
+ struct ag71xx_cfg *cfg;
struct ag71xx *priv;
u32 mac_h, mac_l;
- u32 rd;
+ u32 rd, mask;
+ int ret;
+
+ ret = dev_get_drvdata(dev, (const void **)&cfg);
+ if (ret)
+ return ret;
regs_gmac = dev_request_mem_region_by_name(dev, "gmac");
if (IS_ERR(regs_gmac))
@@ -452,41 +589,42 @@ static int ag71xx_probe(struct device_d *dev)
priv->dev = dev;
priv->regs = regs;
priv->regs_gmac = regs_gmac;
+ priv->cfg = cfg;
miibus->read = ag71xx_ether_mii_read;
miibus->write = ag71xx_ether_mii_write;
miibus->priv = priv;
/* enable switch core */
- rd = __raw_readl((char *)KSEG1ADDR(AR71XX_PLL_BASE + AR933X_ETHSW_CLOCK_CONTROL_REG)) & ~(0x1f);
+ rd = ar7240_reg_rd(AR71XX_PLL_BASE + AR933X_ETHSW_CLOCK_CONTROL_REG);
+ rd &= ~(0x1f);
rd |= 0x10;
- __raw_writel(rd, (char *)KSEG1ADDR(AR71XX_PLL_BASE + AR933X_ETHSW_CLOCK_CONTROL_REG));
+ if ((ar7240_reg_rd(WASP_BOOTSTRAP_REG) & WASP_REF_CLK_25) == 0)
+ rd |= 0x1;
+ ar7240_reg_wr((AR71XX_PLL_BASE + AR933X_ETHSW_CLOCK_CONTROL_REG), rd);
if (ath79_reset_rr(AR933X_RESET_REG_RESET_MODULE) != 0)
ath79_reset_wr(AR933X_RESET_REG_RESET_MODULE, 0);
/* reset GE0 MAC and MDIO */
+ mask = AR933X_RESET_GE0_MAC | AR933X_RESET_GE0_MDIO
+ | AR933X_RESET_SWITCH;
+
rd = ath79_reset_rr(AR933X_RESET_REG_RESET_MODULE);
- rd |= AR933X_RESET_GE0_MAC | AR933X_RESET_GE0_MDIO | AR933X_RESET_SWITCH;
+ rd |= mask;
ath79_reset_wr(AR933X_RESET_REG_RESET_MODULE, rd);
mdelay(100);
rd = ath79_reset_rr(AR933X_RESET_REG_RESET_MODULE);
- rd &= ~(AR933X_RESET_GE0_MAC | AR933X_RESET_GE0_MDIO | AR933X_RESET_SWITCH);
+ rd &= ~(mask);
ath79_reset_wr(AR933X_RESET_REG_RESET_MODULE, rd);
mdelay(100);
ag71xx_wr(priv, AG71XX_REG_MAC_CFG1, (MAC_CFG1_SR | MAC_CFG1_TX_RST | MAC_CFG1_RX_RST));
ag71xx_wr(priv, AG71XX_REG_MAC_CFG1, (MAC_CFG1_RXE | MAC_CFG1_TXE));
- rd = ag71xx_rr(priv, AG71XX_REG_MAC_CFG2);
- rd |= (MAC_CFG2_PAD_CRC_EN | MAC_CFG2_LEN_CHECK | MAC_CFG2_IF_10_100);
- ag71xx_wr(priv, AG71XX_REG_MAC_CFG2, rd);
-
- /* config FIFOs */
- ag71xx_wr(priv, AG71XX_REG_FIFO_CFG0, 0x1f00);
-
- ag71xx_mii_setup(priv);
+ if (cfg->init_mii)
+ cfg->init_mii(priv);
ag71xx_wr(priv, AG71XX_REG_FIFO_CFG1, 0x10ffff);
ag71xx_wr(priv, AG71XX_REG_FIFO_CFG2, 0xAAA0555);
@@ -497,8 +635,10 @@ static int ag71xx_probe(struct device_d *dev)
ag71xx_wr(priv, AG71XX_REG_FIFO_CFG3, 0x1f00140);
priv->rx_buffer = xmemalign(PAGE_SIZE, NO_OF_RX_FIFOS * MAX_RBUFF_SZ);
- priv->fifo_tx = dma_alloc_coherent(NO_OF_TX_FIFOS * sizeof(ag7240_desc_t), DMA_ADDRESS_BROKEN);
- priv->fifo_rx = dma_alloc_coherent(NO_OF_RX_FIFOS * sizeof(ag7240_desc_t), DMA_ADDRESS_BROKEN);
+ priv->fifo_tx = dma_alloc_coherent(NO_OF_TX_FIFOS * sizeof(ag7240_desc_t),
+ &priv->addr_tx);
+ priv->fifo_rx = dma_alloc_coherent(NO_OF_RX_FIFOS * sizeof(ag7240_desc_t),
+ &priv->addr_rx);
priv->next_tx = 0;
mac_l = 0x3344;
@@ -516,11 +656,9 @@ static int ag71xx_probe(struct device_d *dev)
}
static __maybe_unused struct of_device_id ag71xx_dt_ids[] = {
- {
- .compatible = "qca,ar7100-gmac",
- }, {
- /* sentinel */
- }
+ { .compatible = "qca,ar9331-ge0", .data = &ag71xx_cfg_ar9331_ge0, },
+ { .compatible = "qca,ar9344-gmac0", .data = &ag71xx_cfg_ar9344_gmac0, },
+ { /* sentinel */ }
};
static struct driver_d ag71xx_driver = {
--
2.11.0
_______________________________________________
barebox mailing list
barebox@lists.infradead.org
http://lists.infradead.org/mailman/listinfo/barebox
next prev parent reply other threads:[~2017-08-27 7:03 UTC|newest]
Thread overview: 15+ messages / expand[flat|nested] mbox.gz Atom feed top
2017-08-27 7:02 [PATCH v1 00/10] provide ethernet support for ar9344 and TP-LINK TL-WDR4300 Oleksij Rempel
2017-08-27 7:02 ` Oleksij Rempel [this message]
2017-09-12 6:11 ` [PATCH v1 01/10] net: ath79: add ar9344 support Sascha Hauer
2017-08-27 7:02 ` [PATCH v1 02/10] MIPS: dts: ar9331: update compatible for mac0 node Oleksij Rempel
2017-09-12 6:12 ` Sascha Hauer
2017-08-27 7:02 ` [PATCH v1 03/10] MIPS: dts: ar9344: add " Oleksij Rempel
2017-08-27 7:02 ` [PATCH v1 04/10] MIPS: dts: tl_wdr4300: enable mac0 Oleksij Rempel
2017-08-27 7:02 ` [PATCH v1 05/10] MIPS: dts: tl_wdr4300: add alias for spiflash Oleksij Rempel
2017-08-27 7:02 ` [PATCH v1 06/10] MIPS: dts: tl_wdr4300: remove RO flag from barebox partition Oleksij Rempel
2017-08-27 7:02 ` [PATCH v1 07/10] add phy: ar8327 driver Oleksij Rempel
2017-09-12 6:19 ` Sascha Hauer
2017-08-27 7:02 ` [PATCH v1 08/10] rename tplink devicetree Oleksij Rempel
2017-09-12 6:20 ` Sascha Hauer
2017-08-27 7:02 ` [PATCH v1 09/10] MIPS: tplink-wdr4300_defconfig: add network support Oleksij Rempel
2017-08-27 7:02 ` [PATCH v1 10/10] MIPS: dts: ar9344: add APB bus Oleksij Rempel
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20170827070230.13382-2-linux@rempel-privat.de \
--to=linux@rempel-privat.de \
--cc=barebox@lists.infradead.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox