mv643xx_eth.c 72.3 KB
Newer Older
L
Linus Torvalds 已提交
1
/*
2
 * Driver for Marvell Discovery (MV643XX) and Marvell Orion ethernet ports
L
Linus Torvalds 已提交
3 4 5
 * Copyright (C) 2002 Matthew Dharm <mdharm@momenco.com>
 *
 * Based on the 64360 driver from:
6 7
 * Copyright (C) 2002 Rabeeh Khoury <rabeeh@galileo.co.il>
 *		      Rabeeh Khoury <rabeeh@marvell.com>
L
Linus Torvalds 已提交
8 9
 *
 * Copyright (C) 2003 PMC-Sierra, Inc.,
10
 *	written by Manish Lachwani
L
Linus Torvalds 已提交
11 12 13
 *
 * Copyright (C) 2003 Ralf Baechle <ralf@linux-mips.org>
 *
14
 * Copyright (C) 2004-2006 MontaVista Software, Inc.
L
Linus Torvalds 已提交
15 16 17 18 19
 *			   Dale Farnsworth <dale@farnsworth.org>
 *
 * Copyright (C) 2004 Steven J. Hill <sjhill1@rockwellcollins.com>
 *				     <sjhill@realitydiluted.com>
 *
20 21 22
 * Copyright (C) 2007-2008 Marvell Semiconductor
 *			   Lennert Buytenhek <buytenh@marvell.com>
 *
L
Linus Torvalds 已提交
23 24 25 26 27 28 29 30 31 32 33 34 35 36
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License
 * as published by the Free Software Foundation; either version 2
 * of the License, or (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 */
37

38 39
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt

L
Linus Torvalds 已提交
40 41
#include <linux/init.h>
#include <linux/dma-mapping.h>
42
#include <linux/in.h>
43
#include <linux/ip.h>
L
Linus Torvalds 已提交
44 45 46 47 48
#include <linux/tcp.h>
#include <linux/udp.h>
#include <linux/etherdevice.h>
#include <linux/delay.h>
#include <linux/ethtool.h>
49
#include <linux/platform_device.h>
50 51 52 53
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/spinlock.h>
#include <linux/workqueue.h>
54
#include <linux/phy.h>
55
#include <linux/mv643xx_eth.h>
L
Lennert Buytenhek 已提交
56 57
#include <linux/io.h>
#include <linux/types.h>
58
#include <linux/inet_lro.h>
59
#include <linux/slab.h>
60
#include <linux/clk.h>
61

62
static char mv643xx_eth_driver_name[] = "mv643xx_eth";
63
static char mv643xx_eth_driver_version[] = "1.4";
64

65 66 67 68

/*
 * Registers shared between all ports.
 */
69 70
#define PHY_ADDR			0x0000
#define SMI_REG				0x0004
71 72 73 74 75 76 77
#define  SMI_BUSY			0x10000000
#define  SMI_READ_VALID			0x08000000
#define  SMI_OPCODE_READ		0x04000000
#define  SMI_OPCODE_WRITE		0x00000000
#define ERR_INT_CAUSE			0x0080
#define  ERR_INT_SMI_DONE		0x00000010
#define ERR_INT_MASK			0x0084
78 79 80 81 82
#define WINDOW_BASE(w)			(0x0200 + ((w) << 3))
#define WINDOW_SIZE(w)			(0x0204 + ((w) << 3))
#define WINDOW_REMAP_HIGH(w)		(0x0280 + ((w) << 2))
#define WINDOW_BAR_ENABLE		0x0290
#define WINDOW_PROTECT(w)		(0x0294 + ((w) << 4))
83 84

/*
85 86
 * Main per-port registers.  These live at offset 0x0400 for
 * port #0, 0x0800 for port #1, and 0x0c00 for port #2.
87
 */
88
#define PORT_CONFIG			0x0000
89
#define  UNICAST_PROMISCUOUS_MODE	0x00000001
90 91 92 93
#define PORT_CONFIG_EXT			0x0004
#define MAC_ADDR_LOW			0x0014
#define MAC_ADDR_HIGH			0x0018
#define SDMA_CONFIG			0x001c
94 95 96 97 98 99
#define  TX_BURST_SIZE_16_64BIT		0x01000000
#define  TX_BURST_SIZE_4_64BIT		0x00800000
#define  BLM_TX_NO_SWAP			0x00000020
#define  BLM_RX_NO_SWAP			0x00000010
#define  RX_BURST_SIZE_16_64BIT		0x00000008
#define  RX_BURST_SIZE_4_64BIT		0x00000004
100
#define PORT_SERIAL_CONTROL		0x003c
101 102 103 104 105 106 107 108 109 110 111
#define  SET_MII_SPEED_TO_100		0x01000000
#define  SET_GMII_SPEED_TO_1000		0x00800000
#define  SET_FULL_DUPLEX_MODE		0x00200000
#define  MAX_RX_PACKET_9700BYTE		0x000a0000
#define  DISABLE_AUTO_NEG_SPEED_GMII	0x00002000
#define  DO_NOT_FORCE_LINK_FAIL		0x00000400
#define  SERIAL_PORT_CONTROL_RESERVED	0x00000200
#define  DISABLE_AUTO_NEG_FOR_FLOW_CTRL	0x00000008
#define  DISABLE_AUTO_NEG_FOR_DUPLEX	0x00000004
#define  FORCE_LINK_PASS		0x00000002
#define  SERIAL_PORT_ENABLE		0x00000001
112
#define PORT_STATUS			0x0044
113
#define  TX_FIFO_EMPTY			0x00000400
114
#define  TX_IN_PROGRESS			0x00000080
115 116 117 118 119 120
#define  PORT_SPEED_MASK		0x00000030
#define  PORT_SPEED_1000		0x00000010
#define  PORT_SPEED_100			0x00000020
#define  PORT_SPEED_10			0x00000000
#define  FLOW_CONTROL_ENABLED		0x00000008
#define  FULL_DUPLEX			0x00000004
121
#define  LINK_UP			0x00000002
122 123 124 125 126 127
#define TXQ_COMMAND			0x0048
#define TXQ_FIX_PRIO_CONF		0x004c
#define TX_BW_RATE			0x0050
#define TX_BW_MTU			0x0058
#define TX_BW_BURST			0x005c
#define INT_CAUSE			0x0060
128
#define  INT_TX_END			0x07f80000
129
#define  INT_TX_END_0			0x00080000
130
#define  INT_RX				0x000003fc
131
#define  INT_RX_0			0x00000004
132
#define  INT_EXT			0x00000002
133
#define INT_CAUSE_EXT			0x0064
134 135
#define  INT_EXT_LINK_PHY		0x00110000
#define  INT_EXT_TX			0x000000ff
136 137 138
#define INT_MASK			0x0068
#define INT_MASK_EXT			0x006c
#define TX_FIFO_URGENT_THRESHOLD	0x0074
139 140
#define RX_DISCARD_FRAME_CNT		0x0084
#define RX_OVERRUN_FRAME_CNT		0x0088
141 142 143 144 145 146 147 148 149 150 151 152 153 154
#define TXQ_FIX_PRIO_CONF_MOVED		0x00dc
#define TX_BW_RATE_MOVED		0x00e0
#define TX_BW_MTU_MOVED			0x00e8
#define TX_BW_BURST_MOVED		0x00ec
#define RXQ_CURRENT_DESC_PTR(q)		(0x020c + ((q) << 4))
#define RXQ_COMMAND			0x0280
#define TXQ_CURRENT_DESC_PTR(q)		(0x02c0 + ((q) << 2))
#define TXQ_BW_TOKENS(q)		(0x0300 + ((q) << 4))
#define TXQ_BW_CONF(q)			(0x0304 + ((q) << 4))
#define TXQ_BW_WRR_CONF(q)		(0x0308 + ((q) << 4))

/*
 * Misc per-port registers.
 */
155 156 157 158
#define MIB_COUNTERS(p)			(0x1000 + ((p) << 7))
#define SPECIAL_MCAST_TABLE(p)		(0x1400 + ((p) << 10))
#define OTHER_MCAST_TABLE(p)		(0x1500 + ((p) << 10))
#define UNICAST_TABLE(p)		(0x1600 + ((p) << 10))
159

160 161

/*
162
 * SDMA configuration register default value.
163
 */
164 165
#if defined(__BIG_ENDIAN)
#define PORT_SDMA_CONFIG_DEFAULT_VALUE		\
166 167
		(RX_BURST_SIZE_4_64BIT	|	\
		 TX_BURST_SIZE_4_64BIT)
168 169
#elif defined(__LITTLE_ENDIAN)
#define PORT_SDMA_CONFIG_DEFAULT_VALUE		\
170 171 172 173
		(RX_BURST_SIZE_4_64BIT	|	\
		 BLM_RX_NO_SWAP		|	\
		 BLM_TX_NO_SWAP		|	\
		 TX_BURST_SIZE_4_64BIT)
174 175 176 177
#else
#error One of __BIG_ENDIAN or __LITTLE_ENDIAN must be defined
#endif

178 179

/*
180
 * Misc definitions.
181
 */
182 183
#define DEFAULT_RX_QUEUE_SIZE	128
#define DEFAULT_TX_QUEUE_SIZE	256
184
#define SKB_DMA_REALIGN		((PAGE_SIZE - NET_SKB_PAD) % SMP_CACHE_BYTES)
185 186


187 188
/*
 * RX/TX descriptors.
189 190
 */
#if defined(__BIG_ENDIAN)
191
struct rx_desc {
192 193 194 195 196 197 198
	u16 byte_cnt;		/* Descriptor buffer byte count		*/
	u16 buf_size;		/* Buffer size				*/
	u32 cmd_sts;		/* Descriptor command status		*/
	u32 next_desc_ptr;	/* Next descriptor pointer		*/
	u32 buf_ptr;		/* Descriptor buffer pointer		*/
};

199
struct tx_desc {
200 201 202 203 204 205 206
	u16 byte_cnt;		/* buffer byte count			*/
	u16 l4i_chk;		/* CPU provided TCP checksum		*/
	u32 cmd_sts;		/* Command/status field			*/
	u32 next_desc_ptr;	/* Pointer to next descriptor		*/
	u32 buf_ptr;		/* pointer to buffer for this descriptor*/
};
#elif defined(__LITTLE_ENDIAN)
207
struct rx_desc {
208 209 210 211 212 213 214
	u32 cmd_sts;		/* Descriptor command status		*/
	u16 buf_size;		/* Buffer size				*/
	u16 byte_cnt;		/* Descriptor buffer byte count		*/
	u32 buf_ptr;		/* Descriptor buffer pointer		*/
	u32 next_desc_ptr;	/* Next descriptor pointer		*/
};

215
struct tx_desc {
216 217 218 219 220 221 222 223 224 225
	u32 cmd_sts;		/* Command/status field			*/
	u16 l4i_chk;		/* CPU provided TCP checksum		*/
	u16 byte_cnt;		/* buffer byte count			*/
	u32 buf_ptr;		/* pointer to buffer for this descriptor*/
	u32 next_desc_ptr;	/* Pointer to next descriptor		*/
};
#else
#error One of __BIG_ENDIAN or __LITTLE_ENDIAN must be defined
#endif

226
/* RX & TX descriptor command */
227
#define BUFFER_OWNED_BY_DMA		0x80000000
228 229

/* RX & TX descriptor status */
230
#define ERROR_SUMMARY			0x00000001
231 232

/* RX descriptor status */
233 234 235 236
#define LAYER_4_CHECKSUM_OK		0x40000000
#define RX_ENABLE_INTERRUPT		0x20000000
#define RX_FIRST_DESC			0x08000000
#define RX_LAST_DESC			0x04000000
237 238 239 240 241 242
#define RX_IP_HDR_OK			0x02000000
#define RX_PKT_IS_IPV4			0x01000000
#define RX_PKT_IS_ETHERNETV2		0x00800000
#define RX_PKT_LAYER4_TYPE_MASK		0x00600000
#define RX_PKT_LAYER4_TYPE_TCP_IPV4	0x00000000
#define RX_PKT_IS_VLAN_TAGGED		0x00080000
243 244

/* TX descriptor command */
245 246 247 248 249 250 251 252
#define TX_ENABLE_INTERRUPT		0x00800000
#define GEN_CRC				0x00400000
#define TX_FIRST_DESC			0x00200000
#define TX_LAST_DESC			0x00100000
#define ZERO_PADDING			0x00080000
#define GEN_IP_V4_CHECKSUM		0x00040000
#define GEN_TCP_UDP_CHECKSUM		0x00020000
#define UDP_FRAME			0x00010000
253 254
#define MAC_HDR_EXTRA_4_BYTES		0x00008000
#define MAC_HDR_EXTRA_8_BYTES		0x00000200
255

256
#define TX_IHL_SHIFT			11
257 258


259
/* global *******************************************************************/
260
struct mv643xx_eth_shared_private {
L
Lennert Buytenhek 已提交
261 262 263
	/*
	 * Ethernet controller base address.
	 */
264
	void __iomem *base;
265

266 267 268 269 270
	/*
	 * Points at the right SMI instance to use.
	 */
	struct mv643xx_eth_shared_private *smi;

L
Lennert Buytenhek 已提交
271
	/*
272
	 * Provides access to local SMI interface.
L
Lennert Buytenhek 已提交
273
	 */
274
	struct mii_bus *smi_bus;
275

276 277 278 279 280 281 282 283 284
	/*
	 * If we have access to the error interrupt pin (which is
	 * somewhat misnamed as it not only reflects internal errors
	 * but also reflects SMI completion), use that to wait for
	 * SMI access completion instead of polling the SMI busy bit.
	 */
	int err_interrupt;
	wait_queue_head_t smi_busy_wait;

L
Lennert Buytenhek 已提交
285 286 287
	/*
	 * Per-port MBUS window access register value.
	 */
288 289
	u32 win_protect;

L
Lennert Buytenhek 已提交
290 291 292
	/*
	 * Hardware-specific parameters.
	 */
293
	int extended_rx_coal_limit;
294
	int tx_bw_control;
295
	int tx_csum_limit;
296

297 298
};

299 300 301 302
#define TX_BW_CONTROL_ABSENT		0
#define TX_BW_CONTROL_OLD_LAYOUT	1
#define TX_BW_CONTROL_NEW_LAYOUT	2

303 304 305
static int mv643xx_eth_open(struct net_device *dev);
static int mv643xx_eth_stop(struct net_device *dev);

306 307

/* per-port *****************************************************************/
308
struct mib_counters {
309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338
	u64 good_octets_received;
	u32 bad_octets_received;
	u32 internal_mac_transmit_err;
	u32 good_frames_received;
	u32 bad_frames_received;
	u32 broadcast_frames_received;
	u32 multicast_frames_received;
	u32 frames_64_octets;
	u32 frames_65_to_127_octets;
	u32 frames_128_to_255_octets;
	u32 frames_256_to_511_octets;
	u32 frames_512_to_1023_octets;
	u32 frames_1024_to_max_octets;
	u64 good_octets_sent;
	u32 good_frames_sent;
	u32 excessive_collision;
	u32 multicast_frames_sent;
	u32 broadcast_frames_sent;
	u32 unrec_mac_control_received;
	u32 fc_sent;
	u32 good_fc_received;
	u32 bad_fc_received;
	u32 undersize_received;
	u32 fragments_received;
	u32 oversize_received;
	u32 jabber_received;
	u32 mac_receive_error;
	u32 bad_crc_event;
	u32 collision;
	u32 late_collision;
339 340 341
	/* Non MIB hardware counters */
	u32 rx_discard;
	u32 rx_overrun;
342 343
};

344 345 346 347 348 349
struct lro_counters {
	u32 lro_aggregated;
	u32 lro_flushed;
	u32 lro_no_desc;
};

350
struct rx_queue {
351 352
	int index;

353 354 355 356 357 358 359 360 361 362
	int rx_ring_size;

	int rx_desc_count;
	int rx_curr_desc;
	int rx_used_desc;

	struct rx_desc *rx_desc_area;
	dma_addr_t rx_desc_dma;
	int rx_desc_area_size;
	struct sk_buff **rx_skb;
363 364 365

	struct net_lro_mgr lro_mgr;
	struct net_lro_desc lro_arr[8];
366 367
};

