summaryrefslogtreecommitdiffstats
path: root/src/drivers
diff options
context:
space:
mode:
authorMichael Brown2017-06-20 13:10:14 +0200
committerMichael Brown2017-06-24 20:17:55 +0200
commit1e5c5a216351ffc18004e3e190f09fc04a110b52 (patch)
tree913a178c80b2a9f9ddece65bcf72dc3dbcada471 /src/drivers
parent[crypto] Expose pem_asn1() for use with non-image data (diff)
downloadipxe-1e5c5a216351ffc18004e3e190f09fc04a110b52.tar.gz
ipxe-1e5c5a216351ffc18004e3e190f09fc04a110b52.tar.xz
ipxe-1e5c5a216351ffc18004e3e190f09fc04a110b52.zip
[exanic] Add driver for Exablaze ExaNIC cards
Signed-off-by: Michael Brown <mcb30@ipxe.org>
Diffstat (limited to 'src/drivers')
-rw-r--r--src/drivers/net/exanic.c911
-rw-r--r--src/drivers/net/exanic.h257
2 files changed, 1168 insertions, 0 deletions
diff --git a/src/drivers/net/exanic.c b/src/drivers/net/exanic.c
new file mode 100644
index 00000000..62296927
--- /dev/null
+++ b/src/drivers/net/exanic.c
@@ -0,0 +1,911 @@
+/*
+ * Copyright (C) 2017 Michael Brown <mbrown@fensystems.co.uk>.
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License as
+ * published by the Free Software Foundation; either version 2 of the
+ * License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
+ * 02110-1301, USA.
+ *
+ * You can also choose to distribute this program under the terms of
+ * the Unmodified Binary Distribution Licence (as given in the file
+ * COPYING.UBDL), provided that you have satisfied its requirements.
+ */
+
+FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
+
+#include <stdint.h>
+#include <string.h>
+#include <strings.h>
+#include <unistd.h>
+#include <errno.h>
+#include <byteswap.h>
+#include <ipxe/netdevice.h>
+#include <ipxe/ethernet.h>
+#include <ipxe/if_ether.h>
+#include <ipxe/iobuf.h>
+#include <ipxe/malloc.h>
+#include <ipxe/umalloc.h>
+#include <ipxe/pci.h>
+#include "exanic.h"
+
+/** @file
+ *
+ * Exablaze ExaNIC driver
+ *
+ */
+
+/* Disambiguate the various error causes */
+#define EIO_ABORTED __einfo_error ( EINFO_EIO_ABORTED )
+#define EINFO_EIO_ABORTED \
+ __einfo_uniqify ( EINFO_EIO, 0x01, "Frame aborted" )
+#define EIO_CORRUPT __einfo_error ( EINFO_EIO_CORRUPT )
+#define EINFO_EIO_CORRUPT \
+ __einfo_uniqify ( EINFO_EIO, 0x02, "CRC incorrect" )
+#define EIO_HWOVFL __einfo_error ( EINFO_EIO_HWOVFL )
+#define EINFO_EIO_HWOVFL \
+ __einfo_uniqify ( EINFO_EIO, 0x03, "Hardware overflow" )
+#define EIO_STATUS( status ) \
+ EUNIQ ( EINFO_EIO, ( (status) & EXANIC_STATUS_ERROR_MASK ), \
+ EIO_ABORTED, EIO_CORRUPT, EIO_HWOVFL )
+
+/**
+ * Write DMA base address register
+ *
+ * @v addr DMA base address
+ * @v reg Register
+ */
+static void exanic_write_base ( physaddr_t addr, void *reg ) {
+ uint32_t lo;
+ uint32_t hi;
+
+ /* Write high and low registers, setting flags as appropriate */
+ lo = addr;
+ if ( sizeof ( physaddr_t ) > sizeof ( uint32_t ) ) {
+ /* 64-bit build; may be a 32-bit or 64-bit address */
+ hi = ( ( ( uint64_t ) addr ) >> 32 );
+ if ( ! hi )
+ lo |= EXANIC_DMA_32_BIT;
+ } else {
+ /* 32-bit build; always a 32-bit address */
+ hi = 0;
+ lo |= EXANIC_DMA_32_BIT;
+ }
+ writel ( hi, ( reg + 0 ) );
+ writel ( lo, ( reg + 4 ) );
+}
+
+/**
+ * Clear DMA base address register
+ *
+ * @v reg Register
+ */
+static inline void exanic_clear_base ( void *reg ) {
+
+ /* Clear both high and low registers */
+ writel ( 0, ( reg + 0 ) );
+ writel ( 0, ( reg + 4 ) );
+}
+
+/******************************************************************************
+ *
+ * Device reset
+ *
+ ******************************************************************************
+ */
+
+/**
+ * Reset hardware
+ *
+ * @v exanic ExaNIC device
+ */
+static void exanic_reset ( struct exanic *exanic ) {
+ void *port_regs;
+ unsigned int i;
+
+ /* Disable all possible ports */
+ for ( i = 0 ; i < EXANIC_MAX_PORTS ; i++ ) {
+ port_regs = ( exanic->regs + EXANIC_PORT_REGS ( i ) );
+ writel ( 0, ( port_regs + EXANIC_PORT_ENABLE ) );
+ writel ( 0, ( port_regs + EXANIC_PORT_IRQ ) );
+ exanic_clear_base ( port_regs + EXANIC_PORT_RX_BASE );
+ }
+
+ /* Disable transmit feedback */
+ exanic_clear_base ( exanic->regs + EXANIC_TXF_BASE );
+}
+
+/******************************************************************************
+ *
+ * MAC address
+ *
+ ******************************************************************************
+ */
+
+/**
+ * Read I2C line status
+ *
+ * @v basher Bit-bashing interface
+ * @v bit_id Bit number
+ * @ret zero Input is a logic 0
+ * @ret non-zero Input is a logic 1
+ */
+static int exanic_i2c_read_bit ( struct bit_basher *basher,
+ unsigned int bit_id ) {
+ struct exanic *exanic =
+ container_of ( basher, struct exanic, basher.basher );
+ unsigned int shift;
+ uint32_t i2c;
+
+ /* Identify bit */
+ assert ( bit_id == I2C_BIT_SDA );
+ shift = exanic->i2cfg.getsda;
+
+ /* Read I2C register */
+ DBG_DISABLE ( DBGLVL_IO );
+ i2c = readl ( exanic->regs + EXANIC_I2C );
+ DBG_ENABLE ( DBGLVL_IO );
+ return ( ( i2c >> shift ) & 1 );
+}
+
+/**
+ * Write I2C line status
+ *
+ * @v basher Bit-bashing interface
+ * @v bit_id Bit number
+ * @v data Value to write
+ */
+static void exanic_i2c_write_bit ( struct bit_basher *basher,
+ unsigned int bit_id, unsigned long data ) {
+ struct exanic *exanic =
+ container_of ( basher, struct exanic, basher.basher );
+ unsigned int shift;
+ uint32_t mask;
+ uint32_t i2c;
+
+ /* Identify shift */
+ assert ( ( bit_id == I2C_BIT_SCL ) || ( bit_id == I2C_BIT_SDA ) );
+ shift = ( ( bit_id == I2C_BIT_SCL ) ?
+ exanic->i2cfg.setscl : exanic->i2cfg.setsda );
+ mask = ( 1UL << shift );
+
+ /* Modify I2C register */
+ DBG_DISABLE ( DBGLVL_IO );
+ i2c = readl ( exanic->regs + EXANIC_I2C );
+ i2c &= ~mask;
+ if ( ! data )
+ i2c |= mask;
+ writel ( i2c, ( exanic->regs + EXANIC_I2C ) );
+ DBG_ENABLE ( DBGLVL_IO );
+}
+
+/** I2C bit-bashing interface operations */
+static struct bit_basher_operations exanic_i2c_basher_ops = {
+ .read = exanic_i2c_read_bit,
+ .write = exanic_i2c_write_bit,
+};
+
+/** Possible I2C bus configurations */
+static struct exanic_i2c_config exanic_i2cfgs[] = {
+ /* X2/X10 */
+ { .setscl = 7, .setsda = 4, .getsda = 12 },
+ /* X4 */
+ { .setscl = 7, .setsda = 5, .getsda = 13 },
+};
+
+/**
+ * Initialise EEPROM
+ *
+ * @v exanic ExaNIC device
+ * @v i2cfg I2C bus configuration
+ * @ret rc Return status code
+ */
+static int exanic_try_init_eeprom ( struct exanic *exanic,
+ struct exanic_i2c_config *i2cfg ) {
+ int rc;
+
+ /* Configure I2C bus */
+ memcpy ( &exanic->i2cfg, i2cfg, sizeof ( exanic->i2cfg ) );
+
+ /* Initialise I2C bus */
+ if ( ( rc = init_i2c_bit_basher ( &exanic->basher,
+ &exanic_i2c_basher_ops ) ) != 0 ) {
+ DBGC2 ( exanic, "EXANIC %p found no I2C bus via %d/%d/%d\n",
+ exanic, exanic->i2cfg.setscl,
+ exanic->i2cfg.setsda, exanic->i2cfg.getsda );
+ return rc;
+ }
+
+ /* Check for EEPROM presence */
+ init_i2c_eeprom ( &exanic->eeprom, EXANIC_EEPROM_ADDRESS );
+ if ( ( rc = i2c_check_presence ( &exanic->basher.i2c,
+ &exanic->eeprom ) ) != 0 ) {
+ DBGC2 ( exanic, "EXANIC %p found no EEPROM via %d/%d/%d\n",
+ exanic, exanic->i2cfg.setscl,
+ exanic->i2cfg.setsda, exanic->i2cfg.getsda );
+ return rc;
+ }
+
+ DBGC ( exanic, "EXANIC %p found EEPROM via %d/%d/%d\n",
+ exanic, exanic->i2cfg.setscl,
+ exanic->i2cfg.setsda, exanic->i2cfg.getsda );
+ return 0;
+}
+
+/**
+ * Initialise EEPROM
+ *
+ * @v exanic ExaNIC device
+ * @ret rc Return status code
+ */
+static int exanic_init_eeprom ( struct exanic *exanic ) {
+ struct exanic_i2c_config *i2cfg;
+ unsigned int i;
+ int rc;
+
+ /* Try all possible bus configurations */
+ for ( i = 0 ; i < ( sizeof ( exanic_i2cfgs ) /
+ sizeof ( exanic_i2cfgs[0] ) ) ; i++ ) {
+ i2cfg = &exanic_i2cfgs[i];
+ if ( ( rc = exanic_try_init_eeprom ( exanic, i2cfg ) ) == 0 )
+ return 0;
+ }
+
+ DBGC ( exanic, "EXANIC %p found no EEPROM\n", exanic );
+ return -ENODEV;
+}
+
+/**
+ * Fetch base MAC address
+ *
+ * @v exanic ExaNIC device
+ * @ret rc Return status code
+ */
+static int exanic_fetch_mac ( struct exanic *exanic ) {
+ struct i2c_interface *i2c = &exanic->basher.i2c;
+ int rc;
+
+ /* Initialise EEPROM */
+ if ( ( rc = exanic_init_eeprom ( exanic ) ) != 0 )
+ return rc;
+
+ /* Fetch base MAC address */
+ if ( ( rc = i2c->read ( i2c, &exanic->eeprom, 0, exanic->mac,
+ sizeof ( exanic->mac ) ) ) != 0 ) {
+ DBGC ( exanic, "EXANIC %p could not read MAC address: %s\n",
+ exanic, strerror ( rc ) );
+ return rc;
+ }
+
+ return 0;
+}
+
+/******************************************************************************
+ *
+ * Link state
+ *
+ ******************************************************************************
+ */
+
+/**
+ * Check link state
+ *
+ * @v netdev Network device
+ */
+static void exanic_check_link ( struct net_device *netdev ) {
+ struct exanic_port *port = netdev->priv;
+ uint32_t status;
+ uint32_t speed;
+
+ /* Report port status changes */
+ status = readl ( port->regs + EXANIC_PORT_STATUS );
+ speed = readl ( port->regs + EXANIC_PORT_SPEED );
+ if ( status != port->status ) {
+ DBGC ( port, "EXANIC %s port status %#08x speed %dMbps\n",
+ netdev->name, status, speed );
+ if ( status & EXANIC_PORT_STATUS_LINK ) {
+ netdev_link_up ( netdev );
+ } else {
+ netdev_link_down ( netdev );
+ }
+ port->status = status;
+ }
+}
+
+/**
+ * Check link state periodically
+ *
+ * @v retry Link state check timer
+ * @v over Failure indicator
+ */
+static void exanic_expired ( struct retry_timer *timer, int over __unused ) {
+ struct exanic_port *port =
+ container_of ( timer, struct exanic_port, timer );
+ struct net_device *netdev = port->netdev;
+ static const uint32_t speeds[] = {
+ 100, 1000, 10000, 40000, 100000,
+ };
+ unsigned int index;
+
+ /* Restart timer */
+ start_timer_fixed ( timer, EXANIC_LINK_INTERVAL );
+
+ /* Check link state */
+ exanic_check_link ( netdev );
+
+ /* Do nothing further if link is already up */
+ if ( netdev_link_ok ( netdev ) )
+ return;
+
+ /* Do nothing further unless we have a valid list of supported speeds */
+ if ( ! port->speeds )
+ return;
+
+ /* Autonegotiation is not supported; try manually selecting
+ * the next supported link speed.
+ */
+ do {
+ if ( ! port->speed )
+ port->speed = ( 8 * sizeof ( port->speeds ) );
+ port->speed--;
+ } while ( ! ( ( 1UL << port->speed ) & port->speeds ) );
+ index = ( port->speed - ( ffs ( EXANIC_CAPS_SPEED_MASK ) - 1 ) );
+ assert ( index < ( sizeof ( speeds ) / sizeof ( speeds[0] ) ) );
+
+ /* Attempt the selected speed */
+ DBGC ( netdev, "EXANIC %s attempting %dMbps\n",
+ netdev->name, speeds[index] );
+ writel ( speeds[index], ( port->regs + EXANIC_PORT_SPEED ) );
+}
+
+/******************************************************************************
+ *
+ * Network device interface
+ *
+ ******************************************************************************
+ */
+
+/**
+ * Open network device
+ *
+ * @v netdev Network device
+ * @ret rc Return status code
+ */
+static int exanic_open ( struct net_device *netdev ) {
+ struct exanic_port *port = netdev->priv;
+ struct exanic_tx_chunk *tx;
+ unsigned int i;
+
+ /* Reset transmit region contents */
+ for ( i = 0 ; i < port->tx_count ; i++ ) {
+ tx = ( port->tx + ( i * sizeof ( *tx ) ) );
+ writew ( port->txf_slot, &tx->desc.txf_slot );
+ writeb ( EXANIC_TYPE_RAW, &tx->desc.type );
+ writeb ( 0, &tx->desc.flags );
+ writew ( 0, &tx->pad );
+ }
+
+ /* Reset receive region contents */
+ memset_user ( port->rx, 0, 0xff, EXANIC_RX_LEN );
+
+ /* Reset transmit feedback region */
+ *(port->txf) = 0;
+
+ /* Reset counters */
+ port->tx_prod = 0;
+ port->tx_cons = 0;
+ port->rx_cons = 0;
+
+ /* Map receive region */
+ exanic_write_base ( phys_to_bus ( user_to_phys ( port->rx, 0 ) ),
+ ( port->regs + EXANIC_PORT_RX_BASE ) );
+
+ /* Enable promiscuous mode */
+ writel ( EXANIC_PORT_FLAGS_PROMISC,
+ ( port->regs + EXANIC_PORT_FLAGS ) );
+
+ /* Reset to default speed and clear cached status */
+ writel ( port->default_speed, ( port->regs + EXANIC_PORT_SPEED ) );
+ port->speed = 0;
+ port->status = 0;
+
+ /* Enable port */
+ wmb();
+ writel ( EXANIC_PORT_ENABLE_ENABLED,
+ ( port->regs + EXANIC_PORT_ENABLE ) );
+
+ /* Start link state timer */
+ start_timer_fixed ( &port->timer, EXANIC_LINK_INTERVAL );
+
+ return 0;
+}
+
+/**
+ * Close network device
+ *
+ * @v netdev Network device
+ */
+static void exanic_close ( struct net_device *netdev ) {
+ struct exanic_port *port = netdev->priv;
+
+ /* Stop link state timer */
+ stop_timer ( &port->timer );
+
+ /* Disable port */
+ writel ( 0, ( port->regs + EXANIC_PORT_ENABLE ) );
+ wmb();
+
+ /* Clear receive region */
+ exanic_clear_base ( port->regs + EXANIC_PORT_RX_BASE );
+
+ /* Discard any in-progress receive */
+ if ( port->rx_iobuf ) {
+ netdev_rx_err ( netdev, port->rx_iobuf, -ECANCELED );
+ port->rx_iobuf = NULL;
+ }
+}
+
+/**
+ * Transmit packet
+ *
+ * @v netdev Network device
+ * @v iobuf I/O buffer
+ * @ret rc Return status code
+ */
+static int exanic_transmit ( struct net_device *netdev,
+ struct io_buffer *iobuf ) {
+ struct exanic_port *port = netdev->priv;
+ struct exanic_tx_chunk *tx;
+ unsigned int tx_fill;
+ unsigned int tx_index;
+ size_t offset;
+ size_t len;
+ uint8_t *src;
+ uint8_t *dst;
+
+ /* Sanity check */
+ len = iob_len ( iobuf );
+ if ( len > sizeof ( tx->data ) ) {
+ DBGC ( port, "EXANIC %s transmit too large\n", netdev->name );
+ return -ENOTSUP;
+ }
+
+ /* Get next transmit descriptor */
+ tx_fill = ( port->tx_prod - port->tx_cons );
+ if ( tx_fill >= port->tx_count ) {
+ DBGC ( port, "EXANIC %s out of transmit descriptors\n",
+ netdev->name );
+ return -ENOBUFS;
+ }
+ tx_index = ( port->tx_prod & ( port->tx_count - 1 ) );
+ offset = ( tx_index * sizeof ( *tx ) );
+ tx = ( port->tx + offset );
+ DBGC2 ( port, "EXANIC %s TX %04x at [%05zx,%05zx)\n",
+ netdev->name, port->tx_prod, ( port->tx_offset + offset ),
+ ( port->tx_offset + offset +
+ offsetof ( typeof ( *tx ), data ) + len ) );
+ port->tx_prod++;
+
+ /* Populate transmit descriptor */
+ writew ( port->tx_prod, &tx->desc.txf_id );
+ writew ( ( sizeof ( tx->pad ) + len ), &tx->desc.len );
+
+ /* Copy data to transmit region. There is no DMA on the
+ * transmit data path.
+ */
+ src = iobuf->data;
+ dst = tx->data;
+ while ( len-- )
+ writeb ( *(src++), dst++ );
+
+ /* Send transmit command */
+ wmb();
+ writel ( ( port->tx_offset + offset ),
+ ( port->regs + EXANIC_PORT_TX_COMMAND ) );
+
+ return 0;
+}
+
+/**
+ * Poll for completed packets
+ *
+ * @v netdev Network device
+ */
+static void exanic_poll_tx ( struct net_device *netdev ) {
+ struct exanic_port *port = netdev->priv;
+
+ /* Report any completed packets */
+ while ( port->tx_cons != *(port->txf) ) {
+ DBGC2 ( port, "EXANIC %s TX %04x complete\n",
+ netdev->name, port->tx_cons );
+ netdev_tx_complete_next ( netdev );
+ port->tx_cons++;
+ }
+}
+
+/**
+ * Poll for received packets
+ *
+ * @v netdev Network device
+ */
+static void exanic_poll_rx ( struct net_device *netdev ) {
+ struct exanic_port *port = netdev->priv;
+ struct exanic_rx_chunk *rx;
+ struct exanic_rx_descriptor desc;
+ uint8_t current;
+ uint8_t previous;
+ size_t offset;
+ size_t len;
+
+ for ( ; ; port->rx_cons++ ) {
+
+ /* Fetch descriptor */
+ offset = ( ( port->rx_cons * sizeof ( *rx ) ) % EXANIC_RX_LEN );
+ copy_from_user ( &desc, port->rx,
+ ( offset + offsetof ( typeof ( *rx ), desc ) ),
+ sizeof ( desc ) );
+
+ /* Calculate generation */
+ current = ( port->rx_cons / ( EXANIC_RX_LEN / sizeof ( *rx ) ));
+ previous = ( current - 1 );
+
+ /* Do nothing if no chunk is ready */
+ if ( desc.generation == previous )
+ break;
+
+ /* Allocate I/O buffer if needed */
+ if ( ! port->rx_iobuf ) {
+ port->rx_iobuf = alloc_iob ( EXANIC_MAX_RX_LEN );
+ if ( ! port->rx_iobuf ) {
+ /* Wait for next poll */
+ break;
+ }
+ port->rx_rc = 0;
+ }
+
+ /* Calculate chunk length */
+ len = ( desc.len ? desc.len : sizeof ( rx->data ) );
+
+ /* Append data to I/O buffer */
+ if ( len <= iob_tailroom ( port->rx_iobuf ) ) {
+ copy_from_user ( iob_put ( port->rx_iobuf, len ),
+ port->rx,
+ ( offset + offsetof ( typeof ( *rx ),
+ data ) ), len );
+ } else {
+ DBGC ( port, "EXANIC %s RX too large\n",
+ netdev->name );
+ port->rx_rc = -ERANGE;
+ }
+
+ /* Check for overrun */
+ rmb();
+ copy_from_user ( &desc.generation, port->rx,
+ ( offset + offsetof ( typeof ( *rx ),
+ desc.generation ) ),
+ sizeof ( desc.generation ) );
+ if ( desc.generation != current ) {
+ DBGC ( port, "EXANIC %s RX overrun\n", netdev->name );
+ port->rx_rc = -ENOBUFS;
+ continue;
+ }
+
+ /* Wait for end of packet */
+ if ( ! desc.len )
+ continue;
+
+ /* Check for receive errors */
+ if ( desc.status & EXANIC_STATUS_ERROR_MASK ) {
+ port->rx_rc = -EIO_STATUS ( desc.status );
+ DBGC ( port, "EXANIC %s RX %04x error: %s\n",
+ netdev->name, port->rx_cons,
+ strerror ( port->rx_rc ) );
+ } else {
+ DBGC2 ( port, "EXANIC %s RX %04x\n",
+ netdev->name, port->rx_cons );
+ }
+
+ /* Hand off to network stack */
+ if ( port->rx_rc ) {
+ netdev_rx_err ( netdev, port->rx_iobuf, port->rx_rc );
+ } else {
+ iob_unput ( port->rx_iobuf, 4 /* strip CRC */ );
+ netdev_rx ( netdev, port->rx_iobuf );
+ }
+ port->rx_iobuf = NULL;
+ }
+}
+
+/**
+ * Poll for completed and received packets
+ *
+ * @v netdev Network device
+ */
+static void exanic_poll ( struct net_device *netdev ) {
+
+ /* Poll for completed packets */
+ exanic_poll_tx ( netdev );
+
+ /* Poll for received packets */
+ exanic_poll_rx ( netdev );
+}
+
+/** ExaNIC network device operations */
+static struct net_device_operations exanic_operations = {
+ .open = exanic_open,
+ .close = exanic_close,
+ .transmit = exanic_transmit,
+ .poll = exanic_poll,
+};
+
+/******************************************************************************
+ *
+ * PCI interface
+ *
+ ******************************************************************************
+ */
+
+/**
+ * Probe port
+ *
+ * @v exanic ExaNIC device
+ * @v dev Parent device
+ * @v index Port number
+ * @ret rc Return status code
+ */
+static int exanic_probe_port ( struct exanic *exanic, struct device *dev,
+ unsigned int index ) {
+ struct net_device *netdev;
+ struct exanic_port *port;
+ void *port_regs;
+ uint32_t status;
+ size_t tx_len;
+ int rc;
+
+ /* Do nothing if port is not physically present */
+ port_regs = ( exanic->regs + EXANIC_PORT_REGS ( index ) );
+ status = readl ( port_regs + EXANIC_PORT_STATUS );
+ tx_len = readl ( port_regs + EXANIC_PORT_TX_LEN );
+ if ( ( status & EXANIC_PORT_STATUS_ABSENT ) || ( tx_len == 0 ) ) {
+ rc = 0;
+ goto absent;
+ }
+
+ /* Allocate network device */
+ netdev = alloc_etherdev ( sizeof ( *port ) );
+ if ( ! netdev ) {
+ rc = -ENOMEM;
+ goto err_alloc_netdev;
+ }
+ netdev_init ( netdev, &exanic_operations );
+ netdev->dev = dev;
+ port = netdev->priv;
+ memset ( port, 0, sizeof ( *port ) );
+ exanic->port[index] = port;
+ port->netdev = netdev;
+ port->regs = port_regs;
+ timer_init ( &port->timer, exanic_expired, &netdev->refcnt );
+
+ /* Identify transmit region */
+ port->tx_offset = readl ( port->regs + EXANIC_PORT_TX_OFFSET );
+ if ( tx_len > EXANIC_MAX_TX_LEN )
+ tx_len = EXANIC_MAX_TX_LEN;
+ assert ( ! ( tx_len & ( tx_len - 1 ) ) );
+ port->tx = ( exanic->tx + port->tx_offset );
+ port->tx_count = ( tx_len / sizeof ( struct exanic_tx_chunk ) );
+
+ /* Identify transmit feedback region */
+ port->txf_slot = EXANIC_TXF_SLOT ( index );
+ port->txf = ( exanic->txf +
+ ( port->txf_slot * sizeof ( *(port->txf) ) ) );
+
+ /* Allocate receive region (via umalloc()) */
+ port->rx = umalloc ( EXANIC_RX_LEN );
+ if ( ! port->rx ) {
+ rc = -ENOMEM;
+ goto err_alloc_rx;
+ }
+
+ /* Set MAC address */
+ memcpy ( netdev->hw_addr, exanic->mac, ETH_ALEN );
+ netdev->hw_addr[ ETH_ALEN - 1 ] += index;
+
+ /* Record default link speed and supported speeds */
+ port->default_speed = readl ( port->regs + EXANIC_PORT_SPEED );
+ port->speeds = ( exanic->caps & EXANIC_CAPS_SPEED_MASK );
+
+ /* Register network device */
+ if ( ( rc = register_netdev ( netdev ) ) != 0 )
+ goto err_register_netdev;
+ DBGC ( port, "EXANIC %s port %d TX [%#05zx,%#05zx) TXF %#02x RX "
+ "[%#lx,%#lx)\n", netdev->name, index, port->tx_offset,
+ ( port->tx_offset + tx_len ), port->txf_slot,
+ user_to_phys ( port->rx, 0 ),
+ user_to_phys ( port->rx, EXANIC_RX_LEN ) );
+
+ /* Set initial link state */
+ exanic_check_link ( netdev );
+
+ return 0;
+
+ unregister_netdev ( netdev );
+ err_register_netdev:
+ ufree ( port->rx );
+ err_alloc_rx:
+ netdev_nullify ( netdev );
+ netdev_put ( netdev );
+ err_alloc_netdev:
+ absent:
+ return rc;
+}
+
+/**
+ * Probe port
+ *
+ * @v exanic ExaNIC device
+ * @v index Port number
+ */
+static void exanic_remove_port ( struct exanic *exanic, unsigned int index ) {
+ struct exanic_port *port;
+
+ /* Do nothing if port is not physically present */
+ port = exanic->port[index];
+ if ( ! port )
+ return;
+
+ /* Unregister network device */
+ unregister_netdev ( port->netdev );
+
+ /* Free receive region */
+ ufree ( port->rx );
+
+ /* Free network device */
+ netdev_nullify ( port->netdev );
+ netdev_put ( port->netdev );
+}
+
+/**
+ * Probe PCI device
+ *
+ * @v pci PCI device
+ * @ret rc Return status code
+ */
+static int exanic_probe ( struct pci_device *pci ) {
+ struct exanic *exanic;
+ unsigned long regs_bar_start;
+ unsigned long tx_bar_start;
+ size_t tx_bar_len;
+ int i;
+ int rc;
+
+ /* Allocate and initialise structure */
+ exanic = zalloc ( sizeof ( *exanic ) );
+ if ( ! exanic ) {
+ rc = -ENOMEM;
+ goto err_alloc;
+ }
+ pci_set_drvdata ( pci, exanic );
+
+ /* Fix up PCI device */
+ adjust_pci_device ( pci );
+
+ /* Map registers */
+ regs_bar_start = pci_bar_start ( pci, EXANIC_REGS_BAR );
+ exanic->regs = ioremap ( regs_bar_start, EXANIC_REGS_LEN );
+ if ( ! exanic->regs ) {
+ rc = -ENODEV;
+ goto err_ioremap_regs;
+ }
+
+ /* Reset device */
+ exanic_reset ( exanic );
+
+ /* Read capabilities */
+ exanic->caps = readl ( exanic->regs + EXANIC_CAPS );
+
+ /* Fetch base MAC address */
+ if ( ( rc = exanic_fetch_mac ( exanic ) ) != 0 )
+ goto err_fetch_mac;
+ DBGC ( exanic, "EXANIC %p capabilities %#08x base MAC %s\n",
+ exanic, exanic->caps, eth_ntoa ( exanic->mac ) );
+
+ /* Map transmit region */
+ tx_bar_start = pci_bar_start ( pci, EXANIC_TX_BAR );
+ tx_bar_len = pci_bar_size ( pci, EXANIC_TX_BAR );
+ exanic->tx = ioremap ( tx_bar_start, tx_bar_len );
+ if ( ! exanic->tx ) {
+ rc = -ENODEV;
+ goto err_ioremap_tx;
+ }
+
+ /* Allocate transmit feedback region (shared between all ports) */
+ exanic->txf = malloc_dma ( EXANIC_TXF_LEN, EXANIC_ALIGN );
+ if ( ! exanic->txf ) {
+ rc = -ENOMEM;
+ goto err_alloc_txf;
+ }
+ memset ( exanic->txf, 0, EXANIC_TXF_LEN );
+ exanic_write_base ( virt_to_bus ( exanic->txf ),
+ ( exanic->regs + EXANIC_TXF_BASE ) );
+
+ /* Allocate and initialise per-port network devices */
+ for ( i = 0 ; i < EXANIC_MAX_PORTS ; i++ ) {
+ if ( ( rc = exanic_probe_port ( exanic, &pci->dev, i ) ) != 0 )
+ goto err_probe_port;
+ }
+
+ return 0;
+
+ i = EXANIC_MAX_PORTS;
+ err_probe_port:
+ for ( i-- ; i >= 0 ; i-- )
+ exanic_remove_port ( exanic, i );
+ exanic_reset ( exanic );
+ free_dma ( exanic->txf, EXANIC_TXF_LEN );
+ err_alloc_txf:
+ iounmap ( exanic->tx );
+ err_ioremap_tx:
+ iounmap ( exanic->regs );
+ err_fetch_mac:
+ err_ioremap_regs:
+ free ( exanic );
+ err_alloc:
+ return rc;
+}
+
+/**
+ * Remove PCI device
+ *
+ * @v pci PCI device
+ */
+static void exanic_remove ( struct pci_device *pci ) {
+ struct exanic *exanic = pci_get_drvdata ( pci );
+ unsigned int i;
+
+ /* Remove all ports */
+ for ( i = 0 ; i < EXANIC_MAX_PORTS ; i++ )
+ exanic_remove_port ( exanic, i );
+
+ /* Reset device */
+ exanic_reset ( exanic );
+
+ /* Free transmit feedback region */
+ free_dma ( exanic->txf, EXANIC_TXF_LEN );
+
+ /* Unmap transmit region */
+ iounmap ( exanic->tx );
+
+ /* Unmap registers */
+ iounmap ( exanic->regs );
+
+ /* Free device */
+ free ( exanic );
+}
+
+/** ExaNIC PCI device IDs */
+static struct pci_device_id exanic_ids[] = {
+ PCI_ROM ( 0x10ee, 0x2b00, "exanic-old", "ExaNIC (old)", 0 ),
+ PCI_ROM ( 0x1ce4, 0x0001, "exanic-x4", "ExaNIC X4", 0 ),
+ PCI_ROM ( 0x1ce4, 0x0002, "exanic-x2", "ExaNIC X2", 0 ),
+ PCI_ROM ( 0x1ce4, 0x0003, "exanic-x10", "ExaNIC X10", 0 ),
+ PCI_ROM ( 0x1ce4, 0x0004, "exanic-x10gm", "ExaNIC X10 GM", 0 ),
+ PCI_ROM ( 0x1ce4, 0x0005, "exanic-x40", "ExaNIC X40", 0 ),
+ PCI_ROM ( 0x1ce4, 0x0006, "exanic-x10hpt", "ExaNIC X10 HPT", 0 ),
+};
+
+/** ExaNIC PCI driver */
+struct pci_driver exanic_driver __pci_driver = {
+ .ids = exanic_ids,
+ .id_count = ( sizeof ( exanic_ids ) / sizeof ( exanic_ids[0] ) ),
+ .probe = exanic_probe,
+ .remove = exanic_remove,
+};
diff --git a/src/drivers/net/exanic.h b/src/drivers/net/exanic.h
new file mode 100644
index 00000000..fd9f5b8c
--- /dev/null
+++ b/src/drivers/net/exanic.h
@@ -0,0 +1,257 @@
+#ifndef _EXANIC_H
+#define _EXANIC_H
+
+/** @file
+ *
+ * Exablaze ExaNIC driver
+ *
+ */
+
+FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
+
+#include <stdint.h>
+#include <ipxe/pci.h>
+#include <ipxe/ethernet.h>
+#include <ipxe/uaccess.h>
+#include <ipxe/retry.h>
+#include <ipxe/i2c.h>
+#include <ipxe/bitbash.h>
+
+/** Maximum number of ports */
+#define EXANIC_MAX_PORTS 8
+
+/** Register BAR */
+#define EXANIC_REGS_BAR PCI_BASE_ADDRESS_0
+
+/** Transmit region BAR */
+#define EXANIC_TX_BAR PCI_BASE_ADDRESS_2
+
+/** Alignment for DMA regions */
+#define EXANIC_ALIGN 0x1000
+
+/** Flag for 32-bit DMA addresses */
+#define EXANIC_DMA_32_BIT 0x00000001UL
+
+/** Register set length */
+#define EXANIC_REGS_LEN 0x2000
+
+/** Transmit feedback region length */
+#define EXANIC_TXF_LEN 0x1000
+
+/** Transmit feedback slot
+ *
+ * This is a policy decision.
+ */
+#define EXANIC_TXF_SLOT( index ) ( 0x40 * (index) )
+
+/** Receive region length */
+#define EXANIC_RX_LEN 0x200000
+
+/** Transmit feedback base address register */
+#define EXANIC_TXF_BASE 0x0014
+
+/** Capabilities register */
+#define EXANIC_CAPS 0x0038
+#define EXANIC_CAPS_100M 0x01000000UL /**< 100Mbps supported */
+#define EXANIC_CAPS_1G 0x02000000UL /**< 1Gbps supported */
+#define EXANIC_CAPS_10G 0x04000000UL /**< 10Gbps supported */
+#define EXANIC_CAPS_40G 0x08000000UL /**< 40Gbps supported */
+#define EXANIC_CAPS_100G 0x10000000UL /**< 100Gbps supported */
+#define EXANIC_CAPS_SPEED_MASK 0x1f000000UL /**< Supported speeds mask */
+
+/** I2C GPIO register */
+#define EXANIC_I2C 0x012c
+
+/** Port register offset */
+#define EXANIC_PORT_REGS( index ) ( 0x0200 + ( 0x40 * (index) ) )
+
+/** Port enable register */
+#define EXANIC_PORT_ENABLE 0x0000
+#define EXANIC_PORT_ENABLE_ENABLED 0x00000001UL /**< Port is enabled */
+
+/** Port speed register */
+#define EXANIC_PORT_SPEED 0x0004
+
+/** Port status register */
+#define EXANIC_PORT_STATUS 0x0008
+#define EXANIC_PORT_STATUS_LINK 0x00000008UL /**< Link is up */
+#define EXANIC_PORT_STATUS_ABSENT 0x80000000UL /**< Port is not present */
+
+/** Port MAC address (second half) register */
+#define EXANIC_PORT_MAC 0x000c
+
+/** Port flags register */
+#define EXANIC_PORT_FLAGS 0x0010
+#define EXANIC_PORT_FLAGS_PROMISC 0x00000001UL /**< Promiscuous mode */
+
+/** Port receive chunk base address register */
+#define EXANIC_PORT_RX_BASE 0x0014
+
+/** Port transmit command register */
+#define EXANIC_PORT_TX_COMMAND 0x0020
+
+/** Port transmit region offset register */
+#define EXANIC_PORT_TX_OFFSET 0x0024
+
+/** Port transmit region length register */
+#define EXANIC_PORT_TX_LEN 0x0028
+
+/** Port MAC address (first half) register */
+#define EXANIC_PORT_OUI 0x0030
+
+/** Port interrupt configuration register */
+#define EXANIC_PORT_IRQ 0x0034
+
+/** An ExaNIC transmit chunk descriptor */
+struct exanic_tx_descriptor {
+ /** Feedback ID */
+ uint16_t txf_id;
+ /** Feedback slot */
+ uint16_t txf_slot;
+ /** Payload length (including padding */
+ uint16_t len;
+ /** Payload type */
+ uint8_t type;
+ /** Flags */
+ uint8_t flags;
+} __attribute__ (( packed ));
+
+/** An ExaNIC transmit chunk */
+struct exanic_tx_chunk {
+ /** Descriptor */
+ struct exanic_tx_descriptor desc;
+ /** Padding */
+ uint8_t pad[2];
+ /** Payload data */
+ uint8_t data[2038];
+} __attribute__ (( packed ));
+
+/** Raw Ethernet frame type */
+#define EXANIC_TYPE_RAW 0x01
+
+/** An ExaNIC receive chunk descriptor */
+struct exanic_rx_descriptor {
+ /** Timestamp */
+ uint32_t timestamp;
+ /** Status (valid only on final chunk) */
+ uint8_t status;
+ /** Length (zero except on the final chunk) */
+ uint8_t len;
+ /** Filter number */
+ uint8_t filter;
+ /** Generation */
+ uint8_t generation;
+} __attribute__ (( packed ));
+
+/** An ExaNIC receive chunk */
+struct exanic_rx_chunk {
+ /** Payload data */
+ uint8_t data[120];
+ /** Descriptor */
+ struct exanic_rx_descriptor desc;
+} __attribute__ (( packed ));
+
+/** Receive status error mask */
+#define EXANIC_STATUS_ERROR_MASK 0x0f
+
+/** An ExaNIC I2C bus configuration */
+struct exanic_i2c_config {
+ /** GPIO bit for pulling SCL low */
+ uint8_t setscl;
+ /** GPIO bit for pulling SDA low */
+ uint8_t setsda;
+ /** GPIO bit for reading SDA */
+ uint8_t getsda;
+};
+
+/** EEPROM address */
+#define EXANIC_EEPROM_ADDRESS 0x50
+
+/** An ExaNIC port */
+struct exanic_port {
+ /** Network device */
+ struct net_device *netdev;
+ /** Port registers */
+ void *regs;
+
+ /** Transmit region offset */
+ size_t tx_offset;
+ /** Transmit region */
+ void *tx;
+ /** Number of transmit descriptors */
+ uint16_t tx_count;
+ /** Transmit producer counter */
+ uint16_t tx_prod;
+ /** Transmit consumer counter */
+ uint16_t tx_cons;
+ /** Transmit feedback slot */
+ uint16_t txf_slot;
+ /** Transmit feedback region */
+ uint16_t *txf;
+
+ /** Receive region */
+ userptr_t rx;
+ /** Receive consumer counter */
+ unsigned int rx_cons;
+ /** Receive I/O buffer (if any) */
+ struct io_buffer *rx_iobuf;
+ /** Receive status */
+ int rx_rc;
+
+ /** Port status */
+ uint32_t status;
+ /** Default link speed (as raw register value) */
+ uint32_t default_speed;
+ /** Speed capability bitmask */
+ uint32_t speeds;
+ /** Current attempted link speed (as a capability bit index) */
+ unsigned int speed;
+ /** Port status check timer */
+ struct retry_timer timer;
+};
+
+/** An ExaNIC */
+struct exanic {
+ /** Registers */
+ void *regs;
+ /** Transmit region */
+ void *tx;
+ /** Transmit feedback region */
+ void *txf;
+
+ /** I2C bus configuration */
+ struct exanic_i2c_config i2cfg;
+ /** I2C bit-bashing interface */
+ struct i2c_bit_basher basher;
+ /** I2C serial EEPROM */
+ struct i2c_device eeprom;
+
+ /** Capabilities */
+ uint32_t caps;
+ /** Base MAC address */
+ uint8_t mac[ETH_ALEN];
+
+ /** Ports */
+ struct exanic_port *port[EXANIC_MAX_PORTS];
+};
+
+/** Maximum used length of transmit region
+ *
+ * This is a policy decision to avoid overflowing the 16-bit transmit
+ * producer and consumer counters.
+ */
+#define EXANIC_MAX_TX_LEN ( 256 * sizeof ( struct exanic_tx_chunk ) )
+
+/** Maximum length of received packet
+ *
+ * This is a policy decision.
+ */
+#define EXANIC_MAX_RX_LEN ( ETH_FRAME_LEN + 4 /* VLAN */ + 4 /* CRC */ )
+
+/** Interval between link state checks
+ *
+ * This is a policy decision.
+ */
+#define EXANIC_LINK_INTERVAL ( 1 * TICKS_PER_SEC )
+
+#endif /* _EXANIC_H */