Skip to content

Commit

Permalink
e1000: defer packets until BM enabled
Browse files Browse the repository at this point in the history
Some guests seem to set BM for e1000 after
enabling RX.
If packets arrive in the window, device is wedged.
Probably works by luck on real hardware, work around
this by making can_receive depend on BM.

Tested-by: Gabriel Somlo <[email protected]>
Signed-off-by: Michael S. Tsirkin <[email protected]>
Reviewed-by: Jason Wang <[email protected]>
Signed-off-by: Stefan Hajnoczi <[email protected]>
  • Loading branch information
mstsirkin authored and stefanhaRH committed Dec 19, 2014
1 parent 58889fe commit 20302e7
Showing 1 changed file with 20 additions and 1 deletion.
21 changes: 20 additions & 1 deletion hw/net/e1000.c
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@
#include "sysemu/sysemu.h"
#include "sysemu/dma.h"
#include "qemu/iov.h"
#include "qemu/range.h"

#include "e1000_regs.h"

Expand Down Expand Up @@ -923,7 +924,9 @@ e1000_can_receive(NetClientState *nc)
E1000State *s = qemu_get_nic_opaque(nc);

return (s->mac_reg[STATUS] & E1000_STATUS_LU) &&
(s->mac_reg[RCTL] & E1000_RCTL_EN) && e1000_has_rxbufs(s, 1);
(s->mac_reg[RCTL] & E1000_RCTL_EN) &&
(s->parent_obj.config[PCI_COMMAND] & PCI_COMMAND_MASTER) &&
e1000_has_rxbufs(s, 1);
}

static uint64_t rx_desc_base(E1000State *s)
Expand Down Expand Up @@ -1529,6 +1532,20 @@ static NetClientInfo net_e1000_info = {
.link_status_changed = e1000_set_link_status,
};

static void e1000_write_config(PCIDevice *pci_dev, uint32_t address,
uint32_t val, int len)
{
E1000State *s = E1000(pci_dev);

pci_default_write_config(pci_dev, address, val, len);

if (range_covers_byte(address, len, PCI_COMMAND) &&
(pci_dev->config[PCI_COMMAND] & PCI_COMMAND_MASTER)) {
qemu_flush_queued_packets(qemu_get_queue(s->nic));
}
}


static int pci_e1000_init(PCIDevice *pci_dev)
{
DeviceState *dev = DEVICE(pci_dev);
Expand All @@ -1539,6 +1556,8 @@ static int pci_e1000_init(PCIDevice *pci_dev)
int i;
uint8_t *macaddr;

pci_dev->config_write = e1000_write_config;

pci_conf = pci_dev->config;

/* TODO: RST# value should be 0, PCI spec 6.2.4 */
Expand Down

0 comments on commit 20302e7

Please sign in to comment.