368
struct tx_queue {
369 370
	int index;

371
	int tx_ring_size;
372

373 374 375
	int tx_desc_count;
	int tx_curr_desc;
	int tx_used_desc;
376

377
	struct tx_desc *tx_desc_area;
378 379
	dma_addr_t tx_desc_dma;
	int tx_desc_area_size;
380 381

	struct sk_buff_head tx_skb;
382 383 384 385

	unsigned long tx_packets;
	unsigned long tx_bytes;
	unsigned long tx_dropped;
386 387 388 389
};

struct mv643xx_eth_private {
	struct mv643xx_eth_shared_private *shared;
390
	void __iomem *base;
L
Lennert Buytenhek 已提交
391
	int port_num;
392

L
Lennert Buytenhek 已提交
393
	struct net_device *dev;
394

395
	struct phy_device *phy;
396

397 398
	struct timer_list mib_counters_timer;
	spinlock_t mib_counters_lock;
L
Lennert Buytenhek 已提交
399
	struct mib_counters mib_counters;
400

401 402
	struct lro_counters lro_counters;

L
Lennert Buytenhek 已提交
403
	struct work_struct tx_timeout_task;
404

405
	struct napi_struct napi;
406
	u32 int_mask;
407
	u8 oom;
408 409 410 411 412 413
	u8 work_link;
	u8 work_tx;
	u8 work_tx_end;
	u8 work_rx;
	u8 work_rx_refill;

414 415
	int skb_size;

416 417 418
	/*
	 * RX state.
	 */
419
	int rx_ring_size;
420 421
	unsigned long rx_desc_sram_addr;
	int rx_desc_sram_size;
422
	int rxq_count;
423
	struct timer_list rx_oom;
424
	struct rx_queue rxq[8];
425 426 427 428

	/*
	 * TX state.
	 */
429
	int tx_ring_size;
430 431
	unsigned long tx_desc_sram_addr;
	int tx_desc_sram_size;
432
	int txq_count;
433
	struct tx_queue txq[8];
434 435 436 437

	/*
	 * Hardware-specific parameters.
	 */
438
#if defined(CONFIG_HAVE_CLK)
439
	struct clk *clk;
440
#endif
441
	unsigned int t_clk;
442
};
L
Linus Torvalds 已提交
443

444

445
/* port register accessors **************************************************/
446
static inline u32 rdl(struct mv643xx_eth_private *mp, int offset)
447
{
448
	return readl(mp->shared->base + offset);
449
}
450

451 452 453 454 455
static inline u32 rdlp(struct mv643xx_eth_private *mp, int offset)
{
	return readl(mp->base + offset);
}

456
static inline void wrl(struct mv643xx_eth_private *mp, int offset, u32 data)
457
{
458
	writel(data, mp->shared->base + offset);
459
}
460

461 462 463 464 465
static inline void wrlp(struct mv643xx_eth_private *mp, int offset, u32 data)
{
	writel(data, mp->base + offset);
}

466

467
/* rxq/txq helper functions *************************************************/
468
static struct mv643xx_eth_private *rxq_to_mp(struct rx_queue *rxq)
469
{
470
	return container_of(rxq, struct mv643xx_eth_private, rxq[rxq->index]);
471
}
472

473 474
static struct mv643xx_eth_private *txq_to_mp(struct tx_queue *txq)
{
475
	return container_of(txq, struct mv643xx_eth_private, txq[txq->index]);
476 477
}

478
static void rxq_enable(struct rx_queue *rxq)
479
{
480
	struct mv643xx_eth_private *mp = rxq_to_mp(rxq);
481
	wrlp(mp, RXQ_COMMAND, 1 << rxq->index);
482
}
L
Linus Torvalds 已提交
483

484 485 486
static void rxq_disable(struct rx_queue *rxq)
{
	struct mv643xx_eth_private *mp = rxq_to_mp(rxq);
487
	u8 mask = 1 << rxq->index;
L
Linus Torvalds 已提交
488

489 490
	wrlp(mp, RXQ_COMMAND, mask << 8);
	while (rdlp(mp, RXQ_COMMAND) & mask)
491
		udelay(10);
492 493
}

494 495 496 497 498 499 500
static void txq_reset_hw_ptr(struct tx_queue *txq)
{
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
	u32 addr;

	addr = (u32)txq->tx_desc_dma;
	addr += txq->tx_curr_desc * sizeof(struct tx_desc);
501
	wrlp(mp, TXQ_CURRENT_DESC_PTR(txq->index), addr);
502 503
}

504
static void txq_enable(struct tx_queue *txq)
L
Linus Torvalds 已提交
505
{
506
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
507
	wrlp(mp, TXQ_COMMAND, 1 << txq->index);
L
Linus Torvalds 已提交
508 509
}

510
static void txq_disable(struct tx_queue *txq)
L
Linus Torvalds 已提交
511
{
512
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
513
	u8 mask = 1 << txq->index;
514

515 516
	wrlp(mp, TXQ_COMMAND, mask << 8);
	while (rdlp(mp, TXQ_COMMAND) & mask)
517 518 519
		udelay(10);
}

520
static void txq_maybe_wake(struct tx_queue *txq)
521 522
{
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
523
	struct netdev_queue *nq = netdev_get_tx_queue(mp->dev, txq->index);
524

525 526 527 528 529 530
	if (netif_tx_queue_stopped(nq)) {
		__netif_tx_lock(nq, smp_processor_id());
		if (txq->tx_ring_size - txq->tx_desc_count >= MAX_SKB_FRAGS + 1)
			netif_tx_wake_queue(nq);
		__netif_tx_unlock(nq);
	}
L
Linus Torvalds 已提交
531 532
}

533

534
/* rx napi ******************************************************************/
535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560
static int
mv643xx_get_skb_header(struct sk_buff *skb, void **iphdr, void **tcph,
		       u64 *hdr_flags, void *priv)
{
	unsigned long cmd_sts = (unsigned long)priv;

	/*
	 * Make sure that this packet is Ethernet II, is not VLAN
	 * tagged, is IPv4, has a valid IP header, and is TCP.
	 */
	if ((cmd_sts & (RX_IP_HDR_OK | RX_PKT_IS_IPV4 |
		       RX_PKT_IS_ETHERNETV2 | RX_PKT_LAYER4_TYPE_MASK |
		       RX_PKT_IS_VLAN_TAGGED)) !=
	    (RX_IP_HDR_OK | RX_PKT_IS_IPV4 |
	     RX_PKT_IS_ETHERNETV2 | RX_PKT_LAYER4_TYPE_TCP_IPV4))
		return -1;

	skb_reset_network_header(skb);
	skb_set_transport_header(skb, ip_hdrlen(skb));
	*iphdr = ip_hdr(skb);
	*tcph = tcp_hdr(skb);
	*hdr_flags = LRO_IPV4 | LRO_TCP;

	return 0;
}

561
static int rxq_process(struct rx_queue *rxq, int budget)
L
Linus Torvalds 已提交
562
{
563 564
	struct mv643xx_eth_private *mp = rxq_to_mp(rxq);
	struct net_device_stats *stats = &mp->dev->stats;
565
	int lro_flush_needed;
566
	int rx;
L
Linus Torvalds 已提交
567

568
	lro_flush_needed = 0;
569
	rx = 0;
570
	while (rx < budget && rxq->rx_desc_count) {
L
Lennert Buytenhek 已提交
571
		struct rx_desc *rx_desc;
572
		unsigned int cmd_sts;
L
Lennert Buytenhek 已提交
573
		struct sk_buff *skb;
574
		u16 byte_cnt;
575

576
		rx_desc = &rxq->rx_desc_area[rxq->rx_curr_desc];
L
Linus Torvalds 已提交
577

578
		cmd_sts = rx_desc->cmd_sts;
579
		if (cmd_sts & BUFFER_OWNED_BY_DMA)
580 581
			break;
		rmb();
L
Linus Torvalds 已提交
582

583 584
		skb = rxq->rx_skb[rxq->rx_curr_desc];
		rxq->rx_skb[rxq->rx_curr_desc] = NULL;
585

586 587 588
		rxq->rx_curr_desc++;
		if (rxq->rx_curr_desc == rxq->rx_ring_size)
			rxq->rx_curr_desc = 0;
589

590
		dma_unmap_single(mp->dev->dev.parent, rx_desc->buf_ptr,
591
				 rx_desc->buf_size, DMA_FROM_DEVICE);
592 593
		rxq->rx_desc_count--;
		rx++;
594

595 596
		mp->work_rx_refill |= 1 << rxq->index;

597 598
		byte_cnt = rx_desc->byte_cnt;

599 600
		/*
		 * Update statistics.
L
Lennert Buytenhek 已提交
601 602 603 604 605
		 *
		 * Note that the descriptor byte count includes 2 dummy
		 * bytes automatically inserted by the hardware at the
		 * start of the packet (which we don't count), and a 4
		 * byte CRC at the end of the packet (which we do count).
606
		 */
L
Linus Torvalds 已提交
607
		stats->rx_packets++;
608
		stats->rx_bytes += byte_cnt - 2;
609

L
Linus Torvalds 已提交
610
		/*
L
Lennert Buytenhek 已提交
611 612 613
		 * In case we received a packet without first / last bits
		 * on, or the error summary bit is set, the packet needs
		 * to be dropped.
L
Linus Torvalds 已提交
614
		 */
615 616 617 618 619 620 621 622 623 624 625 626 627
		if ((cmd_sts & (RX_FIRST_DESC | RX_LAST_DESC | ERROR_SUMMARY))
			!= (RX_FIRST_DESC | RX_LAST_DESC))
			goto err;

		/*
		 * The -4 is for the CRC in the trailer of the
		 * received packet
		 */
		skb_put(skb, byte_cnt - 2 - 4);

		if (cmd_sts & LAYER_4_CHECKSUM_OK)
			skb->ip_summed = CHECKSUM_UNNECESSARY;
		skb->protocol = eth_type_trans(skb, mp->dev);
628 629 630 631 632 633 634

		if (skb->dev->features & NETIF_F_LRO &&
		    skb->ip_summed == CHECKSUM_UNNECESSARY) {
			lro_receive_skb(&rxq->lro_mgr, skb, (void *)cmd_sts);
			lro_flush_needed = 1;
		} else
			netif_receive_skb(skb);
635 636 637 638 639 640 641 642 643

		continue;

err:
		stats->rx_dropped++;

		if ((cmd_sts & (RX_FIRST_DESC | RX_LAST_DESC)) !=
			(RX_FIRST_DESC | RX_LAST_DESC)) {
			if (net_ratelimit())
644 645
				netdev_err(mp->dev,
					   "received packet spanning multiple descriptors\n");
L
Linus Torvalds 已提交
646
		}
647 648 649 650 651

		if (cmd_sts & ERROR_SUMMARY)
			stats->rx_errors++;

		dev_kfree_skb(skb);
L
Linus Torvalds 已提交
652
	}
L
Lennert Buytenhek 已提交
653

654 655 656
	if (lro_flush_needed)
		lro_flush_all(&rxq->lro_mgr);

657 658 659
	if (rx < budget)
		mp->work_rx &= ~(1 << rxq->index);

660
	return rx;
L
Linus Torvalds 已提交
661 662
}

663
static int rxq_refill(struct rx_queue *rxq, int budget)
664
{
665 666
	struct mv643xx_eth_private *mp = rxq_to_mp(rxq);
	int refilled;
667

668 669 670 671
	refilled = 0;
	while (refilled < budget && rxq->rx_desc_count < rxq->rx_ring_size) {
		struct sk_buff *skb;
		int rx;
672
		struct rx_desc *rx_desc;
673
		int size;
674

E
Eric Dumazet 已提交
675
		skb = netdev_alloc_skb(mp->dev, mp->skb_size);
676

677
		if (skb == NULL) {
678
			mp->oom = 1;
679 680
			goto oom;
		}
681

682 683
		if (SKB_DMA_REALIGN)
			skb_reserve(skb, SKB_DMA_REALIGN);
684

685 686
		refilled++;
		rxq->rx_desc_count++;
687

688 689 690
		rx = rxq->rx_used_desc++;
		if (rxq->rx_used_desc == rxq->rx_ring_size)
			rxq->rx_used_desc = 0;
691

692 693
		rx_desc = rxq->rx_desc_area + rx;

694
		size = skb->end - skb->data;
695
		rx_desc->buf_ptr = dma_map_single(mp->dev->dev.parent,
696
						  skb->data, size,
697
						  DMA_FROM_DEVICE);
698
		rx_desc->buf_size = size;
699 700
		rxq->rx_skb[rx] = skb;
		wmb();
701
		rx_desc->cmd_sts = BUFFER_OWNED_BY_DMA | RX_ENABLE_INTERRUPT;
702
		wmb();
703

704 705 706 707 708 709 710 711 712 713 714 715 716
		/*
		 * The hardware automatically prepends 2 bytes of
		 * dummy data to each received packet, so that the
		 * IP header ends up 16-byte aligned.
		 */
		skb_reserve(skb, 2);
	}

	if (refilled < budget)
		mp->work_rx_refill &= ~(1 << rxq->index);

oom:
	return refilled;
717 718
}

719 720 721

/* tx ***********************************************************************/
static inline unsigned int has_tiny_unaligned_frags(struct sk_buff *skb)
L
Linus Torvalds 已提交
722
{
723
	int frag;
L
Linus Torvalds 已提交
724

725
	for (frag = 0; frag < skb_shinfo(skb)->nr_frags; frag++) {
E
Eric Dumazet 已提交
726 727 728
		const skb_frag_t *fragp = &skb_shinfo(skb)->frags[frag];

		if (skb_frag_size(fragp) <= 8 && fragp->page_offset & 7)
729
			return 1;
L
Linus Torvalds 已提交
730
	}
731

732 733
	return 0;
}
734

735
static void txq_submit_frag_skb(struct tx_queue *txq, struct sk_buff *skb)
736
{
737
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
738
	int nr_frags = skb_shinfo(skb)->nr_frags;
739
	int frag;
L
Linus Torvalds 已提交
740

741 742 743 744 745 746
	for (frag = 0; frag < nr_frags; frag++) {
		skb_frag_t *this_frag;
		int tx_index;
		struct tx_desc *desc;

		this_frag = &skb_shinfo(skb)->frags[frag];
747 748 749
		tx_index = txq->tx_curr_desc++;
		if (txq->tx_curr_desc == txq->tx_ring_size)
			txq->tx_curr_desc = 0;
750 751 752 753 754 755 756 757 758 759 760 761 762 763
		desc = &txq->tx_desc_area[tx_index];

		/*
		 * The last fragment will generate an interrupt
		 * which will free the skb on TX completion.
		 */
		if (frag == nr_frags - 1) {
			desc->cmd_sts = BUFFER_OWNED_BY_DMA |
					ZERO_PADDING | TX_LAST_DESC |
					TX_ENABLE_INTERRUPT;
		} else {
			desc->cmd_sts = BUFFER_OWNED_BY_DMA;
		}

764
		desc->l4i_chk = 0;
E
Eric Dumazet 已提交
765
		desc->byte_cnt = skb_frag_size(this_frag);
766 767
		desc->buf_ptr = skb_frag_dma_map(mp->dev->dev.parent,
						 this_frag, 0,
E
Eric Dumazet 已提交
768
						 skb_frag_size(this_frag),
769
						 DMA_TO_DEVICE);
770
	}
L
Linus Torvalds 已提交
771 772
}

773 774 775 776
static inline __be16 sum16_as_be(__sum16 sum)
{
	return (__force __be16)sum;
}
L
Linus Torvalds 已提交
777

