mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	net: add support for STMicroelectronics Ethernet controllers.
This is the driver for the ST MAC 10/100/1000 on-chip Ethernet controllers (Synopsys IP blocks). Driver documentation: o http://stlinux.com/drupal/kernel/network/stmmac Revisions: o http://stlinux.com/drupal/kernel/network/stmmac-driver-revisions Performances: o http://stlinux.com/drupal/benchmarks/networking/stmmac Signed-off-by: Giuseppe Cavallaro <peppe.cavallaro@st.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
		
							parent
							
								
									47a01a0c94
								
							
						
					
					
						commit
						47dd7a540b
					
				| @ -2483,6 +2483,8 @@ config S6GMAC | |||||||
| 	  To compile this driver as a module, choose M here. The module | 	  To compile this driver as a module, choose M here. The module | ||||||
| 	  will be called s6gmac. | 	  will be called s6gmac. | ||||||
| 
 | 
 | ||||||
|  | source "drivers/net/stmmac/Kconfig" | ||||||
|  | 
 | ||||||
| endif # NETDEV_1000 | endif # NETDEV_1000 | ||||||
| 
 | 
 | ||||||
| # | # | ||||||
|  | |||||||
| @ -100,6 +100,7 @@ obj-$(CONFIG_VIA_VELOCITY) += via-velocity.o | |||||||
| obj-$(CONFIG_ADAPTEC_STARFIRE) += starfire.o | obj-$(CONFIG_ADAPTEC_STARFIRE) += starfire.o | ||||||
| obj-$(CONFIG_RIONET) += rionet.o | obj-$(CONFIG_RIONET) += rionet.o | ||||||
| obj-$(CONFIG_SH_ETH) += sh_eth.o | obj-$(CONFIG_SH_ETH) += sh_eth.o | ||||||
|  | obj-$(CONFIG_STMMAC_ETH) += stmmac/ | ||||||
| 
 | 
 | ||||||
| #
 | #
 | ||||||
| # end link order section
 | # end link order section
 | ||||||
|  | |||||||
							
								
								
									
										53
									
								
								drivers/net/stmmac/Kconfig
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								drivers/net/stmmac/Kconfig
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,53 @@ | |||||||
|  | config STMMAC_ETH | ||||||
|  | 	tristate "STMicroelectronics 10/100/1000 Ethernet driver" | ||||||
|  | 	select MII | ||||||
|  | 	select PHYLIB | ||||||
|  | 	depends on NETDEVICES && CPU_SUBTYPE_ST40 | ||||||
|  | 	help | ||||||
|  | 	  This is the driver for the ST MAC 10/100/1000 on-chip Ethernet | ||||||
|  | 	  controllers. ST Ethernet IPs are built around a Synopsys IP Core. | ||||||
|  | 
 | ||||||
|  | if STMMAC_ETH | ||||||
|  | 
 | ||||||
|  | config STMMAC_DA | ||||||
|  | 	bool "STMMAC DMA arbitration scheme" | ||||||
|  | 	default n | ||||||
|  | 	help | ||||||
|  | 	  Selecting this option, rx has priority over Tx (only for Giga | ||||||
|  | 	  Ethernet device). | ||||||
|  | 	  By default, the DMA arbitration scheme is based on Round-robin | ||||||
|  | 	  (rx:tx priority is 1:1). | ||||||
|  | 
 | ||||||
|  | config STMMAC_DUAL_MAC | ||||||
|  | 	bool "STMMAC: dual mac support (EXPERIMENTAL)" | ||||||
|  | 	default n | ||||||
|  |         depends on EXPERIMENTAL && STMMAC_ETH && !STMMAC_TIMER | ||||||
|  | 	help | ||||||
|  | 	  Some ST SoCs (for example the stx7141 and stx7200c2) have two | ||||||
|  | 	  Ethernet Controllers. This option turns on the second Ethernet | ||||||
|  | 	  device on this kind of platforms. | ||||||
|  | 
 | ||||||
|  | config STMMAC_TIMER | ||||||
|  | 	bool "STMMAC Timer optimisation" | ||||||
|  | 	default n | ||||||
|  | 	help | ||||||
|  | 	  Use an external timer for mitigating the number of network | ||||||
|  | 	  interrupts. | ||||||
|  | 
 | ||||||
|  | choice | ||||||
|  |         prompt "Select Timer device" | ||||||
|  |         depends on STMMAC_TIMER | ||||||
|  | 
 | ||||||
|  | config STMMAC_TMU_TIMER | ||||||
|  |         bool "TMU channel 2" | ||||||
|  |         depends on CPU_SH4 | ||||||
|  | 	help | ||||||
|  | 
 | ||||||
|  | config STMMAC_RTC_TIMER | ||||||
|  |         bool "Real time clock" | ||||||
|  |         depends on RTC_CLASS | ||||||
|  | 	help | ||||||
|  | 
 | ||||||
|  | endchoice | ||||||
|  | 
 | ||||||
|  | endif | ||||||
							
								
								
									
										4
									
								
								drivers/net/stmmac/Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								drivers/net/stmmac/Makefile
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | obj-$(CONFIG_STMMAC_ETH) += stmmac.o | ||||||
|  | stmmac-$(CONFIG_STMMAC_TIMER) += stmmac_timer.o | ||||||
|  | stmmac-objs:= stmmac_main.o stmmac_ethtool.o stmmac_mdio.o \
 | ||||||
|  | 		mac100.o  gmac.o $(stmmac-y) | ||||||
							
								
								
									
										330
									
								
								drivers/net/stmmac/common.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										330
									
								
								drivers/net/stmmac/common.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,330 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   STMMAC Common Header File | ||||||
|  | 
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #include "descs.h" | ||||||
|  | #include <linux/io.h> | ||||||
|  | 
 | ||||||
|  | /* *********************************************
 | ||||||
|  |    DMA CRS Control and Status Register Mapping | ||||||
|  |  * *********************************************/ | ||||||
|  | #define DMA_BUS_MODE		0x00001000	/* Bus Mode */ | ||||||
|  | #define DMA_XMT_POLL_DEMAND	0x00001004	/* Transmit Poll Demand */ | ||||||
|  | #define DMA_RCV_POLL_DEMAND	0x00001008	/* Received Poll Demand */ | ||||||
|  | #define DMA_RCV_BASE_ADDR	0x0000100c	/* Receive List Base */ | ||||||
|  | #define DMA_TX_BASE_ADDR	0x00001010	/* Transmit List Base */ | ||||||
|  | #define DMA_STATUS		0x00001014	/* Status Register */ | ||||||
|  | #define DMA_CONTROL		0x00001018	/* Ctrl (Operational Mode) */ | ||||||
|  | #define DMA_INTR_ENA		0x0000101c	/* Interrupt Enable */ | ||||||
|  | #define DMA_MISSED_FRAME_CTR	0x00001020	/* Missed Frame Counter */ | ||||||
|  | #define DMA_CUR_TX_BUF_ADDR	0x00001050	/* Current Host Tx Buffer */ | ||||||
|  | #define DMA_CUR_RX_BUF_ADDR	0x00001054	/* Current Host Rx Buffer */ | ||||||
|  | 
 | ||||||
|  | /* ********************************
 | ||||||
|  |    DMA Control register defines | ||||||
|  |  * ********************************/ | ||||||
|  | #define DMA_CONTROL_ST		0x00002000	/* Start/Stop Transmission */ | ||||||
|  | #define DMA_CONTROL_SR		0x00000002	/* Start/Stop Receive */ | ||||||
|  | 
 | ||||||
|  | /* **************************************
 | ||||||
|  |    DMA Interrupt Enable register defines | ||||||
|  |  * **************************************/ | ||||||
|  | /**** NORMAL INTERRUPT ****/ | ||||||
|  | #define DMA_INTR_ENA_NIE 0x00010000	/* Normal Summary */ | ||||||
|  | #define DMA_INTR_ENA_TIE 0x00000001	/* Transmit Interrupt */ | ||||||
|  | #define DMA_INTR_ENA_TUE 0x00000004	/* Transmit Buffer Unavailable */ | ||||||
|  | #define DMA_INTR_ENA_RIE 0x00000040	/* Receive Interrupt */ | ||||||
|  | #define DMA_INTR_ENA_ERE 0x00004000	/* Early Receive */ | ||||||
|  | 
 | ||||||
|  | #define DMA_INTR_NORMAL	(DMA_INTR_ENA_NIE | DMA_INTR_ENA_RIE | \ | ||||||
|  | 			DMA_INTR_ENA_TIE) | ||||||
|  | 
 | ||||||
|  | /**** ABNORMAL INTERRUPT ****/ | ||||||
|  | #define DMA_INTR_ENA_AIE 0x00008000	/* Abnormal Summary */ | ||||||
|  | #define DMA_INTR_ENA_FBE 0x00002000	/* Fatal Bus Error */ | ||||||
|  | #define DMA_INTR_ENA_ETE 0x00000400	/* Early Transmit */ | ||||||
|  | #define DMA_INTR_ENA_RWE 0x00000200	/* Receive Watchdog */ | ||||||
|  | #define DMA_INTR_ENA_RSE 0x00000100	/* Receive Stopped */ | ||||||
|  | #define DMA_INTR_ENA_RUE 0x00000080	/* Receive Buffer Unavailable */ | ||||||
|  | #define DMA_INTR_ENA_UNE 0x00000020	/* Tx Underflow */ | ||||||
|  | #define DMA_INTR_ENA_OVE 0x00000010	/* Receive Overflow */ | ||||||
|  | #define DMA_INTR_ENA_TJE 0x00000008	/* Transmit Jabber */ | ||||||
|  | #define DMA_INTR_ENA_TSE 0x00000002	/* Transmit Stopped */ | ||||||
|  | 
 | ||||||
|  | #define DMA_INTR_ABNORMAL	(DMA_INTR_ENA_AIE | DMA_INTR_ENA_FBE | \ | ||||||
|  | 				DMA_INTR_ENA_UNE) | ||||||
|  | 
 | ||||||
|  | /* DMA default interrupt mask */ | ||||||
|  | #define DMA_INTR_DEFAULT_MASK	(DMA_INTR_NORMAL | DMA_INTR_ABNORMAL) | ||||||
|  | 
 | ||||||
|  | /* ****************************
 | ||||||
|  |  *  DMA Status register defines | ||||||
|  |  * ****************************/ | ||||||
|  | #define DMA_STATUS_GPI		0x10000000	/* PMT interrupt */ | ||||||
|  | #define DMA_STATUS_GMI		0x08000000	/* MMC interrupt */ | ||||||
|  | #define DMA_STATUS_GLI		0x04000000	/* GMAC Line interface int. */ | ||||||
|  | #define DMA_STATUS_GMI		0x08000000 | ||||||
|  | #define DMA_STATUS_GLI		0x04000000 | ||||||
|  | #define DMA_STATUS_EB_MASK	0x00380000	/* Error Bits Mask */ | ||||||
|  | #define DMA_STATUS_EB_TX_ABORT	0x00080000	/* Error Bits - TX Abort */ | ||||||
|  | #define DMA_STATUS_EB_RX_ABORT	0x00100000	/* Error Bits - RX Abort */ | ||||||
|  | #define DMA_STATUS_TS_MASK	0x00700000	/* Transmit Process State */ | ||||||
|  | #define DMA_STATUS_TS_SHIFT	20 | ||||||
|  | #define DMA_STATUS_RS_MASK	0x000e0000	/* Receive Process State */ | ||||||
|  | #define DMA_STATUS_RS_SHIFT	17 | ||||||
|  | #define DMA_STATUS_NIS	0x00010000	/* Normal Interrupt Summary */ | ||||||
|  | #define DMA_STATUS_AIS	0x00008000	/* Abnormal Interrupt Summary */ | ||||||
|  | #define DMA_STATUS_ERI	0x00004000	/* Early Receive Interrupt */ | ||||||
|  | #define DMA_STATUS_FBI	0x00002000	/* Fatal Bus Error Interrupt */ | ||||||
|  | #define DMA_STATUS_ETI	0x00000400	/* Early Transmit Interrupt */ | ||||||
|  | #define DMA_STATUS_RWT	0x00000200	/* Receive Watchdog Timeout */ | ||||||
|  | #define DMA_STATUS_RPS	0x00000100	/* Receive Process Stopped */ | ||||||
|  | #define DMA_STATUS_RU	0x00000080	/* Receive Buffer Unavailable */ | ||||||
|  | #define DMA_STATUS_RI	0x00000040	/* Receive Interrupt */ | ||||||
|  | #define DMA_STATUS_UNF	0x00000020	/* Transmit Underflow */ | ||||||
|  | #define DMA_STATUS_OVF	0x00000010	/* Receive Overflow */ | ||||||
|  | #define DMA_STATUS_TJT	0x00000008	/* Transmit Jabber Timeout */ | ||||||
|  | #define DMA_STATUS_TU	0x00000004	/* Transmit Buffer Unavailable */ | ||||||
|  | #define DMA_STATUS_TPS	0x00000002	/* Transmit Process Stopped */ | ||||||
|  | #define DMA_STATUS_TI	0x00000001	/* Transmit Interrupt */ | ||||||
|  | 
 | ||||||
|  | /* Other defines */ | ||||||
|  | #define HASH_TABLE_SIZE 64 | ||||||
|  | #define PAUSE_TIME 0x200 | ||||||
|  | 
 | ||||||
|  | /* Flow Control defines */ | ||||||
|  | #define FLOW_OFF	0 | ||||||
|  | #define FLOW_RX		1 | ||||||
|  | #define FLOW_TX		2 | ||||||
|  | #define FLOW_AUTO	(FLOW_TX | FLOW_RX) | ||||||
|  | 
 | ||||||
|  | /* DMA STORE-AND-FORWARD Operation Mode */ | ||||||
|  | #define SF_DMA_MODE 1 | ||||||
|  | 
 | ||||||
|  | #define HW_CSUM 1 | ||||||
|  | #define NO_HW_CSUM 0 | ||||||
|  | 
 | ||||||
|  | /* GMAC TX FIFO is 8K, Rx FIFO is 16K */ | ||||||
|  | #define BUF_SIZE_16KiB 16384 | ||||||
|  | #define BUF_SIZE_8KiB 8192 | ||||||
|  | #define BUF_SIZE_4KiB 4096 | ||||||
|  | #define BUF_SIZE_2KiB 2048 | ||||||
|  | 
 | ||||||
|  | /* Power Down and WOL */ | ||||||
|  | #define PMT_NOT_SUPPORTED 0 | ||||||
|  | #define PMT_SUPPORTED 1 | ||||||
|  | 
 | ||||||
|  | /* Common MAC defines */ | ||||||
|  | #define MAC_CTRL_REG		0x00000000	/* MAC Control */ | ||||||
|  | #define MAC_ENABLE_TX		0x00000008	/* Transmitter Enable */ | ||||||
|  | #define MAC_RNABLE_RX		0x00000004	/* Receiver Enable */ | ||||||
|  | 
 | ||||||
|  | /* MAC Management Counters register */ | ||||||
|  | #define MMC_CONTROL		0x00000100	/* MMC Control */ | ||||||
|  | #define MMC_HIGH_INTR		0x00000104	/* MMC High Interrupt */ | ||||||
|  | #define MMC_LOW_INTR		0x00000108	/* MMC Low Interrupt */ | ||||||
|  | #define MMC_HIGH_INTR_MASK	0x0000010c	/* MMC High Interrupt Mask */ | ||||||
|  | #define MMC_LOW_INTR_MASK	0x00000110	/* MMC Low Interrupt Mask */ | ||||||
|  | 
 | ||||||
|  | #define MMC_CONTROL_MAX_FRM_MASK	0x0003ff8	/* Maximum Frame Size */ | ||||||
|  | #define MMC_CONTROL_MAX_FRM_SHIFT	3 | ||||||
|  | #define MMC_CONTROL_MAX_FRAME		0x7FF | ||||||
|  | 
 | ||||||
