aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--arch/microblaze/include/asm/system.h11
-rw-r--r--drivers/net/Kconfig2
-rw-r--r--drivers/net/ll_temac.h14
-rw-r--r--drivers/net/ll_temac_main.c137
4 files changed, 38 insertions, 126 deletions
diff --git a/arch/microblaze/include/asm/system.h b/arch/microblaze/include/asm/system.h
index 48c4f0335e3f..59efb3fef957 100644
--- a/arch/microblaze/include/asm/system.h
+++ b/arch/microblaze/include/asm/system.h
@@ -12,7 +12,6 @@
12#include <asm/registers.h> 12#include <asm/registers.h>
13#include <asm/setup.h> 13#include <asm/setup.h>
14#include <asm/irqflags.h> 14#include <asm/irqflags.h>
15#include <asm/cache.h>
16 15
17#include <asm-generic/cmpxchg.h> 16#include <asm-generic/cmpxchg.h>
18#include <asm-generic/cmpxchg-local.h> 17#include <asm-generic/cmpxchg-local.h>
@@ -97,14 +96,4 @@ extern struct dentry *of_debugfs_root;
97 96
98#define arch_align_stack(x) (x) 97#define arch_align_stack(x) (x)
99 98
100/*
101 * MicroBlaze doesn't handle unaligned accesses in hardware.
102 *
103 * Based on this we force the IP header alignment in network drivers.
104 * We also modify NET_SKB_PAD to be a cacheline in size, thus maintaining
105 * cacheline alignment of buffers.
106 */
107#define NET_IP_ALIGN 2
108#define NET_SKB_PAD L1_CACHE_BYTES
109
110#endif /* _ASM_MICROBLAZE_SYSTEM_H */ 99#endif /* _ASM_MICROBLAZE_SYSTEM_H */
diff --git a/drivers/net/Kconfig b/drivers/net/Kconfig
index dbd26f992158..49c372a5e0b7 100644
--- a/drivers/net/Kconfig
+++ b/drivers/net/Kconfig
@@ -2435,8 +2435,8 @@ config MV643XX_ETH
2435 2435
2436config XILINX_LL_TEMAC 2436config XILINX_LL_TEMAC
2437 tristate "Xilinx LL TEMAC (LocalLink Tri-mode Ethernet MAC) driver" 2437 tristate "Xilinx LL TEMAC (LocalLink Tri-mode Ethernet MAC) driver"
2438 depends on PPC || MICROBLAZE
2439 select PHYLIB 2438 select PHYLIB
2439 depends on PPC_DCR_NATIVE
2440 help 2440 help
2441 This driver supports the Xilinx 10/100/1000 LocalLink TEMAC 2441 This driver supports the Xilinx 10/100/1000 LocalLink TEMAC
2442 core used in Xilinx Spartan and Virtex FPGAs 2442 core used in Xilinx Spartan and Virtex FPGAs
diff --git a/drivers/net/ll_temac.h b/drivers/net/ll_temac.h
index c03358434acb..1af66a1e6911 100644
--- a/drivers/net/ll_temac.h
+++ b/drivers/net/ll_temac.h
@@ -5,11 +5,8 @@
5#include <linux/netdevice.h> 5#include <linux/netdevice.h>
6#include <linux/of.h> 6#include <linux/of.h>
7#include <linux/spinlock.h> 7#include <linux/spinlock.h>
8
9#ifdef CONFIG_PPC_DCR
10#include <asm/dcr.h> 8#include <asm/dcr.h>
11#include <asm/dcr-regs.h> 9#include <asm/dcr-regs.h>
12#endif
13 10
14/* packet size info */ 11/* packet size info */
15#define XTE_HDR_SIZE 14 /* size of Ethernet header */ 12#define XTE_HDR_SIZE 14 /* size of Ethernet header */
@@ -293,6 +290,9 @@ This option defaults to enabled (set) */
293 290
294#define TX_CONTROL_CALC_CSUM_MASK 1 291#define TX_CONTROL_CALC_CSUM_MASK 1
295 292
293#define XTE_ALIGN 32
294#define BUFFER_ALIGN(adr) ((XTE_ALIGN - ((u32) adr)) % XTE_ALIGN)
295
296#define MULTICAST_CAM_TABLE_NUM 4 296#define MULTICAST_CAM_TABLE_NUM 4
297 297
298/* TX/RX CURDESC_PTR points to first descriptor */ 298/* TX/RX CURDESC_PTR points to first descriptor */
@@ -335,15 +335,9 @@ struct temac_local {
335 struct mii_bus *mii_bus; /* MII bus reference */ 335 struct mii_bus *mii_bus; /* MII bus reference */
336 int mdio_irqs[PHY_MAX_ADDR]; /* IRQs table for MDIO bus */ 336 int mdio_irqs[PHY_MAX_ADDR]; /* IRQs table for MDIO bus */
337 337
338 /* IO registers, dma functions and IRQs */ 338 /* IO registers and IRQs */
339 void __iomem *regs; 339 void __iomem *regs;
340 void __iomem *sdma_regs;
341#ifdef CONFIG_PPC_DCR
342 dcr_host_t sdma_dcrs; 340 dcr_host_t sdma_dcrs;
343#endif
344 u32 (*dma_in)(struct temac_local *, int);
345 void (*dma_out)(struct temac_local *, int, u32);
346
347 int tx_irq; 341 int tx_irq;
348 int rx_irq; 342 int rx_irq;
349 int emac_num; 343 int emac_num;
diff --git a/drivers/net/ll_temac_main.c b/drivers/net/ll_temac_main.c
index b6edb2fd0a8a..30474d6b15c3 100644
--- a/drivers/net/ll_temac_main.c
+++ b/drivers/net/ll_temac_main.c
@@ -20,6 +20,9 @@
20 * or rx, so this should be okay. 20 * or rx, so this should be okay.
21 * 21 *
22 * TODO: 22 * TODO:
23 * - Fix driver to work on more than just Virtex5. Right now the driver
24 * assumes that the locallink DMA registers are accessed via DCR
25 * instructions.
23 * - Factor out locallink DMA code into separate driver 26 * - Factor out locallink DMA code into separate driver
24 * - Fix multicast assignment. 27 * - Fix multicast assignment.
25 * - Fix support for hardware checksumming. 28 * - Fix support for hardware checksumming.
@@ -112,86 +115,17 @@ void temac_indirect_out32(struct temac_local *lp, int reg, u32 value)
112 temac_iow(lp, XTE_CTL0_OFFSET, CNTLREG_WRITE_ENABLE_MASK | reg); 115 temac_iow(lp, XTE_CTL0_OFFSET, CNTLREG_WRITE_ENABLE_MASK | reg);
113} 116}
114 117
115/**
116 * temac_dma_in32 - Memory mapped DMA read, this function expects a
117 * register input that is based on DCR word addresses which
118 * are then converted to memory mapped byte addresses
119 */
120static u32 temac_dma_in32(struct temac_local *lp, int reg) 118static u32 temac_dma_in32(struct temac_local *lp, int reg)
121{ 119{
122 return in_be32((u32 *)(lp->sdma_regs + (reg << 2)));
123}
124
125/**
126 * temac_dma_out32 - Memory mapped DMA read, this function expects a
127 * register input that is based on DCR word addresses which
128 * are then converted to memory mapped byte addresses
129 */
130static void temac_dma_out32(struct temac_local *lp, int reg, u32 value)
131{
132 out_be32((u32 *)(lp->sdma_regs + (reg << 2)), value);
133}
134
135/* DMA register access functions can be DCR based or memory mapped.
136 * The PowerPC 440 is DCR based, the PowerPC 405 and MicroBlaze are both
137 * memory mapped.
138 */
139#ifdef CONFIG_PPC_DCR
140
141/**
142 * temac_dma_dcr_in32 - DCR based DMA read
143 */
144static u32 temac_dma_dcr_in(struct temac_local *lp, int reg)
145{
146 return dcr_read(lp->sdma_dcrs, reg); 120 return dcr_read(lp->sdma_dcrs, reg);
147} 121}
148 122
149/** 123static void temac_dma_out32(struct temac_local *lp, int reg, u32 value)
150 * temac_dma_dcr_out32 - DCR based DMA write
151 */
152static void temac_dma_dcr_out(struct temac_local *lp, int reg, u32 value)
153{ 124{
154 dcr_write(lp->sdma_dcrs, reg, value); 125 dcr_write(lp->sdma_dcrs, reg, value);
155} 126}
156 127
157/** 128/**
158 * temac_dcr_setup - If the DMA is DCR based, then setup the address and
159 * I/O functions
160 */
161static int temac_dcr_setup(struct temac_local *lp, struct of_device *op,
162 struct device_node *np)
163{
164 unsigned int dcrs;
165
166 /* setup the dcr address mapping if it's in the device tree */
167
168 dcrs = dcr_resource_start(np, 0);
169 if (dcrs != 0) {
170 lp->sdma_dcrs = dcr_map(np, dcrs, dcr_resource_len(np, 0));
171 lp->dma_in = temac_dma_dcr_in;
172 lp->dma_out = temac_dma_dcr_out;
173 dev_dbg(&op->dev, "DCR base: %x\n", dcrs);
174 return 0;
175 }
176 /* no DCR in the device tree, indicate a failure */
177 return -1;
178}
179
180#else
181
182/*
183 * temac_dcr_setup - This is a stub for when DCR is not supported,
184 * such as with MicroBlaze
185 */
186static int temac_dcr_setup(struct temac_local *lp, struct of_device *op,
187 struct device_node *np)
188{
189 return -1;
190}
191
192#endif
193
194/**
195 * temac_dma_bd_init - Setup buffer descriptor rings 129 * temac_dma_bd_init - Setup buffer descriptor rings
196 */ 130 */
197static int temac_dma_bd_init(struct net_device *ndev) 131static int temac_dma_bd_init(struct net_device *ndev)
@@ -221,14 +155,14 @@ static int temac_dma_bd_init(struct net_device *ndev)
221 lp->rx_bd_v[i].next = lp->rx_bd_p + 155 lp->rx_bd_v[i].next = lp->rx_bd_p +
222 sizeof(*lp->rx_bd_v) * ((i + 1) % RX_BD_NUM); 156 sizeof(*lp->rx_bd_v) * ((i + 1) % RX_BD_NUM);
223 157
224 skb = netdev_alloc_skb_ip_align(ndev, 158 skb = alloc_skb(XTE_MAX_JUMBO_FRAME_SIZE
225 XTE_MAX_JUMBO_FRAME_SIZE); 159 + XTE_ALIGN, GFP_ATOMIC);
226
227 if (skb == 0) { 160 if (skb == 0) {
228 dev_err(&ndev->dev, "alloc_skb error %d\n", i); 161 dev_err(&ndev->dev, "alloc_skb error %d\n", i);
229 return -1; 162 return -1;
230 } 163 }
231 lp->rx_skb[i] = skb; 164 lp->rx_skb[i] = skb;
165 skb_reserve(skb, BUFFER_ALIGN(skb->data));
232 /* returns physical address of skb->data */ 166 /* returns physical address of skb->data */
233 lp->rx_bd_v[i].phys = dma_map_single(ndev->dev.parent, 167 lp->rx_bd_v[i].phys = dma_map_single(ndev->dev.parent,
234 skb->data, 168 skb->data,
@@ -238,23 +172,23 @@ static int temac_dma_bd_init(struct net_device *ndev)
238 lp->rx_bd_v[i].app0 = STS_CTRL_APP0_IRQONEND; 172 lp->rx_bd_v[i].app0 = STS_CTRL_APP0_IRQONEND;
239 } 173 }
240 174
241 lp->dma_out(lp, TX_CHNL_CTRL, 0x10220400 | 175 temac_dma_out32(lp, TX_CHNL_CTRL, 0x10220400 |
242 CHNL_CTRL_IRQ_EN | 176 CHNL_CTRL_IRQ_EN |
243 CHNL_CTRL_IRQ_DLY_EN | 177 CHNL_CTRL_IRQ_DLY_EN |
244 CHNL_CTRL_IRQ_COAL_EN); 178 CHNL_CTRL_IRQ_COAL_EN);
245 /* 0x10220483 */ 179 /* 0x10220483 */
246 /* 0x00100483 */ 180 /* 0x00100483 */
247 lp->dma_out(lp, RX_CHNL_CTRL, 0xff010000 | 181 temac_dma_out32(lp, RX_CHNL_CTRL, 0xff010000 |
248 CHNL_CTRL_IRQ_EN | 182 CHNL_CTRL_IRQ_EN |
249 CHNL_CTRL_IRQ_DLY_EN | 183 CHNL_CTRL_IRQ_DLY_EN |
250 CHNL_CTRL_IRQ_COAL_EN | 184 CHNL_CTRL_IRQ_COAL_EN |
251 CHNL_CTRL_IRQ_IOE); 185 CHNL_CTRL_IRQ_IOE);
252 /* 0xff010283 */ 186 /* 0xff010283 */
253 187
254 lp->dma_out(lp, RX_CURDESC_PTR, lp->rx_bd_p); 188 temac_dma_out32(lp, RX_CURDESC_PTR, lp->rx_bd_p);
255 lp->dma_out(lp, RX_TAILDESC_PTR, 189 temac_dma_out32(lp, RX_TAILDESC_PTR,
256 lp->rx_bd_p + (sizeof(*lp->rx_bd_v) * (RX_BD_NUM - 1))); 190 lp->rx_bd_p + (sizeof(*lp->rx_bd_v) * (RX_BD_NUM - 1)));
257 lp->dma_out(lp, TX_CURDESC_PTR, lp->tx_bd_p); 191 temac_dma_out32(lp, TX_CURDESC_PTR, lp->tx_bd_p);
258 192
259 return 0; 193 return 0;
260} 194}
@@ -492,9 +426,9 @@ static void temac_device_reset(struct net_device *ndev)
492 temac_indirect_out32(lp, XTE_RXC1_OFFSET, val & ~XTE_RXC1_RXEN_MASK); 426 temac_indirect_out32(lp, XTE_RXC1_OFFSET, val & ~XTE_RXC1_RXEN_MASK);
493 427
494 /* Reset Local Link (DMA) */ 428 /* Reset Local Link (DMA) */
495 lp->dma_out(lp, DMA_CONTROL_REG, DMA_CONTROL_RST); 429 temac_dma_out32(lp, DMA_CONTROL_REG, DMA_CONTROL_RST);
496 timeout = 1000; 430 timeout = 1000;
497 while (lp->dma_in(lp, DMA_CONTROL_REG) & DMA_CONTROL_RST) { 431 while (temac_dma_in32(lp, DMA_CONTROL_REG) & DMA_CONTROL_RST) {
498 udelay(1); 432 udelay(1);
499 if (--timeout == 0) { 433 if (--timeout == 0) {
500 dev_err(&ndev->dev, 434 dev_err(&ndev->dev,
@@ -502,7 +436,7 @@ static void temac_device_reset(struct net_device *ndev)
502 break; 436 break;
503 } 437 }
504 } 438 }
505 lp->dma_out(lp, DMA_CONTROL_REG, DMA_TAIL_ENABLE); 439 temac_dma_out32(lp, DMA_CONTROL_REG, DMA_TAIL_ENABLE);
506 440
507 temac_dma_bd_init(ndev); 441 temac_dma_bd_init(ndev);
508 442
@@ -663,7 +597,7 @@ static int temac_start_xmit(struct sk_buff *skb, struct net_device *ndev)
663 lp->tx_bd_tail = 0; 597 lp->tx_bd_tail = 0;
664 598
665 /* Kick off the transfer */ 599 /* Kick off the transfer */
666 lp->dma_out(lp, TX_TAILDESC_PTR, tail_p); /* DMA start */ 600 temac_dma_out32(lp, TX_TAILDESC_PTR, tail_p); /* DMA start */
667 601
668 return NETDEV_TX_OK; 602 return NETDEV_TX_OK;
669} 603}
@@ -705,15 +639,16 @@ static void ll_temac_recv(struct net_device *ndev)
705 ndev->stats.rx_packets++; 639 ndev->stats.rx_packets++;
706 ndev->stats.rx_bytes += length; 640 ndev->stats.rx_bytes += length;
707 641
708 new_skb = netdev_alloc_skb_ip_align(ndev, 642 new_skb = alloc_skb(XTE_MAX_JUMBO_FRAME_SIZE + XTE_ALIGN,
709 XTE_MAX_JUMBO_FRAME_SIZE); 643 GFP_ATOMIC);
710
711 if (new_skb == 0) { 644 if (new_skb == 0) {
712 dev_err(&ndev->dev, "no memory for new sk_buff\n"); 645 dev_err(&ndev->dev, "no memory for new sk_buff\n");
713 spin_unlock_irqrestore(&lp->rx_lock, flags); 646 spin_unlock_irqrestore(&lp->rx_lock, flags);
714 return; 647 return;
715 } 648 }
716 649
650 skb_reserve(new_skb, BUFFER_ALIGN(new_skb->data));
651
717 cur_p->app0 = STS_CTRL_APP0_IRQONEND; 652 cur_p->app0 = STS_CTRL_APP0_IRQONEND;
718 cur_p->phys = dma_map_single(ndev->dev.parent, new_skb->data, 653 cur_p->phys = dma_map_single(ndev->dev.parent, new_skb->data,
719 XTE_MAX_JUMBO_FRAME_SIZE, 654 XTE_MAX_JUMBO_FRAME_SIZE,
@@ -728,7 +663,7 @@ static void ll_temac_recv(struct net_device *ndev)
728 cur_p = &lp->rx_bd_v[lp->rx_bd_ci]; 663 cur_p = &lp->rx_bd_v[lp->rx_bd_ci];
729 bdstat = cur_p->app0; 664 bdstat = cur_p->app0;
730 } 665 }
731 lp->dma_out(lp, RX_TAILDESC_PTR, tail_p); 666 temac_dma_out32(lp, RX_TAILDESC_PTR, tail_p);
732 667
733 spin_unlock_irqrestore(&lp->rx_lock, flags); 668 spin_unlock_irqrestore(&lp->rx_lock, flags);
734} 669}
@@ -739,8 +674,8 @@ static irqreturn_t ll_temac_tx_irq(int irq, void *_ndev)
739 struct temac_local *lp = netdev_priv(ndev); 674 struct temac_local *lp = netdev_priv(ndev);
740 unsigned int status; 675 unsigned int status;
741 676
742 status = lp->dma_in(lp, TX_IRQ_REG); 677 status = temac_dma_in32(lp, TX_IRQ_REG);
743 lp->dma_out(lp, TX_IRQ_REG, status); 678 temac_dma_out32(lp, TX_IRQ_REG, status);
744 679
745 if (status & (IRQ_COAL | IRQ_DLY)) 680 if (status & (IRQ_COAL | IRQ_DLY))
746 temac_start_xmit_done(lp->ndev); 681 temac_start_xmit_done(lp->ndev);
@@ -757,8 +692,8 @@ static irqreturn_t ll_temac_rx_irq(int irq, void *_ndev)
757 unsigned int status; 692 unsigned int status;
758 693
759 /* Read and clear the status registers */ 694 /* Read and clear the status registers */
760 status = lp->dma_in(lp, RX_IRQ_REG); 695 status = temac_dma_in32(lp, RX_IRQ_REG);
761 lp->dma_out(lp, RX_IRQ_REG, status); 696 temac_dma_out32(lp, RX_IRQ_REG, status);
762 697
763 if (status & (IRQ_COAL | IRQ_DLY)) 698 if (status & (IRQ_COAL | IRQ_DLY))
764 ll_temac_recv(lp->ndev); 699 ll_temac_recv(lp->ndev);
@@ -859,7 +794,7 @@ static ssize_t temac_show_llink_regs(struct device *dev,
859 int i, len = 0; 794 int i, len = 0;
860 795
861 for (i = 0; i < 0x11; i++) 796 for (i = 0; i < 0x11; i++)
862 len += sprintf(buf + len, "%.8x%s", lp->dma_in(lp, i), 797 len += sprintf(buf + len, "%.8x%s", temac_dma_in32(lp, i),
863 (i % 8) == 7 ? "\n" : " "); 798 (i % 8) == 7 ? "\n" : " ");
864 len += sprintf(buf + len, "\n"); 799 len += sprintf(buf + len, "\n");
865 800
@@ -885,6 +820,7 @@ temac_of_probe(struct of_device *op, const struct of_device_id *match)
885 struct net_device *ndev; 820 struct net_device *ndev;
886 const void *addr; 821 const void *addr;
887 int size, rc = 0; 822 int size, rc = 0;
823 unsigned int dcrs;
888 824
889 /* Init network device structure */ 825 /* Init network device structure */
890 ndev = alloc_etherdev(sizeof(*lp)); 826 ndev = alloc_etherdev(sizeof(*lp));
@@ -934,20 +870,13 @@ temac_of_probe(struct of_device *op, const struct of_device_id *match)
934 goto nodev; 870 goto nodev;
935 } 871 }
936 872
937 /* Setup the DMA register accesses, could be DCR or memory mapped */ 873 dcrs = dcr_resource_start(np, 0);
938 if (temac_dcr_setup(lp, op, np)) { 874 if (dcrs == 0) {
939 875 dev_err(&op->dev, "could not get DMA register address\n");
940 /* no DCR in the device tree, try non-DCR */ 876 goto nodev;
941 lp->sdma_regs = of_iomap(np, 0);
942 if (lp->sdma_regs) {
943 lp->dma_in = temac_dma_in32;
944 lp->dma_out = temac_dma_out32;
945 dev_dbg(&op->dev, "MEM base: %p\n", lp->sdma_regs);
946 } else {
947 dev_err(&op->dev, "unable to map DMA registers\n");
948 goto nodev;
949 }
950 } 877 }
878 lp->sdma_dcrs = dcr_map(np, dcrs, dcr_resource_len(np, 0));
879 dev_dbg(&op->dev, "DCR base: %x\n", dcrs);
951 880
952 lp->rx_irq = irq_of_parse_and_map(np, 0); 881 lp->rx_irq = irq_of_parse_and_map(np, 0);
953 lp->tx_irq = irq_of_parse_and_map(np, 1); 882 lp->tx_irq = irq_of_parse_and_map(np, 1);