778
static int txq_submit_skb(struct tx_queue *txq, struct sk_buff *skb)
L
Linus Torvalds 已提交
779
{
780
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
781
	int nr_frags = skb_shinfo(skb)->nr_frags;
782
	int tx_index;
783
	struct tx_desc *desc;
784
	u32 cmd_sts;
785
	u16 l4i_chk;
786
	int length;
L
Linus Torvalds 已提交
787

788
	cmd_sts = TX_FIRST_DESC | GEN_CRC | BUFFER_OWNED_BY_DMA;
789
	l4i_chk = 0;
790 791

	if (skb->ip_summed == CHECKSUM_PARTIAL) {
792
		int hdr_len;
793
		int tag_bytes;
794 795 796

		BUG_ON(skb->protocol != htons(ETH_P_IP) &&
		       skb->protocol != htons(ETH_P_8021Q));
797

798 799 800 801
		hdr_len = (void *)ip_hdr(skb) - (void *)skb->data;
		tag_bytes = hdr_len - ETH_HLEN;
		if (skb->len - hdr_len > mp->shared->tx_csum_limit ||
		    unlikely(tag_bytes & ~12)) {
802 803 804 805 806
			if (skb_checksum_help(skb) == 0)
				goto no_csum;
			kfree_skb(skb);
			return 1;
		}
807

808
		if (tag_bytes & 4)
809
			cmd_sts |= MAC_HDR_EXTRA_4_BYTES;
810
		if (tag_bytes & 8)
811
			cmd_sts |= MAC_HDR_EXTRA_8_BYTES;
812 813 814 815

		cmd_sts |= GEN_TCP_UDP_CHECKSUM |
			   GEN_IP_V4_CHECKSUM   |
			   ip_hdr(skb)->ihl << TX_IHL_SHIFT;
816

817 818
		switch (ip_hdr(skb)->protocol) {
		case IPPROTO_UDP:
819
			cmd_sts |= UDP_FRAME;
820
			l4i_chk = ntohs(sum16_as_be(udp_hdr(skb)->check));
821 822
			break;
		case IPPROTO_TCP:
823
			l4i_chk = ntohs(sum16_as_be(tcp_hdr(skb)->check));
824 825 826 827 828
			break;
		default:
			BUG();
		}
	} else {
829
no_csum:
830
		/* Errata BTS #50, IHL must be 5 if no HW checksum */
831
		cmd_sts |= 5 << TX_IHL_SHIFT;
832 833
	}

834 835 836
	tx_index = txq->tx_curr_desc++;
	if (txq->tx_curr_desc == txq->tx_ring_size)
		txq->tx_curr_desc = 0;
837 838 839 840 841 842 843 844 845 846 847 848
	desc = &txq->tx_desc_area[tx_index];

	if (nr_frags) {
		txq_submit_frag_skb(txq, skb);
		length = skb_headlen(skb);
	} else {
		cmd_sts |= ZERO_PADDING | TX_LAST_DESC | TX_ENABLE_INTERRUPT;
		length = skb->len;
	}

	desc->l4i_chk = l4i_chk;
	desc->byte_cnt = length;
849 850
	desc->buf_ptr = dma_map_single(mp->dev->dev.parent, skb->data,
				       length, DMA_TO_DEVICE);
851

852 853
	__skb_queue_tail(&txq->tx_skb, skb);

854 855
	skb_tx_timestamp(skb);

856 857 858 859
	/* ensure all other descriptors are written before first cmd_sts */
	wmb();
	desc->cmd_sts = cmd_sts;

860 861
	/* clear TX_END status */
	mp->work_tx_end &= ~(1 << txq->index);
862

863 864
	/* ensure all descriptors are written before poking hardware */
	wmb();
865
	txq_enable(txq);
866

867
	txq->tx_desc_count += nr_frags + 1;
868 869

	return 0;
L
Linus Torvalds 已提交
870 871
}

D
Denis Kirjanov 已提交
872
static netdev_tx_t mv643xx_eth_xmit(struct sk_buff *skb, struct net_device *dev)
L
Linus Torvalds 已提交
873
{
874
	struct mv643xx_eth_private *mp = netdev_priv(dev);
875
	int length, queue;
876
	struct tx_queue *txq;
877
	struct netdev_queue *nq;
878

879 880 881 882
	queue = skb_get_queue_mapping(skb);
	txq = mp->txq + queue;
	nq = netdev_get_tx_queue(dev, queue);

883
	if (has_tiny_unaligned_frags(skb) && __skb_linearize(skb)) {
884
		txq->tx_dropped++;
885 886
		netdev_printk(KERN_DEBUG, dev,
			      "failed to linearize skb with tiny unaligned fragment\n");
887 888 889
		return NETDEV_TX_BUSY;
	}

890
	if (txq->tx_ring_size - txq->tx_desc_count < MAX_SKB_FRAGS + 1) {
891
		if (net_ratelimit())
892
			netdev_err(dev, "tx queue full?!\n");
893 894
		kfree_skb(skb);
		return NETDEV_TX_OK;
895 896
	}

897 898
	length = skb->len;

899 900 901
	if (!txq_submit_skb(txq, skb)) {
		int entries_left;

902
		txq->tx_bytes += length;
903
		txq->tx_packets++;
904

905 906 907 908
		entries_left = txq->tx_ring_size - txq->tx_desc_count;
		if (entries_left < MAX_SKB_FRAGS + 1)
			netif_tx_stop_queue(nq);
	}
909 910

	return NETDEV_TX_OK;
L
Linus Torvalds 已提交
911 912
}

913

914 915 916 917
/* tx napi ******************************************************************/
static void txq_kick(struct tx_queue *txq)
{
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
918
	struct netdev_queue *nq = netdev_get_tx_queue(mp->dev, txq->index);
919 920 921
	u32 hw_desc_ptr;
	u32 expected_ptr;

922
	__netif_tx_lock(nq, smp_processor_id());
923

924
	if (rdlp(mp, TXQ_COMMAND) & (1 << txq->index))
925 926
		goto out;

927
	hw_desc_ptr = rdlp(mp, TXQ_CURRENT_DESC_PTR(txq->index));
928 929 930 931 932 933 934
	expected_ptr = (u32)txq->tx_desc_dma +
				txq->tx_curr_desc * sizeof(struct tx_desc);

	if (hw_desc_ptr != expected_ptr)
		txq_enable(txq);

out:
935
	__netif_tx_unlock(nq);
936 937 938 939 940 941 942

	mp->work_tx_end &= ~(1 << txq->index);
}

static int txq_reclaim(struct tx_queue *txq, int budget, int force)
{
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
943
	struct netdev_queue *nq = netdev_get_tx_queue(mp->dev, txq->index);
944 945
	int reclaimed;

946
	__netif_tx_lock(nq, smp_processor_id());
947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971

	reclaimed = 0;
	while (reclaimed < budget && txq->tx_desc_count > 0) {
		int tx_index;
		struct tx_desc *desc;
		u32 cmd_sts;
		struct sk_buff *skb;

		tx_index = txq->tx_used_desc;
		desc = &txq->tx_desc_area[tx_index];
		cmd_sts = desc->cmd_sts;

		if (cmd_sts & BUFFER_OWNED_BY_DMA) {
			if (!force)
				break;
			desc->cmd_sts = cmd_sts & ~BUFFER_OWNED_BY_DMA;
		}

		txq->tx_used_desc = tx_index + 1;
		if (txq->tx_used_desc == txq->tx_ring_size)
			txq->tx_used_desc = 0;

		reclaimed++;
		txq->tx_desc_count--;

972 973 974
		skb = NULL;
		if (cmd_sts & TX_LAST_DESC)
			skb = __skb_dequeue(&txq->tx_skb);
975 976

		if (cmd_sts & ERROR_SUMMARY) {
977
			netdev_info(mp->dev, "tx error\n");
978 979 980
			mp->dev->stats.tx_errors++;
		}

981
		if (cmd_sts & TX_FIRST_DESC) {
982
			dma_unmap_single(mp->dev->dev.parent, desc->buf_ptr,
983 984
					 desc->byte_cnt, DMA_TO_DEVICE);
		} else {
985
			dma_unmap_page(mp->dev->dev.parent, desc->buf_ptr,
986 987
				       desc->byte_cnt, DMA_TO_DEVICE);
		}
988

E
Eric Dumazet 已提交
989
		dev_kfree_skb(skb);
990 991
	}

992 993
	__netif_tx_unlock(nq);

994 995 996 997 998 999 1000
	if (reclaimed < budget)
		mp->work_tx &= ~(1 << txq->index);

	return reclaimed;
}


1001 1002 1003 1004 1005 1006 1007 1008 1009 1010 1011
/* tx rate control **********************************************************/
/*
 * Set total maximum TX rate (shared by all TX queues for this port)
 * to 'rate' bits per second, with a maximum burst of 'burst' bytes.
 */
static void tx_set_rate(struct mv643xx_eth_private *mp, int rate, int burst)
{
	int token_rate;
	int mtu;
	int bucket_size;

1012
	token_rate = ((rate / 1000) * 64) / (mp->t_clk / 1000);
1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023
	if (token_rate > 1023)
		token_rate = 1023;

	mtu = (mp->dev->mtu + 255) >> 8;
	if (mtu > 63)
		mtu = 63;

	bucket_size = (burst + 255) >> 8;
	if (bucket_size > 65535)
		bucket_size = 65535;

1024 1025
	switch (mp->shared->tx_bw_control) {
	case TX_BW_CONTROL_OLD_LAYOUT:
1026 1027 1028
		wrlp(mp, TX_BW_RATE, token_rate);
		wrlp(mp, TX_BW_MTU, mtu);
		wrlp(mp, TX_BW_BURST, bucket_size);
1029 1030
		break;
	case TX_BW_CONTROL_NEW_LAYOUT:
1031 1032 1033
		wrlp(mp, TX_BW_RATE_MOVED, token_rate);
		wrlp(mp, TX_BW_MTU_MOVED, mtu);
		wrlp(mp, TX_BW_BURST_MOVED, bucket_size);
1034
		break;
1035
	}
1036 1037 1038 1039 1040 1041 1042 1043
}

static void txq_set_rate(struct tx_queue *txq, int rate, int burst)
{
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
	int token_rate;
	int bucket_size;

1044
	token_rate = ((rate / 1000) * 64) / (mp->t_clk / 1000);
1045 1046 1047 1048 1049 1050 1051
	if (token_rate > 1023)
		token_rate = 1023;

	bucket_size = (burst + 255) >> 8;
	if (bucket_size > 65535)
		bucket_size = 65535;

1052 1053
	wrlp(mp, TXQ_BW_TOKENS(txq->index), token_rate << 14);
	wrlp(mp, TXQ_BW_CONF(txq->index), (bucket_size << 10) | token_rate);
1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064
}

static void txq_set_fixed_prio_mode(struct tx_queue *txq)
{
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
	int off;
	u32 val;

	/*
	 * Turn on fixed priority mode.
	 */
1065 1066 1067
	off = 0;
	switch (mp->shared->tx_bw_control) {
	case TX_BW_CONTROL_OLD_LAYOUT:
1068
		off = TXQ_FIX_PRIO_CONF;
1069 1070
		break;
	case TX_BW_CONTROL_NEW_LAYOUT:
1071
		off = TXQ_FIX_PRIO_CONF_MOVED;
1072 1073
		break;
	}
1074

1075
	if (off) {
1076
		val = rdlp(mp, off);
1077
		val |= 1 << txq->index;
1078
		wrlp(mp, off, val);
1079
	}
1080 1081 1082
}


1083
/* mii management interface *************************************************/
1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121 1122
static void mv643xx_adjust_pscr(struct mv643xx_eth_private *mp)
{
	u32 pscr = rdlp(mp, PORT_SERIAL_CONTROL);
	u32 autoneg_disable = FORCE_LINK_PASS |
	             DISABLE_AUTO_NEG_SPEED_GMII |
		     DISABLE_AUTO_NEG_FOR_FLOW_CTRL |
		     DISABLE_AUTO_NEG_FOR_DUPLEX;

	if (mp->phy->autoneg == AUTONEG_ENABLE) {
		/* enable auto negotiation */
		pscr &= ~autoneg_disable;
		goto out_write;
	}

	pscr |= autoneg_disable;

	if (mp->phy->speed == SPEED_1000) {
		/* force gigabit, half duplex not supported */
		pscr |= SET_GMII_SPEED_TO_1000;
		pscr |= SET_FULL_DUPLEX_MODE;
		goto out_write;
	}

	pscr &= ~SET_GMII_SPEED_TO_1000;

	if (mp->phy->speed == SPEED_100)
		pscr |= SET_MII_SPEED_TO_100;
	else
		pscr &= ~SET_MII_SPEED_TO_100;

	if (mp->phy->duplex == DUPLEX_FULL)
		pscr |= SET_FULL_DUPLEX_MODE;
	else
		pscr &= ~SET_FULL_DUPLEX_MODE;

out_write:
	wrlp(mp, PORT_SERIAL_CONTROL, pscr);
}

1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134
static irqreturn_t mv643xx_eth_err_irq(int irq, void *dev_id)
{
	struct mv643xx_eth_shared_private *msp = dev_id;

	if (readl(msp->base + ERR_INT_CAUSE) & ERR_INT_SMI_DONE) {
		writel(~ERR_INT_SMI_DONE, msp->base + ERR_INT_CAUSE);
		wake_up(&msp->smi_busy_wait);
		return IRQ_HANDLED;
	}

	return IRQ_NONE;
}
1135

1136
static int smi_is_done(struct mv643xx_eth_shared_private *msp)
L
Linus Torvalds 已提交
1137
{
1138 1139
	return !(readl(msp->base + SMI_REG) & SMI_BUSY);
}
L
Linus Torvalds 已提交
1140

1141 1142 1143 1144
static int smi_wait_ready(struct mv643xx_eth_shared_private *msp)
{
	if (msp->err_interrupt == NO_IRQ) {
		int i;
1145

1146 1147 1148 1149
		for (i = 0; !smi_is_done(msp); i++) {
			if (i == 10)
				return -ETIMEDOUT;
			msleep(10);
1150
		}
1151 1152 1153 1154

		return 0;
	}

1155 1156 1157 1158 1159 1160
	if (!smi_is_done(msp)) {
		wait_event_timeout(msp->smi_busy_wait, smi_is_done(msp),
				   msecs_to_jiffies(100));
		if (!smi_is_done(msp))
			return -ETIMEDOUT;
	}
1161 1162 1163 1164

	return 0;
}

1165
static int smi_bus_read(struct mii_bus *bus, int addr, int reg)
1166
{
1167
	struct mv643xx_eth_shared_private *msp = bus->priv;
1168 1169 1170 1171
	void __iomem *smi_reg = msp->base + SMI_REG;
	int ret;

	if (smi_wait_ready(msp)) {
1172
		pr_warn("SMI bus busy timeout\n");
1173
		return -ETIMEDOUT;
L
Linus Torvalds 已提交
1174 1175
	}

L
Lennert Buytenhek 已提交
1176
	writel(SMI_OPCODE_READ | (reg << 21) | (addr << 16), smi_reg);
L
Linus Torvalds 已提交
1177

1178
	if (smi_wait_ready(msp)) {
1179
		pr_warn("SMI bus busy timeout\n");
1180
		return -ETIMEDOUT;
1181 1182 1183 1184
	}

	ret = readl(smi_reg);
	if (!(ret & SMI_READ_VALID)) {
1185
		pr_warn("SMI bus read not valid\n");
1186
		return -ENODEV;
1187 1188
	}

1189
	return ret & 0xffff;
L
Linus Torvalds 已提交
1190 1191
}

1192
static int smi_bus_write(struct mii_bus *bus, int addr, int reg, u16 val)
L
Linus Torvalds 已提交
1193
{
1194
	struct mv643xx_eth_shared_private *msp = bus->priv;
1195
	void __iomem *smi_reg = msp->base + SMI_REG;
L
Linus Torvalds 已提交
1196

1197
	if (smi_wait_ready(msp)) {
1198
		pr_warn("SMI bus busy timeout\n");
1199
		return -ETIMEDOUT;
L
Linus Torvalds 已提交
1200 1201
	}

L
Lennert Buytenhek 已提交
1202
	writel(SMI_OPCODE_WRITE | (reg << 21) |
1203
		(addr << 16) | (val & 0xffff), smi_reg);
1204

1205
	if (smi_wait_ready(msp)) {
1206
		pr_warn("SMI bus busy timeout\n");
1207 1208
		return -ETIMEDOUT;
	}
1209 1210

	return 0;
1211
}
L
Linus Torvalds 已提交
1212

1213

1214 1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238
/* statistics ***************************************************************/
static struct net_device_stats *mv643xx_eth_get_stats(struct net_device *dev)
{
	struct mv643xx_eth_private *mp = netdev_priv(dev);
	struct net_device_stats *stats = &dev->stats;
	unsigned long tx_packets = 0;
	unsigned long tx_bytes = 0;
	unsigned long tx_dropped = 0;
	int i;

	for (i = 0; i < mp->txq_count; i++) {
		struct tx_queue *txq = mp->txq + i;

		tx_packets += txq->tx_packets;
		tx_bytes += txq->tx_bytes;
		tx_dropped += txq->tx_dropped;
	}

