David, are you going to pick up this patch, or would you like me to? Thanks, g
On Wed, Mar 17, 2010 at 2:02 PM, Grant Likely <grant.lik...@secretlab.ca> wrote: > On Fri, Mar 12, 2010 at 7:05 PM, John Linn <john.l...@xilinx.com> wrote: >> This patch adds support for using the LL TEMAC Ethernet driver on >> non-Virtex 5 platforms by adding support for accessing the Soft DMA >> registers as if they were memory mapped instead of solely through the >> DCR's (available on the Virtex 5). >> >> The patch also updates the driver so that it runs on the MicroBlaze. >> The changes were tested on the PowerPC 440, PowerPC 405, and the >> MicroBlaze platforms. >> >> Signed-off-by: John Tyner <jty...@cs.ucr.edu> >> Signed-off-by: John Linn <john.l...@xilinx.com> >> --- > > I've not booted this, but it looks right, and it compiles fine. The > issues that Michal raised need to be delt with too, but they are > preexisting bugs unrelated to this change which you should fix up in a > separate patch. > > Acked-by: Grant Likely <grant.lik...@secretlab.ca> > >> >> V2 - Incorporated comments from Grant and added more logic to allow the >> driver >> to work on MicroBlaze. >> >> drivers/net/Kconfig | 1 - >> drivers/net/ll_temac.h | 17 +++++- >> drivers/net/ll_temac_main.c | 124 >> ++++++++++++++++++++++++++++++++++--------- >> 3 files changed, 113 insertions(+), 29 deletions(-) >> >> diff --git a/drivers/net/Kconfig b/drivers/net/Kconfig >> index 9b6efe1..5402105 100644 >> --- a/drivers/net/Kconfig >> +++ b/drivers/net/Kconfig >> @@ -2443,7 +2443,6 @@ config MV643XX_ETH >> config XILINX_LL_TEMAC >> tristate "Xilinx LL TEMAC (LocalLink Tri-mode Ethernet MAC) driver" >> select PHYLIB >> - depends on PPC_DCR_NATIVE >> help >> This driver supports the Xilinx 10/100/1000 LocalLink TEMAC >> core used in Xilinx Spartan and Virtex FPGAs >> diff --git a/drivers/net/ll_temac.h b/drivers/net/ll_temac.h >> index 1af66a1..915aa34 100644 >> --- a/drivers/net/ll_temac.h >> +++ b/drivers/net/ll_temac.h >> @@ -5,8 +5,11 @@ >> #include <linux/netdevice.h> >> #include <linux/of.h> >> #include <linux/spinlock.h> >> + >> +#ifdef CONFIG_PPC_DCR >> #include <asm/dcr.h> >> #include <asm/dcr-regs.h> >> +#endif >> >> /* packet size info */ >> #define XTE_HDR_SIZE 14 /* size of Ethernet header */ >> @@ -290,8 +293,12 @@ This option defaults to enabled (set) */ >> >> #define TX_CONTROL_CALC_CSUM_MASK 1 >> >> +/* Align the IP data in the packet on word boundaries as MicroBlaze >> + * needs it. >> + */ >> + >> #define XTE_ALIGN 32 >> -#define BUFFER_ALIGN(adr) ((XTE_ALIGN - ((u32) adr)) % XTE_ALIGN) >> +#define BUFFER_ALIGN(adr) ((34 - ((u32) adr)) % XTE_ALIGN) >> >> #define MULTICAST_CAM_TABLE_NUM 4 >> >> @@ -335,9 +342,15 @@ struct temac_local { >> struct mii_bus *mii_bus; /* MII bus reference */ >> int mdio_irqs[PHY_MAX_ADDR]; /* IRQs table for MDIO bus */ >> >> - /* IO registers and IRQs */ >> + /* IO registers, dma functions and IRQs */ >> void __iomem *regs; >> + void __iomem *sdma_regs; >> +#ifdef CONFIG_PPC_DCR >> dcr_host_t sdma_dcrs; >> +#endif >> + u32 (*dma_in)(struct temac_local *, int); >> + void (*dma_out)(struct temac_local *, int, u32); >> + >> int tx_irq; >> int rx_irq; >> int emac_num; >> diff --git a/drivers/net/ll_temac_main.c b/drivers/net/ll_temac_main.c >> index a18e348..9aedf9b 100644 >> --- a/drivers/net/ll_temac_main.c >> +++ b/drivers/net/ll_temac_main.c >> @@ -20,9 +20,6 @@ >> * or rx, so this should be okay. >> * >> * TODO: >> - * - Fix driver to work on more than just Virtex5. Right now the driver >> - * assumes that the locallink DMA registers are accessed via DCR >> - * instructions. >> * - Factor out locallink DMA code into separate driver >> * - Fix multicast assignment. >> * - Fix support for hardware checksumming. >> @@ -115,17 +112,86 @@ void temac_indirect_out32(struct temac_local *lp, int >> reg, u32 value) >> temac_iow(lp, XTE_CTL0_OFFSET, CNTLREG_WRITE_ENABLE_MASK | reg); >> } >> >> +/** >> + * temac_dma_in32 - Memory mapped DMA read, this function expects a >> + * register input that is based on DCR word addresses which >> + * are then converted to memory mapped byte addresses >> + */ >> static u32 temac_dma_in32(struct temac_local *lp, int reg) >> { >> - return dcr_read(lp->sdma_dcrs, reg); >> + return in_be32((u32 *)(lp->sdma_regs + (reg << 2))); >> } >> >> +/** >> + * temac_dma_out32 - Memory mapped DMA read, this function expects a >> + * register input that is based on DCR word addresses which >> + * are then converted to memory mapped byte addresses >> + */ >> static void temac_dma_out32(struct temac_local *lp, int reg, u32 value) >> { >> + out_be32((u32 *)(lp->sdma_regs + (reg << 2)), value); >> +} >> + >> +/* DMA register access functions can be DCR based or memory mapped. >> + * The PowerPC 440 is DCR based, the PowerPC 405 and MicroBlaze are both >> + * memory mapped. >> + */ >> +#ifdef CONFIG_PPC_DCR >> + >> +/** >> + * temac_dma_dcr_in32 - DCR based DMA read >> + */ >> +static u32 temac_dma_dcr_in(struct temac_local *lp, int reg) >> +{ >> + return dcr_read(lp->sdma_dcrs, reg); >> +} >> + >> +/** >> + * temac_dma_dcr_out32 - DCR based DMA write >> + */ >> +static void temac_dma_dcr_out(struct temac_local *lp, int reg, u32 value) >> +{ >> dcr_write(lp->sdma_dcrs, reg, value); >> } >> >> /** >> + * temac_dcr_setup - If the DMA is DCR based, then setup the address and >> + * I/O functions >> + */ >> +static int temac_dcr_setup(struct temac_local *lp, struct of_device *op, >> + struct device_node *np) >> +{ >> + unsigned int dcrs; >> + >> + /* setup the dcr address mapping if it's in the device tree */ >> + >> + dcrs = dcr_resource_start(np, 0); >> + if (dcrs != 0) { >> + lp->sdma_dcrs = dcr_map(np, dcrs, dcr_resource_len(np, 0)); >> + lp->dma_in = temac_dma_dcr_in; >> + lp->dma_out = temac_dma_dcr_out; >> + dev_dbg(&op->dev, "DCR base: %x\n", dcrs); >> + return 0; >> + } >> + /* no DCR in the device tree, indicate a failure */ >> + return -1; >> +} >> + >> +#else >> + >> +/* >> + * temac_dcr_setup - This is a stub for when DCR is not supported, >> + * such as with MicroBlaze >> + */ >> +static int temac_dcr_setup(struct temac_local *lp, struct of_device *op, >> + struct device_node *np) >> +{ >> + return -1; >> +} >> + >> +#endif >> + >> +/** >> * temac_dma_bd_init - Setup buffer descriptor rings >> */ >> static int temac_dma_bd_init(struct net_device *ndev) >> @@ -172,23 +238,23 @@ static int temac_dma_bd_init(struct net_device *ndev) >> lp->rx_bd_v[i].app0 = STS_CTRL_APP0_IRQONEND; >> } >> >> - temac_dma_out32(lp, TX_CHNL_CTRL, 0x10220400 | >> + lp->dma_out(lp, TX_CHNL_CTRL, 0x10220400 | >> CHNL_CTRL_IRQ_EN | >> CHNL_CTRL_IRQ_DLY_EN | >> CHNL_CTRL_IRQ_COAL_EN); >> /* 0x10220483 */ >> /* 0x00100483 */ >> - temac_dma_out32(lp, RX_CHNL_CTRL, 0xff010000 | >> + lp->dma_out(lp, RX_CHNL_CTRL, 0xff010000 | >> CHNL_CTRL_IRQ_EN | >> CHNL_CTRL_IRQ_DLY_EN | >> CHNL_CTRL_IRQ_COAL_EN | >> CHNL_CTRL_IRQ_IOE); >> /* 0xff010283 */ >> >> - temac_dma_out32(lp, RX_CURDESC_PTR, lp->rx_bd_p); >> - temac_dma_out32(lp, RX_TAILDESC_PTR, >> + lp->dma_out(lp, RX_CURDESC_PTR, lp->rx_bd_p); >> + lp->dma_out(lp, RX_TAILDESC_PTR, >> lp->rx_bd_p + (sizeof(*lp->rx_bd_v) * (RX_BD_NUM - >> 1))); >> - temac_dma_out32(lp, TX_CURDESC_PTR, lp->tx_bd_p); >> + lp->dma_out(lp, TX_CURDESC_PTR, lp->tx_bd_p); >> >> return 0; >> } >> @@ -426,9 +492,9 @@ static void temac_device_reset(struct net_device *ndev) >> temac_indirect_out32(lp, XTE_RXC1_OFFSET, val & ~XTE_RXC1_RXEN_MASK); >> >> /* Reset Local Link (DMA) */ >> - temac_dma_out32(lp, DMA_CONTROL_REG, DMA_CONTROL_RST); >> + lp->dma_out(lp, DMA_CONTROL_REG, DMA_CONTROL_RST); >> timeout = 1000; >> - while (temac_dma_in32(lp, DMA_CONTROL_REG) & DMA_CONTROL_RST) { >> + while (lp->dma_in(lp, DMA_CONTROL_REG) & DMA_CONTROL_RST) { >> udelay(1); >> if (--timeout == 0) { >> dev_err(&ndev->dev, >> @@ -436,7 +502,7 @@ static void temac_device_reset(struct net_device *ndev) >> break; >> } >> } >> - temac_dma_out32(lp, DMA_CONTROL_REG, DMA_TAIL_ENABLE); >> + lp->dma_out(lp, DMA_CONTROL_REG, DMA_TAIL_ENABLE); >> >> temac_dma_bd_init(ndev); >> >> @@ -597,7 +663,7 @@ static int temac_start_xmit(struct sk_buff *skb, struct >> net_device *ndev) >> lp->tx_bd_tail = 0; >> >> /* Kick off the transfer */ >> - temac_dma_out32(lp, TX_TAILDESC_PTR, tail_p); /* DMA start */ >> + lp->dma_out(lp, TX_TAILDESC_PTR, tail_p); /* DMA start */ >> >> return NETDEV_TX_OK; >> } >> @@ -663,7 +729,7 @@ static void ll_temac_recv(struct net_device *ndev) >> cur_p = &lp->rx_bd_v[lp->rx_bd_ci]; >> bdstat = cur_p->app0; >> } >> - temac_dma_out32(lp, RX_TAILDESC_PTR, tail_p); >> + lp->dma_out(lp, RX_TAILDESC_PTR, tail_p); >> >> spin_unlock_irqrestore(&lp->rx_lock, flags); >> } >> @@ -674,8 +740,8 @@ static irqreturn_t ll_temac_tx_irq(int irq, void *_ndev) >> struct temac_local *lp = netdev_priv(ndev); >> unsigned int status; >> >> - status = temac_dma_in32(lp, TX_IRQ_REG); >> - temac_dma_out32(lp, TX_IRQ_REG, status); >> + status = lp->dma_in(lp, TX_IRQ_REG); >> + lp->dma_out(lp, TX_IRQ_REG, status); >> >> if (status & (IRQ_COAL | IRQ_DLY)) >> temac_start_xmit_done(lp->ndev); >> @@ -692,8 +758,8 @@ static irqreturn_t ll_temac_rx_irq(int irq, void *_ndev) >> unsigned int status; >> >> /* Read and clear the status registers */ >> - status = temac_dma_in32(lp, RX_IRQ_REG); >> - temac_dma_out32(lp, RX_IRQ_REG, status); >> + status = lp->dma_in(lp, RX_IRQ_REG); >> + lp->dma_out(lp, RX_IRQ_REG, status); >> >> if (status & (IRQ_COAL | IRQ_DLY)) >> ll_temac_recv(lp->ndev); >> @@ -794,7 +860,7 @@ static ssize_t temac_show_llink_regs(struct device *dev, >> int i, len = 0; >> >> for (i = 0; i < 0x11; i++) >> - len += sprintf(buf + len, "%.8x%s", temac_dma_in32(lp, i), >> + len += sprintf(buf + len, "%.8x%s", lp->dma_in(lp, i), >> (i % 8) == 7 ? "\n" : " "); >> len += sprintf(buf + len, "\n"); >> >> @@ -820,7 +886,6 @@ temac_of_probe(struct of_device *op, const struct >> of_device_id *match) >> struct net_device *ndev; >> const void *addr; >> int size, rc = 0; >> - unsigned int dcrs; >> >> /* Init network device structure */ >> ndev = alloc_etherdev(sizeof(*lp)); >> @@ -870,13 +935,20 @@ temac_of_probe(struct of_device *op, const struct >> of_device_id *match) >> goto nodev; >> } >> >> - dcrs = dcr_resource_start(np, 0); >> - if (dcrs == 0) { >> - dev_err(&op->dev, "could not get DMA register address\n"); >> - goto nodev; >> + /* Setup the DMA register accesses, could be DCR or memory mapped */ >> + if (temac_dcr_setup(lp, op, np)) { >> + >> + /* no DCR in the device tree, try non-DCR */ >> + lp->sdma_regs = of_iomap(np, 0); >> + if (lp->sdma_regs) { >> + lp->dma_in = temac_dma_in32; >> + lp->dma_out = temac_dma_out32; >> + dev_dbg(&op->dev, "MEM base: %p\n", lp->sdma_regs); >> + } else { >> + dev_err(&op->dev, "unable to map DMA registers\n"); >> + goto nodev; >> + } >> } >> - lp->sdma_dcrs = dcr_map(np, dcrs, dcr_resource_len(np, 0)); >> - dev_dbg(&op->dev, "DCR base: %x\n", dcrs); >> >> lp->rx_irq = irq_of_parse_and_map(np, 0); >> lp->tx_irq = irq_of_parse_and_map(np, 1); >> -- >> 1.6.2.1 >> >> >> >> This email and any attachments are intended for the sole use of the named >> recipient(s) and contain(s) confidential information that may be >> proprietary, privileged or copyrighted under applicable law. If you are not >> the intended recipient, do not read, copy, or forward this email message or >> any attachments. Delete this email message and any attachments immediately. >> >> >> > > > > -- > Grant Likely, B.Sc., P.Eng. > Secret Lab Technologies Ltd. > -- Grant Likely, B.Sc., P.Eng. Secret Lab Technologies Ltd. _______________________________________________ Linuxppc-dev mailing list Linuxppc-dev@lists.ozlabs.org https://lists.ozlabs.org/listinfo/linuxppc-dev