2011-03-14 13:13:55 +03:00
|
|
|
/*
|
|
|
|
* QEMU model of Xilinx AXI-Ethernet.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2011 Edgar E. Iglesias.
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2016-01-26 21:17:11 +03:00
|
|
|
#include "qemu/osdep.h"
|
2019-08-12 08:23:51 +03:00
|
|
|
#include "hw/hw.h"
|
2013-02-04 18:40:22 +04:00
|
|
|
#include "hw/sysbus.h"
|
include/qemu/osdep.h: Don't include qapi/error.h
Commit 57cb38b included qapi/error.h into qemu/osdep.h to get the
Error typedef. Since then, we've moved to include qemu/osdep.h
everywhere. Its file comment explains: "To avoid getting into
possible circular include dependencies, this file should not include
any other QEMU headers, with the exceptions of config-host.h,
compiler.h, os-posix.h and os-win32.h, all of which are doing a
similar job to this file and are under similar constraints."
qapi/error.h doesn't do a similar job, and it doesn't adhere to
similar constraints: it includes qapi-types.h. That's in excess of
100KiB of crap most .c files don't actually need.
Add the typedef to qemu/typedefs.h, and include that instead of
qapi/error.h. Include qapi/error.h in .c files that need it and don't
get it now. Include qapi-types.h in qom/object.h for uint16List.
Update scripts/clean-includes accordingly. Update it further to match
reality: replace config.h by config-target.h, add sysemu/os-posix.h,
sysemu/os-win32.h. Update the list of includes in the qemu/osdep.h
comment quoted above similarly.
This reduces the number of objects depending on qapi/error.h from "all
of them" to less than a third. Unfortunately, the number depending on
qapi-types.h shrinks only a little. More work is needed for that one.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
[Fix compilation without the spice devel packages. - Paolo]
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2016-03-14 11:01:28 +03:00
|
|
|
#include "qapi/error.h"
|
2012-12-17 21:20:00 +04:00
|
|
|
#include "qemu/log.h"
|
2019-05-23 17:35:07 +03:00
|
|
|
#include "qemu/module.h"
|
2012-10-24 10:43:34 +04:00
|
|
|
#include "net/net.h"
|
2011-03-14 13:13:55 +03:00
|
|
|
#include "net/checksum.h"
|
|
|
|
|
2019-08-12 08:23:48 +03:00
|
|
|
#include "hw/hw.h"
|
2019-08-12 08:23:42 +03:00
|
|
|
#include "hw/irq.h"
|
2019-08-12 08:23:51 +03:00
|
|
|
#include "hw/qdev-properties.h"
|
2013-02-04 18:40:22 +04:00
|
|
|
#include "hw/stream.h"
|
2011-03-14 13:13:55 +03:00
|
|
|
|
|
|
|
#define DPHY(x)
|
|
|
|
|
2013-04-16 04:19:27 +04:00
|
|
|
#define TYPE_XILINX_AXI_ENET "xlnx.axi-ethernet"
|
2013-04-16 04:25:18 +04:00
|
|
|
#define TYPE_XILINX_AXI_ENET_DATA_STREAM "xilinx-axienet-data-stream"
|
2013-04-16 04:28:35 +04:00
|
|
|
#define TYPE_XILINX_AXI_ENET_CONTROL_STREAM "xilinx-axienet-control-stream"
|
2013-04-16 04:19:27 +04:00
|
|
|
|
|
|
|
#define XILINX_AXI_ENET(obj) \
|
|
|
|
OBJECT_CHECK(XilinxAXIEnet, (obj), TYPE_XILINX_AXI_ENET)
|
|
|
|
|
2013-04-16 04:25:18 +04:00
|
|
|
#define XILINX_AXI_ENET_DATA_STREAM(obj) \
|
|
|
|
OBJECT_CHECK(XilinxAXIEnetStreamSlave, (obj),\
|
|
|
|
TYPE_XILINX_AXI_ENET_DATA_STREAM)
|
|
|
|
|
2013-04-16 04:28:35 +04:00
|
|
|
#define XILINX_AXI_ENET_CONTROL_STREAM(obj) \
|
|
|
|
OBJECT_CHECK(XilinxAXIEnetStreamSlave, (obj),\
|
|
|
|
TYPE_XILINX_AXI_ENET_CONTROL_STREAM)
|
|
|
|
|
2011-03-14 13:13:55 +03:00
|
|
|
/* Advertisement control register. */
|
|
|
|
#define ADVERTISE_10HALF 0x0020 /* Try for 10mbps half-duplex */
|
|
|
|
#define ADVERTISE_10FULL 0x0040 /* Try for 10mbps full-duplex */
|
|
|
|
#define ADVERTISE_100HALF 0x0080 /* Try for 100mbps half-duplex */
|
|
|
|
#define ADVERTISE_100FULL 0x0100 /* Try for 100mbps full-duplex */
|
|
|
|
|
2013-04-16 04:28:35 +04:00
|
|
|
#define CONTROL_PAYLOAD_WORDS 5
|
|
|
|
#define CONTROL_PAYLOAD_SIZE (CONTROL_PAYLOAD_WORDS * (sizeof(uint32_t)))
|
|
|
|
|
2011-03-14 13:13:55 +03:00
|
|
|
struct PHY {
|
|
|
|
uint32_t regs[32];
|
|
|
|
|
|
|
|
int link;
|
|
|
|
|
|
|
|
unsigned int (*read)(struct PHY *phy, unsigned int req);
|
|
|
|
void (*write)(struct PHY *phy, unsigned int req,
|
|
|
|
unsigned int data);
|
|
|
|
};
|
|
|
|
|
|
|
|
static unsigned int tdk_read(struct PHY *phy, unsigned int req)
|
|
|
|
{
|
|
|
|
int regnum;
|
|
|
|
unsigned r = 0;
|
|
|
|
|
|
|
|
regnum = req & 0x1f;
|
|
|
|
|
|
|
|
switch (regnum) {
|
|
|
|
case 1:
|
|
|
|
if (!phy->link) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* MR1. */
|
|
|
|
/* Speeds and modes. */
|
|
|
|
r |= (1 << 13) | (1 << 14);
|
|
|
|
r |= (1 << 11) | (1 << 12);
|
|
|
|
r |= (1 << 5); /* Autoneg complete. */
|
|
|
|
r |= (1 << 3); /* Autoneg able. */
|
|
|
|
r |= (1 << 2); /* link. */
|
|
|
|
r |= (1 << 1); /* link. */
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
/* Link partner ability.
|
|
|
|
We are kind; always agree with whatever best mode
|
|
|
|
the guest advertises. */
|
|
|
|
r = 1 << 14; /* Success. */
|
|
|
|
/* Copy advertised modes. */
|
|
|
|
r |= phy->regs[4] & (15 << 5);
|
|
|
|
/* Autoneg support. */
|
|
|
|
r |= 1;
|
|
|
|
break;
|
|
|
|
case 17:
|
2014-04-17 21:32:42 +04:00
|
|
|
/* Marvell PHY on many xilinx boards. */
|
2011-03-14 13:13:55 +03:00
|
|
|
r = 0x8000; /* 1000Mb */
|
|
|
|
break;
|
|
|
|
case 18:
|
|
|
|
{
|
|
|
|
/* Diagnostics reg. */
|
|
|
|
int duplex = 0;
|
|
|
|
int speed_100 = 0;
|
|
|
|
|
|
|
|
if (!phy->link) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Are we advertising 100 half or 100 duplex ? */
|
|
|
|
speed_100 = !!(phy->regs[4] & ADVERTISE_100HALF);
|
|
|
|
speed_100 |= !!(phy->regs[4] & ADVERTISE_100FULL);
|
|
|
|
|
|
|
|
/* Are we advertising 10 duplex or 100 duplex ? */
|
|
|
|
duplex = !!(phy->regs[4] & ADVERTISE_100FULL);
|
|
|
|
duplex |= !!(phy->regs[4] & ADVERTISE_10FULL);
|
|
|
|
r = (speed_100 << 10) | (duplex << 11);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
r = phy->regs[regnum];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
DPHY(qemu_log("\n%s %x = reg[%d]\n", __func__, r, regnum));
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
tdk_write(struct PHY *phy, unsigned int req, unsigned int data)
|
|
|
|
{
|
|
|
|
int regnum;
|
|
|
|
|
|
|
|
regnum = req & 0x1f;
|
|
|
|
DPHY(qemu_log("%s reg[%d] = %x\n", __func__, regnum, data));
|
|
|
|
switch (regnum) {
|
|
|
|
default:
|
|
|
|
phy->regs[regnum] = data;
|
|
|
|
break;
|
|
|
|
}
|
2014-04-09 05:52:39 +04:00
|
|
|
|
2020-05-06 11:25:05 +03:00
|
|
|
/* Unconditionally clear regs[BMCR][BMCR_RESET] and auto-neg */
|
|
|
|
phy->regs[0] &= ~0x8200;
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
tdk_init(struct PHY *phy)
|
|
|
|
{
|
|
|
|
phy->regs[0] = 0x3100;
|
|
|
|
/* PHY Id. */
|
|
|
|
phy->regs[2] = 0x0300;
|
|
|
|
phy->regs[3] = 0xe400;
|
|
|
|
/* Autonegotiation advertisement reg. */
|
|
|
|
phy->regs[4] = 0x01E1;
|
|
|
|
phy->link = 1;
|
|
|
|
|
|
|
|
phy->read = tdk_read;
|
|
|
|
phy->write = tdk_write;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct MDIOBus {
|
|
|
|
/* bus. */
|
|
|
|
int mdc;
|
|
|
|
int mdio;
|
|
|
|
|
|
|
|
/* decoder. */
|
|
|
|
enum {
|
|
|
|
PREAMBLE,
|
|
|
|
SOF,
|
|
|
|
OPC,
|
|
|
|
ADDR,
|
|
|
|
REQ,
|
|
|
|
TURNAROUND,
|
|
|
|
DATA
|
|
|
|
} state;
|
|
|
|
unsigned int drive;
|
|
|
|
|
|
|
|
unsigned int cnt;
|
|
|
|
unsigned int addr;
|
|
|
|
unsigned int opc;
|
|
|
|
unsigned int req;
|
|
|
|
unsigned int data;
|
|
|
|
|
|
|
|
struct PHY *devs[32];
|
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
mdio_attach(struct MDIOBus *bus, struct PHY *phy, unsigned int addr)
|
|
|
|
{
|
|
|
|
bus->devs[addr & 0x1f] = phy;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef USE_THIS_DEAD_CODE
|
|
|
|
static void
|
|
|
|
mdio_detach(struct MDIOBus *bus, struct PHY *phy, unsigned int addr)
|
|
|
|
{
|
|
|
|
bus->devs[addr & 0x1f] = NULL;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static uint16_t mdio_read_req(struct MDIOBus *bus, unsigned int addr,
|
|
|
|
unsigned int reg)
|
|
|
|
{
|
|
|
|
struct PHY *phy;
|
|
|
|
uint16_t data;
|
|
|
|
|
|
|
|
phy = bus->devs[addr];
|
|
|
|
if (phy && phy->read) {
|
|
|
|
data = phy->read(phy, reg);
|
|
|
|
} else {
|
|
|
|
data = 0xffff;
|
|
|
|
}
|
|
|
|
DPHY(qemu_log("%s addr=%d reg=%d data=%x\n", __func__, addr, reg, data));
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mdio_write_req(struct MDIOBus *bus, unsigned int addr,
|
|
|
|
unsigned int reg, uint16_t data)
|
|
|
|
{
|
|
|
|
struct PHY *phy;
|
|
|
|
|
|
|
|
DPHY(qemu_log("%s addr=%d reg=%d data=%x\n", __func__, addr, reg, data));
|
|
|
|
phy = bus->devs[addr];
|
|
|
|
if (phy && phy->write) {
|
|
|
|
phy->write(phy, reg, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DENET(x)
|
|
|
|
|
|
|
|
#define R_RAF (0x000 / 4)
|
|
|
|
enum {
|
|
|
|
RAF_MCAST_REJ = (1 << 1),
|
|
|
|
RAF_BCAST_REJ = (1 << 2),
|
|
|
|
RAF_EMCF_EN = (1 << 12),
|
|
|
|
RAF_NEWFUNC_EN = (1 << 11)
|
|
|
|
};
|
|
|
|
|
|
|
|
#define R_IS (0x00C / 4)
|
|
|
|
enum {
|
|
|
|
IS_HARD_ACCESS_COMPLETE = 1,
|
|
|
|
IS_AUTONEG = (1 << 1),
|
|
|
|
IS_RX_COMPLETE = (1 << 2),
|
|
|
|
IS_RX_REJECT = (1 << 3),
|
|
|
|
IS_TX_COMPLETE = (1 << 5),
|
|
|
|
IS_RX_DCM_LOCK = (1 << 6),
|
|
|
|
IS_MGM_RDY = (1 << 7),
|
|
|
|
IS_PHY_RST_DONE = (1 << 8),
|
|
|
|
};
|
|
|
|
|
|
|
|
#define R_IP (0x010 / 4)
|
|
|
|
#define R_IE (0x014 / 4)
|
|
|
|
#define R_UAWL (0x020 / 4)
|
|
|
|
#define R_UAWU (0x024 / 4)
|
|
|
|
#define R_PPST (0x030 / 4)
|
|
|
|
enum {
|
|
|
|
PPST_LINKSTATUS = (1 << 0),
|
|
|
|
PPST_PHY_LINKSTATUS = (1 << 7),
|
|
|
|
};
|
|
|
|
|
|
|
|
#define R_STATS_RX_BYTESL (0x200 / 4)
|
|
|
|
#define R_STATS_RX_BYTESH (0x204 / 4)
|
|
|
|
#define R_STATS_TX_BYTESL (0x208 / 4)
|
|
|
|
#define R_STATS_TX_BYTESH (0x20C / 4)
|
|
|
|
#define R_STATS_RXL (0x290 / 4)
|
|
|
|
#define R_STATS_RXH (0x294 / 4)
|
|
|
|
#define R_STATS_RX_BCASTL (0x2a0 / 4)
|
|
|
|
#define R_STATS_RX_BCASTH (0x2a4 / 4)
|
|
|
|
#define R_STATS_RX_MCASTL (0x2a8 / 4)
|
|
|
|
#define R_STATS_RX_MCASTH (0x2ac / 4)
|
|
|
|
|
|
|
|
#define R_RCW0 (0x400 / 4)
|
|
|
|
#define R_RCW1 (0x404 / 4)
|
|
|
|
enum {
|
|
|
|
RCW1_VLAN = (1 << 27),
|
|
|
|
RCW1_RX = (1 << 28),
|
|
|
|
RCW1_FCS = (1 << 29),
|
|
|
|
RCW1_JUM = (1 << 30),
|
|
|
|
RCW1_RST = (1 << 31),
|
|
|
|
};
|
|
|
|
|
|
|
|
#define R_TC (0x408 / 4)
|
|
|
|
enum {
|
|
|
|
TC_VLAN = (1 << 27),
|
|
|
|
TC_TX = (1 << 28),
|
|
|
|
TC_FCS = (1 << 29),
|
|
|
|
TC_JUM = (1 << 30),
|
|
|
|
TC_RST = (1 << 31),
|
|
|
|
};
|
|
|
|
|
|
|
|
#define R_EMMC (0x410 / 4)
|
|
|
|
enum {
|
|
|
|
EMMC_LINKSPEED_10MB = (0 << 30),
|
|
|
|
EMMC_LINKSPEED_100MB = (1 << 30),
|
|
|
|
EMMC_LINKSPEED_1000MB = (2 << 30),
|
|
|
|
};
|
|
|
|
|
|
|
|
#define R_PHYC (0x414 / 4)
|
|
|
|
|
|
|
|
#define R_MC (0x500 / 4)
|
|
|
|
#define MC_EN (1 << 6)
|
|
|
|
|
|
|
|
#define R_MCR (0x504 / 4)
|
|
|
|
#define R_MWD (0x508 / 4)
|
|
|
|
#define R_MRD (0x50c / 4)
|
|
|
|
#define R_MIS (0x600 / 4)
|
|
|
|
#define R_MIP (0x620 / 4)
|
|
|
|
#define R_MIE (0x640 / 4)
|
|
|
|
#define R_MIC (0x640 / 4)
|
|
|
|
|
|
|
|
#define R_UAW0 (0x700 / 4)
|
|
|
|
#define R_UAW1 (0x704 / 4)
|
|
|
|
#define R_FMI (0x708 / 4)
|
|
|
|
#define R_AF0 (0x710 / 4)
|
|
|
|
#define R_AF1 (0x714 / 4)
|
|
|
|
#define R_MAX (0x34 / 4)
|
|
|
|
|
|
|
|
/* Indirect registers. */
|
|
|
|
struct TEMAC {
|
|
|
|
struct MDIOBus mdio_bus;
|
|
|
|
struct PHY phy;
|
|
|
|
|
|
|
|
void *parent;
|
|
|
|
};
|
|
|
|
|
2013-04-16 04:25:18 +04:00
|
|
|
typedef struct XilinxAXIEnetStreamSlave XilinxAXIEnetStreamSlave;
|
2013-04-16 04:18:47 +04:00
|
|
|
typedef struct XilinxAXIEnet XilinxAXIEnet;
|
|
|
|
|
2013-04-16 04:25:18 +04:00
|
|
|
struct XilinxAXIEnetStreamSlave {
|
|
|
|
Object parent;
|
|
|
|
|
|
|
|
struct XilinxAXIEnet *enet;
|
|
|
|
} ;
|
|
|
|
|
2011-03-14 13:13:55 +03:00
|
|
|
struct XilinxAXIEnet {
|
|
|
|
SysBusDevice busdev;
|
2011-11-15 14:26:54 +04:00
|
|
|
MemoryRegion iomem;
|
2011-03-14 13:13:55 +03:00
|
|
|
qemu_irq irq;
|
2013-04-16 04:28:35 +04:00
|
|
|
StreamSlave *tx_data_dev;
|
|
|
|
StreamSlave *tx_control_dev;
|
2013-04-16 04:25:18 +04:00
|
|
|
XilinxAXIEnetStreamSlave rx_data_dev;
|
2013-04-16 04:28:35 +04:00
|
|
|
XilinxAXIEnetStreamSlave rx_control_dev;
|
2011-03-14 13:13:55 +03:00
|
|
|
NICState *nic;
|
|
|
|
NICConf conf;
|
|
|
|
|
|
|
|
|
|
|
|
uint32_t c_rxmem;
|
|
|
|
uint32_t c_txmem;
|
|
|
|
uint32_t c_phyaddr;
|
|
|
|
|
|
|
|
struct TEMAC TEMAC;
|
|
|
|
|
|
|
|
/* MII regs. */
|
|
|
|
union {
|
|
|
|
uint32_t regs[4];
|
|
|
|
struct {
|
|
|
|
uint32_t mc;
|
|
|
|
uint32_t mcr;
|
|
|
|
uint32_t mwd;
|
|
|
|
uint32_t mrd;
|
|
|
|
};
|
|
|
|
} mii;
|
|
|
|
|
|
|
|
struct {
|
|
|
|
uint64_t rx_bytes;
|
|
|
|
uint64_t tx_bytes;
|
|
|
|
|
|
|
|
uint64_t rx;
|
|
|
|
uint64_t rx_bcast;
|
|
|
|
uint64_t rx_mcast;
|
|
|
|
} stats;
|
|
|
|
|
|
|
|
/* Receive configuration words. */
|
|
|
|
uint32_t rcw[2];
|
|
|
|
/* Transmit config. */
|
|
|
|
uint32_t tc;
|
|
|
|
uint32_t emmc;
|
|
|
|
uint32_t phyc;
|
|
|
|
|
|
|
|
/* Unicast Address Word. */
|
|
|
|
uint32_t uaw[2];
|
|
|
|
/* Unicast address filter used with extended mcast. */
|
|
|
|
uint32_t ext_uaw[2];
|
|
|
|
uint32_t fmi;
|
|
|
|
|
|
|
|
uint32_t regs[R_MAX];
|
|
|
|
|
|
|
|
/* Multicast filter addrs. */
|
|
|
|
uint32_t maddr[4][2];
|
|
|
|
/* 32K x 1 lookup filter. */
|
|
|
|
uint32_t ext_mtable[1024];
|
|
|
|
|
2013-04-16 04:28:35 +04:00
|
|
|
uint32_t hdr[CONTROL_PAYLOAD_WORDS];
|
2011-03-14 13:13:55 +03:00
|
|
|
|
2020-05-06 11:25:10 +03:00
|
|
|
uint8_t *txmem;
|
|
|
|
uint32_t txpos;
|
|
|
|
|
2011-03-14 13:13:55 +03:00
|
|
|
uint8_t *rxmem;
|
2013-04-16 04:27:55 +04:00
|
|
|
uint32_t rxsize;
|
|
|
|
uint32_t rxpos;
|
2013-04-16 04:28:35 +04:00
|
|
|
|
|
|
|
uint8_t rxapp[CONTROL_PAYLOAD_SIZE];
|
|
|
|
uint32_t rxappsize;
|
2015-07-15 13:19:13 +03:00
|
|
|
|
|
|
|
/* Whether axienet_eth_rx_notify should flush incoming queue. */
|
|
|
|
bool need_flush;
|
2011-03-14 13:13:55 +03:00
|
|
|
};
|
|
|
|
|
2013-04-16 04:18:47 +04:00
|
|
|
static void axienet_rx_reset(XilinxAXIEnet *s)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
|
|
|
s->rcw[1] = RCW1_JUM | RCW1_FCS | RCW1_RX | RCW1_VLAN;
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:18:47 +04:00
|
|
|
static void axienet_tx_reset(XilinxAXIEnet *s)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
|
|
|
s->tc = TC_JUM | TC_TX | TC_VLAN;
|
2020-05-06 11:25:10 +03:00
|
|
|
s->txpos = 0;
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
|
2013-04-16 04:18:47 +04:00
|
|
|
static inline int axienet_rx_resetting(XilinxAXIEnet *s)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
|
|
|
return s->rcw[1] & RCW1_RST;
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:18:47 +04:00
|
|
|
static inline int axienet_rx_enabled(XilinxAXIEnet *s)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
|
|
|
return s->rcw[1] & RCW1_RX;
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:18:47 +04:00
|
|
|
static inline int axienet_extmcf_enabled(XilinxAXIEnet *s)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
|
|
|
return !!(s->regs[R_RAF] & RAF_EMCF_EN);
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:18:47 +04:00
|
|
|
static inline int axienet_newfunc_enabled(XilinxAXIEnet *s)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
|
|
|
return !!(s->regs[R_RAF] & RAF_NEWFUNC_EN);
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:20:06 +04:00
|
|
|
static void xilinx_axienet_reset(DeviceState *d)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
2013-04-16 04:20:06 +04:00
|
|
|
XilinxAXIEnet *s = XILINX_AXI_ENET(d);
|
|
|
|
|
2011-03-14 13:13:55 +03:00
|
|
|
axienet_rx_reset(s);
|
|
|
|
axienet_tx_reset(s);
|
|
|
|
|
|
|
|
s->regs[R_PPST] = PPST_LINKSTATUS | PPST_PHY_LINKSTATUS;
|
|
|
|
s->regs[R_IS] = IS_AUTONEG | IS_RX_DCM_LOCK | IS_MGM_RDY | IS_PHY_RST_DONE;
|
|
|
|
|
|
|
|
s->emmc = EMMC_LINKSPEED_100MB;
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:18:47 +04:00
|
|
|
static void enet_update_irq(XilinxAXIEnet *s)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
|
|
|
s->regs[R_IP] = s->regs[R_IS] & s->regs[R_IE];
|
|
|
|
qemu_set_irq(s->irq, !!s->regs[R_IP]);
|
|
|
|
}
|
|
|
|
|
2012-10-23 14:30:10 +04:00
|
|
|
static uint64_t enet_read(void *opaque, hwaddr addr, unsigned size)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
2013-04-16 04:18:47 +04:00
|
|
|
XilinxAXIEnet *s = opaque;
|
2011-03-14 13:13:55 +03:00
|
|
|
uint32_t r = 0;
|
|
|
|
addr >>= 2;
|
|
|
|
|
|
|
|
switch (addr) {
|
|
|
|
case R_RCW0:
|
|
|
|
case R_RCW1:
|
|
|
|
r = s->rcw[addr & 1];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_TC:
|
|
|
|
r = s->tc;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_EMMC:
|
|
|
|
r = s->emmc;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_PHYC:
|
|
|
|
r = s->phyc;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_MCR:
|
|
|
|
r = s->mii.regs[addr & 3] | (1 << 7); /* Always ready. */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_STATS_RX_BYTESL:
|
|
|
|
case R_STATS_RX_BYTESH:
|
|
|
|
r = s->stats.rx_bytes >> (32 * (addr & 1));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_STATS_TX_BYTESL:
|
|
|
|
case R_STATS_TX_BYTESH:
|
|
|
|
r = s->stats.tx_bytes >> (32 * (addr & 1));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_STATS_RXL:
|
|
|
|
case R_STATS_RXH:
|
|
|
|
r = s->stats.rx >> (32 * (addr & 1));
|
|
|
|
break;
|
|
|
|
case R_STATS_RX_BCASTL:
|
|
|
|
case R_STATS_RX_BCASTH:
|
|
|
|
r = s->stats.rx_bcast >> (32 * (addr & 1));
|
|
|
|
break;
|
|
|
|
case R_STATS_RX_MCASTL:
|
|
|
|
case R_STATS_RX_MCASTH:
|
|
|
|
r = s->stats.rx_mcast >> (32 * (addr & 1));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_MC:
|
|
|
|
case R_MWD:
|
|
|
|
case R_MRD:
|
|
|
|
r = s->mii.regs[addr & 3];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_UAW0:
|
|
|
|
case R_UAW1:
|
|
|
|
r = s->uaw[addr & 1];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_UAWU:
|
|
|
|
case R_UAWL:
|
|
|
|
r = s->ext_uaw[addr & 1];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_FMI:
|
|
|
|
r = s->fmi;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_AF0:
|
|
|
|
case R_AF1:
|
|
|
|
r = s->maddr[s->fmi & 3][addr & 1];
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 0x8000 ... 0x83ff:
|
|
|
|
r = s->ext_mtable[addr - 0x8000];
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (addr < ARRAY_SIZE(s->regs)) {
|
|
|
|
r = s->regs[addr];
|
|
|
|
}
|
|
|
|
DENET(qemu_log("%s addr=" TARGET_FMT_plx " v=%x\n",
|
|
|
|
__func__, addr * 4, r));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
|
2012-10-23 14:30:10 +04:00
|
|
|
static void enet_write(void *opaque, hwaddr addr,
|
2011-11-15 14:26:54 +04:00
|
|
|
uint64_t value, unsigned size)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
2013-04-16 04:18:47 +04:00
|
|
|
XilinxAXIEnet *s = opaque;
|
2011-03-14 13:13:55 +03:00
|
|
|
struct TEMAC *t = &s->TEMAC;
|
|
|
|
|
|
|
|
addr >>= 2;
|
|
|
|
switch (addr) {
|
|
|
|
case R_RCW0:
|
|
|
|
case R_RCW1:
|
|
|
|
s->rcw[addr & 1] = value;
|
|
|
|
if ((addr & 1) && value & RCW1_RST) {
|
|
|
|
axienet_rx_reset(s);
|
2013-04-03 08:04:09 +04:00
|
|
|
} else {
|
|
|
|
qemu_flush_queued_packets(qemu_get_queue(s->nic));
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_TC:
|
|
|
|
s->tc = value;
|
|
|
|
if (value & TC_RST) {
|
|
|
|
axienet_tx_reset(s);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_EMMC:
|
|
|
|
s->emmc = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_PHYC:
|
|
|
|
s->phyc = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_MC:
|
2013-06-10 00:56:20 +04:00
|
|
|
value &= ((1 << 7) - 1);
|
2011-03-14 13:13:55 +03:00
|
|
|
|
|
|
|
/* Enable the MII. */
|
|
|
|
if (value & MC_EN) {
|
|
|
|
unsigned int miiclkdiv = value & ((1 << 6) - 1);
|
|
|
|
if (!miiclkdiv) {
|
|
|
|
qemu_log("AXIENET: MDIO enabled but MDIOCLK is zero!\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s->mii.mc = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_MCR: {
|
|
|
|
unsigned int phyaddr = (value >> 24) & 0x1f;
|
|
|
|
unsigned int regaddr = (value >> 16) & 0x1f;
|
|
|
|
unsigned int op = (value >> 14) & 3;
|
|
|
|
unsigned int initiate = (value >> 11) & 1;
|
|
|
|
|
|
|
|
if (initiate) {
|
|
|
|
if (op == 1) {
|
|
|
|
mdio_write_req(&t->mdio_bus, phyaddr, regaddr, s->mii.mwd);
|
|
|
|
} else if (op == 2) {
|
|
|
|
s->mii.mrd = mdio_read_req(&t->mdio_bus, phyaddr, regaddr);
|
|
|
|
} else {
|
|
|
|
qemu_log("AXIENET: invalid MDIOBus OP=%d\n", op);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s->mii.mcr = value;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case R_MWD:
|
|
|
|
case R_MRD:
|
|
|
|
s->mii.regs[addr & 3] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case R_UAW0:
|
|
|
|
case R_UAW1:
|
|
|
|
s->uaw[addr & 1] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_UAWL:
|
|
|
|
case R_UAWU:
|
|
|
|
s->ext_uaw[addr & 1] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_FMI:
|
|
|
|
s->fmi = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case R_AF0:
|
|
|
|
case R_AF1:
|
|
|
|
s->maddr[s->fmi & 3][addr & 1] = value;
|
|
|
|
break;
|
|
|
|
|
2012-12-05 10:53:42 +04:00
|
|
|
case R_IS:
|
|
|
|
s->regs[addr] &= ~value;
|
|
|
|
break;
|
|
|
|
|
2011-03-14 13:13:55 +03:00
|
|
|
case 0x8000 ... 0x83ff:
|
|
|
|
s->ext_mtable[addr - 0x8000] = value;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
DENET(qemu_log("%s addr=" TARGET_FMT_plx " v=%x\n",
|
2011-11-15 14:26:54 +04:00
|
|
|
__func__, addr * 4, (unsigned)value));
|
2011-03-14 13:13:55 +03:00
|
|
|
if (addr < ARRAY_SIZE(s->regs)) {
|
|
|
|
s->regs[addr] = value;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
enet_update_irq(s);
|
|
|
|
}
|
|
|
|
|
2011-11-15 14:26:54 +04:00
|
|
|
static const MemoryRegionOps enet_ops = {
|
|
|
|
.read = enet_read,
|
|
|
|
.write = enet_write,
|
|
|
|
.endianness = DEVICE_LITTLE_ENDIAN,
|
2011-03-14 13:13:55 +03:00
|
|
|
};
|
|
|
|
|
2015-07-15 13:19:13 +03:00
|
|
|
static int eth_can_rx(XilinxAXIEnet *s)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
|
|
|
/* RX enabled? */
|
2013-04-16 04:27:55 +04:00
|
|
|
return !s->rxsize && !axienet_rx_resetting(s) && axienet_rx_enabled(s);
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static int enet_match_addr(const uint8_t *buf, uint32_t f0, uint32_t f1)
|
|
|
|
{
|
|
|
|
int match = 1;
|
|
|
|
|
|
|
|
if (memcmp(buf, &f0, 4)) {
|
|
|
|
match = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (buf[4] != (f1 & 0xff) || buf[5] != ((f1 >> 8) & 0xff)) {
|
|
|
|
match = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return match;
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:27:55 +04:00
|
|
|
static void axienet_eth_rx_notify(void *opaque)
|
|
|
|
{
|
|
|
|
XilinxAXIEnet *s = XILINX_AXI_ENET(opaque);
|
|
|
|
|
2013-04-16 04:28:35 +04:00
|
|
|
while (s->rxappsize && stream_can_push(s->tx_control_dev,
|
|
|
|
axienet_eth_rx_notify, s)) {
|
|
|
|
size_t ret = stream_push(s->tx_control_dev,
|
|
|
|
(void *)s->rxapp + CONTROL_PAYLOAD_SIZE
|
2020-05-06 11:25:09 +03:00
|
|
|
- s->rxappsize, s->rxappsize, true);
|
2013-04-16 04:28:35 +04:00
|
|
|
s->rxappsize -= ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (s->rxsize && stream_can_push(s->tx_data_dev,
|
|
|
|
axienet_eth_rx_notify, s)) {
|
|
|
|
size_t ret = stream_push(s->tx_data_dev, (void *)s->rxmem + s->rxpos,
|
2020-05-06 11:25:09 +03:00
|
|
|
s->rxsize, true);
|
2013-04-16 04:27:55 +04:00
|
|
|
s->rxsize -= ret;
|
|
|
|
s->rxpos += ret;
|
|
|
|
if (!s->rxsize) {
|
|
|
|
s->regs[R_IS] |= IS_RX_COMPLETE;
|
2015-07-15 13:19:13 +03:00
|
|
|
if (s->need_flush) {
|
|
|
|
s->need_flush = false;
|
|
|
|
qemu_flush_queued_packets(qemu_get_queue(s->nic));
|
|
|
|
}
|
2013-04-16 04:27:55 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
enet_update_irq(s);
|
|
|
|
}
|
|
|
|
|
2012-07-24 19:35:13 +04:00
|
|
|
static ssize_t eth_rx(NetClientState *nc, const uint8_t *buf, size_t size)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
2013-04-16 04:18:47 +04:00
|
|
|
XilinxAXIEnet *s = qemu_get_nic_opaque(nc);
|
2011-03-14 13:13:55 +03:00
|
|
|
static const unsigned char sa_bcast[6] = {0xff, 0xff, 0xff,
|
|
|
|
0xff, 0xff, 0xff};
|
|
|
|
static const unsigned char sa_ipmcast[3] = {0x01, 0x00, 0x52};
|
2013-04-16 04:28:35 +04:00
|
|
|
uint32_t app[CONTROL_PAYLOAD_WORDS] = {0};
|
2011-03-14 13:13:55 +03:00
|
|
|
int promisc = s->fmi & (1 << 31);
|
|
|
|
int unicast, broadcast, multicast, ip_multicast = 0;
|
|
|
|
uint32_t csum32;
|
|
|
|
uint16_t csum16;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
DENET(qemu_log("%s: %zd bytes\n", __func__, size));
|
|
|
|
|
2015-07-15 13:19:13 +03:00
|
|
|
if (!eth_can_rx(s)) {
|
|
|
|
s->need_flush = true;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-03-14 13:13:55 +03:00
|
|
|
unicast = ~buf[0] & 0x1;
|
|
|
|
broadcast = memcmp(buf, sa_bcast, 6) == 0;
|
|
|
|
multicast = !unicast && !broadcast;
|
|
|
|
if (multicast && (memcmp(sa_ipmcast, buf, sizeof sa_ipmcast) == 0)) {
|
|
|
|
ip_multicast = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Jumbo or vlan sizes ? */
|
|
|
|
if (!(s->rcw[1] & RCW1_JUM)) {
|
|
|
|
if (size > 1518 && size <= 1522 && !(s->rcw[1] & RCW1_VLAN)) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Basic Address filters. If you want to use the extended filters
|
|
|
|
you'll generally have to place the ethernet mac into promiscuous mode
|
|
|
|
to avoid the basic filtering from dropping most frames. */
|
|
|
|
if (!promisc) {
|
|
|
|
if (unicast) {
|
|
|
|
if (!enet_match_addr(buf, s->uaw[0], s->uaw[1])) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (broadcast) {
|
|
|
|
/* Broadcast. */
|
|
|
|
if (s->regs[R_RAF] & RAF_BCAST_REJ) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
int drop = 1;
|
|
|
|
|
|
|
|
/* Multicast. */
|
|
|
|
if (s->regs[R_RAF] & RAF_MCAST_REJ) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
|
|
if (enet_match_addr(buf, s->maddr[i][0], s->maddr[i][1])) {
|
|
|
|
drop = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (drop) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Extended mcast filtering enabled? */
|
|
|
|
if (axienet_newfunc_enabled(s) && axienet_extmcf_enabled(s)) {
|
|
|
|
if (unicast) {
|
|
|
|
if (!enet_match_addr(buf, s->ext_uaw[0], s->ext_uaw[1])) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (broadcast) {
|
|
|
|
/* Broadcast. ??? */
|
|
|
|
if (s->regs[R_RAF] & RAF_BCAST_REJ) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
int idx, bit;
|
|
|
|
|
|
|
|
/* Multicast. */
|
|
|
|
if (!memcmp(buf, sa_ipmcast, 3)) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
idx = (buf[4] & 0x7f) << 8;
|
|
|
|
idx |= buf[5];
|
|
|
|
|
|
|
|
bit = 1 << (idx & 0x1f);
|
|
|
|
idx >>= 5;
|
|
|
|
|
|
|
|
if (!(s->ext_mtable[idx] & bit)) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size < 12) {
|
|
|
|
s->regs[R_IS] |= IS_RX_REJECT;
|
|
|
|
enet_update_irq(s);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (size > (s->c_rxmem - 4)) {
|
|
|
|
size = s->c_rxmem - 4;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(s->rxmem, buf, size);
|
|
|
|
memset(s->rxmem + size, 0, 4); /* Clear the FCS. */
|
|
|
|
|
|
|
|
if (s->rcw[1] & RCW1_FCS) {
|
|
|
|
size += 4; /* fcs is inband. */
|
|
|
|
}
|
|
|
|
|
|
|
|
app[0] = 5 << 28;
|
|
|
|
csum32 = net_checksum_add(size - 14, (uint8_t *)s->rxmem + 14);
|
|
|
|
/* Fold it once. */
|
|
|
|
csum32 = (csum32 & 0xffff) + (csum32 >> 16);
|
|
|
|
/* And twice to get rid of possible carries. */
|
|
|
|
csum16 = (csum32 & 0xffff) + (csum32 >> 16);
|
|
|
|
app[3] = csum16;
|
|
|
|
app[4] = size & 0xffff;
|
|
|
|
|
|
|
|
s->stats.rx_bytes += size;
|
|
|
|
s->stats.rx++;
|
|
|
|
if (multicast) {
|
|
|
|
s->stats.rx_mcast++;
|
|
|
|
app[2] |= 1 | (ip_multicast << 1);
|
|
|
|
} else if (broadcast) {
|
|
|
|
s->stats.rx_bcast++;
|
|
|
|
app[2] |= 1 << 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Good frame. */
|
|
|
|
app[2] |= 1 << 6;
|
|
|
|
|
2013-04-16 04:27:55 +04:00
|
|
|
s->rxsize = size;
|
|
|
|
s->rxpos = 0;
|
2013-04-16 04:28:35 +04:00
|
|
|
for (i = 0; i < ARRAY_SIZE(app); ++i) {
|
|
|
|
app[i] = cpu_to_le32(app[i]);
|
|
|
|
}
|
|
|
|
s->rxappsize = CONTROL_PAYLOAD_SIZE;
|
|
|
|
memcpy(s->rxapp, app, s->rxappsize);
|
2013-04-16 04:27:55 +04:00
|
|
|
axienet_eth_rx_notify(s);
|
2011-03-14 13:13:55 +03:00
|
|
|
|
|
|
|
enet_update_irq(s);
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:27:16 +04:00
|
|
|
static size_t
|
2020-05-06 11:25:09 +03:00
|
|
|
xilinx_axienet_control_stream_push(StreamSlave *obj, uint8_t *buf, size_t len,
|
|
|
|
bool eop)
|
2013-04-16 04:28:35 +04:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
XilinxAXIEnetStreamSlave *cs = XILINX_AXI_ENET_CONTROL_STREAM(obj);
|
|
|
|
XilinxAXIEnet *s = cs->enet;
|
|
|
|
|
2020-05-06 11:25:09 +03:00
|
|
|
assert(eop);
|
2013-04-16 04:28:35 +04:00
|
|
|
if (len != CONTROL_PAYLOAD_SIZE) {
|
|
|
|
hw_error("AXI Enet requires %d byte control stream payload\n",
|
|
|
|
(int)CONTROL_PAYLOAD_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(s->hdr, buf, len);
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_SIZE(s->hdr); ++i) {
|
|
|
|
s->hdr[i] = le32_to_cpu(s->hdr[i]);
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t
|
2020-05-06 11:25:09 +03:00
|
|
|
xilinx_axienet_data_stream_push(StreamSlave *obj, uint8_t *buf, size_t size,
|
|
|
|
bool eop)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
2013-04-16 04:25:18 +04:00
|
|
|
XilinxAXIEnetStreamSlave *ds = XILINX_AXI_ENET_DATA_STREAM(obj);
|
|
|
|
XilinxAXIEnet *s = ds->enet;
|
2011-03-14 13:13:55 +03:00
|
|
|
|
|
|
|
/* TX enable ? */
|
|
|
|
if (!(s->tc & TC_TX)) {
|
2013-04-16 04:27:16 +04:00
|
|
|
return size;
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
|
2020-05-06 11:25:10 +03:00
|
|
|
if (s->txpos + size > s->c_txmem) {
|
|
|
|
qemu_log_mask(LOG_GUEST_ERROR, "%s: Packet larger than txmem\n",
|
|
|
|
TYPE_XILINX_AXI_ENET);
|
|
|
|
s->txpos = 0;
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->txpos == 0 && eop) {
|
|
|
|
/* Fast path single fragment. */
|
|
|
|
s->txpos = size;
|
|
|
|
} else {
|
|
|
|
memcpy(s->txmem + s->txpos, buf, size);
|
|
|
|
buf = s->txmem;
|
|
|
|
s->txpos += size;
|
|
|
|
|
|
|
|
if (!eop) {
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-03-14 13:13:55 +03:00
|
|
|
/* Jumbo or vlan sizes ? */
|
|
|
|
if (!(s->tc & TC_JUM)) {
|
2020-05-06 11:25:10 +03:00
|
|
|
if (s->txpos > 1518 && s->txpos <= 1522 && !(s->tc & TC_VLAN)) {
|
|
|
|
s->txpos = 0;
|
2013-04-16 04:27:16 +04:00
|
|
|
return size;
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-16 04:28:35 +04:00
|
|
|
if (s->hdr[0] & 1) {
|
|
|
|
unsigned int start_off = s->hdr[1] >> 16;
|
|
|
|
unsigned int write_off = s->hdr[1] & 0xffff;
|
2011-03-14 13:13:55 +03:00
|
|
|
uint32_t tmp_csum;
|
|
|
|
uint16_t csum;
|
|
|
|
|
2020-05-06 11:25:10 +03:00
|
|
|
tmp_csum = net_checksum_add(s->txpos - start_off,
|
2020-05-06 11:25:07 +03:00
|
|
|
buf + start_off);
|
2011-03-14 13:13:55 +03:00
|
|
|
/* Accumulate the seed. */
|
2013-04-16 04:28:35 +04:00
|
|
|
tmp_csum += s->hdr[2] & 0xffff;
|
2011-03-14 13:13:55 +03:00
|
|
|
|
|
|
|
/* Fold the 32bit partial checksum. */
|
|
|
|
csum = net_checksum_finish(tmp_csum);
|
|
|
|
|
|
|
|
/* Writeback. */
|
|
|
|
buf[write_off] = csum >> 8;
|
|
|
|
buf[write_off + 1] = csum & 0xff;
|
|
|
|
}
|
|
|
|
|
2020-05-06 11:25:10 +03:00
|
|
|
qemu_send_packet(qemu_get_queue(s->nic), buf, s->txpos);
|
2011-03-14 13:13:55 +03:00
|
|
|
|
2020-05-06 11:25:10 +03:00
|
|
|
s->stats.tx_bytes += s->txpos;
|
2011-03-14 13:13:55 +03:00
|
|
|
s->regs[R_IS] |= IS_TX_COMPLETE;
|
|
|
|
enet_update_irq(s);
|
2013-04-16 04:27:16 +04:00
|
|
|
|
2020-05-06 11:25:10 +03:00
|
|
|
s->txpos = 0;
|
2013-04-16 04:27:16 +04:00
|
|
|
return size;
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
static NetClientInfo net_xilinx_enet_info = {
|
qapi: Change Netdev into a flat union
This is a mostly-mechanical conversion that creates a new flat
union 'Netdev' QAPI type that covers all the branches of the
former 'NetClientOptions' simple union, where the branches are
now listed in a new 'NetClientDriver' enum rather than generated
from the simple union. The existence of a flat union has no
change to the command line syntax accepted for new code, and
will make it possible for a future patch to switch the QMP
command to parse a boxed union for no change to valid QMP; but
it does have some ripple effect on the C code when dealing with
the new types.
While making the conversion, note that the 'NetLegacy' type
remains unchanged: it applies only to legacy command line options,
and will not be ported to QMP, so it should remain a wrapper
around a simple union; to avoid confusion, the type named
'NetClientOptions' is now gone, and we introduce 'NetLegacyOptions'
in its place. Then, in the C code, we convert from NetLegacy to
Netdev as soon as possible, so that the bulk of the net stack
only has to deal with one QAPI type, not two. Note that since
the old legacy code always rejected 'hubport', we can just omit
that branch from the new 'NetLegacyOptions' simple union.
Based on an idea originally by Zoltán Kővágó <DirtY.iCE.hu@gmail.com>:
Message-Id: <01a527fbf1a5de880091f98cf011616a78adeeee.1441627176.git.DirtY.iCE.hu@gmail.com>
although the sed script in that patch no longer applies due to
other changes in the tree since then, and I also did some manual
cleanups (such as fixing whitespace to keep checkpatch happy).
Signed-off-by: Eric Blake <eblake@redhat.com>
Message-Id: <1468468228-27827-13-git-send-email-eblake@redhat.com>
Reviewed-by: Markus Armbruster <armbru@redhat.com>
[Fixup from Eric squashed in]
Signed-off-by: Markus Armbruster <armbru@redhat.com>
2016-07-14 06:50:23 +03:00
|
|
|
.type = NET_CLIENT_DRIVER_NIC,
|
2011-03-14 13:13:55 +03:00
|
|
|
.size = sizeof(NICState),
|
|
|
|
.receive = eth_rx,
|
|
|
|
};
|
|
|
|
|
2013-04-16 04:20:44 +04:00
|
|
|
static void xilinx_enet_realize(DeviceState *dev, Error **errp)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
2013-04-16 04:19:27 +04:00
|
|
|
XilinxAXIEnet *s = XILINX_AXI_ENET(dev);
|
2013-04-16 04:25:18 +04:00
|
|
|
XilinxAXIEnetStreamSlave *ds = XILINX_AXI_ENET_DATA_STREAM(&s->rx_data_dev);
|
2013-04-16 04:28:35 +04:00
|
|
|
XilinxAXIEnetStreamSlave *cs = XILINX_AXI_ENET_CONTROL_STREAM(
|
|
|
|
&s->rx_control_dev);
|
2014-04-25 14:44:20 +04:00
|
|
|
Error *local_err = NULL;
|
2013-04-16 04:25:18 +04:00
|
|
|
|
|
|
|
object_property_add_link(OBJECT(ds), "enet", "xlnx.axi-ethernet",
|
qom: Make QOM link property unref optional
Some object_property_add_link() callers expect property deletion to
unref the link property object. Other callers expect to manage the
refcount themselves. The former are currently broken and therefore leak
the link property object.
This patch adds a flags argument to object_property_add_link() so the
caller can specify which refcount behavior they require. The new
OBJ_PROP_LINK_UNREF_ON_RELEASE flag causes the link pointer to be
unreferenced when the property is deleted.
This fixes refcount leaks in qdev.c, xilinx_axidma.c, xilinx_axienet.c,
s390-virtio-bus.c, virtio-pci.c, virtio-rng.c, and ui/console.c.
Rationale for refcount behavior:
* hw/core/qdev.c
- bus children are explicitly unreferenced, don't interfere
- parent_bus is essentially a read-only property that doesn't hold a
refcount, don't unref
- hotplug_handler is leaked, do unref
* hw/dma/xilinx_axidma.c
- rx stream "dma" links are set using set_link, therefore they
need unref
- tx streams are set using set_link, therefore they need unref
* hw/net/xilinx_axienet.c
- same reasoning as hw/dma/xilinx_axidma.c
* hw/pcmcia/pxa2xx.c
- pxa2xx bypasses set_link and therefore does not use refcounts
* hw/s390x/s390-virtio-bus.c
* hw/virtio/virtio-pci.c
* hw/virtio/virtio-rng.c
* ui/console.c
- set_link is used and there is no explicit unref, do unref
Cc: Peter Crosthwaite <peter.crosthwaite@petalogix.com>
Cc: Alexander Graf <agraf@suse.de>
Cc: Anthony Liguori <aliguori@amazon.com>
Cc: "Michael S. Tsirkin" <mst@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Andreas Färber <afaerber@suse.de>
2014-03-19 11:58:55 +04:00
|
|
|
(Object **) &ds->enet,
|
2014-03-19 11:58:56 +04:00
|
|
|
object_property_allow_set_link,
|
qom: Drop parameter @errp of object_property_add() & friends
The only way object_property_add() can fail is when a property with
the same name already exists. Since our property names are all
hardcoded, failure is a programming error, and the appropriate way to
handle it is passing &error_abort.
Same for its variants, except for object_property_add_child(), which
additionally fails when the child already has a parent. Parentage is
also under program control, so this is a programming error, too.
We have a bit over 500 callers. Almost half of them pass
&error_abort, slightly fewer ignore errors, one test case handles
errors, and the remaining few callers pass them to their own callers.
The previous few commits demonstrated once again that ignoring
programming errors is a bad idea.
Of the few ones that pass on errors, several violate the Error API.
The Error ** argument must be NULL, &error_abort, &error_fatal, or a
pointer to a variable containing NULL. Passing an argument of the
latter kind twice without clearing it in between is wrong: if the
first call sets an error, it no longer points to NULL for the second
call. ich9_pm_add_properties(), sparc32_ledma_realize(),
sparc32_dma_realize(), xilinx_axidma_realize(), xilinx_enet_realize()
are wrong that way.
When the one appropriate choice of argument is &error_abort, letting
users pick the argument is a bad idea.
Drop parameter @errp and assert the preconditions instead.
There's one exception to "duplicate property name is a programming
error": the way object_property_add() implements the magic (and
undocumented) "automatic arrayification". Don't drop @errp there.
Instead, rename object_property_add() to object_property_try_add(),
and add the obvious wrapper object_property_add().
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Message-Id: <20200505152926.18877-15-armbru@redhat.com>
[Two semantic rebase conflicts resolved]
2020-05-05 18:29:22 +03:00
|
|
|
OBJ_PROP_LINK_STRONG);
|
2013-04-16 04:28:35 +04:00
|
|
|
object_property_add_link(OBJECT(cs), "enet", "xlnx.axi-ethernet",
|
qom: Make QOM link property unref optional
Some object_property_add_link() callers expect property deletion to
unref the link property object. Other callers expect to manage the
refcount themselves. The former are currently broken and therefore leak
the link property object.
This patch adds a flags argument to object_property_add_link() so the
caller can specify which refcount behavior they require. The new
OBJ_PROP_LINK_UNREF_ON_RELEASE flag causes the link pointer to be
unreferenced when the property is deleted.
This fixes refcount leaks in qdev.c, xilinx_axidma.c, xilinx_axienet.c,
s390-virtio-bus.c, virtio-pci.c, virtio-rng.c, and ui/console.c.
Rationale for refcount behavior:
* hw/core/qdev.c
- bus children are explicitly unreferenced, don't interfere
- parent_bus is essentially a read-only property that doesn't hold a
refcount, don't unref
- hotplug_handler is leaked, do unref
* hw/dma/xilinx_axidma.c
- rx stream "dma" links are set using set_link, therefore they
need unref
- tx streams are set using set_link, therefore they need unref
* hw/net/xilinx_axienet.c
- same reasoning as hw/dma/xilinx_axidma.c
* hw/pcmcia/pxa2xx.c
- pxa2xx bypasses set_link and therefore does not use refcounts
* hw/s390x/s390-virtio-bus.c
* hw/virtio/virtio-pci.c
* hw/virtio/virtio-rng.c
* ui/console.c
- set_link is used and there is no explicit unref, do unref
Cc: Peter Crosthwaite <peter.crosthwaite@petalogix.com>
Cc: Alexander Graf <agraf@suse.de>
Cc: Anthony Liguori <aliguori@amazon.com>
Cc: "Michael S. Tsirkin" <mst@redhat.com>
Signed-off-by: Stefan Hajnoczi <stefanha@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Signed-off-by: Andreas Färber <afaerber@suse.de>
2014-03-19 11:58:55 +04:00
|
|
|
(Object **) &cs->enet,
|
2014-03-19 11:58:56 +04:00
|
|
|
object_property_allow_set_link,
|
qom: Drop parameter @errp of object_property_add() & friends
The only way object_property_add() can fail is when a property with
the same name already exists. Since our property names are all
hardcoded, failure is a programming error, and the appropriate way to
handle it is passing &error_abort.
Same for its variants, except for object_property_add_child(), which
additionally fails when the child already has a parent. Parentage is
also under program control, so this is a programming error, too.
We have a bit over 500 callers. Almost half of them pass
&error_abort, slightly fewer ignore errors, one test case handles
errors, and the remaining few callers pass them to their own callers.
The previous few commits demonstrated once again that ignoring
programming errors is a bad idea.
Of the few ones that pass on errors, several violate the Error API.
The Error ** argument must be NULL, &error_abort, &error_fatal, or a
pointer to a variable containing NULL. Passing an argument of the
latter kind twice without clearing it in between is wrong: if the
first call sets an error, it no longer points to NULL for the second
call. ich9_pm_add_properties(), sparc32_ledma_realize(),
sparc32_dma_realize(), xilinx_axidma_realize(), xilinx_enet_realize()
are wrong that way.
When the one appropriate choice of argument is &error_abort, letting
users pick the argument is a bad idea.
Drop parameter @errp and assert the preconditions instead.
There's one exception to "duplicate property name is a programming
error": the way object_property_add() implements the magic (and
undocumented) "automatic arrayification". Don't drop @errp there.
Instead, rename object_property_add() to object_property_try_add(),
and add the obvious wrapper object_property_add().
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Message-Id: <20200505152926.18877-15-armbru@redhat.com>
[Two semantic rebase conflicts resolved]
2020-05-05 18:29:22 +03:00
|
|
|
OBJ_PROP_LINK_STRONG);
|
2014-04-25 14:44:20 +04:00
|
|
|
object_property_set_link(OBJECT(ds), OBJECT(s), "enet", &local_err);
|
|
|
|
object_property_set_link(OBJECT(cs), OBJECT(s), "enet", &local_err);
|
|
|
|
if (local_err) {
|
2013-04-16 04:25:18 +04:00
|
|
|
goto xilinx_enet_realize_fail;
|
|
|
|
}
|
2011-03-14 13:13:55 +03:00
|
|
|
|
|
|
|
qemu_macaddr_default_if_unset(&s->conf.macaddr);
|
|
|
|
s->nic = qemu_new_nic(&net_xilinx_enet_info, &s->conf,
|
2013-04-16 04:20:44 +04:00
|
|
|
object_get_typename(OBJECT(dev)), dev->id, s);
|
2013-01-30 15:12:22 +04:00
|
|
|
qemu_format_nic_info_str(qemu_get_queue(s->nic), s->conf.macaddr.a);
|
2011-03-14 13:13:55 +03:00
|
|
|
|
|
|
|
tdk_init(&s->TEMAC.phy);
|
|
|
|
mdio_attach(&s->TEMAC.mdio_bus, &s->TEMAC.phy, s->c_phyaddr);
|
|
|
|
|
|
|
|
s->TEMAC.parent = s;
|
|
|
|
|
2011-08-21 07:09:37 +04:00
|
|
|
s->rxmem = g_malloc(s->c_rxmem);
|
2020-05-06 11:25:10 +03:00
|
|
|
s->txmem = g_malloc(s->c_txmem);
|
2013-04-16 04:25:18 +04:00
|
|
|
return;
|
|
|
|
|
|
|
|
xilinx_enet_realize_fail:
|
2017-06-08 16:39:02 +03:00
|
|
|
error_propagate(errp, local_err);
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
|
2013-04-16 04:20:44 +04:00
|
|
|
static void xilinx_enet_init(Object *obj)
|
2012-08-10 07:16:11 +04:00
|
|
|
{
|
2013-04-16 04:19:27 +04:00
|
|
|
XilinxAXIEnet *s = XILINX_AXI_ENET(obj);
|
2013-04-16 04:20:44 +04:00
|
|
|
SysBusDevice *sbd = SYS_BUS_DEVICE(obj);
|
2012-08-10 07:16:11 +04:00
|
|
|
|
2019-08-23 17:32:49 +03:00
|
|
|
object_initialize_child(OBJECT(s), "axistream-connected-target",
|
|
|
|
&s->rx_data_dev, sizeof(s->rx_data_dev),
|
|
|
|
TYPE_XILINX_AXI_ENET_DATA_STREAM, &error_abort,
|
|
|
|
NULL);
|
|
|
|
object_initialize_child(OBJECT(s), "axistream-control-connected-target",
|
|
|
|
&s->rx_control_dev, sizeof(s->rx_control_dev),
|
|
|
|
TYPE_XILINX_AXI_ENET_CONTROL_STREAM, &error_abort,
|
|
|
|
NULL);
|
2013-04-16 04:20:44 +04:00
|
|
|
sysbus_init_irq(sbd, &s->irq);
|
|
|
|
|
2013-06-07 05:25:08 +04:00
|
|
|
memory_region_init_io(&s->iomem, OBJECT(s), &enet_ops, s, "enet", 0x40000);
|
2013-04-16 04:20:44 +04:00
|
|
|
sysbus_init_mmio(sbd, &s->iomem);
|
2012-08-10 07:16:11 +04:00
|
|
|
}
|
|
|
|
|
2012-01-24 23:12:29 +04:00
|
|
|
static Property xilinx_enet_properties[] = {
|
2013-04-16 04:18:47 +04:00
|
|
|
DEFINE_PROP_UINT32("phyaddr", XilinxAXIEnet, c_phyaddr, 7),
|
|
|
|
DEFINE_PROP_UINT32("rxmem", XilinxAXIEnet, c_rxmem, 0x1000),
|
|
|
|
DEFINE_PROP_UINT32("txmem", XilinxAXIEnet, c_txmem, 0x1000),
|
|
|
|
DEFINE_NIC_PROPERTIES(XilinxAXIEnet, conf),
|
2017-09-07 15:54:51 +03:00
|
|
|
DEFINE_PROP_LINK("axistream-connected", XilinxAXIEnet,
|
|
|
|
tx_data_dev, TYPE_STREAM_SLAVE, StreamSlave *),
|
|
|
|
DEFINE_PROP_LINK("axistream-control-connected", XilinxAXIEnet,
|
|
|
|
tx_control_dev, TYPE_STREAM_SLAVE, StreamSlave *),
|
2012-01-24 23:12:29 +04:00
|
|
|
DEFINE_PROP_END_OF_LIST(),
|
|
|
|
};
|
|
|
|
|
|
|
|
static void xilinx_enet_class_init(ObjectClass *klass, void *data)
|
|
|
|
{
|
2011-12-08 07:34:16 +04:00
|
|
|
DeviceClass *dc = DEVICE_CLASS(klass);
|
2012-01-24 23:12:29 +04:00
|
|
|
|
2013-04-16 04:20:44 +04:00
|
|
|
dc->realize = xilinx_enet_realize;
|
2020-01-10 18:30:32 +03:00
|
|
|
device_class_set_props(dc, xilinx_enet_properties);
|
2013-04-16 04:20:06 +04:00
|
|
|
dc->reset = xilinx_axienet_reset;
|
2013-04-16 04:25:18 +04:00
|
|
|
}
|
|
|
|
|
2020-05-06 11:25:06 +03:00
|
|
|
static void xilinx_enet_control_stream_class_init(ObjectClass *klass,
|
|
|
|
void *data)
|
2013-04-16 04:25:18 +04:00
|
|
|
{
|
|
|
|
StreamSlaveClass *ssc = STREAM_SLAVE_CLASS(klass);
|
|
|
|
|
2020-05-06 11:25:06 +03:00
|
|
|
ssc->push = xilinx_axienet_control_stream_push;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void xilinx_enet_data_stream_class_init(ObjectClass *klass, void *data)
|
|
|
|
{
|
|
|
|
StreamSlaveClass *ssc = STREAM_SLAVE_CLASS(klass);
|
|
|
|
|
|
|
|
ssc->push = xilinx_axienet_data_stream_push;
|
2012-01-24 23:12:29 +04:00
|
|
|
}
|
|
|
|
|
2013-01-10 19:19:07 +04:00
|
|
|
static const TypeInfo xilinx_enet_info = {
|
2013-04-16 04:19:27 +04:00
|
|
|
.name = TYPE_XILINX_AXI_ENET,
|
2011-12-08 07:34:16 +04:00
|
|
|
.parent = TYPE_SYS_BUS_DEVICE,
|
2013-04-16 04:18:47 +04:00
|
|
|
.instance_size = sizeof(XilinxAXIEnet),
|
2011-12-08 07:34:16 +04:00
|
|
|
.class_init = xilinx_enet_class_init,
|
2013-04-16 04:20:44 +04:00
|
|
|
.instance_init = xilinx_enet_init,
|
2013-04-16 04:25:18 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
static const TypeInfo xilinx_enet_data_stream_info = {
|
|
|
|
.name = TYPE_XILINX_AXI_ENET_DATA_STREAM,
|
|
|
|
.parent = TYPE_OBJECT,
|
|
|
|
.instance_size = sizeof(struct XilinxAXIEnetStreamSlave),
|
2020-05-06 11:25:06 +03:00
|
|
|
.class_init = xilinx_enet_data_stream_class_init,
|
2012-08-10 07:16:11 +04:00
|
|
|
.interfaces = (InterfaceInfo[]) {
|
|
|
|
{ TYPE_STREAM_SLAVE },
|
|
|
|
{ }
|
|
|
|
}
|
2011-03-14 13:13:55 +03:00
|
|
|
};
|
2012-02-09 18:20:55 +04:00
|
|
|
|
2013-04-16 04:28:35 +04:00
|
|
|
static const TypeInfo xilinx_enet_control_stream_info = {
|
|
|
|
.name = TYPE_XILINX_AXI_ENET_CONTROL_STREAM,
|
|
|
|
.parent = TYPE_OBJECT,
|
|
|
|
.instance_size = sizeof(struct XilinxAXIEnetStreamSlave),
|
2020-05-06 11:25:06 +03:00
|
|
|
.class_init = xilinx_enet_control_stream_class_init,
|
2013-04-16 04:28:35 +04:00
|
|
|
.interfaces = (InterfaceInfo[]) {
|
|
|
|
{ TYPE_STREAM_SLAVE },
|
|
|
|
{ }
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-02-09 18:20:55 +04:00
|
|
|
static void xilinx_enet_register_types(void)
|
2011-03-14 13:13:55 +03:00
|
|
|
{
|
2011-12-08 07:34:16 +04:00
|
|
|
type_register_static(&xilinx_enet_info);
|
2013-04-16 04:25:18 +04:00
|
|
|
type_register_static(&xilinx_enet_data_stream_info);
|
2013-04-16 04:28:35 +04:00
|
|
|
type_register_static(&xilinx_enet_control_stream_info);
|
2011-03-14 13:13:55 +03:00
|
|
|
}
|
|
|
|
|
2012-02-09 18:20:55 +04:00
|
|
|
type_init(xilinx_enet_register_types)
|