	stats->tx_packets = tx_packets;
	stats->tx_bytes = tx_bytes;
	stats->tx_dropped = tx_dropped;

	return stats;
}

1239 1240 1241 1242 1243 1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258
static void mv643xx_eth_grab_lro_stats(struct mv643xx_eth_private *mp)
{
	u32 lro_aggregated = 0;
	u32 lro_flushed = 0;
	u32 lro_no_desc = 0;
	int i;

	for (i = 0; i < mp->rxq_count; i++) {
		struct rx_queue *rxq = mp->rxq + i;

		lro_aggregated += rxq->lro_mgr.stats.aggregated;
		lro_flushed += rxq->lro_mgr.stats.flushed;
		lro_no_desc += rxq->lro_mgr.stats.no_desc;
	}

	mp->lro_counters.lro_aggregated = lro_aggregated;
	mp->lro_counters.lro_flushed = lro_flushed;
	mp->lro_counters.lro_no_desc = lro_no_desc;
}

L
Lennert Buytenhek 已提交
1259
static inline u32 mib_read(struct mv643xx_eth_private *mp, int offset)
1260
{
L
Lennert Buytenhek 已提交
1261
	return rdl(mp, MIB_COUNTERS(mp->port_num) + offset);
L
Linus Torvalds 已提交
1262 1263
}

L
Lennert Buytenhek 已提交
1264
static void mib_counters_clear(struct mv643xx_eth_private *mp)
1265
{
L
Lennert Buytenhek 已提交
1266 1267 1268 1269
	int i;

	for (i = 0; i < 0x80; i += 4)
		mib_read(mp, i);
1270 1271 1272 1273

	/* Clear non MIB hw counters also */
	rdlp(mp, RX_DISCARD_FRAME_CNT);
	rdlp(mp, RX_OVERRUN_FRAME_CNT);
1274
}
1275

L
Lennert Buytenhek 已提交
1276
static void mib_counters_update(struct mv643xx_eth_private *mp)
1277
{
1278
	struct mib_counters *p = &mp->mib_counters;
1279

1280
	spin_lock_bh(&mp->mib_counters_lock);
L
Lennert Buytenhek 已提交
1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310
	p->good_octets_received += mib_read(mp, 0x00);
	p->bad_octets_received += mib_read(mp, 0x08);
	p->internal_mac_transmit_err += mib_read(mp, 0x0c);
	p->good_frames_received += mib_read(mp, 0x10);
	p->bad_frames_received += mib_read(mp, 0x14);
	p->broadcast_frames_received += mib_read(mp, 0x18);
	p->multicast_frames_received += mib_read(mp, 0x1c);
	p->frames_64_octets += mib_read(mp, 0x20);
	p->frames_65_to_127_octets += mib_read(mp, 0x24);
	p->frames_128_to_255_octets += mib_read(mp, 0x28);
	p->frames_256_to_511_octets += mib_read(mp, 0x2c);
	p->frames_512_to_1023_octets += mib_read(mp, 0x30);
	p->frames_1024_to_max_octets += mib_read(mp, 0x34);
	p->good_octets_sent += mib_read(mp, 0x38);
	p->good_frames_sent += mib_read(mp, 0x40);
	p->excessive_collision += mib_read(mp, 0x44);
	p->multicast_frames_sent += mib_read(mp, 0x48);
	p->broadcast_frames_sent += mib_read(mp, 0x4c);
	p->unrec_mac_control_received += mib_read(mp, 0x50);
	p->fc_sent += mib_read(mp, 0x54);
	p->good_fc_received += mib_read(mp, 0x58);
	p->bad_fc_received += mib_read(mp, 0x5c);
	p->undersize_received += mib_read(mp, 0x60);
	p->fragments_received += mib_read(mp, 0x64);
	p->oversize_received += mib_read(mp, 0x68);
	p->jabber_received += mib_read(mp, 0x6c);
	p->mac_receive_error += mib_read(mp, 0x70);
	p->bad_crc_event += mib_read(mp, 0x74);
	p->collision += mib_read(mp, 0x78);
	p->late_collision += mib_read(mp, 0x7c);
1311 1312 1313
	/* Non MIB hardware counters */
	p->rx_discard += rdlp(mp, RX_DISCARD_FRAME_CNT);
	p->rx_overrun += rdlp(mp, RX_OVERRUN_FRAME_CNT);
1314
	spin_unlock_bh(&mp->mib_counters_lock);
1315 1316 1317 1318 1319 1320 1321 1322 1323

	mod_timer(&mp->mib_counters_timer, jiffies + 30 * HZ);
}

static void mib_counters_timer_wrapper(unsigned long _mp)
{
	struct mv643xx_eth_private *mp = (void *)_mp;

	mib_counters_update(mp);
1324 1325
}

1326

1327 1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349
/* interrupt coalescing *****************************************************/
/*
 * Hardware coalescing parameters are set in units of 64 t_clk
 * cycles.  I.e.:
 *
 *	coal_delay_in_usec = 64000000 * register_value / t_clk_rate
 *
 *	register_value = coal_delay_in_usec * t_clk_rate / 64000000
 *
 * In the ->set*() methods, we round the computed register value
 * to the nearest integer.
 */
static unsigned int get_rx_coal(struct mv643xx_eth_private *mp)
{
	u32 val = rdlp(mp, SDMA_CONFIG);
	u64 temp;

	if (mp->shared->extended_rx_coal_limit)
		temp = ((val & 0x02000000) >> 10) | ((val & 0x003fff80) >> 7);
	else
		temp = (val & 0x003fff00) >> 8;

	temp *= 64000000;
1350
	do_div(temp, mp->t_clk);
1351 1352 1353 1354 1355 1356 1357 1358 1359

	return (unsigned int)temp;
}

static void set_rx_coal(struct mv643xx_eth_private *mp, unsigned int usec)
{
	u64 temp;
	u32 val;

1360
	temp = (u64)usec * mp->t_clk;
1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385
	temp += 31999999;
	do_div(temp, 64000000);

	val = rdlp(mp, SDMA_CONFIG);
	if (mp->shared->extended_rx_coal_limit) {
		if (temp > 0xffff)
			temp = 0xffff;
		val &= ~0x023fff80;
		val |= (temp & 0x8000) << 10;
		val |= (temp & 0x7fff) << 7;
	} else {
		if (temp > 0x3fff)
			temp = 0x3fff;
		val &= ~0x003fff00;
		val |= (temp & 0x3fff) << 8;
	}
	wrlp(mp, SDMA_CONFIG, val);
}

static unsigned int get_tx_coal(struct mv643xx_eth_private *mp)
{
	u64 temp;

	temp = (rdlp(mp, TX_FIFO_URGENT_THRESHOLD) & 0x3fff0) >> 4;
	temp *= 64000000;
1386
	do_div(temp, mp->t_clk);
1387 1388 1389 1390 1391 1392 1393 1394

	return (unsigned int)temp;
}

static void set_tx_coal(struct mv643xx_eth_private *mp, unsigned int usec)
{
	u64 temp;

1395
	temp = (u64)usec * mp->t_clk;
1396 1397 1398 1399 1400 1401 1402 1403 1404 1405
	temp += 31999999;
	do_div(temp, 64000000);

	if (temp > 0x3fff)
		temp = 0x3fff;

	wrlp(mp, TX_FIFO_URGENT_THRESHOLD, temp << 4);
}


1406
/* ethtool ******************************************************************/
1407
struct mv643xx_eth_stats {
1408 1409
	char stat_string[ETH_GSTRING_LEN];
	int sizeof_stat;
1410 1411
	int netdev_off;
	int mp_off;
1412 1413
};

1414 1415 1416 1417 1418 1419 1420 1421
#define SSTAT(m)						\
	{ #m, FIELD_SIZEOF(struct net_device_stats, m),		\
	  offsetof(struct net_device, stats.m), -1 }

#define MIBSTAT(m)						\
	{ #m, FIELD_SIZEOF(struct mib_counters, m),		\
	  -1, offsetof(struct mv643xx_eth_private, mib_counters.m) }

1422 1423 1424 1425
#define LROSTAT(m)						\
	{ #m, FIELD_SIZEOF(struct lro_counters, m),		\
	  -1, offsetof(struct mv643xx_eth_private, lro_counters.m) }

1426 1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438 1439 1440 1441 1442 1443 1444 1445 1446 1447 1448 1449 1450 1451 1452 1453 1454 1455 1456 1457 1458 1459 1460 1461 1462 1463 1464
static const struct mv643xx_eth_stats mv643xx_eth_stats[] = {
	SSTAT(rx_packets),
	SSTAT(tx_packets),
	SSTAT(rx_bytes),
	SSTAT(tx_bytes),
	SSTAT(rx_errors),
	SSTAT(tx_errors),
	SSTAT(rx_dropped),
	SSTAT(tx_dropped),
	MIBSTAT(good_octets_received),
	MIBSTAT(bad_octets_received),
	MIBSTAT(internal_mac_transmit_err),
	MIBSTAT(good_frames_received),
	MIBSTAT(bad_frames_received),
	MIBSTAT(broadcast_frames_received),
	MIBSTAT(multicast_frames_received),
	MIBSTAT(frames_64_octets),
	MIBSTAT(frames_65_to_127_octets),
	MIBSTAT(frames_128_to_255_octets),
	MIBSTAT(frames_256_to_511_octets),
	MIBSTAT(frames_512_to_1023_octets),
	MIBSTAT(frames_1024_to_max_octets),
	MIBSTAT(good_octets_sent),
	MIBSTAT(good_frames_sent),
	MIBSTAT(excessive_collision),
	MIBSTAT(multicast_frames_sent),
	MIBSTAT(broadcast_frames_sent),
	MIBSTAT(unrec_mac_control_received),
	MIBSTAT(fc_sent),
	MIBSTAT(good_fc_received),
	MIBSTAT(bad_fc_received),
	MIBSTAT(undersize_received),
	MIBSTAT(fragments_received),
	MIBSTAT(oversize_received),
	MIBSTAT(jabber_received),
	MIBSTAT(mac_receive_error),
	MIBSTAT(bad_crc_event),
	MIBSTAT(collision),
	MIBSTAT(late_collision),
1465 1466
	MIBSTAT(rx_discard),
	MIBSTAT(rx_overrun),
1467 1468 1469
	LROSTAT(lro_aggregated),
	LROSTAT(lro_flushed),
	LROSTAT(lro_no_desc),
1470 1471
};

L
Lennert Buytenhek 已提交
1472
static int
1473 1474
mv643xx_eth_get_settings_phy(struct mv643xx_eth_private *mp,
			     struct ethtool_cmd *cmd)
1475 1476 1477
{
	int err;

1478 1479 1480
	err = phy_read_status(mp->phy);
	if (err == 0)
		err = phy_ethtool_gset(mp->phy, cmd);
1481

L
Lennert Buytenhek 已提交
1482 1483 1484
	/*
	 * The MAC does not support 1000baseT_Half.
	 */
1485 1486 1487 1488 1489 1490
	cmd->supported &= ~SUPPORTED_1000baseT_Half;
	cmd->advertising &= ~ADVERTISED_1000baseT_Half;

	return err;
}

L
Lennert Buytenhek 已提交
1491
static int
1492
mv643xx_eth_get_settings_phyless(struct mv643xx_eth_private *mp,
L
Lennert Buytenhek 已提交
1493
				 struct ethtool_cmd *cmd)
1494
{
1495 1496
	u32 port_status;

1497
	port_status = rdlp(mp, PORT_STATUS);
1498

1499 1500
	cmd->supported = SUPPORTED_MII;
	cmd->advertising = ADVERTISED_MII;
1501 1502
	switch (port_status & PORT_SPEED_MASK) {
	case PORT_SPEED_10:
1503
		ethtool_cmd_speed_set(cmd, SPEED_10);
1504 1505
		break;
	case PORT_SPEED_100:
1506
		ethtool_cmd_speed_set(cmd, SPEED_100);
1507 1508
		break;
	case PORT_SPEED_1000:
1509
		ethtool_cmd_speed_set(cmd, SPEED_1000);
1510 1511 1512 1513 1514 1515
		break;
	default:
		cmd->speed = -1;
		break;
	}
	cmd->duplex = (port_status & FULL_DUPLEX) ? DUPLEX_FULL : DUPLEX_HALF;
1516 1517 1518 1519 1520 1521 1522 1523 1524 1525
	cmd->port = PORT_MII;
	cmd->phy_address = 0;
	cmd->transceiver = XCVR_INTERNAL;
	cmd->autoneg = AUTONEG_DISABLE;
	cmd->maxtxpkt = 1;
	cmd->maxrxpkt = 1;

	return 0;
}

1526 1527 1528 1529 1530 1531 1532 1533 1534 1535 1536
static int
mv643xx_eth_get_settings(struct net_device *dev, struct ethtool_cmd *cmd)
{
	struct mv643xx_eth_private *mp = netdev_priv(dev);

	if (mp->phy != NULL)
		return mv643xx_eth_get_settings_phy(mp, cmd);
	else
		return mv643xx_eth_get_settings_phyless(mp, cmd);
}

L
Lennert Buytenhek 已提交
1537 1538
static int
mv643xx_eth_set_settings(struct net_device *dev, struct ethtool_cmd *cmd)
L
Linus Torvalds 已提交
1539
{
1540
	struct mv643xx_eth_private *mp = netdev_priv(dev);
1541
	int ret;
1542

1543 1544 1545
	if (mp->phy == NULL)
		return -EINVAL;

L
Lennert Buytenhek 已提交
1546 1547 1548 1549 1550
	/*
	 * The MAC does not support 1000baseT_Half.
	 */
	cmd->advertising &= ~ADVERTISED_1000baseT_Half;

1551 1552 1553 1554
	ret = phy_ethtool_sset(mp->phy, cmd);
	if (!ret)
		mv643xx_adjust_pscr(mp);
	return ret;
1555
}
L
Linus Torvalds 已提交
1556

L
Lennert Buytenhek 已提交
1557 1558
static void mv643xx_eth_get_drvinfo(struct net_device *dev,
				    struct ethtool_drvinfo *drvinfo)
1559
{
A
Axel Lin 已提交
1560 1561
	strlcpy(drvinfo->driver, mv643xx_eth_driver_name,
		sizeof(drvinfo->driver));
1562
	strlcpy(drvinfo->version, mv643xx_eth_driver_version,
A
Axel Lin 已提交
1563 1564 1565
		sizeof(drvinfo->version));
	strlcpy(drvinfo->fw_version, "N/A", sizeof(drvinfo->fw_version));
	strlcpy(drvinfo->bus_info, "platform", sizeof(drvinfo->bus_info));
1566
	drvinfo->n_stats = ARRAY_SIZE(mv643xx_eth_stats);
1567
}
L
Linus Torvalds 已提交
1568

L
Lennert Buytenhek 已提交
1569
static int mv643xx_eth_nway_reset(struct net_device *dev)
1570
{
1571
	struct mv643xx_eth_private *mp = netdev_priv(dev);
L
Linus Torvalds 已提交
1572

1573 1574
	if (mp->phy == NULL)
		return -EINVAL;
L
Linus Torvalds 已提交
1575

1576
	return genphy_restart_aneg(mp->phy);
1577 1578
}

1579 1580 1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595 1596 1597 1598 1599 1600
static int
mv643xx_eth_get_coalesce(struct net_device *dev, struct ethtool_coalesce *ec)
{
	struct mv643xx_eth_private *mp = netdev_priv(dev);

	ec->rx_coalesce_usecs = get_rx_coal(mp);
	ec->tx_coalesce_usecs = get_tx_coal(mp);

	return 0;
}

static int
mv643xx_eth_set_coalesce(struct net_device *dev, struct ethtool_coalesce *ec)
{
	struct mv643xx_eth_private *mp = netdev_priv(dev);

	set_rx_coal(mp, ec->rx_coalesce_usecs);
	set_tx_coal(mp, ec->tx_coalesce_usecs);

	return 0;
}

1601 1602 1603 1604 1605 1606 1607 1608 1609 1610 1611 1612 1613 1614 1615 1616 1617 1618 1619 1620 1621 1622 1623 1624 1625 1626
static void
mv643xx_eth_get_ringparam(struct net_device *dev, struct ethtool_ringparam *er)
{
	struct mv643xx_eth_private *mp = netdev_priv(dev);

	er->rx_max_pending = 4096;
	er->tx_max_pending = 4096;

	er->rx_pending = mp->rx_ring_size;
	er->tx_pending = mp->tx_ring_size;
}