|  | struct stmmac_extra_stats { | ||||||
|  | 	/* Transmit errors */ | ||||||
|  | 	unsigned long tx_underflow ____cacheline_aligned; | ||||||
|  | 	unsigned long tx_carrier; | ||||||
|  | 	unsigned long tx_losscarrier; | ||||||
|  | 	unsigned long tx_heartbeat; | ||||||
|  | 	unsigned long tx_deferred; | ||||||
|  | 	unsigned long tx_vlan; | ||||||
|  | 	unsigned long tx_jabber; | ||||||
|  | 	unsigned long tx_frame_flushed; | ||||||
|  | 	unsigned long tx_payload_error; | ||||||
|  | 	unsigned long tx_ip_header_error; | ||||||
|  | 	/* Receive errors */ | ||||||
|  | 	unsigned long rx_desc; | ||||||
|  | 	unsigned long rx_partial; | ||||||
|  | 	unsigned long rx_runt; | ||||||
|  | 	unsigned long rx_toolong; | ||||||
|  | 	unsigned long rx_collision; | ||||||
|  | 	unsigned long rx_crc; | ||||||
|  | 	unsigned long rx_lenght; | ||||||
|  | 	unsigned long rx_mii; | ||||||
|  | 	unsigned long rx_multicast; | ||||||
|  | 	unsigned long rx_gmac_overflow; | ||||||
|  | 	unsigned long rx_watchdog; | ||||||
|  | 	unsigned long da_rx_filter_fail; | ||||||
|  | 	unsigned long sa_rx_filter_fail; | ||||||
|  | 	unsigned long rx_missed_cntr; | ||||||
|  | 	unsigned long rx_overflow_cntr; | ||||||
|  | 	unsigned long rx_vlan; | ||||||
|  | 	/* Tx/Rx IRQ errors */ | ||||||
|  | 	unsigned long tx_undeflow_irq; | ||||||
|  | 	unsigned long tx_process_stopped_irq; | ||||||
|  | 	unsigned long tx_jabber_irq; | ||||||
|  | 	unsigned long rx_overflow_irq; | ||||||
|  | 	unsigned long rx_buf_unav_irq; | ||||||
|  | 	unsigned long rx_process_stopped_irq; | ||||||
|  | 	unsigned long rx_watchdog_irq; | ||||||
|  | 	unsigned long tx_early_irq; | ||||||
|  | 	unsigned long fatal_bus_error_irq; | ||||||
|  | 	/* Extra info */ | ||||||
|  | 	unsigned long threshold; | ||||||
|  | 	unsigned long tx_pkt_n; | ||||||
|  | 	unsigned long rx_pkt_n; | ||||||
|  | 	unsigned long poll_n; | ||||||
|  | 	unsigned long sched_timer_n; | ||||||
|  | 	unsigned long normal_irq_n; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* GMAC core can compute the checksums in HW. */ | ||||||
|  | enum rx_frame_status { | ||||||
|  | 	good_frame = 0, | ||||||
|  | 	discard_frame = 1, | ||||||
|  | 	csum_none = 2, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | static inline void stmmac_set_mac_addr(unsigned long ioaddr, u8 addr[6], | ||||||
|  | 			 unsigned int high, unsigned int low) | ||||||
|  | { | ||||||
|  | 	unsigned long data; | ||||||
|  | 
 | ||||||
|  | 	data = (addr[5] << 8) | addr[4]; | ||||||
|  | 	writel(data, ioaddr + high); | ||||||
|  | 	data = (addr[3] << 24) | (addr[2] << 16) | (addr[1] << 8) | addr[0]; | ||||||
|  | 	writel(data, ioaddr + low); | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static inline void stmmac_get_mac_addr(unsigned long ioaddr, | ||||||
|  | 				unsigned char *addr, unsigned int high, | ||||||
|  | 				unsigned int low) | ||||||
|  | { | ||||||
|  | 	unsigned int hi_addr, lo_addr; | ||||||
|  | 
 | ||||||
|  | 	/* Read the MAC address from the hardware */ | ||||||
|  | 	hi_addr = readl(ioaddr + high); | ||||||
|  | 	lo_addr = readl(ioaddr + low); | ||||||
|  | 
 | ||||||
|  | 	/* Extract the MAC address from the high and low words */ | ||||||
|  | 	addr[0] = lo_addr & 0xff; | ||||||
|  | 	addr[1] = (lo_addr >> 8) & 0xff; | ||||||
|  | 	addr[2] = (lo_addr >> 16) & 0xff; | ||||||
|  | 	addr[3] = (lo_addr >> 24) & 0xff; | ||||||
|  | 	addr[4] = hi_addr & 0xff; | ||||||
|  | 	addr[5] = (hi_addr >> 8) & 0xff; | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct stmmac_ops { | ||||||
|  | 	/* MAC core initialization */ | ||||||
|  | 	void (*core_init) (unsigned long ioaddr) ____cacheline_aligned; | ||||||
|  | 	/* DMA core initialization */ | ||||||
|  | 	int (*dma_init) (unsigned long ioaddr, int pbl, u32 dma_tx, u32 dma_rx); | ||||||
|  | 	/* Dump MAC registers */ | ||||||
|  | 	void (*dump_mac_regs) (unsigned long ioaddr); | ||||||
|  | 	/* Dump DMA registers */ | ||||||
|  | 	void (*dump_dma_regs) (unsigned long ioaddr); | ||||||
|  | 	/* Set tx/rx threshold in the csr6 register
 | ||||||
|  | 	 * An invalid value enables the store-and-forward mode */ | ||||||
|  | 	void (*dma_mode) (unsigned long ioaddr, int txmode, int rxmode); | ||||||
|  | 	/* To track extra statistic (if supported) */ | ||||||
|  | 	void (*dma_diagnostic_fr) (void *data, struct stmmac_extra_stats *x, | ||||||
|  | 				   unsigned long ioaddr); | ||||||
|  | 	/* RX descriptor ring initialization */ | ||||||
|  | 	void (*init_rx_desc) (struct dma_desc *p, unsigned int ring_size, | ||||||
|  | 				int disable_rx_ic); | ||||||
|  | 	/* TX descriptor ring initialization */ | ||||||
|  | 	void (*init_tx_desc) (struct dma_desc *p, unsigned int ring_size); | ||||||
|  | 
 | ||||||
|  | 	/* Invoked by the xmit function to prepare the tx descriptor */ | ||||||
|  | 	void (*prepare_tx_desc) (struct dma_desc *p, int is_fs, int len, | ||||||
|  | 				 int csum_flag); | ||||||
|  | 	/* Set/get the owner of the descriptor */ | ||||||
|  | 	void (*set_tx_owner) (struct dma_desc *p); | ||||||
|  | 	int (*get_tx_owner) (struct dma_desc *p); | ||||||
|  | 	/* Invoked by the xmit function to close the tx descriptor */ | ||||||
|  | 	void (*close_tx_desc) (struct dma_desc *p); | ||||||
|  | 	/* Clean the tx descriptor as soon as the tx irq is received */ | ||||||
|  | 	void (*release_tx_desc) (struct dma_desc *p); | ||||||
|  | 	/* Clear interrupt on tx frame completion. When this bit is
 | ||||||
|  | 	 * set an interrupt happens as soon as the frame is transmitted */ | ||||||
|  | 	void (*clear_tx_ic) (struct dma_desc *p); | ||||||
|  | 	/* Last tx segment reports the transmit status */ | ||||||
|  | 	int (*get_tx_ls) (struct dma_desc *p); | ||||||
|  | 	/* Return the transmit status looking at the TDES1 */ | ||||||
|  | 	int (*tx_status) (void *data, struct stmmac_extra_stats *x, | ||||||
|  | 			  struct dma_desc *p, unsigned long ioaddr); | ||||||
|  | 	/* Get the buffer size from the descriptor */ | ||||||
|  | 	int (*get_tx_len) (struct dma_desc *p); | ||||||
|  | 	/* Handle extra events on specific interrupts hw dependent */ | ||||||
|  | 	void (*host_irq_status) (unsigned long ioaddr); | ||||||
|  | 	int (*get_rx_owner) (struct dma_desc *p); | ||||||
|  | 	void (*set_rx_owner) (struct dma_desc *p); | ||||||
|  | 	/* Get the receive frame size */ | ||||||
|  | 	int (*get_rx_frame_len) (struct dma_desc *p); | ||||||
|  | 	/* Return the reception status looking at the RDES1 */ | ||||||
|  | 	int (*rx_status) (void *data, struct stmmac_extra_stats *x, | ||||||
|  | 			  struct dma_desc *p); | ||||||
|  | 	/* Multicast filter setting */ | ||||||
|  | 	void (*set_filter) (struct net_device *dev); | ||||||
|  | 	/* Flow control setting */ | ||||||
|  | 	void (*flow_ctrl) (unsigned long ioaddr, unsigned int duplex, | ||||||
|  | 			   unsigned int fc, unsigned int pause_time); | ||||||
|  | 	/* Set power management mode (e.g. magic frame) */ | ||||||
|  | 	void (*pmt) (unsigned long ioaddr, unsigned long mode); | ||||||
|  | 	/* Set/Get Unicast MAC addresses */ | ||||||
|  | 	void (*set_umac_addr) (unsigned long ioaddr, unsigned char *addr, | ||||||
|  | 			     unsigned int reg_n); | ||||||
|  | 	void (*get_umac_addr) (unsigned long ioaddr, unsigned char *addr, | ||||||
|  | 			     unsigned int reg_n); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct mac_link { | ||||||
|  | 	int port; | ||||||
|  | 	int duplex; | ||||||
|  | 	int speed; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct mii_regs { | ||||||
|  | 	unsigned int addr;	/* MII Address */ | ||||||
|  | 	unsigned int data;	/* MII Data */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct hw_cap { | ||||||
|  | 	unsigned int version;	/* Core Version register (GMAC) */ | ||||||
|  | 	unsigned int pmt;	/* Power-Down mode (GMAC) */ | ||||||
|  | 	struct mac_link link; | ||||||
|  | 	struct mii_regs mii; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct mac_device_info { | ||||||
|  | 	struct hw_cap hw; | ||||||
|  | 	struct stmmac_ops *ops; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct mac_device_info *gmac_setup(unsigned long addr); | ||||||
|  | struct mac_device_info *mac100_setup(unsigned long addr); | ||||||
							
								
								
									
										163
									
								
								drivers/net/stmmac/descs.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										163
									
								
								drivers/net/stmmac/descs.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,163 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   Header File to describe the DMA descriptors | ||||||
|  |   Use enhanced descriptors in case of GMAC Cores. | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | struct dma_desc { | ||||||
|  | 	/* Receive descriptor */ | ||||||
|  | 	union { | ||||||
|  | 		struct { | ||||||
|  | 			/* RDES0 */ | ||||||
|  | 			u32 reserved1:1; | ||||||
|  | 			u32 crc_error:1; | ||||||
|  | 			u32 dribbling:1; | ||||||
|  | 			u32 mii_error:1; | ||||||
|  | 			u32 receive_watchdog:1; | ||||||
|  | 			u32 frame_type:1; | ||||||
|  | 			u32 collision:1; | ||||||
|  | 			u32 frame_too_long:1; | ||||||
|  | 			u32 last_descriptor:1; | ||||||
|  | 			u32 first_descriptor:1; | ||||||
|  | 			u32 multicast_frame:1; | ||||||
|  | 			u32 run_frame:1; | ||||||
|  | 			u32 length_error:1; | ||||||
|  | 			u32 partial_frame_error:1; | ||||||
|  | 			u32 descriptor_error:1; | ||||||
|  | 			u32 error_summary:1; | ||||||
|  | 			u32 frame_length:14; | ||||||
|  | 			u32 filtering_fail:1; | ||||||
|  | 			u32 own:1; | ||||||
|  | 			/* RDES1 */ | ||||||
|  | 			u32 buffer1_size:11; | ||||||
|  | 			u32 buffer2_size:11; | ||||||
|  | 			u32 reserved2:2; | ||||||
|  | 			u32 second_address_chained:1; | ||||||
|  | 			u32 end_ring:1; | ||||||
|  | 			u32 reserved3:5; | ||||||
|  | 			u32 disable_ic:1; | ||||||
|  | 		} rx; | ||||||
|  | 		struct { | ||||||
|  | 			/* RDES0 */ | ||||||
|  | 			u32 payload_csum_error:1; | ||||||
|  | 			u32 crc_error:1; | ||||||
|  | 			u32 dribbling:1; | ||||||
|  | 			u32 error_gmii:1; | ||||||
|  | 			u32 receive_watchdog:1; | ||||||
|  | 			u32 frame_type:1; | ||||||
|  | 			u32 late_collision:1; | ||||||
|  | 			u32 ipc_csum_error:1; | ||||||
|  | 			u32 last_descriptor:1; | ||||||
|  | 			u32 first_descriptor:1; | ||||||
|  | 			u32 vlan_tag:1; | ||||||
|  | 			u32 overflow_error:1; | ||||||
|  | 			u32 length_error:1; | ||||||
|  | 			u32 sa_filter_fail:1; | ||||||
|  | 			u32 descriptor_error:1; | ||||||
|  | 			u32 error_summary:1; | ||||||
|  | 			u32 frame_length:14; | ||||||
|  | 			u32 da_filter_fail:1; | ||||||
|  | 			u32 own:1; | ||||||
|  | 			/* RDES1 */ | ||||||
|  | 			u32 buffer1_size:13; | ||||||
|  | 			u32 reserved1:1; | ||||||
|  | 			u32 second_address_chained:1; | ||||||
|  | 			u32 end_ring:1; | ||||||
|  | 			u32 buffer2_size:13; | ||||||
|  | 			u32 reserved2:2; | ||||||
|  | 			u32 disable_ic:1; | ||||||
|  | 		} erx;		/* -- enhanced -- */ | ||||||
|  | 
 | ||||||
|  | 		/* Transmit descriptor */ | ||||||
|  | 		struct { | ||||||
|  | 			/* TDES0 */ | ||||||
|  | 			u32 deferred:1; | ||||||
|  | 			u32 underflow_error:1; | ||||||
|  | 			u32 excessive_deferral:1; | ||||||
|  | 			u32 collision_count:4; | ||||||
|  | 			u32 heartbeat_fail:1; | ||||||
|  | 			u32 excessive_collisions:1; | ||||||
|  | 			u32 late_collision:1; | ||||||
|  | 			u32 no_carrier:1; | ||||||
|  | 			u32 loss_carrier:1; | ||||||
|  | 			u32 reserved1:3; | ||||||
|  | 			u32 error_summary:1; | ||||||
|  | 			u32 reserved2:15; | ||||||
|  | 			u32 own:1; | ||||||
|  | 			/* TDES1 */ | ||||||
|  | 			u32 buffer1_size:11; | ||||||
|  | 			u32 buffer2_size:11; | ||||||
|  | 			u32 reserved3:1; | ||||||
|  | 			u32 disable_padding:1; | ||||||
|  | 			u32 second_address_chained:1; | ||||||
|  | 			u32 end_ring:1; | ||||||
|  | 			u32 crc_disable:1; | ||||||
|  | 			u32 reserved4:2; | ||||||
|  | 			u32 first_segment:1; | ||||||
|  | 			u32 last_segment:1; | ||||||
|  | 			u32 interrupt:1; | ||||||
|  | 		} tx; | ||||||
|  | 		struct { | ||||||
|  | 			/* TDES0 */ | ||||||
|  | 			u32 deferred:1; | ||||||
|  | 			u32 underflow_error:1; | ||||||
|  | 			u32 excessive_deferral:1; | ||||||
|  | 			u32 collision_count:4; | ||||||
|  | 			u32 vlan_frame:1; | ||||||
|  | 			u32 excessive_collisions:1; | ||||||
|  | 			u32 late_collision:1; | ||||||
|  | 			u32 no_carrier:1; | ||||||
|  | 			u32 loss_carrier:1; | ||||||
|  | 			u32 payload_error:1; | ||||||
|  | 			u32 frame_flushed:1; | ||||||
|  | 			u32 jabber_timeout:1; | ||||||
|  | 			u32 error_summary:1; | ||||||
|  | 			u32 ip_header_error:1; | ||||||
|  | 			u32 time_stamp_status:1; | ||||||
|  | 			u32 reserved1:2; | ||||||
|  | 			u32 second_address_chained:1; | ||||||
|  | 			u32 end_ring:1; | ||||||
|  | 			u32 checksum_insertion:2; | ||||||
|  | 			u32 reserved2:1; | ||||||
|  | 			u32 time_stamp_enable:1; | ||||||
|  | 			u32 disable_padding:1; | ||||||
|  | 			u32 crc_disable:1; | ||||||
|  | 			u32 first_segment:1; | ||||||
|  | 			u32 last_segment:1; | ||||||
|  | 			u32 interrupt:1; | ||||||
|  | 			u32 own:1; | ||||||
|  | 			/* TDES1 */ | ||||||
|  | 			u32 buffer1_size:13; | ||||||
|  | 			u32 reserved3:3; | ||||||
|  | 			u32 buffer2_size:13; | ||||||
|  | 			u32 reserved4:3; | ||||||
|  | 		} etx;		/* -- enhanced -- */ | ||||||
|  | 	} des01; | ||||||
|  | 	unsigned int des2; | ||||||
|  | 	unsigned int des3; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Transmit checksum insertion control */ | ||||||
|  | enum tdes_csum_insertion { | ||||||
|  | 	cic_disabled = 0,	/* Checksum Insertion Control */ | ||||||
|  | 	cic_only_ip = 1,	/* Only IP header */ | ||||||
|  | 	cic_no_pseudoheader = 2,	/* IP header but pseudoheader
 | ||||||
|  | 					 * is not calculated */ | ||||||
|  | 	cic_full = 3,		/* IP header and pseudoheader */ | ||||||
|  | }; | ||||||
							
								
								
									
										693
									
								
								drivers/net/stmmac/gmac.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										693
									
								
								drivers/net/stmmac/gmac.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,693 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   This is the driver for the GMAC on-chip Ethernet controller for ST SoCs. | ||||||
|  |   DWC Ether MAC 10/100/1000 Universal version 3.41a  has been used for | ||||||
|  |   developing this code. | ||||||
|  | 
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #include <linux/netdevice.h> | ||||||
|  | #include <linux/crc32.h> | ||||||
|  | #include <linux/mii.h> | ||||||
|  | #include <linux/phy.h> | ||||||
|  | 
 | ||||||
|  | #include "stmmac.h" | ||||||
|  | #include "gmac.h" | ||||||
|  | 
 | ||||||
|  | #undef GMAC_DEBUG | ||||||
|  | /*#define GMAC_DEBUG*/ | ||||||
|  | #undef FRAME_FILTER_DEBUG | ||||||
|  | /*#define FRAME_FILTER_DEBUG*/ | ||||||
|  | #ifdef GMAC_DEBUG | ||||||
|  | #define DBG(fmt, args...)  printk(fmt, ## args) | ||||||
|  | #else | ||||||
|  | #define DBG(fmt, args...)  do { } while (0) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | static void gmac_dump_regs(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	pr_info("\t----------------------------------------------\n" | ||||||
|  | 	       "\t  GMAC registers (base addr = 0x%8x)\n" | ||||||
|  | 	       "\t----------------------------------------------\n", | ||||||
|  | 	       (unsigned int)ioaddr); | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; i < 55; i++) { | ||||||
|  | 		int offset = i * 4; | ||||||
|  | 		pr_info("\tReg No. %d (offset 0x%x): 0x%08x\n", i, | ||||||
|  | 		       offset, readl(ioaddr + offset)); | ||||||
|  | 	} | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_dma_init(unsigned long ioaddr, int pbl, u32 dma_tx, u32 dma_rx) | ||||||
|  | { | ||||||
|  | 	u32 value = readl(ioaddr + DMA_BUS_MODE); | ||||||
|  | 	/* DMA SW reset */ | ||||||
|  | 	value |= DMA_BUS_MODE_SFT_RESET; | ||||||
|  | 	writel(value, ioaddr + DMA_BUS_MODE); | ||||||
|  | 	do {} while ((readl(ioaddr + DMA_BUS_MODE) & DMA_BUS_MODE_SFT_RESET)); | ||||||
|  | 
 | ||||||
|  | 	value = /* DMA_BUS_MODE_FB | */ DMA_BUS_MODE_4PBL | | ||||||
|  | 	    ((pbl << DMA_BUS_MODE_PBL_SHIFT) | | ||||||
|  | 	     (pbl << DMA_BUS_MODE_RPBL_SHIFT)); | ||||||
|  | 
 | ||||||
|  | #ifdef CONFIG_STMMAC_DA | ||||||
|  | 	value |= DMA_BUS_MODE_DA;	/* Rx has priority over tx */ | ||||||
|  | #endif | ||||||
|  | 	writel(value, ioaddr + DMA_BUS_MODE); | ||||||
|  | 
 | ||||||
|  | 	/* Mask interrupts by writing to CSR7 */ | ||||||
|  | 	writel(DMA_INTR_DEFAULT_MASK, ioaddr + DMA_INTR_ENA); | ||||||
|  | 
 | ||||||
|  | 	/* The base address of the RX/TX descriptor lists must be written into
 | ||||||
|  | 	 * DMA CSR3 and CSR4, respectively. */ | ||||||
|  | 	writel(dma_tx, ioaddr + DMA_TX_BASE_ADDR); | ||||||
|  | 	writel(dma_rx, ioaddr + DMA_RCV_BASE_ADDR); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Transmit FIFO flush operation */ | ||||||
|  | static void gmac_flush_tx_fifo(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	u32 csr6 = readl(ioaddr + DMA_CONTROL); | ||||||
|  | 	writel((csr6 | DMA_CONTROL_FTF), ioaddr + DMA_CONTROL); | ||||||
|  | 
 | ||||||
|  | 	do {} while ((readl(ioaddr + DMA_CONTROL) & DMA_CONTROL_FTF)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_dma_operation_mode(unsigned long ioaddr, int txmode, | ||||||
|  | 				    int rxmode) | ||||||
|  | { | ||||||
|  | 	u32 csr6 = readl(ioaddr + DMA_CONTROL); | ||||||
|  | 
 | ||||||
|  | 	if (txmode == SF_DMA_MODE) { | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: enabling TX store and forward mode\n"); | ||||||
|  | 		/* Transmit COE type 2 cannot be done in cut-through mode. */ | ||||||
|  | 		csr6 |= DMA_CONTROL_TSF; | ||||||
|  | 		/* Operating on second frame increase the performance
 | ||||||
|  | 		 * especially when transmit store-and-forward is used.*/ | ||||||
|  | 		csr6 |= DMA_CONTROL_OSF; | ||||||
|  | 	} else { | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: disabling TX store and forward mode" | ||||||
|  | 			      " (threshold = %d)\n", txmode); | ||||||
|  | 		csr6 &= ~DMA_CONTROL_TSF; | ||||||
|  | 		csr6 &= DMA_CONTROL_TC_TX_MASK; | ||||||
|  | 		/* Set the transmit threashold */ | ||||||
|  | 		if (txmode <= 32) | ||||||
|  | 			csr6 |= DMA_CONTROL_TTC_32; | ||||||
|  | 		else if (txmode <= 64) | ||||||
|  | 			csr6 |= DMA_CONTROL_TTC_64; | ||||||
|  | 		else if (txmode <= 128) | ||||||
|  | 			csr6 |= DMA_CONTROL_TTC_128; | ||||||
|  | 		else if (txmode <= 192) | ||||||
|  | 			csr6 |= DMA_CONTROL_TTC_192; | ||||||
|  | 		else | ||||||
|  | 			csr6 |= DMA_CONTROL_TTC_256; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (rxmode == SF_DMA_MODE) { | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: enabling RX store and forward mode\n"); | ||||||
|  | 		csr6 |= DMA_CONTROL_RSF; | ||||||
|  | 	} else { | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: disabling RX store and forward mode" | ||||||
|  | 			      " (threshold = %d)\n", rxmode); | ||||||
|  | 		csr6 &= ~DMA_CONTROL_RSF; | ||||||
|  | 		csr6 &= DMA_CONTROL_TC_RX_MASK; | ||||||
|  | 		if (rxmode <= 32) | ||||||
|  | 			csr6 |= DMA_CONTROL_RTC_32; | ||||||
|  | 		else if (rxmode <= 64) | ||||||
|  | 			csr6 |= DMA_CONTROL_RTC_64; | ||||||
|  | 		else if (rxmode <= 96) | ||||||
|  | 			csr6 |= DMA_CONTROL_RTC_96; | ||||||
|  | 		else | ||||||
|  | 			csr6 |= DMA_CONTROL_RTC_128; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	writel(csr6, ioaddr + DMA_CONTROL); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Not yet implemented --- no RMON module */ | ||||||
|  | static void gmac_dma_diagnostic_fr(void *data, struct stmmac_extra_stats *x, | ||||||
|  | 				   unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_dump_dma_regs(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	pr_info(" DMA registers\n"); | ||||||
|  | 	for (i = 0; i < 22; i++) { | ||||||
|  | 		if ((i < 9) || (i > 17)) { | ||||||
|  | 			int offset = i * 4; | ||||||
|  | 			pr_err("\t Reg No. %d (offset 0x%x): 0x%08x\n", i, | ||||||
|  | 			       (DMA_BUS_MODE + offset), | ||||||
|  | 			       readl(ioaddr + DMA_BUS_MODE + offset)); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_get_tx_frame_status(void *data, struct stmmac_extra_stats *x, | ||||||
|  | 				    struct dma_desc *p, unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	int ret = 0; | ||||||
|  | 	struct net_device_stats *stats = (struct net_device_stats *)data; | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(p->des01.etx.error_summary)) { | ||||||
|  | 		DBG(KERN_ERR "GMAC TX error... 0x%08x\n", p->des01.etx); | ||||||
|  | 		if (unlikely(p->des01.etx.jabber_timeout)) { | ||||||
|  | 			DBG(KERN_ERR "\tjabber_timeout error\n"); | ||||||
|  | 			x->tx_jabber++; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (unlikely(p->des01.etx.frame_flushed)) { | ||||||
|  | 			DBG(KERN_ERR "\tframe_flushed error\n"); | ||||||
|  | 			x->tx_frame_flushed++; | ||||||
|  | 			gmac_flush_tx_fifo(ioaddr); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (unlikely(p->des01.etx.loss_carrier)) { | ||||||
|  | 			DBG(KERN_ERR "\tloss_carrier error\n"); | ||||||
|  | 			x->tx_losscarrier++; | ||||||
|  | 			stats->tx_carrier_errors++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.etx.no_carrier)) { | ||||||
|  | 			DBG(KERN_ERR "\tno_carrier error\n"); | ||||||
|  | 			x->tx_carrier++; | ||||||
|  | 			stats->tx_carrier_errors++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.etx.late_collision)) { | ||||||
|  | 			DBG(KERN_ERR "\tlate_collision error\n"); | ||||||
|  | 			stats->collisions += p->des01.etx.collision_count; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.etx.excessive_collisions)) { | ||||||
|  | 			DBG(KERN_ERR "\texcessive_collisions\n"); | ||||||
|  | 			stats->collisions += p->des01.etx.collision_count; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.etx.excessive_deferral)) { | ||||||
|  | 			DBG(KERN_INFO "\texcessive tx_deferral\n"); | ||||||
|  | 			x->tx_deferred++; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (unlikely(p->des01.etx.underflow_error)) { | ||||||
|  | 			DBG(KERN_ERR "\tunderflow error\n"); | ||||||
|  | 			gmac_flush_tx_fifo(ioaddr); | ||||||
|  | 			x->tx_underflow++; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (unlikely(p->des01.etx.ip_header_error)) { | ||||||
|  | 			DBG(KERN_ERR "\tTX IP header csum error\n"); | ||||||
|  | 			x->tx_ip_header_error++; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (unlikely(p->des01.etx.payload_error)) { | ||||||
|  | 			DBG(KERN_ERR "\tAddr/Payload csum error\n"); | ||||||
|  | 			x->tx_payload_error++; | ||||||
|  | 			gmac_flush_tx_fifo(ioaddr); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		ret = -1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(p->des01.etx.deferred)) { | ||||||
|  | 		DBG(KERN_INFO "GMAC TX status: tx deferred\n"); | ||||||
|  | 		x->tx_deferred++; | ||||||
|  | 	} | ||||||
|  | #ifdef STMMAC_VLAN_TAG_USED | ||||||
|  | 	if (p->des01.etx.vlan_frame) { | ||||||
|  | 		DBG(KERN_INFO "GMAC TX status: VLAN frame\n"); | ||||||
|  | 		x->tx_vlan++; | ||||||
|  | 	} | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_get_tx_len(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.etx.buffer1_size; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_coe_rdes0(int ipc_err, int type, int payload_err) | ||||||
|  | { | ||||||
|  | 	int ret = good_frame; | ||||||
|  | 	u32 status = (type << 2 | ipc_err << 1 | payload_err) & 0x7; | ||||||
|  | 
 | ||||||
|  | 	/* bits 5 7 0 | Frame status
 | ||||||
|  | 	 * ---------------------------------------------------------- | ||||||
|  | 	 *      0 0 0 | IEEE 802.3 Type frame (lenght < 1536 octects) | ||||||
|  | 	 *      1 0 0 | IPv4/6 No CSUM errorS. | ||||||
|  | 	 *      1 0 1 | IPv4/6 CSUM PAYLOAD error | ||||||
|  | 	 *      1 1 0 | IPv4/6 CSUM IP HR error | ||||||
|  | 	 *      1 1 1 | IPv4/6 IP PAYLOAD AND HEADER errorS | ||||||
|  | 	 *      0 0 1 | IPv4/6 unsupported IP PAYLOAD | ||||||
|  | 	 *      0 1 1 | COE bypassed.. no IPv4/6 frame | ||||||
|  | 	 *      0 1 0 | Reserved. | ||||||
|  | 	 */ | ||||||
|  | 	if (status == 0x0) { | ||||||
|  | 		DBG(KERN_INFO "RX Des0 status: IEEE 802.3 Type frame.\n"); | ||||||
|  | 		ret = good_frame; | ||||||
|  | 	} else if (status == 0x4) { | ||||||
|  | 		DBG(KERN_INFO "RX Des0 status: IPv4/6 No CSUM errorS.\n"); | ||||||
|  | 		ret = good_frame; | ||||||
|  | 	} else if (status == 0x5) { | ||||||
|  | 		DBG(KERN_ERR "RX Des0 status: IPv4/6 Payload Error.\n"); | ||||||
|  | 		ret = csum_none; | ||||||
|  | 	} else if (status == 0x6) { | ||||||
|  | 		DBG(KERN_ERR "RX Des0 status: IPv4/6 Header Error.\n"); | ||||||
|  | 		ret = csum_none; | ||||||
|  | 	} else if (status == 0x7) { | ||||||
|  | 		DBG(KERN_ERR | ||||||
|  | 		    "RX Des0 status: IPv4/6 Header and Payload Error.\n"); | ||||||
|  | 		ret = csum_none; | ||||||
|  | 	} else if (status == 0x1) { | ||||||
|  | 		DBG(KERN_ERR | ||||||
|  | 		    "RX Des0 status: IPv4/6 unsupported IP PAYLOAD.\n"); | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} else if (status == 0x3) { | ||||||
|  | 		DBG(KERN_ERR "RX Des0 status: No IPv4, IPv6 frame.\n"); | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_get_rx_frame_status(void *data, struct stmmac_extra_stats *x, | ||||||
|  | 				    struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	int ret = good_frame; | ||||||
|  | 	struct net_device_stats *stats = (struct net_device_stats *)data; | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(p->des01.erx.error_summary)) { | ||||||
|  | 		DBG(KERN_ERR "GMAC RX Error Summary... 0x%08x\n", p->des01.erx); | ||||||
|  | 		if (unlikely(p->des01.erx.descriptor_error)) { | ||||||
|  | 			DBG(KERN_ERR "\tdescriptor error\n"); | ||||||
|  | 			x->rx_desc++; | ||||||
|  | 			stats->rx_length_errors++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.erx.overflow_error)) { | ||||||
|  | 			DBG(KERN_ERR "\toverflow error\n"); | ||||||
|  | 			x->rx_gmac_overflow++; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (unlikely(p->des01.erx.ipc_csum_error)) | ||||||
|  | 			DBG(KERN_ERR "\tIPC Csum Error/Giant frame\n"); | ||||||
|  | 
 | ||||||
|  | 		if (unlikely(p->des01.erx.late_collision)) { | ||||||
|  | 			DBG(KERN_ERR "\tlate_collision error\n"); | ||||||
|  | 			stats->collisions++; | ||||||
|  | 			stats->collisions++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.erx.receive_watchdog)) { | ||||||
|  | 			DBG(KERN_ERR "\treceive_watchdog error\n"); | ||||||
|  | 			x->rx_watchdog++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.erx.error_gmii)) { | ||||||
|  | 			DBG(KERN_ERR "\tReceive Error\n"); | ||||||
|  | 			x->rx_mii++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.erx.crc_error)) { | ||||||
|  | 			DBG(KERN_ERR "\tCRC error\n"); | ||||||
|  | 			x->rx_crc++; | ||||||
|  | 			stats->rx_crc_errors++; | ||||||
|  | 		} | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* After a payload csum error, the ES bit is set.
 | ||||||
|  | 	 * It doesn't match with the information reported into the databook. | ||||||
|  | 	 * At any rate, we need to understand if the CSUM hw computation is ok | ||||||
|  | 	 * and report this info to the upper layers. */ | ||||||
|  | 	ret = gmac_coe_rdes0(p->des01.erx.ipc_csum_error, | ||||||
|  | 		p->des01.erx.frame_type, p->des01.erx.payload_csum_error); | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(p->des01.erx.dribbling)) { | ||||||
|  | 		DBG(KERN_ERR "GMAC RX: dribbling error\n"); | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | 	if (unlikely(p->des01.erx.sa_filter_fail)) { | ||||||
|  | 		DBG(KERN_ERR "GMAC RX : Source Address filter fail\n"); | ||||||
|  | 		x->sa_rx_filter_fail++; | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | 	if (unlikely(p->des01.erx.da_filter_fail)) { | ||||||
|  | 		DBG(KERN_ERR "GMAC RX : Destination Address filter fail\n"); | ||||||
|  | 		x->da_rx_filter_fail++; | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | 	if (unlikely(p->des01.erx.length_error)) { | ||||||
|  | 		DBG(KERN_ERR "GMAC RX: length_error error\n"); | ||||||
|  | 		x->rx_lenght++; | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | #ifdef STMMAC_VLAN_TAG_USED | ||||||
|  | 	if (p->des01.erx.vlan_tag) { | ||||||
|  | 		DBG(KERN_INFO "GMAC RX: VLAN frame tagged\n"); | ||||||
|  | 		x->rx_vlan++; | ||||||
|  | 	} | ||||||
|  | #endif | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_irq_status(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	u32 intr_status = readl(ioaddr + GMAC_INT_STATUS); | ||||||
|  | 
 | ||||||
|  | 	/* Not used events (e.g. MMC interrupts) are not handled. */ | ||||||
|  | 	if ((intr_status & mmc_tx_irq)) | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: MMC tx interrupt: 0x%08x\n", | ||||||
|  | 		    readl(ioaddr + GMAC_MMC_TX_INTR)); | ||||||
|  | 	if (unlikely(intr_status & mmc_rx_irq)) | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: MMC rx interrupt: 0x%08x\n", | ||||||
|  | 		    readl(ioaddr + GMAC_MMC_RX_INTR)); | ||||||
|  | 	if (unlikely(intr_status & mmc_rx_csum_offload_irq)) | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: MMC rx csum offload: 0x%08x\n", | ||||||
|  | 		    readl(ioaddr + GMAC_MMC_RX_CSUM_OFFLOAD)); | ||||||
|  | 	if (unlikely(intr_status & pmt_irq)) { | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: received Magic frame\n"); | ||||||
|  | 		/* clear the PMT bits 5 and 6 by reading the PMT
 | ||||||
|  | 		 * status register. */ | ||||||
|  | 		readl(ioaddr + GMAC_PMT); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_core_init(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	u32 value = readl(ioaddr + GMAC_CONTROL); | ||||||
|  | 	value |= GMAC_CORE_INIT; | ||||||
|  | 	writel(value, ioaddr + GMAC_CONTROL); | ||||||
|  | 
 | ||||||
|  | 	/* STBus Bridge Configuration */ | ||||||
|  | 	/*writel(0xc5608, ioaddr + 0x00007000);*/ | ||||||
|  | 
 | ||||||
|  | 	/* Freeze MMC counters */ | ||||||
|  | 	writel(0x8, ioaddr + GMAC_MMC_CTRL); | ||||||
|  | 	/* Mask GMAC interrupts */ | ||||||
|  | 	writel(0x207, ioaddr + GMAC_INT_MASK); | ||||||
|  | 
 | ||||||
|  | #ifdef STMMAC_VLAN_TAG_USED | ||||||
|  | 	/* Tag detection without filtering */ | ||||||
|  | 	writel(0x0, ioaddr + GMAC_VLAN_TAG); | ||||||
|  | #endif | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_set_umac_addr(unsigned long ioaddr, unsigned char *addr, | ||||||
|  | 				unsigned int reg_n) | ||||||
|  | { | ||||||
|  | 	stmmac_set_mac_addr(ioaddr, addr, GMAC_ADDR_HIGH(reg_n), | ||||||
|  | 				GMAC_ADDR_LOW(reg_n)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_get_umac_addr(unsigned long ioaddr, unsigned char *addr, | ||||||
|  | 				unsigned int reg_n) | ||||||
|  | { | ||||||
|  | 	stmmac_get_mac_addr(ioaddr, addr, GMAC_ADDR_HIGH(reg_n), | ||||||
|  | 				GMAC_ADDR_LOW(reg_n)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_set_filter(struct net_device *dev) | ||||||
|  | { | ||||||
|  | 	unsigned long ioaddr = dev->base_addr; | ||||||
|  | 	unsigned int value = 0; | ||||||
|  | 
 | ||||||
|  | 	DBG(KERN_INFO "%s: # mcasts %d, # unicast %d\n", | ||||||
|  | 	    __func__, dev->mc_count, dev->uc_count); | ||||||
|  | 
 | ||||||
|  | 	if (dev->flags & IFF_PROMISC) | ||||||
|  | 		value = GMAC_FRAME_FILTER_PR; | ||||||
|  | 	else if ((dev->mc_count > HASH_TABLE_SIZE) | ||||||
|  | 		   || (dev->flags & IFF_ALLMULTI)) { | ||||||
|  | 		value = GMAC_FRAME_FILTER_PM;	/* pass all multi */ | ||||||
|  | 		writel(0xffffffff, ioaddr + GMAC_HASH_HIGH); | ||||||
|  | 		writel(0xffffffff, ioaddr + GMAC_HASH_LOW); | ||||||
|  | 	} else if (dev->mc_count > 0) { | ||||||
|  | 		int i; | ||||||
|  | 		u32 mc_filter[2]; | ||||||
|  | 		struct dev_mc_list *mclist; | ||||||
|  | 
 | ||||||
|  | 		/* Hash filter for multicast */ | ||||||
|  | 		value = GMAC_FRAME_FILTER_HMC; | ||||||
|  | 
 | ||||||
|  | 		memset(mc_filter, 0, sizeof(mc_filter)); | ||||||
|  | 		for (i = 0, mclist = dev->mc_list; | ||||||
|  | 		     mclist && i < dev->mc_count; i++, mclist = mclist->next) { | ||||||
|  | 			/* The upper 6 bits of the calculated CRC are used to
 | ||||||
|  | 			   index the contens of the hash table */ | ||||||
|  | 			int bit_nr = | ||||||
|  | 			    bitrev32(~crc32_le(~0, mclist->dmi_addr, 6)) >> 26; | ||||||
|  | 			/* The most significant bit determines the register to
 | ||||||
|  | 			 * use (H/L) while the other 5 bits determine the bit | ||||||
|  | 			 * within the register. */ | ||||||
|  | 			mc_filter[bit_nr >> 5] |= 1 << (bit_nr & 31); | ||||||
|  | 		} | ||||||
|  | 		writel(mc_filter[0], ioaddr + GMAC_HASH_LOW); | ||||||
|  | 		writel(mc_filter[1], ioaddr + GMAC_HASH_HIGH); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Handle multiple unicast addresses (perfect filtering)*/ | ||||||
|  | 	if (dev->uc_count > GMAC_MAX_UNICAST_ADDRESSES) | ||||||
|  | 		/* Switch to promiscuous mode is more than 16 addrs
 | ||||||
|  | 		   are required */ | ||||||
|  | 		value |= GMAC_FRAME_FILTER_PR; | ||||||
|  | 	else { | ||||||
|  | 		int i; | ||||||
|  | 		struct dev_addr_list *uc_ptr = dev->uc_list; | ||||||
|  | 
 | ||||||
|  | 			for (i = 0; i < dev->uc_count; i++) { | ||||||
|  | 				gmac_set_umac_addr(ioaddr, uc_ptr->da_addr, | ||||||
|  | 						i + 1); | ||||||
|  | 
 | ||||||
|  | 				DBG(KERN_INFO "\t%d " | ||||||
|  | 				"- Unicast addr %02x:%02x:%02x:%02x:%02x:" | ||||||
|  | 				"%02x\n", i + 1, | ||||||
|  | 				uc_ptr->da_addr[0], uc_ptr->da_addr[1], | ||||||
|  | 				uc_ptr->da_addr[2], uc_ptr->da_addr[3], | ||||||
|  | 				uc_ptr->da_addr[4], uc_ptr->da_addr[5]); | ||||||
|  | 				uc_ptr = uc_ptr->next; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | #ifdef FRAME_FILTER_DEBUG | ||||||
|  | 	/* Enable Receive all mode (to debug filtering_fail errors) */ | ||||||
|  | 	value |= GMAC_FRAME_FILTER_RA; | ||||||
|  | #endif | ||||||
|  | 	writel(value, ioaddr + GMAC_FRAME_FILTER); | ||||||
|  | 
 | ||||||
|  | 	DBG(KERN_INFO "\tFrame Filter reg: 0x%08x\n\tHash regs: " | ||||||
|  | 	    "HI 0x%08x, LO 0x%08x\n", readl(ioaddr + GMAC_FRAME_FILTER), | ||||||
|  | 	    readl(ioaddr + GMAC_HASH_HIGH), readl(ioaddr + GMAC_HASH_LOW)); | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_flow_ctrl(unsigned long ioaddr, unsigned int duplex, | ||||||
|  | 			   unsigned int fc, unsigned int pause_time) | ||||||
|  | { | ||||||
|  | 	unsigned int flow = 0; | ||||||
|  | 
 | ||||||
|  | 	DBG(KERN_DEBUG "GMAC Flow-Control:\n"); | ||||||
|  | 	if (fc & FLOW_RX) { | ||||||
|  | 		DBG(KERN_DEBUG "\tReceive Flow-Control ON\n"); | ||||||
|  | 		flow |= GMAC_FLOW_CTRL_RFE; | ||||||
|  | 	} | ||||||
|  | 	if (fc & FLOW_TX) { | ||||||
|  | 		DBG(KERN_DEBUG "\tTransmit Flow-Control ON\n"); | ||||||
|  | 		flow |= GMAC_FLOW_CTRL_TFE; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (duplex) { | ||||||
|  | 		DBG(KERN_DEBUG "\tduplex mode: pause time: %d\n", pause_time); | ||||||
|  | 		flow |= (pause_time << GMAC_FLOW_CTRL_PT_SHIFT); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	writel(flow, ioaddr + GMAC_FLOW_CTRL); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_pmt(unsigned long ioaddr, unsigned long mode) | ||||||
|  | { | ||||||
|  | 	unsigned int pmt = 0; | ||||||
|  | 
 | ||||||
|  | 	if (mode == WAKE_MAGIC) { | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: WOL Magic frame\n"); | ||||||
|  | 		pmt |= power_down | magic_pkt_en; | ||||||
|  | 	} else if (mode == WAKE_UCAST) { | ||||||
|  | 		DBG(KERN_DEBUG "GMAC: WOL on global unicast\n"); | ||||||
|  | 		pmt |= global_unicast; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	writel(pmt, ioaddr + GMAC_PMT); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_init_rx_desc(struct dma_desc *p, unsigned int ring_size, | ||||||
|  | 				int disable_rx_ic) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	for (i = 0; i < ring_size; i++) { | ||||||
|  | 		p->des01.erx.own = 1; | ||||||
|  | 		p->des01.erx.buffer1_size = BUF_SIZE_8KiB - 1; | ||||||
|  | 		/* To support jumbo frames */ | ||||||
|  | 		p->des01.erx.buffer2_size = BUF_SIZE_8KiB - 1; | ||||||
|  | 		if (i == ring_size - 1) | ||||||
|  | 			p->des01.erx.end_ring = 1; | ||||||
|  | 		if (disable_rx_ic) | ||||||
|  | 			p->des01.erx.disable_ic = 1; | ||||||
|  | 		p++; | ||||||
|  | 	} | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_init_tx_desc(struct dma_desc *p, unsigned int ring_size) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; i < ring_size; i++) { | ||||||
|  | 		p->des01.etx.own = 0; | ||||||
|  | 		if (i == ring_size - 1) | ||||||
|  | 			p->des01.etx.end_ring = 1; | ||||||
|  | 		p++; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_get_tx_owner(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.etx.own; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_get_rx_owner(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.erx.own; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_set_tx_owner(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	p->des01.etx.own = 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_set_rx_owner(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	p->des01.erx.own = 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_get_tx_ls(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.etx.last_segment; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_release_tx_desc(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	int ter = p->des01.etx.end_ring; | ||||||
|  | 
 | ||||||
|  | 	memset(p, 0, sizeof(struct dma_desc)); | ||||||
|  | 	p->des01.etx.end_ring = ter; | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_prepare_tx_desc(struct dma_desc *p, int is_fs, int len, | ||||||
|  | 				 int csum_flag) | ||||||
|  | { | ||||||
|  | 	p->des01.etx.first_segment = is_fs; | ||||||
|  | 	if (unlikely(len > BUF_SIZE_4KiB)) { | ||||||
|  | 		p->des01.etx.buffer1_size = BUF_SIZE_4KiB; | ||||||
|  | 		p->des01.etx.buffer2_size = len - BUF_SIZE_4KiB; | ||||||
|  | 	} else { | ||||||
|  | 		p->des01.etx.buffer1_size = len; | ||||||
|  | 	} | ||||||
|  | 	if (likely(csum_flag)) | ||||||
|  | 		p->des01.etx.checksum_insertion = cic_full; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_clear_tx_ic(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	p->des01.etx.interrupt = 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void gmac_close_tx_desc(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	p->des01.etx.last_segment = 1; | ||||||
|  | 	p->des01.etx.interrupt = 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int gmac_get_rx_frame_len(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.erx.frame_length; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct stmmac_ops gmac_driver = { | ||||||
|  | 	.core_init = gmac_core_init, | ||||||
|  | 	.dump_mac_regs = gmac_dump_regs, | ||||||
|  | 	.dma_init = gmac_dma_init, | ||||||
|  | 	.dump_dma_regs = gmac_dump_dma_regs, | ||||||
|  | 	.dma_mode = gmac_dma_operation_mode, | ||||||
|  | 	.dma_diagnostic_fr = gmac_dma_diagnostic_fr, | ||||||
|  | 	.tx_status = gmac_get_tx_frame_status, | ||||||
|  | 	.rx_status = gmac_get_rx_frame_status, | ||||||
|  | 	.get_tx_len = gmac_get_tx_len, | ||||||
|  | 	.set_filter = gmac_set_filter, | ||||||
|  | 	.flow_ctrl = gmac_flow_ctrl, | ||||||
|  | 	.pmt = gmac_pmt, | ||||||
|  | 	.init_rx_desc = gmac_init_rx_desc, | ||||||
|  | 	.init_tx_desc = gmac_init_tx_desc, | ||||||
|  | 	.get_tx_owner = gmac_get_tx_owner, | ||||||
|  | 	.get_rx_owner = gmac_get_rx_owner, | ||||||
|  | 	.release_tx_desc = gmac_release_tx_desc, | ||||||
|  | 	.prepare_tx_desc = gmac_prepare_tx_desc, | ||||||
|  | 	.clear_tx_ic = gmac_clear_tx_ic, | ||||||
|  | 	.close_tx_desc = gmac_close_tx_desc, | ||||||
|  | 	.get_tx_ls = gmac_get_tx_ls, | ||||||
|  | 	.set_tx_owner = gmac_set_tx_owner, | ||||||
|  | 	.set_rx_owner = gmac_set_rx_owner, | ||||||
|  | 	.get_rx_frame_len = gmac_get_rx_frame_len, | ||||||
|  | 	.host_irq_status = gmac_irq_status, | ||||||
|  | 	.set_umac_addr = gmac_set_umac_addr, | ||||||
|  | 	.get_umac_addr = gmac_get_umac_addr, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct mac_device_info *gmac_setup(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	struct mac_device_info *mac; | ||||||
|  | 	u32 uid = readl(ioaddr + GMAC_VERSION); | ||||||
|  | 
 | ||||||
|  | 	pr_info("\tGMAC - user ID: 0x%x, Synopsys ID: 0x%x\n", | ||||||
|  | 	       ((uid & 0x0000ff00) >> 8), (uid & 0x000000ff)); | ||||||
|  | 
 | ||||||
|  | 	mac = kzalloc(sizeof(const struct mac_device_info), GFP_KERNEL); | ||||||
|  | 
 | ||||||
|  | 	mac->ops = &gmac_driver; | ||||||
|  | 	mac->hw.pmt = PMT_SUPPORTED; | ||||||
|  | 	mac->hw.link.port = GMAC_CONTROL_PS; | ||||||
|  | 	mac->hw.link.duplex = GMAC_CONTROL_DM; | ||||||
|  | 	mac->hw.link.speed = GMAC_CONTROL_FES; | ||||||
|  | 	mac->hw.mii.addr = GMAC_MII_ADDR; | ||||||
|  | 	mac->hw.mii.data = GMAC_MII_DATA; | ||||||
|  | 
 | ||||||
|  | 	return mac; | ||||||
|  | } | ||||||
							
								
								
									
										204
									
								
								drivers/net/stmmac/gmac.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										204
									
								
								drivers/net/stmmac/gmac.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,204 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #define GMAC_CONTROL		0x00000000	/* Configuration */ | ||||||
|  | #define GMAC_FRAME_FILTER	0x00000004	/* Frame Filter */ | ||||||
|  | #define GMAC_HASH_HIGH		0x00000008	/* Multicast Hash Table High */ | ||||||
|  | #define GMAC_HASH_LOW		0x0000000c	/* Multicast Hash Table Low */ | ||||||
|  | #define GMAC_MII_ADDR		0x00000010	/* MII Address */ | ||||||
|  | #define GMAC_MII_DATA		0x00000014	/* MII Data */ | ||||||
|  | #define GMAC_FLOW_CTRL		0x00000018	/* Flow Control */ | ||||||
|  | #define GMAC_VLAN_TAG		0x0000001c	/* VLAN Tag */ | ||||||
|  | #define GMAC_VERSION		0x00000020	/* GMAC CORE Version */ | ||||||
|  | #define GMAC_WAKEUP_FILTER	0x00000028	/* Wake-up Frame Filter */ | ||||||
|  | 
 | ||||||
|  | #define GMAC_INT_STATUS		0x00000038	/* interrupt status register */ | ||||||
|  | enum gmac_irq_status { | ||||||
|  | 	time_stamp_irq = 0x0200, | ||||||
|  | 	mmc_rx_csum_offload_irq = 0x0080, | ||||||
|  | 	mmc_tx_irq = 0x0040, | ||||||
|  | 	mmc_rx_irq = 0x0020, | ||||||
|  | 	mmc_irq = 0x0010, | ||||||
|  | 	pmt_irq = 0x0008, | ||||||
|  | 	pcs_ane_irq = 0x0004, | ||||||
|  | 	pcs_link_irq = 0x0002, | ||||||
|  | 	rgmii_irq = 0x0001, | ||||||
|  | }; | ||||||
|  | #define GMAC_INT_MASK		0x0000003c	/* interrupt mask register */ | ||||||
|  | 
 | ||||||
|  | /* PMT Control and Status */ | ||||||
|  | #define GMAC_PMT		0x0000002c | ||||||
|  | enum power_event { | ||||||
|  | 	pointer_reset = 0x80000000, | ||||||
|  | 	global_unicast = 0x00000200, | ||||||
|  | 	wake_up_rx_frame = 0x00000040, | ||||||
|  | 	magic_frame = 0x00000020, | ||||||
|  | 	wake_up_frame_en = 0x00000004, | ||||||
|  | 	magic_pkt_en = 0x00000002, | ||||||
|  | 	power_down = 0x00000001, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* GMAC HW ADDR regs */ | ||||||
|  | #define GMAC_ADDR_HIGH(reg)		(0x00000040+(reg * 8)) | ||||||
|  | #define GMAC_ADDR_LOW(reg)		(0x00000044+(reg * 8)) | ||||||
|  | #define GMAC_MAX_UNICAST_ADDRESSES	16 | ||||||
|  | 
 | ||||||
|  | #define GMAC_AN_CTRL	0x000000c0	/* AN control */ | ||||||
|  | #define GMAC_AN_STATUS	0x000000c4	/* AN status */ | ||||||
|  | #define GMAC_ANE_ADV	0x000000c8	/* Auto-Neg. Advertisement */ | ||||||
|  | #define GMAC_ANE_LINK	0x000000cc	/* Auto-Neg. link partener ability */ | ||||||
|  | #define GMAC_ANE_EXP	0x000000d0	/* ANE expansion */ | ||||||
|  | #define GMAC_TBI	0x000000d4	/* TBI extend status */ | ||||||
|  | #define GMAC_GMII_STATUS 0x000000d8	/* S/R-GMII status */ | ||||||
|  | 
 | ||||||
|  | /* GMAC Configuration defines */ | ||||||
|  | #define GMAC_CONTROL_TC	0x01000000	/* Transmit Conf. in RGMII/SGMII */ | ||||||
|  | #define GMAC_CONTROL_WD	0x00800000	/* Disable Watchdog on receive */ | ||||||
|  | #define GMAC_CONTROL_JD	0x00400000	/* Jabber disable */ | ||||||
|  | #define GMAC_CONTROL_BE	0x00200000	/* Frame Burst Enable */ | ||||||
|  | #define GMAC_CONTROL_JE	0x00100000	/* Jumbo frame */ | ||||||
|  | enum inter_frame_gap { | ||||||
|  | 	GMAC_CONTROL_IFG_88 = 0x00040000, | ||||||
|  | 	GMAC_CONTROL_IFG_80 = 0x00020000, | ||||||
|  | 	GMAC_CONTROL_IFG_40 = 0x000e0000, | ||||||
|  | }; | ||||||
|  | #define GMAC_CONTROL_DCRS	0x00010000 /* Disable carrier sense during tx */ | ||||||
|  | #define GMAC_CONTROL_PS		0x00008000 /* Port Select 0:GMI 1:MII */ | ||||||
|  | #define GMAC_CONTROL_FES	0x00004000 /* Speed 0:10 1:100 */ | ||||||
|  | #define GMAC_CONTROL_DO		0x00002000 /* Disable Rx Own */ | ||||||
|  | #define GMAC_CONTROL_LM		0x00001000 /* Loop-back mode */ | ||||||
|  | #define GMAC_CONTROL_DM		0x00000800 /* Duplex Mode */ | ||||||
|  | #define GMAC_CONTROL_IPC	0x00000400 /* Checksum Offload */ | ||||||
|  | #define GMAC_CONTROL_DR		0x00000200 /* Disable Retry */ | ||||||
|  | #define GMAC_CONTROL_LUD	0x00000100 /* Link up/down */ | ||||||
|  | #define GMAC_CONTROL_ACS	0x00000080 /* Automatic Pad Stripping */ | ||||||
|  | #define GMAC_CONTROL_DC		0x00000010 /* Deferral Check */ | ||||||
|  | #define GMAC_CONTROL_TE		0x00000008 /* Transmitter Enable */ | ||||||
|  | #define GMAC_CONTROL_RE		0x00000004 /* Receiver Enable */ | ||||||
|  | 
 | ||||||
|  | #define GMAC_CORE_INIT (GMAC_CONTROL_JD | GMAC_CONTROL_PS | GMAC_CONTROL_ACS | \ | ||||||
|  | 			GMAC_CONTROL_IPC | GMAC_CONTROL_JE | GMAC_CONTROL_BE) | ||||||
|  | 
 | ||||||
|  | /* GMAC Frame Filter defines */ | ||||||
|  | #define GMAC_FRAME_FILTER_PR	0x00000001	/* Promiscuous Mode */ | ||||||
|  | #define GMAC_FRAME_FILTER_HUC	0x00000002	/* Hash Unicast */ | ||||||
|  | #define GMAC_FRAME_FILTER_HMC	0x00000004	/* Hash Multicast */ | ||||||
|  | #define GMAC_FRAME_FILTER_DAIF	0x00000008	/* DA Inverse Filtering */ | ||||||
|  | #define GMAC_FRAME_FILTER_PM	0x00000010	/* Pass all multicast */ | ||||||
|  | #define GMAC_FRAME_FILTER_DBF	0x00000020	/* Disable Broadcast frames */ | ||||||
|  | #define GMAC_FRAME_FILTER_SAIF	0x00000100	/* Inverse Filtering */ | ||||||
|  | #define GMAC_FRAME_FILTER_SAF	0x00000200	/* Source Address Filter */ | ||||||
|  | #define GMAC_FRAME_FILTER_HPF	0x00000400	/* Hash or perfect Filter */ | ||||||
|  | #define GMAC_FRAME_FILTER_RA	0x80000000	/* Receive all mode */ | ||||||
|  | /* GMII ADDR  defines */ | ||||||
|  | #define GMAC_MII_ADDR_WRITE	0x00000002	/* MII Write */ | ||||||
|  | #define GMAC_MII_ADDR_BUSY	0x00000001	/* MII Busy */ | ||||||
|  | /* GMAC FLOW CTRL defines */ | ||||||
|  | #define GMAC_FLOW_CTRL_PT_MASK	0xffff0000	/* Pause Time Mask */ | ||||||
|  | #define GMAC_FLOW_CTRL_PT_SHIFT	16 | ||||||
|  | #define GMAC_FLOW_CTRL_RFE	0x00000004	/* Rx Flow Control Enable */ | ||||||
|  | #define GMAC_FLOW_CTRL_TFE	0x00000002	/* Tx Flow Control Enable */ | ||||||
|  | #define GMAC_FLOW_CTRL_FCB_BPA	0x00000001	/* Flow Control Busy ... */ | ||||||
|  | 
 | ||||||
|  | /*--- DMA BLOCK defines ---*/ | ||||||
|  | /* DMA Bus Mode register defines */ | ||||||
|  | #define DMA_BUS_MODE_SFT_RESET	0x00000001	/* Software Reset */ | ||||||
|  | #define DMA_BUS_MODE_DA		0x00000002	/* Arbitration scheme */ | ||||||
|  | #define DMA_BUS_MODE_DSL_MASK	0x0000007c	/* Descriptor Skip Length */ | ||||||
|  | #define DMA_BUS_MODE_DSL_SHIFT	2	/*   (in DWORDS)      */ | ||||||
|  | /* Programmable burst length (passed thorugh platform)*/ | ||||||
|  | #define DMA_BUS_MODE_PBL_MASK	0x00003f00	/* Programmable Burst Len */ | ||||||
|  | #define DMA_BUS_MODE_PBL_SHIFT	8 | ||||||
|  | 
 | ||||||
|  | enum rx_tx_priority_ratio { | ||||||
|  | 	double_ratio = 0x00004000,	/*2:1 */ | ||||||
|  | 	triple_ratio = 0x00008000,	/*3:1 */ | ||||||
|  | 	quadruple_ratio = 0x0000c000,	/*4:1 */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define DMA_BUS_MODE_FB		0x00010000	/* Fixed burst */ | ||||||
|  | #define DMA_BUS_MODE_RPBL_MASK	0x003e0000	/* Rx-Programmable Burst Len */ | ||||||
|  | #define DMA_BUS_MODE_RPBL_SHIFT	17 | ||||||
|  | #define DMA_BUS_MODE_USP	0x00800000 | ||||||
|  | #define DMA_BUS_MODE_4PBL	0x01000000 | ||||||
|  | #define DMA_BUS_MODE_AAL	0x02000000 | ||||||
|  | 
 | ||||||
|  | /* DMA CRS Control and Status Register Mapping */ | ||||||
|  | #define DMA_HOST_TX_DESC	  0x00001048	/* Current Host Tx descriptor */ | ||||||
|  | #define DMA_HOST_RX_DESC	  0x0000104c	/* Current Host Rx descriptor */ | ||||||
|  | /*  DMA Bus Mode register defines */ | ||||||
|  | #define DMA_BUS_PR_RATIO_MASK	  0x0000c000	/* Rx/Tx priority ratio */ | ||||||
|  | #define DMA_BUS_PR_RATIO_SHIFT	  14 | ||||||
|  | #define DMA_BUS_FB	  	  0x00010000	/* Fixed Burst */ | ||||||
|  | 
 | ||||||
|  | /* DMA operation mode defines (start/stop tx/rx are placed in common header)*/ | ||||||
|  | #define DMA_CONTROL_DT		0x04000000 /* Disable Drop TCP/IP csum error */ | ||||||
|  | #define DMA_CONTROL_RSF		0x02000000 /* Receive Store and Forward */ | ||||||
|  | #define DMA_CONTROL_DFF		0x01000000 /* Disaable flushing */ | ||||||
|  | /* Theshold for Activating the FC */ | ||||||
|  | enum rfa { | ||||||
|  | 	act_full_minus_1 = 0x00800000, | ||||||
|  | 	act_full_minus_2 = 0x00800200, | ||||||
|  | 	act_full_minus_3 = 0x00800400, | ||||||
|  | 	act_full_minus_4 = 0x00800600, | ||||||
|  | }; | ||||||
|  | /* Theshold for Deactivating the FC */ | ||||||
|  | enum rfd { | ||||||
|  | 	deac_full_minus_1 = 0x00400000, | ||||||
|  | 	deac_full_minus_2 = 0x00400800, | ||||||
|  | 	deac_full_minus_3 = 0x00401000, | ||||||
|  | 	deac_full_minus_4 = 0x00401800, | ||||||
|  | }; | ||||||
|  | #define DMA_CONTROL_TSF		0x00200000 /* Transmit  Store and Forward */ | ||||||
|  | #define DMA_CONTROL_FTF		0x00100000 /* Flush transmit FIFO */ | ||||||
|  | 
 | ||||||
|  | enum ttc_control { | ||||||
|  | 	DMA_CONTROL_TTC_64 = 0x00000000, | ||||||
|  | 	DMA_CONTROL_TTC_128 = 0x00004000, | ||||||
|  | 	DMA_CONTROL_TTC_192 = 0x00008000, | ||||||
|  | 	DMA_CONTROL_TTC_256 = 0x0000c000, | ||||||
|  | 	DMA_CONTROL_TTC_40 = 0x00010000, | ||||||
|  | 	DMA_CONTROL_TTC_32 = 0x00014000, | ||||||
|  | 	DMA_CONTROL_TTC_24 = 0x00018000, | ||||||
|  | 	DMA_CONTROL_TTC_16 = 0x0001c000, | ||||||
|  | }; | ||||||
|  | #define DMA_CONTROL_TC_TX_MASK	0xfffe3fff | ||||||
|  | 
 | ||||||
|  | #define DMA_CONTROL_EFC		0x00000100 | ||||||
|  | #define DMA_CONTROL_FEF		0x00000080 | ||||||
|  | #define DMA_CONTROL_FUF		0x00000040 | ||||||
|  | 
 | ||||||
|  | enum rtc_control { | ||||||
|  | 	DMA_CONTROL_RTC_64 = 0x00000000, | ||||||
|  | 	DMA_CONTROL_RTC_32 = 0x00000008, | ||||||
|  | 	DMA_CONTROL_RTC_96 = 0x00000010, | ||||||
|  | 	DMA_CONTROL_RTC_128 = 0x00000018, | ||||||
|  | }; | ||||||
|  | #define DMA_CONTROL_TC_RX_MASK	0xffffffe7 | ||||||
|  | 
 | ||||||
|  | #define DMA_CONTROL_OSF	0x00000004	/* Operate on second frame */ | ||||||
|  | 
 | ||||||
|  | /* MMC registers offset */ | ||||||
|  | #define GMAC_MMC_CTRL      0x100 | ||||||
|  | #define GMAC_MMC_RX_INTR   0x104 | ||||||
|  | #define GMAC_MMC_TX_INTR   0x108 | ||||||
|  | #define GMAC_MMC_RX_CSUM_OFFLOAD   0x208 | ||||||
							
								
								
									
										517
									
								
								drivers/net/stmmac/mac100.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										517
									
								
								drivers/net/stmmac/mac100.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,517 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   This is the driver for the MAC 10/100 on-chip Ethernet controller | ||||||
|  |   currently tested on all the ST boards based on STb7109 and stx7200 SoCs. | ||||||
|  | 
 | ||||||
|  |   DWC Ether MAC 10/100 Universal version 4.0 has been used for developing | ||||||
|  |   this code. | ||||||
|  | 
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #include <linux/netdevice.h> | ||||||
|  | #include <linux/crc32.h> | ||||||
|  | #include <linux/mii.h> | ||||||
|  | #include <linux/phy.h> | ||||||
|  | 
 | ||||||
|  | #include "common.h" | ||||||
|  | #include "mac100.h" | ||||||
|  | 
 | ||||||
|  | #undef MAC100_DEBUG | ||||||
|  | /*#define MAC100_DEBUG*/ | ||||||
|  | #ifdef MAC100_DEBUG | ||||||
|  | #define DBG(fmt, args...)  printk(fmt, ## args) | ||||||
|  | #else | ||||||
|  | #define DBG(fmt, args...)  do { } while (0) | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | static void mac100_core_init(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	u32 value = readl(ioaddr + MAC_CONTROL); | ||||||
|  | 
 | ||||||
|  | 	writel((value | MAC_CORE_INIT), ioaddr + MAC_CONTROL); | ||||||
|  | 
 | ||||||
|  | #ifdef STMMAC_VLAN_TAG_USED | ||||||
|  | 	writel(ETH_P_8021Q, ioaddr + MAC_VLAN1); | ||||||
|  | #endif | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_dump_mac_regs(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	pr_info("\t----------------------------------------------\n" | ||||||
|  | 	       "\t  MAC100 CSR (base addr = 0x%8x)\n" | ||||||
|  | 	       "\t----------------------------------------------\n", | ||||||
|  | 	       (unsigned int)ioaddr); | ||||||
|  | 	pr_info("\tcontrol reg (offset 0x%x): 0x%08x\n", MAC_CONTROL, | ||||||
|  | 	       readl(ioaddr + MAC_CONTROL)); | ||||||
|  | 	pr_info("\taddr HI (offset 0x%x): 0x%08x\n ", MAC_ADDR_HIGH, | ||||||
|  | 	       readl(ioaddr + MAC_ADDR_HIGH)); | ||||||
|  | 	pr_info("\taddr LO (offset 0x%x): 0x%08x\n", MAC_ADDR_LOW, | ||||||
|  | 	       readl(ioaddr + MAC_ADDR_LOW)); | ||||||
|  | 	pr_info("\tmulticast hash HI (offset 0x%x): 0x%08x\n", | ||||||
|  | 			MAC_HASH_HIGH, readl(ioaddr + MAC_HASH_HIGH)); | ||||||
|  | 	pr_info("\tmulticast hash LO (offset 0x%x): 0x%08x\n", | ||||||
|  | 			MAC_HASH_LOW, readl(ioaddr + MAC_HASH_LOW)); | ||||||
|  | 	pr_info("\tflow control (offset 0x%x): 0x%08x\n", | ||||||
|  | 		MAC_FLOW_CTRL, readl(ioaddr + MAC_FLOW_CTRL)); | ||||||
|  | 	pr_info("\tVLAN1 tag (offset 0x%x): 0x%08x\n", MAC_VLAN1, | ||||||
|  | 	       readl(ioaddr + MAC_VLAN1)); | ||||||
|  | 	pr_info("\tVLAN2 tag (offset 0x%x): 0x%08x\n", MAC_VLAN2, | ||||||
|  | 	       readl(ioaddr + MAC_VLAN2)); | ||||||
|  | 	pr_info("\n\tMAC management counter registers\n"); | ||||||
|  | 	pr_info("\t MMC crtl (offset 0x%x): 0x%08x\n", | ||||||
|  | 	       MMC_CONTROL, readl(ioaddr + MMC_CONTROL)); | ||||||
|  | 	pr_info("\t MMC High Interrupt (offset 0x%x): 0x%08x\n", | ||||||
|  | 	       MMC_HIGH_INTR, readl(ioaddr + MMC_HIGH_INTR)); | ||||||
|  | 	pr_info("\t MMC Low Interrupt (offset 0x%x): 0x%08x\n", | ||||||
|  | 	       MMC_LOW_INTR, readl(ioaddr + MMC_LOW_INTR)); | ||||||
|  | 	pr_info("\t MMC High Interrupt Mask (offset 0x%x): 0x%08x\n", | ||||||
|  | 	       MMC_HIGH_INTR_MASK, readl(ioaddr + MMC_HIGH_INTR_MASK)); | ||||||
|  | 	pr_info("\t MMC Low Interrupt Mask (offset 0x%x): 0x%08x\n", | ||||||
|  | 	       MMC_LOW_INTR_MASK, readl(ioaddr + MMC_LOW_INTR_MASK)); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int mac100_dma_init(unsigned long ioaddr, int pbl, u32 dma_tx, | ||||||
|  | 			   u32 dma_rx) | ||||||
|  | { | ||||||
|  | 	u32 value = readl(ioaddr + DMA_BUS_MODE); | ||||||
|  | 	/* DMA SW reset */ | ||||||
|  | 	value |= DMA_BUS_MODE_SFT_RESET; | ||||||
|  | 	writel(value, ioaddr + DMA_BUS_MODE); | ||||||
|  | 	do {} while ((readl(ioaddr + DMA_BUS_MODE) & DMA_BUS_MODE_SFT_RESET)); | ||||||
|  | 
 | ||||||
|  | 	/* Enable Application Access by writing to DMA CSR0 */ | ||||||
|  | 	writel(DMA_BUS_MODE_DEFAULT | (pbl << DMA_BUS_MODE_PBL_SHIFT), | ||||||
|  | 	       ioaddr + DMA_BUS_MODE); | ||||||
|  | 
 | ||||||
|  | 	/* Mask interrupts by writing to CSR7 */ | ||||||
|  | 	writel(DMA_INTR_DEFAULT_MASK, ioaddr + DMA_INTR_ENA); | ||||||
|  | 
 | ||||||
|  | 	/* The base address of the RX/TX descriptor lists must be written into
 | ||||||
|  | 	 * DMA CSR3 and CSR4, respectively. */ | ||||||
|  | 	writel(dma_tx, ioaddr + DMA_TX_BASE_ADDR); | ||||||
|  | 	writel(dma_rx, ioaddr + DMA_RCV_BASE_ADDR); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Store and Forward capability is not used at all..
 | ||||||
|  |  * The transmit threshold can be programmed by | ||||||
|  |  * setting the TTC bits in the DMA control register.*/ | ||||||
|  | static void mac100_dma_operation_mode(unsigned long ioaddr, int txmode, | ||||||
|  | 				      int rxmode) | ||||||
|  | { | ||||||
|  | 	u32 csr6 = readl(ioaddr + DMA_CONTROL); | ||||||
|  | 
 | ||||||
|  | 	if (txmode <= 32) | ||||||
|  | 		csr6 |= DMA_CONTROL_TTC_32; | ||||||
|  | 	else if (txmode <= 64) | ||||||
|  | 		csr6 |= DMA_CONTROL_TTC_64; | ||||||
|  | 	else | ||||||
|  | 		csr6 |= DMA_CONTROL_TTC_128; | ||||||
|  | 
 | ||||||
|  | 	writel(csr6, ioaddr + DMA_CONTROL); | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_dump_dma_regs(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 
 | ||||||
|  | 	DBG(KERN_DEBUG "MAC100 DMA CSR \n"); | ||||||
|  | 	for (i = 0; i < 9; i++) | ||||||
|  | 		pr_debug("\t CSR%d (offset 0x%x): 0x%08x\n", i, | ||||||
|  | 		       (DMA_BUS_MODE + i * 4), | ||||||
|  | 		       readl(ioaddr + DMA_BUS_MODE + i * 4)); | ||||||
|  | 	DBG(KERN_DEBUG "\t CSR20 (offset 0x%x): 0x%08x\n", | ||||||
|  | 	    DMA_CUR_TX_BUF_ADDR, readl(ioaddr + DMA_CUR_TX_BUF_ADDR)); | ||||||
|  | 	DBG(KERN_DEBUG "\t CSR21 (offset 0x%x): 0x%08x\n", | ||||||
|  | 	    DMA_CUR_RX_BUF_ADDR, readl(ioaddr + DMA_CUR_RX_BUF_ADDR)); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* DMA controller has two counters to track the number of
 | ||||||
|  |    the receive missed frames. */ | ||||||
|  | static void mac100_dma_diagnostic_fr(void *data, struct stmmac_extra_stats *x, | ||||||
|  | 				     unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	struct net_device_stats *stats = (struct net_device_stats *)data; | ||||||
|  | 	u32 csr8 = readl(ioaddr + DMA_MISSED_FRAME_CTR); | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(csr8)) { | ||||||
|  | 		if (csr8 & DMA_MISSED_FRAME_OVE) { | ||||||
|  | 			stats->rx_over_errors += 0x800; | ||||||
|  | 			x->rx_overflow_cntr += 0x800; | ||||||
|  | 		} else { | ||||||
|  | 			unsigned int ove_cntr; | ||||||
|  | 			ove_cntr = ((csr8 & DMA_MISSED_FRAME_OVE_CNTR) >> 17); | ||||||
|  | 			stats->rx_over_errors += ove_cntr; | ||||||
|  | 			x->rx_overflow_cntr += ove_cntr; | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		if (csr8 & DMA_MISSED_FRAME_OVE_M) { | ||||||
|  | 			stats->rx_missed_errors += 0xffff; | ||||||
|  | 			x->rx_missed_cntr += 0xffff; | ||||||
|  | 		} else { | ||||||
|  | 			unsigned int miss_f = (csr8 & DMA_MISSED_FRAME_M_CNTR); | ||||||
|  | 			stats->rx_missed_errors += miss_f; | ||||||
|  | 			x->rx_missed_cntr += miss_f; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int mac100_get_tx_frame_status(void *data, struct stmmac_extra_stats *x, | ||||||
|  | 				      struct dma_desc *p, unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	int ret = 0; | ||||||
|  | 	struct net_device_stats *stats = (struct net_device_stats *)data; | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(p->des01.tx.error_summary)) { | ||||||
|  | 		if (unlikely(p->des01.tx.underflow_error)) { | ||||||
|  | 			x->tx_underflow++; | ||||||
|  | 			stats->tx_fifo_errors++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.tx.no_carrier)) { | ||||||
|  | 			x->tx_carrier++; | ||||||
|  | 			stats->tx_carrier_errors++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.tx.loss_carrier)) { | ||||||
|  | 			x->tx_losscarrier++; | ||||||
|  | 			stats->tx_carrier_errors++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely((p->des01.tx.excessive_deferral) || | ||||||
|  | 			     (p->des01.tx.excessive_collisions) || | ||||||
|  | 			     (p->des01.tx.late_collision))) | ||||||
|  | 			stats->collisions += p->des01.tx.collision_count; | ||||||
|  | 		ret = -1; | ||||||
|  | 	} | ||||||
|  | 	if (unlikely(p->des01.tx.heartbeat_fail)) { | ||||||
|  | 		x->tx_heartbeat++; | ||||||
|  | 		stats->tx_heartbeat_errors++; | ||||||
|  | 		ret = -1; | ||||||
|  | 	} | ||||||
|  | 	if (unlikely(p->des01.tx.deferred)) | ||||||
|  | 		x->tx_deferred++; | ||||||
|  | 
 | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int mac100_get_tx_len(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.tx.buffer1_size; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* This function verifies if each incoming frame has some errors
 | ||||||
|  |  * and, if required, updates the multicast statistics. | ||||||
|  |  * In case of success, it returns csum_none becasue the device | ||||||
|  |  * is not able to compute the csum in HW. */ | ||||||
|  | static int mac100_get_rx_frame_status(void *data, struct stmmac_extra_stats *x, | ||||||
|  | 				      struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	int ret = csum_none; | ||||||
|  | 	struct net_device_stats *stats = (struct net_device_stats *)data; | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(p->des01.rx.last_descriptor == 0)) { | ||||||
|  | 		pr_warning("mac100 Error: Oversized Ethernet " | ||||||
|  | 			   "frame spanned multiple buffers\n"); | ||||||
|  | 		stats->rx_length_errors++; | ||||||
|  | 		return discard_frame; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(p->des01.rx.error_summary)) { | ||||||
|  | 		if (unlikely(p->des01.rx.descriptor_error)) | ||||||
|  | 			x->rx_desc++; | ||||||
|  | 		if (unlikely(p->des01.rx.partial_frame_error)) | ||||||
|  | 			x->rx_partial++; | ||||||
|  | 		if (unlikely(p->des01.rx.run_frame)) | ||||||
|  | 			x->rx_runt++; | ||||||
|  | 		if (unlikely(p->des01.rx.frame_too_long)) | ||||||
|  | 			x->rx_toolong++; | ||||||
|  | 		if (unlikely(p->des01.rx.collision)) { | ||||||
|  | 			x->rx_collision++; | ||||||
|  | 			stats->collisions++; | ||||||
|  | 		} | ||||||
|  | 		if (unlikely(p->des01.rx.crc_error)) { | ||||||
|  | 			x->rx_crc++; | ||||||
|  | 			stats->rx_crc_errors++; | ||||||
|  | 		} | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | 	if (unlikely(p->des01.rx.dribbling)) | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 
 | ||||||
|  | 	if (unlikely(p->des01.rx.length_error)) { | ||||||
|  | 		x->rx_lenght++; | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | 	if (unlikely(p->des01.rx.mii_error)) { | ||||||
|  | 		x->rx_mii++; | ||||||
|  | 		ret = discard_frame; | ||||||
|  | 	} | ||||||
|  | 	if (p->des01.rx.multicast_frame) { | ||||||
|  | 		x->rx_multicast++; | ||||||
|  | 		stats->multicast++; | ||||||
|  | 	} | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_irq_status(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_set_umac_addr(unsigned long ioaddr, unsigned char *addr, | ||||||
|  | 			  unsigned int reg_n) | ||||||
|  | { | ||||||
|  | 	stmmac_set_mac_addr(ioaddr, addr, MAC_ADDR_HIGH, MAC_ADDR_LOW); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_get_umac_addr(unsigned long ioaddr, unsigned char *addr, | ||||||
|  | 			  unsigned int reg_n) | ||||||
|  | { | ||||||
|  | 	stmmac_get_mac_addr(ioaddr, addr, MAC_ADDR_HIGH, MAC_ADDR_LOW); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_set_filter(struct net_device *dev) | ||||||
|  | { | ||||||
|  | 	unsigned long ioaddr = dev->base_addr; | ||||||
|  | 	u32 value = readl(ioaddr + MAC_CONTROL); | ||||||
|  | 
 | ||||||
|  | 	if (dev->flags & IFF_PROMISC) { | ||||||
|  | 		value |= MAC_CONTROL_PR; | ||||||
|  | 		value &= ~(MAC_CONTROL_PM | MAC_CONTROL_IF | MAC_CONTROL_HO | | ||||||
|  | 			   MAC_CONTROL_HP); | ||||||
|  | 	} else if ((dev->mc_count > HASH_TABLE_SIZE) | ||||||
|  | 		   || (dev->flags & IFF_ALLMULTI)) { | ||||||
|  | 		value |= MAC_CONTROL_PM; | ||||||
|  | 		value &= ~(MAC_CONTROL_PR | MAC_CONTROL_IF | MAC_CONTROL_HO); | ||||||
|  | 		writel(0xffffffff, ioaddr + MAC_HASH_HIGH); | ||||||
|  | 		writel(0xffffffff, ioaddr + MAC_HASH_LOW); | ||||||
|  | 	} else if (dev->mc_count == 0) {	/* no multicast */ | ||||||
|  | 		value &= ~(MAC_CONTROL_PM | MAC_CONTROL_PR | MAC_CONTROL_IF | | ||||||
|  | 			   MAC_CONTROL_HO | MAC_CONTROL_HP); | ||||||
|  | 	} else { | ||||||
|  | 		int i; | ||||||
|  | 		u32 mc_filter[2]; | ||||||
|  | 		struct dev_mc_list *mclist; | ||||||
|  | 
 | ||||||
|  | 		/* Perfect filter mode for physical address and Hash
 | ||||||
|  | 		   filter for multicast */ | ||||||
|  | 		value |= MAC_CONTROL_HP; | ||||||
|  | 		value &= ~(MAC_CONTROL_PM | MAC_CONTROL_PR | MAC_CONTROL_IF | ||||||
|  | 			   | MAC_CONTROL_HO); | ||||||
|  | 
 | ||||||
|  | 		memset(mc_filter, 0, sizeof(mc_filter)); | ||||||
|  | 		for (i = 0, mclist = dev->mc_list; | ||||||
|  | 		     mclist && i < dev->mc_count; i++, mclist = mclist->next) { | ||||||
|  | 			/* The upper 6 bits of the calculated CRC are used to
 | ||||||
|  | 			 * index the contens of the hash table */ | ||||||
|  | 			int bit_nr = | ||||||
|  | 			    ether_crc(ETH_ALEN, mclist->dmi_addr) >> 26; | ||||||
|  | 			/* The most significant bit determines the register to
 | ||||||
|  | 			 * use (H/L) while the other 5 bits determine the bit | ||||||
|  | 			 * within the register. */ | ||||||
|  | 			mc_filter[bit_nr >> 5] |= 1 << (bit_nr & 31); | ||||||
|  | 		} | ||||||
|  | 		writel(mc_filter[0], ioaddr + MAC_HASH_LOW); | ||||||
|  | 		writel(mc_filter[1], ioaddr + MAC_HASH_HIGH); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	writel(value, ioaddr + MAC_CONTROL); | ||||||
|  | 
 | ||||||
|  | 	DBG(KERN_INFO "%s: CTRL reg: 0x%08x Hash regs: " | ||||||
|  | 	    "HI 0x%08x, LO 0x%08x\n", | ||||||
|  | 	    __func__, readl(ioaddr + MAC_CONTROL), | ||||||
|  | 	    readl(ioaddr + MAC_HASH_HIGH), readl(ioaddr + MAC_HASH_LOW)); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_flow_ctrl(unsigned long ioaddr, unsigned int duplex, | ||||||
|  | 			     unsigned int fc, unsigned int pause_time) | ||||||
|  | { | ||||||
|  | 	unsigned int flow = MAC_FLOW_CTRL_ENABLE; | ||||||
|  | 
 | ||||||
|  | 	if (duplex) | ||||||
|  | 		flow |= (pause_time << MAC_FLOW_CTRL_PT_SHIFT); | ||||||
|  | 	writel(flow, ioaddr + MAC_FLOW_CTRL); | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* No PMT module supported in our SoC  for the Ethernet Controller. */ | ||||||
|  | static void mac100_pmt(unsigned long ioaddr, unsigned long mode) | ||||||
|  | { | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_init_rx_desc(struct dma_desc *p, unsigned int ring_size, | ||||||
|  | 				int disable_rx_ic) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	for (i = 0; i < ring_size; i++) { | ||||||
|  | 		p->des01.rx.own = 1; | ||||||
|  | 		p->des01.rx.buffer1_size = BUF_SIZE_2KiB - 1; | ||||||
|  | 		if (i == ring_size - 1) | ||||||
|  | 			p->des01.rx.end_ring = 1; | ||||||
|  | 		if (disable_rx_ic) | ||||||
|  | 			p->des01.rx.disable_ic = 1; | ||||||
|  | 		p++; | ||||||
|  | 	} | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_init_tx_desc(struct dma_desc *p, unsigned int ring_size) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	for (i = 0; i < ring_size; i++) { | ||||||
|  | 		p->des01.tx.own = 0; | ||||||
|  | 		if (i == ring_size - 1) | ||||||
|  | 			p->des01.tx.end_ring = 1; | ||||||
|  | 		p++; | ||||||
|  | 	} | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int mac100_get_tx_owner(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.tx.own; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int mac100_get_rx_owner(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.rx.own; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_set_tx_owner(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	p->des01.tx.own = 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_set_rx_owner(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	p->des01.rx.own = 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int mac100_get_tx_ls(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.tx.last_segment; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_release_tx_desc(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	int ter = p->des01.tx.end_ring; | ||||||
|  | 
 | ||||||
|  | 	/* clean field used within the xmit */ | ||||||
|  | 	p->des01.tx.first_segment = 0; | ||||||
|  | 	p->des01.tx.last_segment = 0; | ||||||
|  | 	p->des01.tx.buffer1_size = 0; | ||||||
|  | 
 | ||||||
|  | 	/* clean status reported */ | ||||||
|  | 	p->des01.tx.error_summary = 0; | ||||||
|  | 	p->des01.tx.underflow_error = 0; | ||||||
|  | 	p->des01.tx.no_carrier = 0; | ||||||
|  | 	p->des01.tx.loss_carrier = 0; | ||||||
|  | 	p->des01.tx.excessive_deferral = 0; | ||||||
|  | 	p->des01.tx.excessive_collisions = 0; | ||||||
|  | 	p->des01.tx.late_collision = 0; | ||||||
|  | 	p->des01.tx.heartbeat_fail = 0; | ||||||
|  | 	p->des01.tx.deferred = 0; | ||||||
|  | 
 | ||||||
|  | 	/* set termination field */ | ||||||
|  | 	p->des01.tx.end_ring = ter; | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_prepare_tx_desc(struct dma_desc *p, int is_fs, int len, | ||||||
|  | 				   int csum_flag) | ||||||
|  | { | ||||||
|  | 	p->des01.tx.first_segment = is_fs; | ||||||
|  | 	p->des01.tx.buffer1_size = len; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_clear_tx_ic(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	p->des01.tx.interrupt = 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void mac100_close_tx_desc(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	p->des01.tx.last_segment = 1; | ||||||
|  | 	p->des01.tx.interrupt = 1; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int mac100_get_rx_frame_len(struct dma_desc *p) | ||||||
|  | { | ||||||
|  | 	return p->des01.rx.frame_length; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | struct stmmac_ops mac100_driver = { | ||||||
|  | 	.core_init = mac100_core_init, | ||||||
|  | 	.dump_mac_regs = mac100_dump_mac_regs, | ||||||
|  | 	.dma_init = mac100_dma_init, | ||||||
|  | 	.dump_dma_regs = mac100_dump_dma_regs, | ||||||
|  | 	.dma_mode = mac100_dma_operation_mode, | ||||||
|  | 	.dma_diagnostic_fr = mac100_dma_diagnostic_fr, | ||||||
|  | 	.tx_status = mac100_get_tx_frame_status, | ||||||
|  | 	.rx_status = mac100_get_rx_frame_status, | ||||||
|  | 	.get_tx_len = mac100_get_tx_len, | ||||||
|  | 	.set_filter = mac100_set_filter, | ||||||
|  | 	.flow_ctrl = mac100_flow_ctrl, | ||||||
|  | 	.pmt = mac100_pmt, | ||||||
|  | 	.init_rx_desc = mac100_init_rx_desc, | ||||||
|  | 	.init_tx_desc = mac100_init_tx_desc, | ||||||
|  | 	.get_tx_owner = mac100_get_tx_owner, | ||||||
|  | 	.get_rx_owner = mac100_get_rx_owner, | ||||||
|  | 	.release_tx_desc = mac100_release_tx_desc, | ||||||
|  | 	.prepare_tx_desc = mac100_prepare_tx_desc, | ||||||
|  | 	.clear_tx_ic = mac100_clear_tx_ic, | ||||||
|  | 	.close_tx_desc = mac100_close_tx_desc, | ||||||
|  | 	.get_tx_ls = mac100_get_tx_ls, | ||||||
|  | 	.set_tx_owner = mac100_set_tx_owner, | ||||||
|  | 	.set_rx_owner = mac100_set_rx_owner, | ||||||
|  | 	.get_rx_frame_len = mac100_get_rx_frame_len, | ||||||
|  | 	.host_irq_status = mac100_irq_status, | ||||||
|  | 	.set_umac_addr = mac100_set_umac_addr, | ||||||
|  | 	.get_umac_addr = mac100_get_umac_addr, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | struct mac_device_info *mac100_setup(unsigned long ioaddr) | ||||||
|  | { | ||||||
|  | 	struct mac_device_info *mac; | ||||||
|  | 
 | ||||||
|  | 	mac = kzalloc(sizeof(const struct mac_device_info), GFP_KERNEL); | ||||||
|  | 
 | ||||||
|  | 	pr_info("\tMAC 10/100\n"); | ||||||
|  | 
 | ||||||
|  | 	mac->ops = &mac100_driver; | ||||||
|  | 	mac->hw.pmt = PMT_NOT_SUPPORTED; | ||||||
|  | 	mac->hw.link.port = MAC_CONTROL_PS; | ||||||
|  | 	mac->hw.link.duplex = MAC_CONTROL_F; | ||||||
|  | 	mac->hw.link.speed = 0; | ||||||
|  | 	mac->hw.mii.addr = MAC_MII_ADDR; | ||||||
|  | 	mac->hw.mii.data = MAC_MII_DATA; | ||||||
|  | 
 | ||||||
|  | 	return mac; | ||||||
|  | } | ||||||
							
								
								
									
										116
									
								
								drivers/net/stmmac/mac100.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										116
									
								
								drivers/net/stmmac/mac100.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,116 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   MAC 10/100 Header File | ||||||
|  | 
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | /*----------------------------------------------------------------------------
 | ||||||
|  |  *	 			MAC BLOCK defines | ||||||
|  |  *---------------------------------------------------------------------------*/ | ||||||
|  | /* MAC CSR offset */ | ||||||
|  | #define MAC_CONTROL	0x00000000	/* MAC Control */ | ||||||
|  | #define MAC_ADDR_HIGH	0x00000004	/* MAC Address High */ | ||||||
|  | #define MAC_ADDR_LOW	0x00000008	/* MAC Address Low */ | ||||||
|  | #define MAC_HASH_HIGH	0x0000000c	/* Multicast Hash Table High */ | ||||||
|  | #define MAC_HASH_LOW	0x00000010	/* Multicast Hash Table Low */ | ||||||
|  | #define MAC_MII_ADDR	0x00000014	/* MII Address */ | ||||||
|  | #define MAC_MII_DATA	0x00000018	/* MII Data */ | ||||||
|  | #define MAC_FLOW_CTRL	0x0000001c	/* Flow Control */ | ||||||
|  | #define MAC_VLAN1	0x00000020	/* VLAN1 Tag */ | ||||||
|  | #define MAC_VLAN2	0x00000024	/* VLAN2 Tag */ | ||||||
|  | 
 | ||||||
|  | /* MAC CTRL defines */ | ||||||
|  | #define MAC_CONTROL_RA	0x80000000	/* Receive All Mode */ | ||||||
|  | #define MAC_CONTROL_BLE	0x40000000	/* Endian Mode */ | ||||||
|  | #define MAC_CONTROL_HBD	0x10000000	/* Heartbeat Disable */ | ||||||
|  | #define MAC_CONTROL_PS	0x08000000	/* Port Select */ | ||||||
|  | #define MAC_CONTROL_DRO	0x00800000	/* Disable Receive Own */ | ||||||
|  | #define MAC_CONTROL_EXT_LOOPBACK 0x00400000	/* Reserved (ext loopback?) */ | ||||||
|  | #define MAC_CONTROL_OM	0x00200000	/* Loopback Operating Mode */ | ||||||
|  | #define MAC_CONTROL_F	0x00100000	/* Full Duplex Mode */ | ||||||
|  | #define MAC_CONTROL_PM	0x00080000	/* Pass All Multicast */ | ||||||
|  | #define MAC_CONTROL_PR	0x00040000	/* Promiscuous Mode */ | ||||||
|  | #define MAC_CONTROL_IF	0x00020000	/* Inverse Filtering */ | ||||||
|  | #define MAC_CONTROL_PB	0x00010000	/* Pass Bad Frames */ | ||||||
|  | #define MAC_CONTROL_HO	0x00008000	/* Hash Only Filtering Mode */ | ||||||
|  | #define MAC_CONTROL_HP	0x00002000	/* Hash/Perfect Filtering Mode */ | ||||||
|  | #define MAC_CONTROL_LCC	0x00001000	/* Late Collision Control */ | ||||||
|  | #define MAC_CONTROL_DBF	0x00000800	/* Disable Broadcast Frames */ | ||||||
|  | #define MAC_CONTROL_DRTY	0x00000400	/* Disable Retry */ | ||||||
|  | #define MAC_CONTROL_ASTP	0x00000100	/* Automatic Pad Stripping */ | ||||||
|  | #define MAC_CONTROL_BOLMT_10	0x00000000	/* Back Off Limit 10 */ | ||||||
|  | #define MAC_CONTROL_BOLMT_8	0x00000040	/* Back Off Limit 8 */ | ||||||
|  | #define MAC_CONTROL_BOLMT_4	0x00000080	/* Back Off Limit 4 */ | ||||||
|  | #define MAC_CONTROL_BOLMT_1	0x000000c0	/* Back Off Limit 1 */ | ||||||
|  | #define MAC_CONTROL_DC		0x00000020	/* Deferral Check */ | ||||||
|  | #define MAC_CONTROL_TE		0x00000008	/* Transmitter Enable */ | ||||||
|  | #define MAC_CONTROL_RE		0x00000004	/* Receiver Enable */ | ||||||
|  | 
 | ||||||
|  | #define MAC_CORE_INIT (MAC_CONTROL_HBD | MAC_CONTROL_ASTP) | ||||||
|  | 
 | ||||||
|  | /* MAC FLOW CTRL defines */ | ||||||
|  | #define MAC_FLOW_CTRL_PT_MASK	0xffff0000	/* Pause Time Mask */ | ||||||
|  | #define MAC_FLOW_CTRL_PT_SHIFT	16 | ||||||
|  | #define MAC_FLOW_CTRL_PASS	0x00000004	/* Pass Control Frames */ | ||||||
|  | #define MAC_FLOW_CTRL_ENABLE	0x00000002	/* Flow Control Enable */ | ||||||
|  | #define MAC_FLOW_CTRL_PAUSE	0x00000001	/* Flow Control Busy ... */ | ||||||
|  | 
 | ||||||
|  | /* MII ADDR  defines */ | ||||||
|  | #define MAC_MII_ADDR_WRITE	0x00000002	/* MII Write */ | ||||||
|  | #define MAC_MII_ADDR_BUSY	0x00000001	/* MII Busy */ | ||||||
|  | 
 | ||||||
|  | /*----------------------------------------------------------------------------
 | ||||||
|  |  * 				DMA BLOCK defines | ||||||
|  |  *---------------------------------------------------------------------------*/ | ||||||
|  | 
 | ||||||
|  | /* DMA Bus Mode register defines */ | ||||||
|  | #define DMA_BUS_MODE_DBO	0x00100000	/* Descriptor Byte Ordering */ | ||||||
|  | #define DMA_BUS_MODE_BLE	0x00000080	/* Big Endian/Little Endian */ | ||||||
|  | #define DMA_BUS_MODE_PBL_MASK	0x00003f00	/* Programmable Burst Len */ | ||||||
|  | #define DMA_BUS_MODE_PBL_SHIFT	8 | ||||||
|  | #define DMA_BUS_MODE_DSL_MASK	0x0000007c	/* Descriptor Skip Length */ | ||||||
|  | #define DMA_BUS_MODE_DSL_SHIFT	2	/*   (in DWORDS)      */ | ||||||
|  | #define DMA_BUS_MODE_BAR_BUS	0x00000002	/* Bar-Bus Arbitration */ | ||||||
|  | #define DMA_BUS_MODE_SFT_RESET	0x00000001	/* Software Reset */ | ||||||
|  | #define DMA_BUS_MODE_DEFAULT	0x00000000 | ||||||
|  | 
 | ||||||
|  | /* DMA Control register defines */ | ||||||
|  | #define DMA_CONTROL_SF		0x00200000	/* Store And Forward */ | ||||||
|  | 
 | ||||||
|  | /* Transmit Threshold Control */ | ||||||
|  | enum ttc_control { | ||||||
|  | 	DMA_CONTROL_TTC_DEFAULT = 0x00000000,	/* Threshold is 32 DWORDS */ | ||||||
|  | 	DMA_CONTROL_TTC_64 = 0x00004000,	/* Threshold is 64 DWORDS */ | ||||||
|  | 	DMA_CONTROL_TTC_128 = 0x00008000,	/* Threshold is 128 DWORDS */ | ||||||
|  | 	DMA_CONTROL_TTC_256 = 0x0000c000,	/* Threshold is 256 DWORDS */ | ||||||
|  | 	DMA_CONTROL_TTC_18 = 0x00400000,	/* Threshold is 18 DWORDS */ | ||||||
|  | 	DMA_CONTROL_TTC_24 = 0x00404000,	/* Threshold is 24 DWORDS */ | ||||||
|  | 	DMA_CONTROL_TTC_32 = 0x00408000,	/* Threshold is 32 DWORDS */ | ||||||
|  | 	DMA_CONTROL_TTC_40 = 0x0040c000,	/* Threshold is 40 DWORDS */ | ||||||
|  | 	DMA_CONTROL_SE = 0x00000008,	/* Stop On Empty */ | ||||||
|  | 	DMA_CONTROL_OSF = 0x00000004,	/* Operate On 2nd Frame */ | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* STMAC110 DMA Missed Frame Counter register defines */ | ||||||
|  | #define DMA_MISSED_FRAME_OVE	0x10000000	/* FIFO Overflow Overflow */ | ||||||
|  | #define DMA_MISSED_FRAME_OVE_CNTR 0x0ffe0000	/* Overflow Frame Counter */ | ||||||
|  | #define DMA_MISSED_FRAME_OVE_M	0x00010000	/* Missed Frame Overflow */ | ||||||
|  | #define DMA_MISSED_FRAME_M_CNTR	0x0000ffff	/* Missed Frame Couinter */ | ||||||
							
								
								
									
										98
									
								
								drivers/net/stmmac/stmmac.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								drivers/net/stmmac/stmmac.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,98 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #define DRV_MODULE_VERSION	"Oct_09" | ||||||
|  | 
 | ||||||
|  | #if defined(CONFIG_VLAN_8021Q) || defined(CONFIG_VLAN_8021Q_MODULE) | ||||||
|  | #define STMMAC_VLAN_TAG_USED | ||||||
|  | #include <linux/if_vlan.h> | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | #include "common.h" | ||||||
|  | #ifdef CONFIG_STMMAC_TIMER | ||||||
|  | #include "stmmac_timer.h" | ||||||
|  | #endif | ||||||
|  | 
 | ||||||
|  | struct stmmac_priv { | ||||||
|  | 	/* Frequently used values are kept adjacent for cache effect */ | ||||||
|  | 	struct dma_desc *dma_tx ____cacheline_aligned; | ||||||
|  | 	dma_addr_t dma_tx_phy; | ||||||
|  | 	struct sk_buff **tx_skbuff; | ||||||
|  | 	unsigned int cur_tx; | ||||||
|  | 	unsigned int dirty_tx; | ||||||
|  | 	unsigned int dma_tx_size; | ||||||
|  | 	int tx_coe; | ||||||
|  | 	int tx_coalesce; | ||||||
|  | 
 | ||||||
|  | 	struct dma_desc *dma_rx ; | ||||||
|  | 	unsigned int cur_rx; | ||||||
|  | 	unsigned int dirty_rx; | ||||||
|  | 	struct sk_buff **rx_skbuff; | ||||||
|  | 	dma_addr_t *rx_skbuff_dma; | ||||||
|  | 	struct sk_buff_head rx_recycle; | ||||||
|  | 
 | ||||||
|  | 	struct net_device *dev; | ||||||
|  | 	int is_gmac; | ||||||
|  | 	dma_addr_t dma_rx_phy; | ||||||
|  | 	unsigned int dma_rx_size; | ||||||
|  | 	int rx_csum; | ||||||
|  | 	unsigned int dma_buf_sz; | ||||||
|  | 	struct device *device; | ||||||
|  | 	struct mac_device_info *mac_type; | ||||||
|  | 
 | ||||||
|  | 	struct stmmac_extra_stats xstats; | ||||||
|  | 	struct napi_struct napi; | ||||||
|  | 
 | ||||||
|  | 	phy_interface_t phy_interface; | ||||||
|  | 	int pbl; | ||||||
|  | 	int bus_id; | ||||||
|  | 	int phy_addr; | ||||||
|  | 	int phy_mask; | ||||||
|  | 	int (*phy_reset) (void *priv); | ||||||
|  | 	void (*fix_mac_speed) (void *priv, unsigned int speed); | ||||||
|  | 	void *bsp_priv; | ||||||
|  | 
 | ||||||
|  | 	int phy_irq; | ||||||
|  | 	struct phy_device *phydev; | ||||||
|  | 	int oldlink; | ||||||
|  | 	int speed; | ||||||
|  | 	int oldduplex; | ||||||
|  | 	unsigned int flow_ctrl; | ||||||
|  | 	unsigned int pause; | ||||||
|  | 	struct mii_bus *mii; | ||||||
|  | 
 | ||||||
|  | 	u32 msg_enable; | ||||||
|  | 	spinlock_t lock; | ||||||
|  | 	int wolopts; | ||||||
|  | 	int wolenabled; | ||||||
|  | 	int shutdown; | ||||||
|  | #ifdef CONFIG_STMMAC_TIMER | ||||||
|  | 	struct stmmac_timer *tm; | ||||||
|  | #endif | ||||||
|  | #ifdef STMMAC_VLAN_TAG_USED | ||||||
|  | 	struct vlan_group *vlgrp; | ||||||
|  | #endif | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | extern int stmmac_mdio_unregister(struct net_device *ndev); | ||||||
|  | extern int stmmac_mdio_register(struct net_device *ndev); | ||||||
|  | extern void stmmac_set_ethtool_ops(struct net_device *netdev); | ||||||
							
								
								
									
										395
									
								
								drivers/net/stmmac/stmmac_ethtool.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										395
									
								
								drivers/net/stmmac/stmmac_ethtool.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,395 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   STMMAC Ethtool support | ||||||
|  | 
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #include <linux/etherdevice.h> | ||||||
|  | #include <linux/ethtool.h> | ||||||
|  | #include <linux/mii.h> | ||||||
|  | #include <linux/phy.h> | ||||||
|  | 
 | ||||||
|  | #include "stmmac.h" | ||||||
|  | 
 | ||||||
|  | #define REG_SPACE_SIZE	0x1054 | ||||||
|  | #define MAC100_ETHTOOL_NAME	"st_mac100" | ||||||
|  | #define GMAC_ETHTOOL_NAME	"st_gmac" | ||||||
|  | 
 | ||||||
|  | struct stmmac_stats { | ||||||
|  | 	char stat_string[ETH_GSTRING_LEN]; | ||||||
|  | 	int sizeof_stat; | ||||||
|  | 	int stat_offset; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | #define STMMAC_STAT(m)	\ | ||||||
|  | 	{ #m, FIELD_SIZEOF(struct stmmac_extra_stats, m),	\ | ||||||
|  | 	offsetof(struct stmmac_priv, xstats.m)} | ||||||
|  | 
 | ||||||
|  | static const struct  stmmac_stats stmmac_gstrings_stats[] = { | ||||||
|  | 	STMMAC_STAT(tx_underflow), | ||||||
|  | 	STMMAC_STAT(tx_carrier), | ||||||
|  | 	STMMAC_STAT(tx_losscarrier), | ||||||
|  | 	STMMAC_STAT(tx_heartbeat), | ||||||
|  | 	STMMAC_STAT(tx_deferred), | ||||||
|  | 	STMMAC_STAT(tx_vlan), | ||||||
|  | 	STMMAC_STAT(rx_vlan), | ||||||
|  | 	STMMAC_STAT(tx_jabber), | ||||||
|  | 	STMMAC_STAT(tx_frame_flushed), | ||||||
|  | 	STMMAC_STAT(tx_payload_error), | ||||||
|  | 	STMMAC_STAT(tx_ip_header_error), | ||||||
|  | 	STMMAC_STAT(rx_desc), | ||||||
|  | 	STMMAC_STAT(rx_partial), | ||||||
|  | 	STMMAC_STAT(rx_runt), | ||||||
|  | 	STMMAC_STAT(rx_toolong), | ||||||
|  | 	STMMAC_STAT(rx_collision), | ||||||
|  | 	STMMAC_STAT(rx_crc), | ||||||
|  | 	STMMAC_STAT(rx_lenght), | ||||||
|  | 	STMMAC_STAT(rx_mii), | ||||||
|  | 	STMMAC_STAT(rx_multicast), | ||||||
|  | 	STMMAC_STAT(rx_gmac_overflow), | ||||||
|  | 	STMMAC_STAT(rx_watchdog), | ||||||
|  | 	STMMAC_STAT(da_rx_filter_fail), | ||||||
|  | 	STMMAC_STAT(sa_rx_filter_fail), | ||||||
|  | 	STMMAC_STAT(rx_missed_cntr), | ||||||
|  | 	STMMAC_STAT(rx_overflow_cntr), | ||||||
|  | 	STMMAC_STAT(tx_undeflow_irq), | ||||||
|  | 	STMMAC_STAT(tx_process_stopped_irq), | ||||||
|  | 	STMMAC_STAT(tx_jabber_irq), | ||||||
|  | 	STMMAC_STAT(rx_overflow_irq), | ||||||
|  | 	STMMAC_STAT(rx_buf_unav_irq), | ||||||
|  | 	STMMAC_STAT(rx_process_stopped_irq), | ||||||
|  | 	STMMAC_STAT(rx_watchdog_irq), | ||||||
|  | 	STMMAC_STAT(tx_early_irq), | ||||||
|  | 	STMMAC_STAT(fatal_bus_error_irq), | ||||||
|  | 	STMMAC_STAT(threshold), | ||||||
|  | 	STMMAC_STAT(tx_pkt_n), | ||||||
|  | 	STMMAC_STAT(rx_pkt_n), | ||||||
|  | 	STMMAC_STAT(poll_n), | ||||||
|  | 	STMMAC_STAT(sched_timer_n), | ||||||
|  | 	STMMAC_STAT(normal_irq_n), | ||||||
|  | }; | ||||||
|  | #define STMMAC_STATS_LEN ARRAY_SIZE(stmmac_gstrings_stats) | ||||||
|  | 
 | ||||||
|  | void stmmac_ethtool_getdrvinfo(struct net_device *dev, | ||||||
|  | 			       struct ethtool_drvinfo *info) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 
 | ||||||
|  | 	if (!priv->is_gmac) | ||||||
|  | 		strcpy(info->driver, MAC100_ETHTOOL_NAME); | ||||||
|  | 	else | ||||||
|  | 		strcpy(info->driver, GMAC_ETHTOOL_NAME); | ||||||
|  | 
 | ||||||
|  | 	strcpy(info->version, DRV_MODULE_VERSION); | ||||||
|  | 	info->fw_version[0] = '\0'; | ||||||
|  | 	info->n_stats = STMMAC_STATS_LEN; | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_ethtool_getsettings(struct net_device *dev, struct ethtool_cmd *cmd) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 	struct phy_device *phy = priv->phydev; | ||||||
|  | 	int rc; | ||||||
|  | 	if (phy == NULL) { | ||||||
|  | 		pr_err("%s: %s: PHY is not registered\n", | ||||||
|  | 		       __func__, dev->name); | ||||||
|  | 		return -ENODEV; | ||||||
|  | 	} | ||||||
|  | 	if (!netif_running(dev)) { | ||||||
|  | 		pr_err("%s: interface is disabled: we cannot track " | ||||||
|  | 		"link speed / duplex setting\n", dev->name); | ||||||
|  | 		return -EBUSY; | ||||||
|  | 	} | ||||||
|  | 	cmd->transceiver = XCVR_INTERNAL; | ||||||
|  | 	spin_lock_irq(&priv->lock); | ||||||
|  | 	rc = phy_ethtool_gset(phy, cmd); | ||||||
|  | 	spin_unlock_irq(&priv->lock); | ||||||
|  | 	return rc; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_ethtool_setsettings(struct net_device *dev, struct ethtool_cmd *cmd) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 	struct phy_device *phy = priv->phydev; | ||||||
|  | 	int rc; | ||||||
|  | 
 | ||||||
|  | 	spin_lock(&priv->lock); | ||||||
|  | 	rc = phy_ethtool_sset(phy, cmd); | ||||||
|  | 	spin_unlock(&priv->lock); | ||||||
|  | 
 | ||||||
|  | 	return rc; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u32 stmmac_ethtool_getmsglevel(struct net_device *dev) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 	return priv->msg_enable; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void stmmac_ethtool_setmsglevel(struct net_device *dev, u32 level) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 	priv->msg_enable = level; | ||||||
|  | 
 | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_check_if_running(struct net_device *dev) | ||||||
|  | { | ||||||
|  | 	if (!netif_running(dev)) | ||||||
|  | 		return -EBUSY; | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_ethtool_get_regs_len(struct net_device *dev) | ||||||
|  | { | ||||||
|  | 	return REG_SPACE_SIZE; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void stmmac_ethtool_gregs(struct net_device *dev, | ||||||
|  | 			  struct ethtool_regs *regs, void *space) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	u32 *reg_space = (u32 *) space; | ||||||
|  | 
 | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 
 | ||||||
|  | 	memset(reg_space, 0x0, REG_SPACE_SIZE); | ||||||
|  | 
 | ||||||
|  | 	if (!priv->is_gmac) { | ||||||
|  | 		/* MAC registers */ | ||||||
|  | 		for (i = 0; i < 12; i++) | ||||||
|  | 			reg_space[i] = readl(dev->base_addr + (i * 4)); | ||||||
|  | 		/* DMA registers */ | ||||||
|  | 		for (i = 0; i < 9; i++) | ||||||
|  | 			reg_space[i + 12] = | ||||||
|  | 			    readl(dev->base_addr + (DMA_BUS_MODE + (i * 4))); | ||||||
|  | 		reg_space[22] = readl(dev->base_addr + DMA_CUR_TX_BUF_ADDR); | ||||||
|  | 		reg_space[23] = readl(dev->base_addr + DMA_CUR_RX_BUF_ADDR); | ||||||
|  | 	} else { | ||||||
|  | 		/* MAC registers */ | ||||||
|  | 		for (i = 0; i < 55; i++) | ||||||
|  | 			reg_space[i] = readl(dev->base_addr + (i * 4)); | ||||||
|  | 		/* DMA registers */ | ||||||
|  | 		for (i = 0; i < 22; i++) | ||||||
|  | 			reg_space[i + 55] = | ||||||
|  | 			    readl(dev->base_addr + (DMA_BUS_MODE + (i * 4))); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_ethtool_set_tx_csum(struct net_device *netdev, u32 data) | ||||||
|  | { | ||||||
|  | 	if (data) | ||||||
|  | 		netdev->features |= NETIF_F_HW_CSUM; | ||||||
|  | 	else | ||||||
|  | 		netdev->features &= ~NETIF_F_HW_CSUM; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | u32 stmmac_ethtool_get_rx_csum(struct net_device *dev) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 
 | ||||||
|  | 	return priv->rx_csum; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void | ||||||
|  | stmmac_get_pauseparam(struct net_device *netdev, | ||||||
|  | 		      struct ethtool_pauseparam *pause) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(netdev); | ||||||
|  | 
 | ||||||
|  | 	spin_lock(&priv->lock); | ||||||
|  | 
 | ||||||
|  | 	pause->rx_pause = 0; | ||||||
|  | 	pause->tx_pause = 0; | ||||||
|  | 	pause->autoneg = priv->phydev->autoneg; | ||||||
|  | 
 | ||||||
|  | 	if (priv->flow_ctrl & FLOW_RX) | ||||||
|  | 		pause->rx_pause = 1; | ||||||
|  | 	if (priv->flow_ctrl & FLOW_TX) | ||||||
|  | 		pause->tx_pause = 1; | ||||||
|  | 
 | ||||||
|  | 	spin_unlock(&priv->lock); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int | ||||||
|  | stmmac_set_pauseparam(struct net_device *netdev, | ||||||
|  | 		      struct ethtool_pauseparam *pause) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(netdev); | ||||||
|  | 	struct phy_device *phy = priv->phydev; | ||||||
|  | 	int new_pause = FLOW_OFF; | ||||||
|  | 	int ret = 0; | ||||||
|  | 
 | ||||||
|  | 	spin_lock(&priv->lock); | ||||||
|  | 
 | ||||||
|  | 	if (pause->rx_pause) | ||||||
|  | 		new_pause |= FLOW_RX; | ||||||
|  | 	if (pause->tx_pause) | ||||||
|  | 		new_pause |= FLOW_TX; | ||||||
|  | 
 | ||||||
|  | 	priv->flow_ctrl = new_pause; | ||||||
|  | 
 | ||||||
|  | 	if (phy->autoneg) { | ||||||
|  | 		if (netif_running(netdev)) { | ||||||
|  | 			struct ethtool_cmd cmd; | ||||||
|  | 			/* auto-negotiation automatically restarted */ | ||||||
|  | 			cmd.cmd = ETHTOOL_NWAY_RST; | ||||||
|  | 			cmd.supported = phy->supported; | ||||||
|  | 			cmd.advertising = phy->advertising; | ||||||
|  | 			cmd.autoneg = phy->autoneg; | ||||||
|  | 			cmd.speed = phy->speed; | ||||||
|  | 			cmd.duplex = phy->duplex; | ||||||
|  | 			cmd.phy_address = phy->addr; | ||||||
|  | 			ret = phy_ethtool_sset(phy, &cmd); | ||||||
|  | 		} | ||||||
|  | 	} else { | ||||||
|  | 		unsigned long ioaddr = netdev->base_addr; | ||||||
|  | 		priv->mac_type->ops->flow_ctrl(ioaddr, phy->duplex, | ||||||
|  | 					       priv->flow_ctrl, priv->pause); | ||||||
|  | 	} | ||||||
|  | 	spin_unlock(&priv->lock); | ||||||
|  | 	return ret; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void stmmac_get_ethtool_stats(struct net_device *dev, | ||||||
|  | 				 struct ethtool_stats *dummy, u64 *data) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 	unsigned long ioaddr = dev->base_addr; | ||||||
|  | 	int i; | ||||||
|  | 
 | ||||||
|  | 	/* Update HW stats if supported */ | ||||||
|  | 	priv->mac_type->ops->dma_diagnostic_fr(&dev->stats, &priv->xstats, | ||||||
|  | 					       ioaddr); | ||||||
|  | 
 | ||||||
|  | 	for (i = 0; i < STMMAC_STATS_LEN; i++) { | ||||||
|  | 		char *p = (char *)priv + stmmac_gstrings_stats[i].stat_offset; | ||||||
|  | 		data[i] = (stmmac_gstrings_stats[i].sizeof_stat == | ||||||
|  | 		sizeof(u64)) ? (*(u64 *)p) : (*(u32 *)p); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int stmmac_get_sset_count(struct net_device *netdev, int sset) | ||||||
|  | { | ||||||
|  | 	switch (sset) { | ||||||
|  | 	case ETH_SS_STATS: | ||||||
|  | 		return STMMAC_STATS_LEN; | ||||||
|  | 	default: | ||||||
|  | 		return -EOPNOTSUPP; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void stmmac_get_strings(struct net_device *dev, u32 stringset, u8 *data) | ||||||
|  | { | ||||||
|  | 	int i; | ||||||
|  | 	u8 *p = data; | ||||||
|  | 
 | ||||||
|  | 	switch (stringset) { | ||||||
|  | 	case ETH_SS_STATS: | ||||||
|  | 		for (i = 0; i < STMMAC_STATS_LEN; i++) { | ||||||
|  | 			memcpy(p, stmmac_gstrings_stats[i].stat_string, | ||||||
|  | 				ETH_GSTRING_LEN); | ||||||
|  | 			p += ETH_GSTRING_LEN; | ||||||
|  | 		} | ||||||
|  | 		break; | ||||||
|  | 	default: | ||||||
|  | 		WARN_ON(1); | ||||||
|  | 		break; | ||||||
|  | 	} | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /* Currently only support WOL through Magic packet. */ | ||||||
|  | static void stmmac_get_wol(struct net_device *dev, struct ethtool_wolinfo *wol) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 
 | ||||||
|  | 	spin_lock_irq(&priv->lock); | ||||||
|  | 	if (priv->wolenabled == PMT_SUPPORTED) { | ||||||
|  | 		wol->supported = WAKE_MAGIC; | ||||||
|  | 		wol->wolopts = priv->wolopts; | ||||||
|  | 	} | ||||||
|  | 	spin_unlock_irq(&priv->lock); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static int stmmac_set_wol(struct net_device *dev, struct ethtool_wolinfo *wol) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(dev); | ||||||
|  | 	u32 support = WAKE_MAGIC; | ||||||
|  | 
 | ||||||
|  | 	if (priv->wolenabled == PMT_NOT_SUPPORTED) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
|  | 	if (wol->wolopts & ~support) | ||||||
|  | 		return -EINVAL; | ||||||
|  | 
 | ||||||
|  | 	if (wol->wolopts == 0) | ||||||
|  | 		device_set_wakeup_enable(priv->device, 0); | ||||||
|  | 	else | ||||||
|  | 		device_set_wakeup_enable(priv->device, 1); | ||||||
|  | 
 | ||||||
|  | 	spin_lock_irq(&priv->lock); | ||||||
|  | 	priv->wolopts = wol->wolopts; | ||||||
|  | 	spin_unlock_irq(&priv->lock); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static struct ethtool_ops stmmac_ethtool_ops = { | ||||||
|  | 	.begin = stmmac_check_if_running, | ||||||
|  | 	.get_drvinfo = stmmac_ethtool_getdrvinfo, | ||||||
|  | 	.get_settings = stmmac_ethtool_getsettings, | ||||||
|  | 	.set_settings = stmmac_ethtool_setsettings, | ||||||
|  | 	.get_msglevel = stmmac_ethtool_getmsglevel, | ||||||
|  | 	.set_msglevel = stmmac_ethtool_setmsglevel, | ||||||
|  | 	.get_regs = stmmac_ethtool_gregs, | ||||||
|  | 	.get_regs_len = stmmac_ethtool_get_regs_len, | ||||||
|  | 	.get_link = ethtool_op_get_link, | ||||||
|  | 	.get_rx_csum = stmmac_ethtool_get_rx_csum, | ||||||
|  | 	.get_tx_csum = ethtool_op_get_tx_csum, | ||||||
|  | 	.set_tx_csum = stmmac_ethtool_set_tx_csum, | ||||||
|  | 	.get_sg = ethtool_op_get_sg, | ||||||
|  | 	.set_sg = ethtool_op_set_sg, | ||||||
|  | 	.get_pauseparam = stmmac_get_pauseparam, | ||||||
|  | 	.set_pauseparam = stmmac_set_pauseparam, | ||||||
|  | 	.get_ethtool_stats = stmmac_get_ethtool_stats, | ||||||
|  | 	.get_strings = stmmac_get_strings, | ||||||
|  | 	.get_wol = stmmac_get_wol, | ||||||
|  | 	.set_wol = stmmac_set_wol, | ||||||
|  | 	.get_sset_count	= stmmac_get_sset_count, | ||||||
|  | #ifdef NETIF_F_TSO | ||||||
|  | 	.get_tso = ethtool_op_get_tso, | ||||||
|  | 	.set_tso = ethtool_op_set_tso, | ||||||
|  | #endif | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | void stmmac_set_ethtool_ops(struct net_device *netdev) | ||||||
|  | { | ||||||
|  | 	SET_ETHTOOL_OPS(netdev, &stmmac_ethtool_ops); | ||||||
|  | } | ||||||
							
								
								
									
										2204
									
								
								drivers/net/stmmac/stmmac_main.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2204
									
								
								drivers/net/stmmac/stmmac_main.c
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										217
									
								
								drivers/net/stmmac/stmmac_mdio.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										217
									
								
								drivers/net/stmmac/stmmac_mdio.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,217 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   STMMAC Ethernet Driver -- MDIO bus implementation | ||||||
|  |   Provides Bus interface for MII registers | ||||||
|  | 
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Carl Shaw <carl.shaw@st.com> | ||||||
|  |   Maintainer: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #include <linux/netdevice.h> | ||||||
|  | #include <linux/mii.h> | ||||||
|  | #include <linux/phy.h> | ||||||
|  | 
 | ||||||
|  | #include "stmmac.h" | ||||||
|  | 
 | ||||||
|  | #define MII_BUSY 0x00000001 | ||||||
|  | #define MII_WRITE 0x00000002 | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * stmmac_mdio_read | ||||||
|  |  * @bus: points to the mii_bus structure | ||||||
|  |  * @phyaddr: MII addr reg bits 15-11 | ||||||
|  |  * @phyreg: MII addr reg bits 10-6 | ||||||
|  |  * Description: it reads data from the MII register from within the phy device. | ||||||
|  |  * For the 7111 GMAC, we must set the bit 0 in the MII address register while | ||||||
|  |  * accessing the PHY registers. | ||||||
|  |  * Fortunately, it seems this has no drawback for the 7109 MAC. | ||||||
|  |  */ | ||||||
|  | static int stmmac_mdio_read(struct mii_bus *bus, int phyaddr, int phyreg) | ||||||
|  | { | ||||||
|  | 	struct net_device *ndev = bus->priv; | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(ndev); | ||||||
|  | 	unsigned long ioaddr = ndev->base_addr; | ||||||
|  | 	unsigned int mii_address = priv->mac_type->hw.mii.addr; | ||||||
|  | 	unsigned int mii_data = priv->mac_type->hw.mii.data; | ||||||
|  | 
 | ||||||
|  | 	int data; | ||||||
|  | 	u16 regValue = (((phyaddr << 11) & (0x0000F800)) | | ||||||
|  | 			((phyreg << 6) & (0x000007C0))); | ||||||
|  | 	regValue |= MII_BUSY;	/* in case of GMAC */ | ||||||
|  | 
 | ||||||
|  | 	do {} while (((readl(ioaddr + mii_address)) & MII_BUSY) == 1); | ||||||
|  | 	writel(regValue, ioaddr + mii_address); | ||||||
|  | 	do {} while (((readl(ioaddr + mii_address)) & MII_BUSY) == 1); | ||||||
|  | 
 | ||||||
|  | 	/* Read the data from the MII data register */ | ||||||
|  | 	data = (int)readl(ioaddr + mii_data); | ||||||
|  | 
 | ||||||
|  | 	return data; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * stmmac_mdio_write | ||||||
|  |  * @bus: points to the mii_bus structure | ||||||
|  |  * @phyaddr: MII addr reg bits 15-11 | ||||||
|  |  * @phyreg: MII addr reg bits 10-6 | ||||||
|  |  * @phydata: phy data | ||||||
|  |  * Description: it writes the data into the MII register from within the device. | ||||||
|  |  */ | ||||||
|  | static int stmmac_mdio_write(struct mii_bus *bus, int phyaddr, int phyreg, | ||||||
|  | 			     u16 phydata) | ||||||
|  | { | ||||||
|  | 	struct net_device *ndev = bus->priv; | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(ndev); | ||||||
|  | 	unsigned long ioaddr = ndev->base_addr; | ||||||
|  | 	unsigned int mii_address = priv->mac_type->hw.mii.addr; | ||||||
|  | 	unsigned int mii_data = priv->mac_type->hw.mii.data; | ||||||
|  | 
 | ||||||
|  | 	u16 value = | ||||||
|  | 	    (((phyaddr << 11) & (0x0000F800)) | ((phyreg << 6) & (0x000007C0))) | ||||||
|  | 	    | MII_WRITE; | ||||||
|  | 
 | ||||||
|  | 	value |= MII_BUSY; | ||||||
|  | 
 | ||||||
|  | 	/* Wait until any existing MII operation is complete */ | ||||||
|  | 	do {} while (((readl(ioaddr + mii_address)) & MII_BUSY) == 1); | ||||||
|  | 
 | ||||||
|  | 	/* Set the MII address register to write */ | ||||||
|  | 	writel(phydata, ioaddr + mii_data); | ||||||
|  | 	writel(value, ioaddr + mii_address); | ||||||
|  | 
 | ||||||
|  | 	/* Wait until any existing MII operation is complete */ | ||||||
|  | 	do {} while (((readl(ioaddr + mii_address)) & MII_BUSY) == 1); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * stmmac_mdio_reset | ||||||
|  |  * @bus: points to the mii_bus structure | ||||||
|  |  * Description: reset the MII bus | ||||||
|  |  */ | ||||||
|  | static int stmmac_mdio_reset(struct mii_bus *bus) | ||||||
|  | { | ||||||
|  | 	struct net_device *ndev = bus->priv; | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(ndev); | ||||||
|  | 	unsigned long ioaddr = ndev->base_addr; | ||||||
|  | 	unsigned int mii_address = priv->mac_type->hw.mii.addr; | ||||||
|  | 
 | ||||||
|  | 	if (priv->phy_reset) { | ||||||
|  | 		pr_debug("stmmac_mdio_reset: calling phy_reset\n"); | ||||||
|  | 		priv->phy_reset(priv->bsp_priv); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* This is a workaround for problems with the STE101P PHY.
 | ||||||
|  | 	 * It doesn't complete its reset until at least one clock cycle | ||||||
|  | 	 * on MDC, so perform a dummy mdio read. | ||||||
|  | 	 */ | ||||||
|  | 	writel(0, ioaddr + mii_address); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * stmmac_mdio_register | ||||||
|  |  * @ndev: net device structure | ||||||
|  |  * Description: it registers the MII bus | ||||||
|  |  */ | ||||||
|  | int stmmac_mdio_register(struct net_device *ndev) | ||||||
|  | { | ||||||
|  | 	int err = 0; | ||||||
|  | 	struct mii_bus *new_bus; | ||||||
|  | 	int *irqlist; | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(ndev); | ||||||
|  | 	int addr, found; | ||||||
|  | 
 | ||||||
|  | 	new_bus = mdiobus_alloc(); | ||||||
|  | 	if (new_bus == NULL) | ||||||
|  | 		return -ENOMEM; | ||||||
|  | 
 | ||||||
|  | 	irqlist = kzalloc(sizeof(int) * PHY_MAX_ADDR, GFP_KERNEL); | ||||||
|  | 	if (irqlist == NULL) { | ||||||
|  | 		err = -ENOMEM; | ||||||
|  | 		goto irqlist_alloc_fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	/* Assign IRQ to phy at address phy_addr */ | ||||||
|  | 	if (priv->phy_addr != -1) | ||||||
|  | 		irqlist[priv->phy_addr] = priv->phy_irq; | ||||||
|  | 
 | ||||||
|  | 	new_bus->name = "STMMAC MII Bus"; | ||||||
|  | 	new_bus->read = &stmmac_mdio_read; | ||||||
|  | 	new_bus->write = &stmmac_mdio_write; | ||||||
|  | 	new_bus->reset = &stmmac_mdio_reset; | ||||||
|  | 	snprintf(new_bus->id, MII_BUS_ID_SIZE, "%x", priv->bus_id); | ||||||
|  | 	new_bus->priv = ndev; | ||||||
|  | 	new_bus->irq = irqlist; | ||||||
|  | 	new_bus->phy_mask = priv->phy_mask; | ||||||
|  | 	new_bus->parent = priv->device; | ||||||
|  | 	err = mdiobus_register(new_bus); | ||||||
|  | 	if (err != 0) { | ||||||
|  | 		pr_err("%s: Cannot register as MDIO bus\n", new_bus->name); | ||||||
|  | 		goto bus_register_fail; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	priv->mii = new_bus; | ||||||
|  | 
 | ||||||
|  | 	found = 0; | ||||||
|  | 	for (addr = 0; addr < 32; addr++) { | ||||||
|  | 		struct phy_device *phydev = new_bus->phy_map[addr]; | ||||||
|  | 		if (phydev) { | ||||||
|  | 			if (priv->phy_addr == -1) { | ||||||
|  | 				priv->phy_addr = addr; | ||||||
|  | 				phydev->irq = priv->phy_irq; | ||||||
|  | 				irqlist[addr] = priv->phy_irq; | ||||||
|  | 			} | ||||||
|  | 			pr_info("%s: PHY ID %08x at %d IRQ %d (%s)%s\n", | ||||||
|  | 			       ndev->name, phydev->phy_id, addr, | ||||||
|  | 			       phydev->irq, dev_name(&phydev->dev), | ||||||
|  | 			       (addr == priv->phy_addr) ? " active" : ""); | ||||||
|  | 			found = 1; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	if (!found) | ||||||
|  | 		pr_warning("%s: No PHY found\n", ndev->name); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | bus_register_fail: | ||||||
|  | 	kfree(irqlist); | ||||||
|  | irqlist_alloc_fail: | ||||||
|  | 	kfree(new_bus); | ||||||
|  | 	return err; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /**
 | ||||||
|  |  * stmmac_mdio_unregister | ||||||
|  |  * @ndev: net device structure | ||||||
|  |  * Description: it unregisters the MII bus | ||||||
|  |  */ | ||||||
|  | int stmmac_mdio_unregister(struct net_device *ndev) | ||||||
|  | { | ||||||
|  | 	struct stmmac_priv *priv = netdev_priv(ndev); | ||||||
|  | 
 | ||||||
|  | 	mdiobus_unregister(priv->mii); | ||||||
|  | 	priv->mii->priv = NULL; | ||||||
|  | 	kfree(priv->mii); | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
							
								
								
									
										140
									
								
								drivers/net/stmmac/stmmac_timer.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								drivers/net/stmmac/stmmac_timer.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,140 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   STMMAC external timer support. | ||||||
|  | 
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | #include <linux/kernel.h> | ||||||
|  | #include <linux/etherdevice.h> | ||||||
|  | #include "stmmac_timer.h" | ||||||
|  | 
 | ||||||
|  | static void stmmac_timer_handler(void *data) | ||||||
|  | { | ||||||
|  | 	struct net_device *dev = (struct net_device *)data; | ||||||
|  | 
 | ||||||
|  | 	stmmac_schedule(dev); | ||||||
|  | 
 | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #define STMMAC_TIMER_MSG(timer, freq) \ | ||||||
|  | printk(KERN_INFO "stmmac_timer: %s Timer ON (freq %dHz)\n", timer, freq); | ||||||
|  | 
 | ||||||
|  | #if defined(CONFIG_STMMAC_RTC_TIMER) | ||||||
|  | #include <linux/rtc.h> | ||||||
|  | static struct rtc_device *stmmac_rtc; | ||||||
|  | static rtc_task_t stmmac_task; | ||||||
|  | 
 | ||||||
|  | static void stmmac_rtc_start(unsigned int new_freq) | ||||||
|  | { | ||||||
|  | 	rtc_irq_set_freq(stmmac_rtc, &stmmac_task, new_freq); | ||||||
|  | 	rtc_irq_set_state(stmmac_rtc, &stmmac_task, 1); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void stmmac_rtc_stop(void) | ||||||
|  | { | ||||||
|  | 	rtc_irq_set_state(stmmac_rtc, &stmmac_task, 0); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_open_ext_timer(struct net_device *dev, struct stmmac_timer *tm) | ||||||
|  | { | ||||||
|  | 	stmmac_task.private_data = dev; | ||||||
|  | 	stmmac_task.func = stmmac_timer_handler; | ||||||
|  | 
 | ||||||
|  | 	stmmac_rtc = rtc_class_open(CONFIG_RTC_HCTOSYS_DEVICE); | ||||||
|  | 	if (stmmac_rtc == NULL) { | ||||||
|  | 		pr_error("open rtc device failed\n"); | ||||||
|  | 		return -ENODEV; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	rtc_irq_register(stmmac_rtc, &stmmac_task); | ||||||
|  | 
 | ||||||
|  | 	/* Periodic mode is not supported */ | ||||||
|  | 	if ((rtc_irq_set_freq(stmmac_rtc, &stmmac_task, tm->freq) < 0)) { | ||||||
|  | 		pr_error("set periodic failed\n"); | ||||||
|  | 		rtc_irq_unregister(stmmac_rtc, &stmmac_task); | ||||||
|  | 		rtc_class_close(stmmac_rtc); | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	STMMAC_TIMER_MSG(CONFIG_RTC_HCTOSYS_DEVICE, tm->freq); | ||||||
|  | 
 | ||||||
|  | 	tm->timer_start = stmmac_rtc_start; | ||||||
|  | 	tm->timer_stop = stmmac_rtc_stop; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_close_ext_timer(void) | ||||||
|  | { | ||||||
|  | 	rtc_irq_set_state(stmmac_rtc, &stmmac_task, 0); | ||||||
|  | 	rtc_irq_unregister(stmmac_rtc, &stmmac_task); | ||||||
|  | 	rtc_class_close(stmmac_rtc); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #elif defined(CONFIG_STMMAC_TMU_TIMER) | ||||||
|  | #include <linux/clk.h> | ||||||
|  | #define TMU_CHANNEL "tmu2_clk" | ||||||
|  | static struct clk *timer_clock; | ||||||
|  | 
 | ||||||
|  | static void stmmac_tmu_start(unsigned int new_freq) | ||||||
|  | { | ||||||
|  | 	clk_set_rate(timer_clock, new_freq); | ||||||
|  | 	clk_enable(timer_clock); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | static void stmmac_tmu_stop(void) | ||||||
|  | { | ||||||
|  | 	clk_disable(timer_clock); | ||||||
|  | 	return; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_open_ext_timer(struct net_device *dev, struct stmmac_timer *tm) | ||||||
|  | { | ||||||
|  | 	timer_clock = clk_get(NULL, TMU_CHANNEL); | ||||||
|  | 
 | ||||||
|  | 	if (timer_clock == NULL) | ||||||
|  | 		return -1; | ||||||
|  | 
 | ||||||
|  | 	if (tmu2_register_user(stmmac_timer_handler, (void *)dev) < 0) { | ||||||
|  | 		timer_clock = NULL; | ||||||
|  | 		return -1; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	STMMAC_TIMER_MSG("TMU2", tm->freq); | ||||||
|  | 	tm->timer_start = stmmac_tmu_start; | ||||||
|  | 	tm->timer_stop = stmmac_tmu_stop; | ||||||
|  | 
 | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | int stmmac_close_ext_timer(void) | ||||||
|  | { | ||||||
|  | 	clk_disable(timer_clock); | ||||||
|  | 	tmu2_unregister_user(); | ||||||
|  | 	clk_put(timer_clock); | ||||||
|  | 	return 0; | ||||||
|  | } | ||||||
|  | #endif | ||||||
							
								
								
									
										41
									
								
								drivers/net/stmmac/stmmac_timer.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								drivers/net/stmmac/stmmac_timer.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,41 @@ | |||||||
|  | /*******************************************************************************
 | ||||||
|  |   STMMAC external timer Header File. | ||||||
|  | 
 | ||||||
|  |   Copyright (C) 2007-2009  STMicroelectronics Ltd | ||||||
|  | 
 | ||||||
|  |   This program is free software; you can redistribute it and/or modify it | ||||||
|  |   under the terms and conditions of the GNU General Public License, | ||||||
|  |   version 2, as published by the Free Software Foundation. | ||||||
|  | 
 | ||||||
|  |   This program is distributed in the hope 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., | ||||||
|  |   51 Franklin St - Fifth Floor, Boston, MA 02110-1301 USA. | ||||||
|  | 
 | ||||||
|  |   The full GNU General Public License is included in this distribution in | ||||||
|  |   the file called "COPYING". | ||||||
|  | 
 | ||||||
|  |   Author: Giuseppe Cavallaro <peppe.cavallaro@st.com> | ||||||
|  | *******************************************************************************/ | ||||||
|  | 
 | ||||||
|  | struct stmmac_timer { | ||||||
|  | 	void (*timer_start) (unsigned int new_freq); | ||||||
|  | 	void (*timer_stop) (void); | ||||||
|  | 	unsigned int freq; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /* Open the HW timer device and return 0 in case of success */ | ||||||
|  | int stmmac_open_ext_timer(struct net_device *dev, struct stmmac_timer *tm); | ||||||
|  | /* Stop the timer and release it */ | ||||||
|  | int stmmac_close_ext_timer(void); | ||||||
|  | /* Function used for scheduling task within the stmmac */ | ||||||
|  | void stmmac_schedule(struct net_device *dev); | ||||||
|  | 
 | ||||||
|  | #if defined(CONFIG_STMMAC_TMU_TIMER) | ||||||
|  | extern int tmu2_register_user(void *fnt, void *data); | ||||||
|  | extern void tmu2_unregister_user(void); | ||||||
|  | #endif | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Giuseppe Cavallaro
						Giuseppe Cavallaro