In MSI-X mode, if there are interrupts already notified but not cleared and a new interrupt arrives, e1000e incorrectly notifies the notified ones again along with the new one. To fix this issue, replace e1000e_update_interrupt_state() with two new functions: e1000e_raise_interrupts() and e1000e_lower_interrupts(). These functions don't only raise or lower interrupts, but it also performs register writes which updates the interrupt state. Before it performs a register write, these function determines the interrupts already raised, and compares with the interrupts raised after the register write to determine the interrupts to notify. The introduction of these functions made tracepoints which assumes that the caller of e1000e_update_interrupt_state() performs register writes obsolete. These tracepoints are now removed, and alternative ones are added to the new functions. Signed-off-by: Akihiko Odaki <akihiko.odaki@daynix.com> Signed-off-by: Jason Wang <jasowang@redhat.com>
		
			
				
	
	
		
			157 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			157 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Core code for QEMU e1000e emulation
 | |
|  *
 | |
|  * Software developer's manuals:
 | |
|  * http://www.intel.com/content/dam/doc/datasheet/82574l-gbe-controller-datasheet.pdf
 | |
|  *
 | |
|  * Copyright (c) 2015 Ravello Systems LTD (http://ravellosystems.com)
 | |
|  * Developed by Daynix Computing LTD (http://www.daynix.com)
 | |
|  *
 | |
|  * Authors:
 | |
|  * Dmitry Fleytman <dmitry@daynix.com>
 | |
|  * Leonid Bloch <leonid@daynix.com>
 | |
|  * Yan Vugenfirer <yan@daynix.com>
 | |
|  *
 | |
|  * Based on work done by:
 | |
|  * Nir Peleg, Tutis Systems Ltd. for Qumranet Inc.
 | |
|  * Copyright (c) 2008 Qumranet
 | |
|  * Based on work done by:
 | |
|  * Copyright (c) 2007 Dan Aloni
 | |
|  * Copyright (c) 2004 Antony T Curtis
 | |
|  *
 | |
|  * This library is free software; you can redistribute it and/or
 | |
|  * modify it under the terms of the GNU Lesser General Public
 | |
|  * License as published by the Free Software Foundation; either
 | |
|  * version 2.1 of the License, or (at your option) any later version.
 | |
|  *
 | |
|  * This library 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
 | |
|  * Lesser General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU Lesser General Public
 | |
|  * License along with this library; if not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| #ifndef HW_NET_E1000E_CORE_H
 | |
| #define HW_NET_E1000E_CORE_H
 | |
| 
 | |
| #define E1000E_PHY_PAGE_SIZE    (0x20)
 | |
| #define E1000E_PHY_PAGES        (0x07)
 | |
| #define E1000E_MAC_SIZE         (0x8000)
 | |
| #define E1000E_EEPROM_SIZE      (64)
 | |
| #define E1000E_MSIX_VEC_NUM     (5)
 | |
| #define E1000E_NUM_QUEUES       (2)
 | |
| 
 | |
| typedef struct E1000Core E1000ECore;
 | |
| 
 | |
| enum { PHY_R = BIT(0),
 | |
|        PHY_W = BIT(1),
 | |
|        PHY_RW = PHY_R | PHY_W,
 | |
|        PHY_ANYPAGE = BIT(2) };
 | |
| 
 | |
| typedef struct E1000IntrDelayTimer_st {
 | |
|     QEMUTimer *timer;
 | |
|     bool running;
 | |
|     uint32_t delay_reg;
 | |
|     uint32_t delay_resolution_ns;
 | |
|     E1000ECore *core;
 | |
| } E1000IntrDelayTimer;
 | |
| 
 | |
| struct E1000Core {
 | |
|     uint32_t mac[E1000E_MAC_SIZE];
 | |
|     uint16_t phy[E1000E_PHY_PAGES][E1000E_PHY_PAGE_SIZE];
 | |
|     uint16_t eeprom[E1000E_EEPROM_SIZE];
 | |
| 
 | |
|     uint32_t rxbuf_sizes[E1000_PSRCTL_BUFFS_PER_DESC];
 | |
|     uint32_t rx_desc_buf_size;
 | |
|     uint32_t rxbuf_min_shift;
 | |
|     uint8_t rx_desc_len;
 | |
| 
 | |
|     QEMUTimer *autoneg_timer;
 | |
| 
 | |
|     struct e1000e_tx {
 | |
|         e1000x_txd_props props;
 | |
| 
 | |
|         bool skip_cp;
 | |
|         unsigned char sum_needed;
 | |
|         bool cptse;
 | |
|         struct NetTxPkt *tx_pkt;
 | |
|     } tx[E1000E_NUM_QUEUES];
 | |
| 
 | |
|     struct NetRxPkt *rx_pkt;
 | |
| 
 | |
|     bool has_vnet;
 | |
|     int max_queue_num;
 | |
| 
 | |
|     /* Interrupt moderation management */
 | |
|     uint32_t delayed_causes;
 | |
| 
 | |
|     E1000IntrDelayTimer radv;
 | |
|     E1000IntrDelayTimer rdtr;
 | |
|     E1000IntrDelayTimer raid;
 | |
| 
 | |
|     E1000IntrDelayTimer tadv;
 | |
|     E1000IntrDelayTimer tidv;
 | |
| 
 | |
|     E1000IntrDelayTimer itr;
 | |
| 
 | |
|     E1000IntrDelayTimer eitr[E1000E_MSIX_VEC_NUM];
 | |
| 
 | |
|     VMChangeStateEntry *vmstate;
 | |
| 
 | |
|     uint32_t itr_guest_value;
 | |
|     uint32_t eitr_guest_value[E1000E_MSIX_VEC_NUM];
 | |
| 
 | |
|     uint16_t vet;
 | |
| 
 | |
|     uint8_t permanent_mac[ETH_ALEN];
 | |
| 
 | |
|     NICState *owner_nic;
 | |
|     PCIDevice *owner;
 | |
|     void (*owner_start_recv)(PCIDevice *d);
 | |
| 
 | |
|     int64_t timadj;
 | |
| };
 | |
| 
 | |
| void
 | |
| e1000e_core_write(E1000ECore *core, hwaddr addr, uint64_t val, unsigned size);
 | |
| 
 | |
| uint64_t
 | |
| e1000e_core_read(E1000ECore *core, hwaddr addr, unsigned size);
 | |
| 
 | |
| void
 | |
| e1000e_core_pci_realize(E1000ECore      *regs,
 | |
|                        const uint16_t *eeprom_templ,
 | |
|                        uint32_t        eeprom_size,
 | |
|                        const uint8_t  *macaddr);
 | |
| 
 | |
| void
 | |
| e1000e_core_reset(E1000ECore *core);
 | |
| 
 | |
| void
 | |
| e1000e_core_pre_save(E1000ECore *core);
 | |
| 
 | |
| int
 | |
| e1000e_core_post_load(E1000ECore *core);
 | |
| 
 | |
| void
 | |
| e1000e_core_set_link_status(E1000ECore *core);
 | |
| 
 | |
| void
 | |
| e1000e_core_pci_uninit(E1000ECore *core);
 | |
| 
 | |
| bool
 | |
| e1000e_can_receive(E1000ECore *core);
 | |
| 
 | |
| ssize_t
 | |
| e1000e_receive(E1000ECore *core, const uint8_t *buf, size_t size);
 | |
| 
 | |
| ssize_t
 | |
| e1000e_receive_iov(E1000ECore *core, const struct iovec *iov, int iovcnt);
 | |
| 
 | |
| void
 | |
| e1000e_start_recv(E1000ECore *core);
 | |
| 
 | |
| #endif
 |