static int
mv643xx_eth_set_ringparam(struct net_device *dev, struct ethtool_ringparam *er)
{
	struct mv643xx_eth_private *mp = netdev_priv(dev);

	if (er->rx_mini_pending || er->rx_jumbo_pending)
		return -EINVAL;

	mp->rx_ring_size = er->rx_pending < 4096 ? er->rx_pending : 4096;
	mp->tx_ring_size = er->tx_pending < 4096 ? er->tx_pending : 4096;

	if (netif_running(dev)) {
		mv643xx_eth_stop(dev);
		if (mv643xx_eth_open(dev)) {
1627 1628
			netdev_err(dev,
				   "fatal error on re-opening device after ring param change\n");
1629 1630 1631 1632 1633 1634 1635
			return -ENOMEM;
		}
	}

	return 0;
}

1636 1637

static int
1638
mv643xx_eth_set_features(struct net_device *dev, netdev_features_t features)
1639 1640
{
	struct mv643xx_eth_private *mp = netdev_priv(dev);
1641
	bool rx_csum = features & NETIF_F_RXCSUM;
1642 1643 1644 1645 1646 1647

	wrlp(mp, PORT_CONFIG, rx_csum ? 0x02000000 : 0x00000000);

	return 0;
}

L
Lennert Buytenhek 已提交
1648 1649
static void mv643xx_eth_get_strings(struct net_device *dev,
				    uint32_t stringset, uint8_t *data)
1650 1651
{
	int i;
L
Linus Torvalds 已提交
1652

L
Lennert Buytenhek 已提交
1653 1654
	if (stringset == ETH_SS_STATS) {
		for (i = 0; i < ARRAY_SIZE(mv643xx_eth_stats); i++) {
1655
			memcpy(data + i * ETH_GSTRING_LEN,
1656
				mv643xx_eth_stats[i].stat_string,
1657
				ETH_GSTRING_LEN);
1658 1659 1660
		}
	}
}
L
Linus Torvalds 已提交
1661

L
Lennert Buytenhek 已提交
1662 1663 1664
static void mv643xx_eth_get_ethtool_stats(struct net_device *dev,
					  struct ethtool_stats *stats,
					  uint64_t *data)
1665
{
1666
	struct mv643xx_eth_private *mp = netdev_priv(dev);
1667
	int i;
L
Linus Torvalds 已提交
1668

1669
	mv643xx_eth_get_stats(dev);
L
Lennert Buytenhek 已提交
1670
	mib_counters_update(mp);
1671
	mv643xx_eth_grab_lro_stats(mp);
L
Linus Torvalds 已提交
1672

1673 1674 1675 1676 1677 1678 1679 1680 1681 1682 1683 1684 1685
	for (i = 0; i < ARRAY_SIZE(mv643xx_eth_stats); i++) {
		const struct mv643xx_eth_stats *stat;
		void *p;

		stat = mv643xx_eth_stats + i;

		if (stat->netdev_off >= 0)
			p = ((void *)mp->dev) + stat->netdev_off;
		else
			p = ((void *)mp) + stat->mp_off;

		data[i] = (stat->sizeof_stat == 8) ?
				*(uint64_t *)p : *(uint32_t *)p;
L
Linus Torvalds 已提交
1686
	}
1687
}
L
Linus Torvalds 已提交
1688

L
Lennert Buytenhek 已提交
1689
static int mv643xx_eth_get_sset_count(struct net_device *dev, int sset)
1690
{
L
Lennert Buytenhek 已提交
1691
	if (sset == ETH_SS_STATS)
1692
		return ARRAY_SIZE(mv643xx_eth_stats);
L
Lennert Buytenhek 已提交
1693 1694

	return -EOPNOTSUPP;
1695
}
L
Linus Torvalds 已提交
1696

1697
static const struct ethtool_ops mv643xx_eth_ethtool_ops = {
L
Lennert Buytenhek 已提交
1698 1699 1700 1701
	.get_settings		= mv643xx_eth_get_settings,
	.set_settings		= mv643xx_eth_set_settings,
	.get_drvinfo		= mv643xx_eth_get_drvinfo,
	.nway_reset		= mv643xx_eth_nway_reset,
1702
	.get_link		= ethtool_op_get_link,
1703 1704
	.get_coalesce		= mv643xx_eth_get_coalesce,
	.set_coalesce		= mv643xx_eth_set_coalesce,
1705 1706
	.get_ringparam		= mv643xx_eth_get_ringparam,
	.set_ringparam		= mv643xx_eth_set_ringparam,
L
Lennert Buytenhek 已提交
1707 1708
	.get_strings		= mv643xx_eth_get_strings,
	.get_ethtool_stats	= mv643xx_eth_get_ethtool_stats,
1709
	.get_sset_count		= mv643xx_eth_get_sset_count,
1710
	.get_ts_info		= ethtool_op_get_ts_info,
1711
};
L
Linus Torvalds 已提交
1712

1713

1714
/* address handling *********************************************************/
1715
static void uc_addr_get(struct mv643xx_eth_private *mp, unsigned char *addr)
1716
{
1717 1718
	unsigned int mac_h = rdlp(mp, MAC_ADDR_HIGH);
	unsigned int mac_l = rdlp(mp, MAC_ADDR_LOW);
L
Linus Torvalds 已提交
1719

1720 1721 1722 1723 1724 1725
	addr[0] = (mac_h >> 24) & 0xff;
	addr[1] = (mac_h >> 16) & 0xff;
	addr[2] = (mac_h >> 8) & 0xff;
	addr[3] = mac_h & 0xff;
	addr[4] = (mac_l >> 8) & 0xff;
	addr[5] = mac_l & 0xff;
1726
}
L
Linus Torvalds 已提交
1727

1728
static void uc_addr_set(struct mv643xx_eth_private *mp, unsigned char *addr)
1729
{
1730 1731 1732
	wrlp(mp, MAC_ADDR_HIGH,
		(addr[0] << 24) | (addr[1] << 16) | (addr[2] << 8) | addr[3]);
	wrlp(mp, MAC_ADDR_LOW, (addr[4] << 8) | addr[5]);
1733
}
1734

1735
static u32 uc_addr_filter_mask(struct net_device *dev)
1736
{
J
Jiri Pirko 已提交
1737
	struct netdev_hw_addr *ha;
1738
	u32 nibbles;
L
Linus Torvalds 已提交
1739

1740 1741
	if (dev->flags & IFF_PROMISC)
		return 0;
L
Linus Torvalds 已提交
1742

1743
	nibbles = 1 << (dev->dev_addr[5] & 0x0f);
1744
	netdev_for_each_uc_addr(ha, dev) {
J
Jiri Pirko 已提交
1745
		if (memcmp(dev->dev_addr, ha->addr, 5))
1746
			return 0;
J
Jiri Pirko 已提交
1747
		if ((dev->dev_addr[5] ^ ha->addr[5]) & 0xf0)
1748
			return 0;
1749

J
Jiri Pirko 已提交
1750
		nibbles |= 1 << (ha->addr[5] & 0x0f);
1751
	}
L
Linus Torvalds 已提交
1752

1753
	return nibbles;
L
Linus Torvalds 已提交
1754 1755
}

1756
static void mv643xx_eth_program_unicast_filter(struct net_device *dev)
L
Linus Torvalds 已提交
1757
{
1758
	struct mv643xx_eth_private *mp = netdev_priv(dev);
1759 1760 1761
	u32 port_config;
	u32 nibbles;
	int i;
L
Linus Torvalds 已提交
1762

1763
	uc_addr_set(mp, dev->dev_addr);
L
Linus Torvalds 已提交
1764

1765 1766
	port_config = rdlp(mp, PORT_CONFIG) & ~UNICAST_PROMISCUOUS_MODE;

1767 1768 1769
	nibbles = uc_addr_filter_mask(dev);
	if (!nibbles) {
		port_config |= UNICAST_PROMISCUOUS_MODE;
1770
		nibbles = 0xffff;
1771 1772 1773 1774 1775 1776 1777 1778 1779 1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790 1791
	}

	for (i = 0; i < 16; i += 4) {
		int off = UNICAST_TABLE(mp->port_num) + i;
		u32 v;

		v = 0;
		if (nibbles & 1)
			v |= 0x00000001;
		if (nibbles & 2)
			v |= 0x00000100;
		if (nibbles & 4)
			v |= 0x00010000;
		if (nibbles & 8)
			v |= 0x01000000;
		nibbles >>= 4;

		wrl(mp, off, v);
	}

	wrlp(mp, PORT_CONFIG, port_config);
L
Linus Torvalds 已提交
1792 1793
}

1794 1795 1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808 1809 1810 1811
static int addr_crc(unsigned char *addr)
{
	int crc = 0;
	int i;

	for (i = 0; i < 6; i++) {
		int j;

		crc = (crc ^ addr[i]) << 8;
		for (j = 7; j >= 0; j--) {
			if (crc & (0x100 << j))
				crc ^= 0x107 << j;
		}
	}

	return crc;
}

1812
static void mv643xx_eth_program_multicast_filter(struct net_device *dev)
L
Linus Torvalds 已提交
1813
{
L
Lennert Buytenhek 已提交
1814
	struct mv643xx_eth_private *mp = netdev_priv(dev);
1815 1816
	u32 *mc_spec;
	u32 *mc_other;
1817
	struct netdev_hw_addr *ha;
L
Lennert Buytenhek 已提交
1818
	int i;
1819

L
Lennert Buytenhek 已提交
1820
	if (dev->flags & (IFF_PROMISC | IFF_ALLMULTI)) {
1821 1822
		int port_num;
		u32 accept;
1823

1824 1825 1826
oom:
		port_num = mp->port_num;
		accept = 0x01010101;
L
Lennert Buytenhek 已提交
1827 1828 1829
		for (i = 0; i < 0x100; i += 4) {
			wrl(mp, SPECIAL_MCAST_TABLE(port_num) + i, accept);
			wrl(mp, OTHER_MCAST_TABLE(port_num) + i, accept);
1830 1831 1832
		}
		return;
	}
1833

1834
	mc_spec = kmalloc(0x200, GFP_ATOMIC);
1835 1836 1837 1838 1839 1840
	if (mc_spec == NULL)
		goto oom;
	mc_other = mc_spec + (0x100 >> 2);

	memset(mc_spec, 0, 0x100);
	memset(mc_other, 0, 0x100);
L
Linus Torvalds 已提交
1841

1842 1843
	netdev_for_each_mc_addr(ha, dev) {
		u8 *a = ha->addr;
1844 1845
		u32 *table;
		int entry;
L
Linus Torvalds 已提交
1846

L
Lennert Buytenhek 已提交
1847
		if (memcmp(a, "\x01\x00\x5e\x00\x00", 5) == 0) {
1848 1849
			table = mc_spec;
			entry = a[5];
L
Lennert Buytenhek 已提交
1850
		} else {
1851 1852
			table = mc_other;
			entry = addr_crc(a);
L
Lennert Buytenhek 已提交
1853
		}
1854

1855
		table[entry >> 2] |= 1 << (8 * (entry & 3));
L
Lennert Buytenhek 已提交
1856
	}
1857 1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868 1869 1870 1871 1872 1873 1874 1875

	for (i = 0; i < 0x100; i += 4) {
		wrl(mp, SPECIAL_MCAST_TABLE(mp->port_num) + i, mc_spec[i >> 2]);
		wrl(mp, OTHER_MCAST_TABLE(mp->port_num) + i, mc_other[i >> 2]);
	}

	kfree(mc_spec);
}

static void mv643xx_eth_set_rx_mode(struct net_device *dev)
{
	mv643xx_eth_program_unicast_filter(dev);
	mv643xx_eth_program_multicast_filter(dev);
}

static int mv643xx_eth_set_mac_address(struct net_device *dev, void *addr)
{
	struct sockaddr *sa = addr;

1876
	if (!is_valid_ether_addr(sa->sa_data))
1877
		return -EADDRNOTAVAIL;
1878

1879 1880 1881 1882 1883 1884 1885
	memcpy(dev->dev_addr, sa->sa_data, ETH_ALEN);

	netif_addr_lock_bh(dev);
	mv643xx_eth_program_unicast_filter(dev);
	netif_addr_unlock_bh(dev);

	return 0;
1886
}
1887 1888


1889
/* rx/tx queue initialisation ***********************************************/
1890
static int rxq_init(struct mv643xx_eth_private *mp, int index)
1891
{
1892
	struct rx_queue *rxq = mp->rxq + index;
1893 1894
	struct rx_desc *rx_desc;
	int size;
1895 1896
	int i;

1897 1898
	rxq->index = index;

1899
	rxq->rx_ring_size = mp->rx_ring_size;
1900 1901 1902 1903 1904 1905 1906

	rxq->rx_desc_count = 0;
	rxq->rx_curr_desc = 0;
	rxq->rx_used_desc = 0;

	size = rxq->rx_ring_size * sizeof(struct rx_desc);

1907
	if (index == 0 && size <= mp->rx_desc_sram_size) {
1908 1909 1910 1911
		rxq->rx_desc_area = ioremap(mp->rx_desc_sram_addr,
						mp->rx_desc_sram_size);
		rxq->rx_desc_dma = mp->rx_desc_sram_addr;
	} else {
1912 1913 1914
		rxq->rx_desc_area = dma_alloc_coherent(mp->dev->dev.parent,
						       size, &rxq->rx_desc_dma,
						       GFP_KERNEL);
1915 1916
	}

1917
	if (rxq->rx_desc_area == NULL) {
1918
		netdev_err(mp->dev,
1919 1920 1921 1922
			   "can't allocate rx ring (%d bytes)\n", size);
		goto out;
	}
	memset(rxq->rx_desc_area, 0, size);
L
Linus Torvalds 已提交
1923

1924
	rxq->rx_desc_area_size = size;
1925 1926 1927
	rxq->rx_skb = kmalloc_array(rxq->rx_ring_size, sizeof(*rxq->rx_skb),
				    GFP_KERNEL);
	if (rxq->rx_skb == NULL)
1928 1929
		goto out_free;

1930
	rx_desc = rxq->rx_desc_area;
1931
	for (i = 0; i < rxq->rx_ring_size; i++) {
1932 1933 1934 1935 1936 1937
		int nexti;

		nexti = i + 1;
		if (nexti == rxq->rx_ring_size)
			nexti = 0;

1938 1939 1940 1941
		rx_desc[i].next_desc_ptr = rxq->rx_desc_dma +
					nexti * sizeof(struct rx_desc);
	}

1942 1943 1944 1945 1946 1947 1948 1949 1950 1951 1952 1953 1954
	rxq->lro_mgr.dev = mp->dev;
	memset(&rxq->lro_mgr.stats, 0, sizeof(rxq->lro_mgr.stats));
	rxq->lro_mgr.features = LRO_F_NAPI;
	rxq->lro_mgr.ip_summed = CHECKSUM_UNNECESSARY;
	rxq->lro_mgr.ip_summed_aggr = CHECKSUM_UNNECESSARY;
	rxq->lro_mgr.max_desc = ARRAY_SIZE(rxq->lro_arr);
	rxq->lro_mgr.max_aggr = 32;
	rxq->lro_mgr.frag_align_pad = 0;
	rxq->lro_mgr.lro_arr = rxq->lro_arr;
	rxq->lro_mgr.get_skb_header = mv643xx_get_skb_header;

	memset(&rxq->lro_arr, 0, sizeof(rxq->lro_arr));

1955 1956 1957 1958
	return 0;


out_free:
1959
	if (index == 0 && size <= mp->rx_desc_sram_size)
1960 1961
		iounmap(rxq->rx_desc_area);
	else
1962
		dma_free_coherent(mp->dev->dev.parent, size,
1963 1964 1965 1966 1967
				  rxq->rx_desc_area,
				  rxq->rx_desc_dma);

out:
	return -ENOMEM;
1968
}
1969

