summaryrefslogblamecommitdiffstats
path: root/src/drivers/net/icplus.h
blob: 35fa422adf394a80c454f5cdfdc539e774f25a5a (plain) (tree)













































































































































































































                                                                                    
#ifndef _ICPLUS_H
#define _ICPLUS_H

/** @file
 *
 * IC+ network driver
 *
 */

#include <ipxe/nvs.h>
#include <ipxe/mii_bit.h>

FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );

/** BAR size */
#define ICP_BAR_SIZE 0x200

/** Alignment requirement */
#define ICP_ALIGN 0x8

/** Base address low register offset */
#define ICP_BASE_LO 0x0

/** Base address high register offset */
#define ICP_BASE_HI 0x4

/** ASIC control register (double word) */
#define ICP_ASICCTRL 0x30
#define ICP_ASICCTRL_PHYSPEED1000	0x00000040UL	/**< PHY speed 1000 */
#define ICP_ASICCTRL_GLOBALRESET	0x00010000UL	/**< Global reset */
#define ICP_ASICCTRL_DMA		0x00080000UL	/**< DMA */
#define ICP_ASICCTRL_FIFO		0x00100000UL	/**< FIFO */
#define ICP_ASICCTRL_NETWORK		0x00200000UL	/**< Network */
#define ICP_ASICCTRL_HOST		0x00400000UL	/**< Host */
#define ICP_ASICCTRL_AUTOINIT		0x00800000UL	/**< Auto init */
#define ICP_ASICCTRL_RESETBUSY		0x04000000UL	/**< Reset busy */

/** Maximum time to wait for reset */
#define ICP_RESET_MAX_WAIT_MS 1000

/** DMA control register (word/double word) */
#define ICP_DMACTRL 0x00
#define ICP_DMACTRL_RXPOLLNOW		0x0010		/**< Receive poll now */
#define ICP_DMACTRL_TXPOLLNOW 		0x1000		/**< Transmit poll now */

/** EEPROM control register (word) */
#define ICP_EEPROMCTRL 0x4a
#define ICP_EEPROMCTRL_ADDRESS( x )	( (x) << 0 )	/**< Address */
#define ICP_EEPROMCTRL_OPCODE( x )	( (x) << 8 )	/**< Opcode */
#define ICP_EEPROMCTRL_OPCODE_READ \
	ICP_EEPROMCTRL_OPCODE ( 2 )			/**< Read register */
#define ICP_EEPROMCTRL_BUSY		0x8000		/**< EEPROM busy */

/** Maximum time to wait for reading EEPROM */
#define ICP_EEPROM_MAX_WAIT_MS 1000

/** EEPROM word length */
#define ICP_EEPROM_WORD_LEN_LOG2 1

/** Minimum EEPROM size, in words */
#define ICP_EEPROM_MIN_SIZE_WORDS 0x20

/** Address of MAC address within EEPROM */
#define ICP_EEPROM_MAC 0x10

/** EEPROM data register (word) */
#define ICP_EEPROMDATA 0x48

/** Interupt status register (word) */
#define ICP_INTSTATUS 0x5e
#define ICP_INTSTATUS_TXCOMPLETE	0x0004		/**< TX complete */
#define ICP_INTSTATUS_LINKEVENT		0x0100		/**< Link event */
#define ICP_INTSTATUS_RXDMACOMPLETE	0x0400		/**< RX DMA complete */

/** MAC control register (double word) */
#define ICP_MACCTRL 0x6c
#define ICP_MACCTRL_DUPLEX		0x00000020UL	/**< Duplex select */
#define ICP_MACCTRL_TXENABLE		0x01000000UL	/**< TX enable */
#define ICP_MACCTRL_TXDISABLE		0x02000000UL	/**< TX disable */
#define ICP_MACCTRL_RXENABLE		0x08000000UL	/**< RX enable */
#define ICP_MACCTRL_RXDISABLE		0x10000000UL	/**< RX disable */

/** PHY control register (byte) */
#define ICP_PHYCTRL 0x76
#define ICP_PHYCTRL_MGMTCLK		0x01		/**< Management clock */
#define ICP_PHYCTRL_MGMTDATA		0x02		/**< Management data */
#define ICP_PHYCTRL_MGMTDIR		0x04		/**< Management direction */
#define ICP_PHYCTRL_LINKSPEED		0xc0		/**< Link speed */

/** Receive mode register (word) */
#define ICP_RXMODE 0x88
#define ICP_RXMODE_UNICAST		0x0001		/**< Receive unicast */
#define ICP_RXMODE_MULTICAST		0x0002		/**< Receice multicast */
#define ICP_RXMODE_BROADCAST		0x0004		/**< Receive broadcast */
#define ICP_RXMODE_ALLFRAMES		0x0008		/**< Receive all frames */

/** List pointer receive register */
#define ICP_RFDLISTPTR 0x1c

/** List pointer transmit register */
#define ICP_TFDLISTPTR 0x10

/** Transmit status register */
#define ICP_TXSTATUS 0x60
#define ICP_TXSTATUS_ERROR		0x00000001UL	/**< TX error */

/** Data fragment */
union icplus_fragment {
	/** Address of data */
	uint64_t address;
	/** Length */
	struct {
		/** Reserved */
		uint8_t reserved[6];
		/** Length of data */
		uint16_t len;
	};
};

/** Transmit or receive descriptor */
struct icplus_descriptor {
	/** Address of next descriptor */
	uint64_t next;
	/** Actual length */
	uint16_t len;
	/** Flags */
	uint8_t flags;
	/** Control */
	uint8_t control;
	/** VLAN */
	uint16_t vlan;
	/** Reserved */
	uint16_t reserved_a;
	/** Data buffer */
	union icplus_fragment data;
	/** Reserved */
	uint8_t reserved_b[8];
};

/** Descriptor complete */
#define ICP_DONE 0x80

/** Transmit alignment disabled */
#define ICP_TX_UNALIGN 0x01

/** Request transmit completion */
#define ICP_TX_INDICATE 0x40

/** Sole transmit fragment */
#define ICP_TX_SOLE_FRAG 0x01

/** Recieve frame overrun error */
#define ICP_RX_ERR_OVERRUN 0x01

/** Receive runt frame error */
#define ICP_RX_ERR_RUNT 0x02

/** Receive alignment error */
#define ICP_RX_ERR_ALIGN 0x04

/** Receive FCS error */
#define ICP_RX_ERR_FCS 0x08

/** Receive oversized frame error */
#define ICP_RX_ERR_OVERSIZED 0x10

/** Recieve length error */
#define ICP_RX_ERR_LEN 0x20

/** Descriptor ring */
struct icplus_ring {
	/** Producer counter */
	unsigned int prod;
	/** Consumer counter */
	unsigned int cons;
	/** Ring entries */
	struct icplus_descriptor *entry;
	/* List pointer register */
	unsigned int listptr;
};

/** Number of descriptors */
#define ICP_NUM_DESC 4

/** Maximum receive packet length */
#define ICP_RX_MAX_LEN ETH_FRAME_LEN

/** An IC+ network card */
struct icplus_nic {
	/** Registers */
	void *regs;
	/** EEPROM */
	struct nvs_device eeprom;
	/** MII bit bashing interface */
	struct mii_bit_basher miibit;
	/** MII device */
	struct mii_device mii;
	/** Transmit descriptor ring */
	struct icplus_ring tx;
	/** Receive descriptor ring */
	struct icplus_ring rx;
	/** Receive I/O buffers */
	struct io_buffer *rx_iobuf[ICP_NUM_DESC];
};

#endif /* _ICPLUS_H */