1970
static void rxq_deinit(struct rx_queue *rxq)
1971
{
1972 1973 1974 1975
	struct mv643xx_eth_private *mp = rxq_to_mp(rxq);
	int i;

	rxq_disable(rxq);
1976

1977 1978 1979 1980
	for (i = 0; i < rxq->rx_ring_size; i++) {
		if (rxq->rx_skb[i]) {
			dev_kfree_skb(rxq->rx_skb[i]);
			rxq->rx_desc_count--;
L
Linus Torvalds 已提交
1981
		}
1982
	}
L
Linus Torvalds 已提交
1983

1984
	if (rxq->rx_desc_count) {
1985
		netdev_err(mp->dev, "error freeing rx ring -- %d skbs stuck\n",
1986 1987 1988
			   rxq->rx_desc_count);
	}

1989
	if (rxq->index == 0 &&
1990
	    rxq->rx_desc_area_size <= mp->rx_desc_sram_size)
1991
		iounmap(rxq->rx_desc_area);
1992
	else
1993
		dma_free_coherent(mp->dev->dev.parent, rxq->rx_desc_area_size,
1994 1995 1996
				  rxq->rx_desc_area, rxq->rx_desc_dma);

	kfree(rxq->rx_skb);
1997
}
L
Linus Torvalds 已提交
1998

1999
static int txq_init(struct mv643xx_eth_private *mp, int index)
2000
{
2001
	struct tx_queue *txq = mp->txq + index;
2002 2003
	struct tx_desc *tx_desc;
	int size;
2004
	int i;
L
Linus Torvalds 已提交
2005

2006 2007
	txq->index = index;

2008
	txq->tx_ring_size = mp->tx_ring_size;
2009 2010 2011 2012 2013 2014 2015

	txq->tx_desc_count = 0;
	txq->tx_curr_desc = 0;
	txq->tx_used_desc = 0;

	size = txq->tx_ring_size * sizeof(struct tx_desc);

2016
	if (index == 0 && size <= mp->tx_desc_sram_size) {
2017 2018 2019 2020
		txq->tx_desc_area = ioremap(mp->tx_desc_sram_addr,
						mp->tx_desc_sram_size);
		txq->tx_desc_dma = mp->tx_desc_sram_addr;
	} else {
2021 2022 2023
		txq->tx_desc_area = dma_alloc_coherent(mp->dev->dev.parent,
						       size, &txq->tx_desc_dma,
						       GFP_KERNEL);
2024 2025 2026
	}

	if (txq->tx_desc_area == NULL) {
2027
		netdev_err(mp->dev,
2028
			   "can't allocate tx ring (%d bytes)\n", size);
2029
		return -ENOMEM;
2030
	}
2031 2032 2033 2034
	memset(txq->tx_desc_area, 0, size);

	txq->tx_desc_area_size = size;

2035
	tx_desc = txq->tx_desc_area;
2036
	for (i = 0; i < txq->tx_ring_size; i++) {
2037
		struct tx_desc *txd = tx_desc + i;
2038 2039 2040 2041 2042
		int nexti;

		nexti = i + 1;
		if (nexti == txq->tx_ring_size)
			nexti = 0;
2043 2044 2045

		txd->cmd_sts = 0;
		txd->next_desc_ptr = txq->tx_desc_dma +
2046 2047 2048
					nexti * sizeof(struct tx_desc);
	}

2049
	skb_queue_head_init(&txq->tx_skb);
2050

2051
	return 0;
2052
}
L
Linus Torvalds 已提交
2053

2054
static void txq_deinit(struct tx_queue *txq)
2055
{
2056
	struct mv643xx_eth_private *mp = txq_to_mp(txq);
2057

2058
	txq_disable(txq);
2059
	txq_reclaim(txq, txq->tx_ring_size, 1);
L
Linus Torvalds 已提交
2060

2061
	BUG_ON(txq->tx_used_desc != txq->tx_curr_desc);
L
Linus Torvalds 已提交
2062

2063
	if (txq->index == 0 &&
2064
	    txq->tx_desc_area_size <= mp->tx_desc_sram_size)
2065
		iounmap(txq->tx_desc_area);
2066
	else
2067
		dma_free_coherent(mp->dev->dev.parent, txq->tx_desc_area_size,
2068
				  txq->tx_desc_area, txq->tx_desc_dma);
2069
}
L
Linus Torvalds 已提交
2070 2071


2072
/* netdev ops and related ***************************************************/
2073 2074 2075 2076 2077
static int mv643xx_eth_collect_events(struct mv643xx_eth_private *mp)
{
	u32 int_cause;
	u32 int_cause_ext;

2078
	int_cause = rdlp(mp, INT_CAUSE) & mp->int_mask;
2079 2080 2081 2082
	if (int_cause == 0)
		return 0;

	int_cause_ext = 0;
2083 2084
	if (int_cause & INT_EXT) {
		int_cause &= ~INT_EXT;
2085
		int_cause_ext = rdlp(mp, INT_CAUSE_EXT);
2086
	}
2087 2088

	if (int_cause) {
2089
		wrlp(mp, INT_CAUSE, ~int_cause);
2090
		mp->work_tx_end |= ((int_cause & INT_TX_END) >> 19) &
2091
				~(rdlp(mp, TXQ_COMMAND) & 0xff);
2092 2093 2094 2095 2096
		mp->work_rx |= (int_cause & INT_RX) >> 2;
	}

	int_cause_ext &= INT_EXT_LINK_PHY | INT_EXT_TX;
	if (int_cause_ext) {
2097
		wrlp(mp, INT_CAUSE_EXT, ~int_cause_ext);
2098 2099 2100 2101 2102 2103 2104 2105 2106 2107 2108 2109 2110 2111 2112 2113
		if (int_cause_ext & INT_EXT_LINK_PHY)
			mp->work_link = 1;
		mp->work_tx |= int_cause_ext & INT_EXT_TX;
	}

	return 1;
}

static irqreturn_t mv643xx_eth_irq(int irq, void *dev_id)
{
	struct net_device *dev = (struct net_device *)dev_id;
	struct mv643xx_eth_private *mp = netdev_priv(dev);

	if (unlikely(!mv643xx_eth_collect_events(mp)))
		return IRQ_NONE;

2114
	wrlp(mp, INT_MASK, 0);
2115 2116 2117 2118 2119
	napi_schedule(&mp->napi);

	return IRQ_HANDLED;
}

2120 2121 2122 2123 2124 2125 2126 2127
static void handle_link_event(struct mv643xx_eth_private *mp)
{
	struct net_device *dev = mp->dev;
	u32 port_status;
	int speed;
	int duplex;
	int fc;

2128
	port_status = rdlp(mp, PORT_STATUS);
2129 2130 2131 2132
	if (!(port_status & LINK_UP)) {
		if (netif_carrier_ok(dev)) {
			int i;

2133
			netdev_info(dev, "link down\n");
2134 2135 2136

			netif_carrier_off(dev);

2137
			for (i = 0; i < mp->txq_count; i++) {
2138 2139
				struct tx_queue *txq = mp->txq + i;

2140
				txq_reclaim(txq, txq->tx_ring_size, 1);
2141
				txq_reset_hw_ptr(txq);
2142 2143 2144 2145 2146 2147 2148 2149 2150 2151 2152 2153 2154 2155 2156 2157 2158 2159 2160 2161 2162 2163
			}
		}
		return;
	}

	switch (port_status & PORT_SPEED_MASK) {
	case PORT_SPEED_10:
		speed = 10;
		break;
	case PORT_SPEED_100:
		speed = 100;
		break;
	case PORT_SPEED_1000:
		speed = 1000;
		break;
	default:
		speed = -1;
		break;
	}
	duplex = (port_status & FULL_DUPLEX) ? 1 : 0;
	fc = (port_status & FLOW_CONTROL_ENABLED) ? 1 : 0;

2164 2165
	netdev_info(dev, "link up, %d Mb/s, %s duplex, flow control %sabled\n",
		    speed, duplex ? "full" : "half", fc ? "en" : "dis");
2166

2167
	if (!netif_carrier_ok(dev))
2168 2169 2170
		netif_carrier_on(dev);
}

2171
static int mv643xx_eth_poll(struct napi_struct *napi, int budget)
2172
{
2173 2174
	struct mv643xx_eth_private *mp;
	int work_done;
2175

2176
	mp = container_of(napi, struct mv643xx_eth_private, napi);
L
Lennert Buytenhek 已提交
2177

2178 2179 2180 2181
	if (unlikely(mp->oom)) {
		mp->oom = 0;
		del_timer(&mp->rx_oom);
	}
L
Linus Torvalds 已提交
2182

2183 2184 2185 2186 2187 2188 2189 2190 2191
	work_done = 0;
	while (work_done < budget) {
		u8 queue_mask;
		int queue;
		int work_tbd;

		if (mp->work_link) {
			mp->work_link = 0;
			handle_link_event(mp);
2192
			work_done++;
2193 2194
			continue;
		}
L
Linus Torvalds 已提交
2195

2196 2197 2198 2199
		queue_mask = mp->work_tx | mp->work_tx_end | mp->work_rx;
		if (likely(!mp->oom))
			queue_mask |= mp->work_rx_refill;

2200 2201 2202 2203 2204
		if (!queue_mask) {
			if (mv643xx_eth_collect_events(mp))
				continue;
			break;
		}
L
Linus Torvalds 已提交
2205

2206 2207 2208 2209 2210 2211 2212 2213 2214 2215 2216 2217 2218 2219
		queue = fls(queue_mask) - 1;
		queue_mask = 1 << queue;

		work_tbd = budget - work_done;
		if (work_tbd > 16)
			work_tbd = 16;

		if (mp->work_tx_end & queue_mask) {
			txq_kick(mp->txq + queue);
		} else if (mp->work_tx & queue_mask) {
			work_done += txq_reclaim(mp->txq + queue, work_tbd, 0);
			txq_maybe_wake(mp->txq + queue);
		} else if (mp->work_rx & queue_mask) {
			work_done += rxq_process(mp->rxq + queue, work_tbd);
2220
		} else if (!mp->oom && (mp->work_rx_refill & queue_mask)) {
2221 2222 2223 2224
			work_done += rxq_refill(mp->rxq + queue, work_tbd);
		} else {
			BUG();
		}
2225
	}
L
Lennert Buytenhek 已提交
2226

2227
	if (work_done < budget) {
2228
		if (mp->oom)
2229 2230
			mod_timer(&mp->rx_oom, jiffies + (HZ / 10));
		napi_complete(napi);
2231
		wrlp(mp, INT_MASK, mp->int_mask);
2232
	}
2233

2234 2235
	return work_done;
}
2236

2237 2238 2239
static inline void oom_timer_wrapper(unsigned long data)
{
	struct mv643xx_eth_private *mp = (void *)data;
L
Linus Torvalds 已提交
2240

2241
	napi_schedule(&mp->napi);
L
Linus Torvalds 已提交
2242 2243
}

2244
static void phy_reset(struct mv643xx_eth_private *mp)
L
Linus Torvalds 已提交
2245
{
2246 2247
	int data;

2248
	data = phy_read(mp->phy, MII_BMCR);
2249 2250
	if (data < 0)
		return;
L
Linus Torvalds 已提交
2251

2252
	data |= BMCR_RESET;
2253
	if (phy_write(mp->phy, MII_BMCR, data) < 0)
2254
		return;
L
Linus Torvalds 已提交
2255

2256
	do {
2257
		data = phy_read(mp->phy, MII_BMCR);
2258
	} while (data >= 0 && data & BMCR_RESET);
L
Linus Torvalds 已提交
2259 2260
}

L
Lennert Buytenhek 已提交
2261
static void port_start(struct mv643xx_eth_private *mp)
L
Linus Torvalds 已提交
2262
{
2263
	u32 pscr;
2264
	int i;
L
Linus Torvalds 已提交
2265

2266 2267 2268
	/*
	 * Perform PHY reset, if there is a PHY.
	 */
2269
	if (mp->phy != NULL) {
2270 2271 2272 2273 2274 2275
		struct ethtool_cmd cmd;

		mv643xx_eth_get_settings(mp->dev, &cmd);
		phy_reset(mp);
		mv643xx_eth_set_settings(mp->dev, &cmd);
	}
L
Linus Torvalds 已提交
2276

2277 2278 2279
	/*
	 * Configure basic link parameters.
	 */
2280
	pscr = rdlp(mp, PORT_SERIAL_CONTROL);
2281 2282

	pscr |= SERIAL_PORT_ENABLE;
2283
	wrlp(mp, PORT_SERIAL_CONTROL, pscr);
2284 2285

	pscr |= DO_NOT_FORCE_LINK_FAIL;
2286
	if (mp->phy == NULL)
2287
		pscr |= FORCE_LINK_PASS;
2288
	wrlp(mp, PORT_SERIAL_CONTROL, pscr);
2289

2290 2291 2292
	/*
	 * Configure TX path and queues.
	 */
2293
	tx_set_rate(mp, 1000000000, 16777216);
2294
	for (i = 0; i < mp->txq_count; i++) {
2295
		struct tx_queue *txq = mp->txq + i;
2296

2297
		txq_reset_hw_ptr(txq);
2298 2299
		txq_set_rate(txq, 1000000000, 16777216);
		txq_set_fixed_prio_mode(txq);
2300 2301
	}

2302 2303
	/*
	 * Receive all unmatched unicast, TCP, UDP, BPDU and broadcast
2304 2305
	 * frames to RX queue #0, and include the pseudo-header when
	 * calculating receive checksums.
2306
	 */
2307
	mv643xx_eth_set_features(mp->dev, mp->dev->features);
2308

2309 2310 2311
	/*
	 * Treat BPDUs as normal multicasts, and disable partition mode.
	 */
2312
	wrlp(mp, PORT_CONFIG_EXT, 0x00000000);
2313

2314 2315 2316 2317 2318
	/*
	 * Add configured unicast addresses to address filter table.
	 */
	mv643xx_eth_program_unicast_filter(mp->dev);

2319
	/*
2320
	 * Enable the receive queues.
2321
	 */
2322
	for (i = 0; i < mp->rxq_count; i++) {
2323
		struct rx_queue *rxq = mp->rxq + i;
2324
		u32 addr;
L
Linus Torvalds 已提交
2325

2326 2327
		addr = (u32)rxq->rx_desc_dma;
		addr += rxq->rx_curr_desc * sizeof(struct rx_desc);
2328
		wrlp(mp, RXQ_CURRENT_DESC_PTR(i), addr);
L
Linus Torvalds 已提交
2329

2330 2331
		rxq_enable(rxq);
	}
L
Linus Torvalds 已提交
2332 2333
}

2334 2335 2336 2337 2338 2339 2340 2341 2342 2343 2344 2345 2346 2347 2348 2349 2350 2351
static void mv643xx_eth_recalc_skb_size(struct mv643xx_eth_private *mp)
{
	int skb_size;

	/*
	 * Reserve 2+14 bytes for an ethernet header (the hardware
	 * automatically prepends 2 bytes of dummy data to each
	 * received packet), 16 bytes for up to four VLAN tags, and
	 * 4 bytes for the trailing FCS -- 36 bytes total.
	 */
	skb_size = mp->dev->mtu + 36;

	/*
	 * Make sure that the skb size is a multiple of 8 bytes, as
	 * the lower three bits of the receive descriptor's buffer
	 * size field are ignored by the hardware.
	 */
	mp->skb_size = (skb_size + 7) & ~7;
2352 2353 2354 2355 2356 2357 2358 2359

	/*
	 * If NET_SKB_PAD is smaller than a cache line,
	 * netdev_alloc_skb() will cause skb->data to be misaligned
	 * to a cache line boundary.  If this is the case, include
	 * some extra space to allow re-aligning the data area.
	 */
	mp->skb_size += SKB_DMA_REALIGN;
2360 2361
}

2362
static int mv643xx_eth_open(struct net_device *dev)
2363
{
2364
	struct mv643xx_eth_private *mp = netdev_priv(dev);
2365
	int err;
2366
	int i;
2367

2368 2369 2370
	wrlp(mp, INT_CAUSE, 0);
	wrlp(mp, INT_CAUSE_EXT, 0);
	rdlp(mp, INT_CAUSE_EXT);
2371

L
Lennert Buytenhek 已提交
2372
	err = request_irq(dev->irq, mv643xx_eth_irq,
2373
			  IRQF_SHARED, dev->name, dev);
2374
	if (err) {
2375
		netdev_err(dev, "can't assign irq\n");
2376
		return -EAGAIN;
2377 2378
	}

2379 2380
	mv643xx_eth_recalc_skb_size(mp);

2381 2382
	napi_enable(&mp->napi);

2383 2384
	mp->int_mask = INT_EXT;

2385
	for (i = 0; i < mp->rxq_count; i++) {
2386 2387 2388
		err = rxq_init(mp, i);
		if (err) {
			while (--i >= 0)
2389
				rxq_deinit(mp->rxq + i);
2390 2391 2392
			goto out;
		}

2393
		rxq_refill(mp->rxq + i, INT_MAX);
2394
		mp->int_mask |= INT_RX_0 << i;
2395 2396
	}

2397
	if (mp->oom) {
2398 2399
		mp->rx_oom.expires = jiffies + (HZ / 10);
		add_timer(&mp->rx_oom);
2400
	}
2401

2402
	for (i = 0; i < mp->txq_count; i++) {
2403 2404 2405
		err = txq_init(mp, i);
		if (err) {
			while (--i >= 0)
2406
				txq_deinit(mp->txq + i);
2407 2408
			goto out_free;
		}
2409
		mp->int_mask |= INT_TX_END_0 << i;
2410
	}
2411

L
Lennert Buytenhek 已提交
2412
	port_start(mp);
2413

2414
	wrlp(mp, INT_MASK_EXT, INT_EXT_LINK_PHY | INT_EXT_TX);
2415
	wrlp(mp, INT_MASK, mp->int_mask);
2416

2417 2418
	return 0;

2419

L
Lennert Buytenhek 已提交
2420
out_free:
2421 2422
	for (i = 0; i < mp->rxq_count; i++)
		rxq_deinit(mp->rxq + i);
L
Lennert Buytenhek 已提交
2423
out:
2424 2425 2426
	free_irq(dev->irq, dev);

	return err;
2427 2428
}

2429
static void port_reset(struct mv643xx_eth_private *mp)
L
Linus Torvalds 已提交
2430
{
L
Lennert Buytenhek 已提交
2431
	unsigned int data;
2432
	int i;
L
Linus Torvalds 已提交
2433

2434 2435 2436 2437
	for (i = 0; i < mp->rxq_count; i++)
		rxq_disable(mp->rxq + i);
	for (i = 0; i < mp->txq_count; i++)
		txq_disable(mp->txq + i);
2438 2439

	while (1) {
2440
		u32 ps = rdlp(mp, PORT_STATUS);
2441 2442 2443

		if ((ps & (TX_IN_PROGRESS | TX_FIFO_EMPTY)) == TX_FIFO_EMPTY)
			break;
2444
		udelay(10);
2445
	}
L
Linus Torvalds 已提交
2446

2447
	/* Reset the Enable bit in the Configuration Register */
2448
	data = rdlp(mp, PORT_SERIAL_CONTROL);
L
Lennert Buytenhek 已提交
2449 2450 2451
	data &= ~(SERIAL_PORT_ENABLE		|
		  DO_NOT_FORCE_LINK_FAIL	|
		  FORCE_LINK_PASS);
2452
	wrlp(mp, PORT_SERIAL_CONTROL, data);
L
Linus Torvalds 已提交
2453 2454
}

2455
static int mv643xx_eth_stop(struct net_device *dev)
L
Linus Torvalds 已提交
2456
{
2457
	struct mv643xx_eth_private *mp = netdev_priv(dev);
2458
	int i;
L
Linus Torvalds 已提交
2459

2460
	wrlp(mp, INT_MASK_EXT, 0x00000000);
2461 2462
	wrlp(mp, INT_MASK, 0x00000000);
	rdlp(mp, INT_MASK);
L
Linus Torvalds 已提交
2463

2464
	napi_disable(&mp->napi);
2465

2466 2467
	del_timer_sync(&mp->rx_oom);

2468
	netif_carrier_off(dev);
L
Linus Torvalds 已提交
2469

L
Lennert Buytenhek 已提交
2470 2471
	free_irq(dev->irq, dev);

2472
	port_reset(mp);
2473
	mv643xx_eth_get_stats(dev);
L
Lennert Buytenhek 已提交
2474
	mib_counters_update(mp);
2475
	del_timer_sync(&mp->mib_counters_timer);
L
Linus Torvalds 已提交
2476

2477 2478 2479 2480
	for (i = 0; i < mp->rxq_count; i++)
		rxq_deinit(mp->rxq + i);
	for (i = 0; i < mp->txq_count; i++)
		txq_deinit(mp->txq + i);
L
Linus Torvalds 已提交
2481

2482
	return 0;
L
Linus Torvalds 已提交
2483 2484
}

L
Lennert Buytenhek 已提交
2485
static int mv643xx_eth_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
L
Linus Torvalds 已提交
2486
{
2487
	struct mv643xx_eth_private *mp = netdev_priv(dev);
2488
	int ret;
L
Linus Torvalds 已提交
2489

2490 2491
	if (mp->phy == NULL)
		return -ENOTSUPP;
2492

2493 2494 2495 2496
	ret = phy_mii_ioctl(mp->phy, ifr, cmd);
	if (!ret)
		mv643xx_adjust_pscr(mp);
	return ret;
L
Linus Torvalds 已提交
2497 2498
}

2499
static int mv643xx_eth_change_mtu(struct net_device *dev, int new_mtu)
L
Linus Torvalds 已提交
2500
{
2501 2502
	struct mv643xx_eth_private *mp = netdev_priv(dev);

L
Lennert Buytenhek 已提交
2503
	if (new_mtu < 64 || new_mtu > 9500)
2504
		return -EINVAL;
L
Linus Torvalds 已提交
2505

2506
	dev->mtu = new_mtu;
2507
	mv643xx_eth_recalc_skb_size(mp);
2508 2509
	tx_set_rate(mp, 1000000000, 16777216);

2510 2511
	if (!netif_running(dev))
		return 0;
L
Linus Torvalds 已提交
2512

2513 2514 2515 2516
	/*
	 * Stop and then re-open the interface. This will allocate RX
	 * skbs of the new MTU.
	 * There is a possible danger that the open will not succeed,
L
Lennert Buytenhek 已提交
2517
	 * due to memory being full.
2518 2519 2520
	 */
	mv643xx_eth_stop(dev);
	if (mv643xx_eth_open(dev)) {
2521 2522
		netdev_err(dev,
			   "fatal error on re-opening device after MTU change\n");
2523 2524 2525
	}

	return 0;
L
Linus Torvalds 已提交
2526 2527
}

L
Lennert Buytenhek 已提交
2528
static void tx_timeout_task(struct work_struct *ugly)
L
Linus Torvalds 已提交
2529
{
L
Lennert Buytenhek 已提交
2530
	struct mv643xx_eth_private *mp;
L
Linus Torvalds 已提交
2531

L
Lennert Buytenhek 已提交
2532 2533
	mp = container_of(ugly, struct mv643xx_eth_private, tx_timeout_task);
	if (netif_running(mp->dev)) {
2534
		netif_tx_stop_all_queues(mp->dev);
L
Lennert Buytenhek 已提交
2535 2536
		port_reset(mp);
		port_start(mp);
2537
		netif_tx_wake_all_queues(mp->dev);
L
Lennert Buytenhek 已提交
2538
	}
2539 2540 2541
}

static void mv643xx_eth_tx_timeout(struct net_device *dev)
L
Linus Torvalds 已提交
2542
{
2543
	struct mv643xx_eth_private *mp = netdev_priv(dev);
L
Linus Torvalds 已提交
2544

2545
	netdev_info(dev, "tx timeout\n");
2546

2547
	schedule_work(&mp->tx_timeout_task);
L
Linus Torvalds 已提交
2548 2549
}

2550
#ifdef CONFIG_NET_POLL_CONTROLLER
L
Lennert Buytenhek 已提交
2551
static void mv643xx_eth_netpoll(struct net_device *dev)
2552
{
L
Lennert Buytenhek 已提交
2553
	struct mv643xx_eth_private *mp = netdev_priv(dev);
2554

2555 2556
	wrlp(mp, INT_MASK, 0x00000000);
	rdlp(mp, INT_MASK);
2557

L
Lennert Buytenhek 已提交
2558
	mv643xx_eth_irq(dev->irq, dev);
2559

2560
	wrlp(mp, INT_MASK, mp->int_mask);
2561
}
2562
#endif
2563 2564


2565
/* platform glue ************************************************************/
2566 2567
static void
mv643xx_eth_conf_mbus_windows(struct mv643xx_eth_shared_private *msp,
2568
			      const struct mbus_dram_target_info *dram)
2569
{
2570
	void __iomem *base = msp->base;
2571 2572 2573
	u32 win_enable;
	u32 win_protect;
	int i;
2574

2575 2576 2577 2578 2579
	for (i = 0; i < 6; i++) {
		writel(0, base + WINDOW_BASE(i));
		writel(0, base + WINDOW_SIZE(i));
		if (i < 4)
			writel(0, base + WINDOW_REMAP_HIGH(i));
2580 2581
	}

2582 2583 2584 2585
	win_enable = 0x3f;
	win_protect = 0;

	for (i = 0; i < dram->num_cs; i++) {
2586
		const struct mbus_dram_window *cs = dram->cs + i;
2587 2588 2589 2590 2591 2592 2593 2594 2595 2596 2597 2598

		writel((cs->base & 0xffff0000) |
			(cs->mbus_attr << 8) |
			dram->mbus_dram_target_id, base + WINDOW_BASE(i));
		writel((cs->size - 1) & 0xffff0000, base + WINDOW_SIZE(i));

		win_enable &= ~(1 << i);
		win_protect |= 3 << (2 * i);
	}

	writel(win_enable, base + WINDOW_BAR_ENABLE);
	msp->win_protect = win_protect;
2599 2600
}

2601 2602 2603 2604 2605 2606 2607
static void infer_hw_params(struct mv643xx_eth_shared_private *msp)
{
	/*
	 * Check whether we have a 14-bit coal limit field in bits
	 * [21:8], or a 16-bit coal limit in bits [25,21:7] of the
	 * SDMA config register.
	 */
2608 2609
	writel(0x02000000, msp->base + 0x0400 + SDMA_CONFIG);
	if (readl(msp->base + 0x0400 + SDMA_CONFIG) & 0x02000000)
2610 2611 2612
		msp->extended_rx_coal_limit = 1;
	else
		msp->extended_rx_coal_limit = 0;
2613 2614

	/*
2615 2616 2617
	 * Check whether the MAC supports TX rate control, and if
	 * yes, whether its associated registers are in the old or
	 * the new place.
2618
	 */
2619 2620
	writel(1, msp->base + 0x0400 + TX_BW_MTU_MOVED);
	if (readl(msp->base + 0x0400 + TX_BW_MTU_MOVED) & 1) {
2621 2622
		msp->tx_bw_control = TX_BW_CONTROL_NEW_LAYOUT;
	} else {
2623 2624
		writel(7, msp->base + 0x0400 + TX_BW_RATE);
		if (readl(msp->base + 0x0400 + TX_BW_RATE) & 7)
2625 2626 2627 2628
			msp->tx_bw_control = TX_BW_CONTROL_OLD_LAYOUT;
		else
			msp->tx_bw_control = TX_BW_CONTROL_ABSENT;
	}
2629 2630
}

2631
static int mv643xx_eth_shared_probe(struct platform_device *pdev)
2632
{
L
Lennert Buytenhek 已提交
2633
	static int mv643xx_eth_version_printed;
2634
	struct mv643xx_eth_shared_platform_data *pd = pdev->dev.platform_data;
2635
	struct mv643xx_eth_shared_private *msp;
2636
	const struct mbus_dram_target_info *dram;
2637 2638
	struct resource *res;
	int ret;
2639

2640
	if (!mv643xx_eth_version_printed++)
2641 2642
		pr_notice("MV-643xx 10/100/1000 ethernet driver version %s\n",
			  mv643xx_eth_driver_version);
2643

2644 2645 2646 2647
	ret = -EINVAL;
	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
	if (res == NULL)
		goto out;
2648

2649
	ret = -ENOMEM;
J
Julia Lawall 已提交
2650
	msp = kzalloc(sizeof(*msp), GFP_KERNEL);
2651 2652 2653
	if (msp == NULL)
		goto out;

2654
	msp->base = ioremap(res->start, resource_size(res));
2655
	if (msp->base == NULL)
2656 2657
		goto out_free;

2658 2659 2660 2661
	/*
	 * Set up and register SMI bus.
	 */
	if (pd == NULL || pd->shared_smi == NULL) {
2662 2663
		msp->smi_bus = mdiobus_alloc();
		if (msp->smi_bus == NULL)
2664
			goto out_unmap;
2665 2666 2667 2668 2669

		msp->smi_bus->priv = msp;
		msp->smi_bus->name = "mv643xx_eth smi";
		msp->smi_bus->read = smi_bus_read;
		msp->smi_bus->write = smi_bus_write,
2670 2671
		snprintf(msp->smi_bus->id, MII_BUS_ID_SIZE, "%s-%d",
			pdev->name, pdev->id);
2672 2673 2674 2675
		msp->smi_bus->parent = &pdev->dev;
		msp->smi_bus->phy_mask = 0xffffffff;
		if (mdiobus_register(msp->smi_bus) < 0)
			goto out_free_mii_bus;
2676 2677
		msp->smi = msp;
	} else {
2678
		msp->smi = platform_get_drvdata(pd->shared_smi);
2679
	}
2680

2681 2682 2683 2684 2685 2686 2687 2688 2689 2690 2691 2692 2693 2694 2695 2696 2697 2698
	msp->err_interrupt = NO_IRQ;
	init_waitqueue_head(&msp->smi_busy_wait);

	/*
	 * Check whether the error interrupt is hooked up.
	 */
	res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
	if (res != NULL) {
		int err;

		err = request_irq(res->start, mv643xx_eth_err_irq,
				  IRQF_SHARED, "mv643xx_eth", msp);
		if (!err) {
			writel(ERR_INT_SMI_DONE, msp->base + ERR_INT_MASK);
			msp->err_interrupt = res->start;
		}
	}

2699 2700 2701
	/*
	 * (Re-)program MBUS remapping windows if we are asked to.
	 */
2702 2703 2704
	dram = mv_mbus_dram_info();
	if (dram)
		mv643xx_eth_conf_mbus_windows(msp, dram);
2705

2706 2707
	msp->tx_csum_limit = (pd != NULL && pd->tx_csum_limit) ?
					pd->tx_csum_limit : 9 * 1024;
2708
	infer_hw_params(msp);
L
Lennert Buytenhek 已提交
2709 2710 2711

	platform_set_drvdata(pdev, msp);

2712 2713
	return 0;

2714 2715
out_free_mii_bus:
	mdiobus_free(msp->smi_bus);
2716 2717
out_unmap:
	iounmap(msp->base);
2718 2719 2720 2721 2722 2723 2724 2725
out_free:
	kfree(msp);
out:
	return ret;
}

static int mv643xx_eth_shared_remove(struct platform_device *pdev)
{
2726
	struct mv643xx_eth_shared_private *msp = platform_get_drvdata(pdev);
2727
	struct mv643xx_eth_shared_platform_data *pd = pdev->dev.platform_data;
2728

2729 2730
	if (pd == NULL || pd->shared_smi == NULL) {
		mdiobus_unregister(msp->smi_bus);
2731
		mdiobus_free(msp->smi_bus);
2732
	}
2733 2734
	if (msp->err_interrupt != NO_IRQ)
		free_irq(msp->err_interrupt, msp);
2735
	iounmap(msp->base);
2736 2737 2738
	kfree(msp);

	return 0;
2739 2740
}

2741
static struct platform_driver mv643xx_eth_shared_driver = {
L
Lennert Buytenhek 已提交
2742 2743
	.probe		= mv643xx_eth_shared_probe,
	.remove		= mv643xx_eth_shared_remove,
2744
	.driver = {
L
Lennert Buytenhek 已提交
2745
		.name	= MV643XX_ETH_SHARED_NAME,
2746 2747 2748 2749
		.owner	= THIS_MODULE,
	},
};

2750
static void phy_addr_set(struct mv643xx_eth_private *mp, int phy_addr)
L
Linus Torvalds 已提交
2751
{
2752
	int addr_shift = 5 * mp->port_num;
L
Lennert Buytenhek 已提交
2753
	u32 data;
L
Linus Torvalds 已提交
2754

L
Lennert Buytenhek 已提交
2755 2756 2757 2758
	data = rdl(mp, PHY_ADDR);
	data &= ~(0x1f << addr_shift);
	data |= (phy_addr & 0x1f) << addr_shift;
	wrl(mp, PHY_ADDR, data);
L
Linus Torvalds 已提交
2759 2760
}

2761
static int phy_addr_get(struct mv643xx_eth_private *mp)
L
Linus Torvalds 已提交
2762
{
L
Lennert Buytenhek 已提交
2763 2764 2765 2766 2767 2768 2769 2770 2771 2772 2773 2774 2775 2776 2777 2778 2779
	unsigned int data;

	data = rdl(mp, PHY_ADDR);

	return (data >> (5 * mp->port_num)) & 0x1f;
}

static void set_params(struct mv643xx_eth_private *mp,
		       struct mv643xx_eth_platform_data *pd)
{
	struct net_device *dev = mp->dev;

	if (is_valid_ether_addr(pd->mac_addr))
		memcpy(dev->dev_addr, pd->mac_addr, 6);
	else
		uc_addr_get(mp, dev->dev_addr);

2780
	mp->rx_ring_size = DEFAULT_RX_QUEUE_SIZE;
L
Lennert Buytenhek 已提交
2781
	if (pd->rx_queue_size)
2782
		mp->rx_ring_size = pd->rx_queue_size;
L
Lennert Buytenhek 已提交
2783 2784
	mp->rx_desc_sram_addr = pd->rx_sram_addr;
	mp->rx_desc_sram_size = pd->rx_sram_size;
L
Linus Torvalds 已提交
2785

2786
	mp->rxq_count = pd->rx_queue_count ? : 1;
2787

2788
	mp->tx_ring_size = DEFAULT_TX_QUEUE_SIZE;
L
Lennert Buytenhek 已提交
2789
	if (pd->tx_queue_size)
2790
		mp->tx_ring_size = pd->tx_queue_size;
L
Lennert Buytenhek 已提交
2791 2792
	mp->tx_desc_sram_addr = pd->tx_sram_addr;
	mp->tx_desc_sram_size = pd->tx_sram_size;
2793

2794
	mp->txq_count = pd->tx_queue_count ? : 1;
L
Linus Torvalds 已提交
2795 2796
}

2797 2798
static struct phy_device *phy_scan(struct mv643xx_eth_private *mp,
				   int phy_addr)
L
Linus Torvalds 已提交
2799
{
2800
	struct mii_bus *bus = mp->shared->smi->smi_bus;
2801 2802 2803 2804
	struct phy_device *phydev;
	int start;
	int num;
	int i;
2805

2806 2807 2808 2809 2810 2811 2812
	if (phy_addr == MV643XX_ETH_PHY_ADDR_DEFAULT) {
		start = phy_addr_get(mp) & 0x1f;
		num = 32;
	} else {
		start = phy_addr & 0x1f;
		num = 1;
	}
2813

2814 2815 2816
	phydev = NULL;
	for (i = 0; i < num; i++) {
		int addr = (start + i) & 0x1f;
L
Lennert Buytenhek 已提交
2817

2818 2819
		if (bus->phy_map[addr] == NULL)
			mdiobus_scan(bus, addr);
L
Linus Torvalds 已提交
2820

2821 2822 2823 2824 2825 2826
		if (phydev == NULL) {
			phydev = bus->phy_map[addr];
			if (phydev != NULL)
				phy_addr_set(mp, addr);
		}
	}
L
Linus Torvalds 已提交
2827

2828
	return phydev;
L
Linus Torvalds 已提交
2829 2830
}

2831
static void phy_init(struct mv643xx_eth_private *mp, int speed, int duplex)
2832
{
2833
	struct phy_device *phy = mp->phy;
2834

L
Lennert Buytenhek 已提交
2835 2836
	phy_reset(mp);

2837
	phy_attach(mp->dev, dev_name(&phy->dev), PHY_INTERFACE_MODE_GMII);
2838 2839 2840 2841 2842 2843

	if (speed == 0) {
		phy->autoneg = AUTONEG_ENABLE;
		phy->speed = 0;
		phy->duplex = 0;
		phy->advertising = phy->supported | ADVERTISED_Autoneg;
2844
	} else {
2845 2846 2847 2848
		phy->autoneg = AUTONEG_DISABLE;
		phy->advertising = 0;
		phy->speed = speed;
		phy->duplex = duplex;
2849
	}
2850
	phy_start_aneg(phy);
2851 2852
}

2853 2854 2855 2856
static void init_pscr(struct mv643xx_eth_private *mp, int speed, int duplex)
{
	u32 pscr;

2857
	pscr = rdlp(mp, PORT_SERIAL_CONTROL);
2858 2859
	if (pscr & SERIAL_PORT_ENABLE) {
		pscr &= ~SERIAL_PORT_ENABLE;
2860
		wrlp(mp, PORT_SERIAL_CONTROL, pscr);
2861 2862 2863
	}

	pscr = MAX_RX_PACKET_9700BYTE | SERIAL_PORT_CONTROL_RESERVED;
2864
	if (mp->phy == NULL) {
2865 2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876 2877
		pscr |= DISABLE_AUTO_NEG_SPEED_GMII;
		if (speed == SPEED_1000)
			pscr |= SET_GMII_SPEED_TO_1000;
		else if (speed == SPEED_100)
			pscr |= SET_MII_SPEED_TO_100;

		pscr |= DISABLE_AUTO_NEG_FOR_FLOW_CTRL;

		pscr |= DISABLE_AUTO_NEG_FOR_DUPLEX;
		if (duplex == DUPLEX_FULL)
			pscr |= SET_FULL_DUPLEX_MODE;
	}

2878
	wrlp(mp, PORT_SERIAL_CONTROL, pscr);
2879 2880
}

2881 2882 2883 2884 2885 2886
static const struct net_device_ops mv643xx_eth_netdev_ops = {
	.ndo_open		= mv643xx_eth_open,
	.ndo_stop		= mv643xx_eth_stop,
	.ndo_start_xmit		= mv643xx_eth_xmit,
	.ndo_set_rx_mode	= mv643xx_eth_set_rx_mode,
	.ndo_set_mac_address	= mv643xx_eth_set_mac_address,
2887
	.ndo_validate_addr	= eth_validate_addr,
2888 2889
	.ndo_do_ioctl		= mv643xx_eth_ioctl,
	.ndo_change_mtu		= mv643xx_eth_change_mtu,
2890
	.ndo_set_features	= mv643xx_eth_set_features,
2891 2892 2893 2894 2895 2896 2897
	.ndo_tx_timeout		= mv643xx_eth_tx_timeout,
	.ndo_get_stats		= mv643xx_eth_get_stats,
#ifdef CONFIG_NET_POLL_CONTROLLER
	.ndo_poll_controller	= mv643xx_eth_netpoll,
#endif
};

2898
static int mv643xx_eth_probe(struct platform_device *pdev)
L
Linus Torvalds 已提交
2899
{
2900
	struct mv643xx_eth_platform_data *pd;
2901
	struct mv643xx_eth_private *mp;
2902 2903
	struct net_device *dev;
	struct resource *res;
L
Lennert Buytenhek 已提交
2904
	int err;
L
Linus Torvalds 已提交
2905

2906 2907
	pd = pdev->dev.platform_data;
	if (pd == NULL) {
2908
		dev_err(&pdev->dev, "no mv643xx_eth_platform_data\n");
2909 2910
		return -ENODEV;
	}
L
Linus Torvalds 已提交
2911

2912
	if (pd->shared == NULL) {
2913
		dev_err(&pdev->dev, "no mv643xx_eth_platform_data->shared\n");
2914 2915
		return -ENODEV;
	}
2916

2917
	dev = alloc_etherdev_mq(sizeof(struct mv643xx_eth_private), 8);
2918 2919
	if (!dev)
		return -ENOMEM;
L
Linus Torvalds 已提交
2920

2921
	mp = netdev_priv(dev);
L
Lennert Buytenhek 已提交
2922 2923 2924
	platform_set_drvdata(pdev, mp);

	mp->shared = platform_get_drvdata(pd->shared);
2925
	mp->base = mp->shared->base + 0x0400 + (pd->port_number << 10);
L
Lennert Buytenhek 已提交
2926 2927
	mp->port_num = pd->port_number;

2928
	mp->dev = dev;
2929

2930
	/*
2931 2932
	 * Start with a default rate, and if there is a clock, allow
	 * it to override the default.
2933
	 */
2934 2935
	mp->t_clk = 133000000;
#if defined(CONFIG_HAVE_CLK)
2936 2937 2938 2939 2940
	mp->clk = clk_get(&pdev->dev, (pdev->id ? "1" : "0"));
	if (!IS_ERR(mp->clk)) {
		clk_prepare_enable(mp->clk);
		mp->t_clk = clk_get_rate(mp->clk);
	}
2941
#endif
L
Lennert Buytenhek 已提交
2942
	set_params(mp, pd);
2943 2944
	netif_set_real_num_tx_queues(dev, mp->txq_count);
	netif_set_real_num_rx_queues(dev, mp->rxq_count);
L
Lennert Buytenhek 已提交
2945

2946 2947
	if (pd->phy_addr != MV643XX_ETH_PHY_NONE)
		mp->phy = phy_scan(mp, pd->phy_addr);
2948

2949
	if (mp->phy != NULL)
2950
		phy_init(mp, pd->speed, pd->duplex);
2951 2952

	SET_ETHTOOL_OPS(dev, &mv643xx_eth_ethtool_ops);
2953

2954
	init_pscr(mp, pd->speed, pd->duplex);
L
Lennert Buytenhek 已提交
2955

2956 2957 2958 2959 2960 2961 2962 2963 2964 2965 2966 2967 2968

	mib_counters_clear(mp);

	init_timer(&mp->mib_counters_timer);
	mp->mib_counters_timer.data = (unsigned long)mp;
	mp->mib_counters_timer.function = mib_counters_timer_wrapper;
	mp->mib_counters_timer.expires = jiffies + 30 * HZ;
	add_timer(&mp->mib_counters_timer);

	spin_lock_init(&mp->mib_counters_lock);

	INIT_WORK(&mp->tx_timeout_task, tx_timeout_task);

2969 2970 2971 2972 2973 2974
	netif_napi_add(dev, &mp->napi, mv643xx_eth_poll, 128);

	init_timer(&mp->rx_oom);
	mp->rx_oom.data = (unsigned long)mp;
	mp->rx_oom.function = oom_timer_wrapper;

L
Lennert Buytenhek 已提交
2975

2976 2977 2978
	res = platform_get_resource(pdev, IORESOURCE_IRQ, 0);
	BUG_ON(!res);
	dev->irq = res->start;
L
Linus Torvalds 已提交
2979

2980 2981
	dev->netdev_ops = &mv643xx_eth_netdev_ops;

2982 2983
	dev->watchdog_timeo = 2 * HZ;
	dev->base_addr = 0;
L
Linus Torvalds 已提交
2984

2985 2986 2987
	dev->hw_features = NETIF_F_SG | NETIF_F_IP_CSUM |
		NETIF_F_RXCSUM | NETIF_F_LRO;
	dev->features = NETIF_F_SG | NETIF_F_IP_CSUM | NETIF_F_RXCSUM;
2988
	dev->vlan_features = NETIF_F_SG | NETIF_F_IP_CSUM;
L
Linus Torvalds 已提交
2989

2990 2991
	dev->priv_flags |= IFF_UNICAST_FLT;

L
Lennert Buytenhek 已提交
2992
	SET_NETDEV_DEV(dev, &pdev->dev);
2993

2994
	if (mp->shared->win_protect)
L
Lennert Buytenhek 已提交
2995
		wrl(mp, WINDOW_PROTECT(mp->port_num), mp->shared->win_protect);
L
Linus Torvalds 已提交
2996

2997 2998
	netif_carrier_off(dev);

2999 3000
	wrlp(mp, SDMA_CONFIG, PORT_SDMA_CONFIG_DEFAULT_VALUE);

3001
	set_rx_coal(mp, 250);
3002 3003
	set_tx_coal(mp, 0);

3004 3005 3006
	err = register_netdev(dev);
	if (err)
		goto out;
L
Linus Torvalds 已提交
3007

3008 3009
	netdev_notice(dev, "port %d with MAC address %pM\n",
		      mp->port_num, dev->dev_addr);
L
Linus Torvalds 已提交
3010

3011
	if (mp->tx_desc_sram_size > 0)
3012
		netdev_notice(dev, "configured with sram\n");
L
Linus Torvalds 已提交
3013

3014
	return 0;
L
Linus Torvalds 已提交
3015

3016
out:
3017 3018 3019 3020 3021 3022
#if defined(CONFIG_HAVE_CLK)
	if (!IS_ERR(mp->clk)) {
		clk_disable_unprepare(mp->clk);
		clk_put(mp->clk);
	}
#endif
3023
	free_netdev(dev);
L
Linus Torvalds 已提交
3024

3025
	return err;
L
Linus Torvalds 已提交
3026 3027
}

3028
static int mv643xx_eth_remove(struct platform_device *pdev)
L
Linus Torvalds 已提交
3029
{
L
Lennert Buytenhek 已提交
3030
	struct mv643xx_eth_private *mp = platform_get_drvdata(pdev);
L
Linus Torvalds 已提交
3031

L
Lennert Buytenhek 已提交
3032
	unregister_netdev(mp->dev);
3033 3034
	if (mp->phy != NULL)
		phy_detach(mp->phy);
3035
	cancel_work_sync(&mp->tx_timeout_task);
3036

3037
#if defined(CONFIG_HAVE_CLK)
3038 3039 3040 3041
	if (!IS_ERR(mp->clk)) {
		clk_disable_unprepare(mp->clk);
		clk_put(mp->clk);
	}
3042 3043
#endif

L
Lennert Buytenhek 已提交
3044
	free_netdev(mp->dev);
3045 3046

	platform_set_drvdata(pdev, NULL);
L
Lennert Buytenhek 已提交
3047

3048
	return 0;
L
Linus Torvalds 已提交
3049 3050
}

3051
static void mv643xx_eth_shutdown(struct platform_device *pdev)
3052
{
L
Lennert Buytenhek 已提交
3053
	struct mv643xx_eth_private *mp = platform_get_drvdata(pdev);
3054

3055
	/* Mask all interrupts on ethernet port */
3056 3057
	wrlp(mp, INT_MASK, 0);
	rdlp(mp, INT_MASK);
3058

L
Lennert Buytenhek 已提交
3059 3060
	if (netif_running(mp->dev))
		port_reset(mp);
3061 3062
}

3063
static struct platform_driver mv643xx_eth_driver = {
L
Lennert Buytenhek 已提交
3064 3065 3066
	.probe		= mv643xx_eth_probe,
	.remove		= mv643xx_eth_remove,
	.shutdown	= mv643xx_eth_shutdown,
3067
	.driver = {
L
Lennert Buytenhek 已提交
3068
		.name	= MV643XX_ETH_NAME,
3069 3070 3071 3072
		.owner	= THIS_MODULE,
	},
};

3073
static int __init mv643xx_eth_init_module(void)
3074
{
3075
	int rc;
3076

3077 3078 3079 3080 3081 3082
	rc = platform_driver_register(&mv643xx_eth_shared_driver);
	if (!rc) {
		rc = platform_driver_register(&mv643xx_eth_driver);
		if (rc)
			platform_driver_unregister(&mv643xx_eth_shared_driver);
	}
L
Lennert Buytenhek 已提交
3083

3084
	return rc;
3085
}
L
Lennert Buytenhek 已提交
3086
module_init(mv643xx_eth_init_module);
3087

3088
static void __exit mv643xx_eth_cleanup_module(void)
3089
{
3090 3091
	platform_driver_unregister(&mv643xx_eth_driver);
	platform_driver_unregister(&mv643xx_eth_shared_driver);
3092
}
3093
module_exit(mv643xx_eth_cleanup_module);
L
Linus Torvalds 已提交
3094

3095 3096
MODULE_AUTHOR("Rabeeh Khoury, Assaf Hoffman, Matthew Dharm, "
	      "Manish Lachwani, Dale Farnsworth and Lennert Buytenhek");
3097
MODULE_DESCRIPTION("Ethernet driver for Marvell MV643XX");
L
Lennert Buytenhek 已提交
3098
MODULE_LICENSE("GPL");
3099
MODULE_ALIAS("platform:" MV643XX_ETH_SHARED_NAME);
L
Lennert Buytenhek 已提交
3100
MODULE_ALIAS("platform:" MV643XX_ETH_NAME);