@ -10,6 +10,7 @@ use core::sync::atomic::{fence, Ordering};
use alloc ::collections ::BTreeMap ;
use bitflags ::* ;
use isomorphic_drivers ::net ::ethernet ::intel ::ixgbe ;
use log ::* ;
use rcore_memory ::paging ::PageTable ;
use rcore_memory ::PAGE_SIZE ;
@ -28,119 +29,10 @@ use crate::sync::SpinNoIrqLock as Mutex;
use crate ::sync ::{ MutexGuard , SpinNoIrq } ;
use crate ::HEAP_ALLOCATOR ;
use super ::super ::{ DeviceType , Driver , DRIVERS , NET_DRIVERS , SOCKET_ACTIVITY } ;
// At the beginning, all transmit descriptors have there status non-zero,
// so we need to track whether we are using the descriptor for the first time.
// When the descriptors wrap around, we set first_trans to false,
// and lookup status instead for checking whether it is empty.
pub struct IXGBE {
header : usize ,
size : usize ,
mac : EthernetAddress ,
send_page : usize ,
send_buffers : Vec < usize > ,
recv_page : usize ,
recv_buffers : Vec < usize > ,
first_trans : bool ,
}
use super ::super ::{ provider ::Provider , DeviceType , Driver , DRIVERS , NET_DRIVERS , SOCKET_ACTIVITY } ;
#[ derive(Clone) ]
pub struct IXGBEDriver ( Arc < Mutex < IXGBE > > ) ;
// On linux, use ip link set <dev> mtu <MTU - 18>
const IXGBE_MTU : usize = 8000 ;
const IXGBE_BUFFER_SIZE : usize = PAGE_SIZE * 2 ;
const IXGBE_CTRL : usize = 0x00000 / 4 ;
const IXGBE_STATUS : usize = 0x00008 / 4 ;
const IXGBE_CTRL_EXT : usize = 0x00018 / 4 ;
const IXGBE_EICR : usize = 0x00800 / 4 ;
const IXGBE_EITR : usize = 0x00820 / 4 ;
const IXGBE_EIMS : usize = 0x00880 / 4 ;
const IXGBE_EIMC : usize = 0x00888 / 4 ;
const IXGBE_GPIE : usize = 0x00898 / 4 ;
const IXGBE_IVAR : usize = 0x00900 / 4 ;
const IXGBE_EIMC1 : usize = 0x00A90 / 4 ;
const IXGBE_EIMC2 : usize = 0x00A91 / 4 ;
const IXGBE_RDBAL : usize = 0x01000 / 4 ;
const IXGBE_RDBAH : usize = 0x01004 / 4 ;
const IXGBE_RDLEN : usize = 0x01008 / 4 ;
const IXGBE_DCA_RXCTRL : usize = 0x0100C / 4 ;
const IXGBE_RDH : usize = 0x01010 / 4 ;
const IXGBE_SRRCTL : usize = 0x01014 / 4 ;
const IXGBE_RDT : usize = 0x01018 / 4 ;
const IXGBE_RXDCTL : usize = 0x01028 / 4 ;
const IXGBE_RTRPT4C : usize = 0x02140 / 4 ;
const IXGBE_RTRPT4C_END : usize = 0x02160 / 4 ;
const IXGBE_RTRPCS : usize = 0x02430 / 4 ;
const IXGBE_RDRXCTL : usize = 0x02F00 / 4 ;
const IXGBE_PFQDE : usize = 0x02F04 / 4 ;
const IXGBE_RXCTRL : usize = 0x03000 / 4 ;
const IXGBE_RTRUP2TC : usize = 0x03020 / 4 ;
const IXGBE_FCTTV : usize = 0x03200 / 4 ;
const IXGBE_FCTTV_END : usize = 0x03210 / 4 ;
const IXGBE_FCRTL : usize = 0x03220 / 4 ;
const IXGBE_FCRTL_END : usize = 0x03240 / 4 ;
const IXGBE_FCRTH : usize = 0x03260 / 4 ;
const IXGBE_FCRTH_END : usize = 0x03280 / 4 ;
const IXGBE_FCRTV : usize = 0x032A0 / 4 ;
const IXGBE_RXPBSIZE : usize = 0x03C00 / 4 ;
const IXGBE_RXPBSIZE_END : usize = 0x03C20 / 4 ;
const IXGBE_FCCFG : usize = 0x03D00 / 4 ;
const IXGBE_HLREG0 : usize = 0x04240 / 4 ;
const IXGBE_MAXFRS : usize = 0x04268 / 4 ;
const IXGBE_MFLCN : usize = 0x04294 / 4 ;
const IXGBE_AUTOC : usize = 0x042A0 / 4 ;
const IXGBE_LINKS : usize = 0x042A4 / 4 ;
const IXGBE_AUTOC2 : usize = 0x04324 / 4 ;
const IXGBE_RTTDCS : usize = 0x04900 / 4 ;
const IXGBE_RTTDT1C : usize = 0x04908 / 4 ;
const IXGBE_RTTDT2C : usize = 0x04910 / 4 ;
const IXGBE_RTTDT2C_END : usize = 0x04930 / 4 ;
const IXGBE_RTTDQSEL : usize = 0x04904 / 4 ;
const IXGBE_RTTDC1C : usize = 0x04908 / 4 ;
const IXGBE_TXPBTHRESH : usize = 0x04950 / 4 ;
const IXGBE_TXPBTHRESH_END : usize = 0x04970 / 4 ;
const IXGBE_DMATXCTL : usize = 0x04A80 / 4 ;
const IXGBE_RXCSUM : usize = 0x05000 / 4 ;
const IXGBE_FCTRL : usize = 0x05080 / 4 ;
const IXGBE_PFVTCTL : usize = 0x051B0 / 4 ;
const IXGBE_MTA : usize = 0x05200 / 4 ;
const IXGBE_MTA_END : usize = 0x05400 / 4 ;
const IXGBE_MRQC : usize = 0x05818 / 4 ;
const IXGBE_TDBAL : usize = 0x06000 / 4 ;
const IXGBE_TDBAH : usize = 0x06004 / 4 ;
const IXGBE_TDLEN : usize = 0x06008 / 4 ;
const IXGBE_TDH : usize = 0x06010 / 4 ;
const IXGBE_TDT : usize = 0x06018 / 4 ;
const IXGBE_TXDCTL : usize = 0x06028 / 4 ;
const IXGBE_DTXMXSZRQ : usize = 0x08100 / 4 ;
const IXGBE_MTQC : usize = 0x08120 / 4 ;
const IXGBE_SECRXCTRL : usize = 0x08D00 / 4 ;
const IXGBE_SECRXSTAT : usize = 0x08D04 / 4 ;
const IXGBE_VFTA : usize = 0x0A000 / 4 ;
const IXGBE_VFTA_END : usize = 0x0A200 / 4 ;
const IXGBE_RAL : usize = 0x0A200 / 4 ;
const IXGBE_RAH : usize = 0x0A204 / 4 ;
const IXGBE_MPSAR : usize = 0x0A600 / 4 ;
const IXGBE_MPSAR_END : usize = 0x0A800 / 4 ;
const IXGBE_RTTUP2TC : usize = 0x0C800 / 4 ;
const IXGBE_TXPBSIZE : usize = 0x0CC00 / 4 ;
const IXGBE_TXPBSIZE_END : usize = 0x0CC20 / 4 ;
const IXGBE_RTTPCS : usize = 0x0CD00 / 4 ;
const IXGBE_RTTPT2C : usize = 0x0CD20 / 4 ;
const IXGBE_RTTPT2C_END : usize = 0x0CD40 / 4 ;
const IXGBE_RTTPT2S : usize = 0x0CD40 / 4 ;
const IXGBE_RTTPT2S_END : usize = 0x0CD60 / 4 ;
const IXGBE_PFVLVF : usize = 0x0F100 / 4 ;
const IXGBE_PFVLVF_END : usize = 0x0F200 / 4 ;
const IXGBE_PFVLVFB : usize = 0x0F200 / 4 ;
const IXGBE_PFVLVFB_END : usize = 0x0F400 / 4 ;
const IXGBE_PFUTA : usize = 0x0F400 / 4 ;
const IXGBE_PFUTA_END : usize = 0x0F600 / 4 ;
const IXGBE_EEC : usize = 0x10010 / 4 ;
struct IXGBEDriver ( ixgbe ::IXGBEDriver ) ;
pub struct IXGBEInterface {
iface : Mutex < EthernetInterface < 'static , 'static , 'static , IXGBEDriver > > ,
@ -148,6 +40,8 @@ pub struct IXGBEInterface {
ifname : String ,
irq : Option < u32 > ,
id : String ,
header : usize ,
size : usize ,
}
impl Driver for IXGBEInterface {
@ -157,48 +51,19 @@ impl Driver for IXGBEInterface {
return false ;
}
let ( handled , rx ) = {
let driver = self . driver . 0. lock ( ) ;
if let None = active_table ( ) . get_entry ( driver . header ) {
let mut current_addr = driver . header ;
while current_addr < driver . header + driver . size {
let handled = {
if let None = active_table ( ) . get_entry ( self . header ) {
let mut current_addr = self . header ;
while current_addr < self . header + self . size {
active_table ( ) . map_if_not_exists ( current_addr , current_addr ) ;
current_addr = current_addr + PAGE_SIZE ;
}
}
let ixgbe = unsafe {
slice ::from_raw_parts_mut ( driver . header as * mut Volatile < u32 > , driver . size / 4 )
} ;
let icr = ixgbe [ IXGBE_EICR ] . read ( ) ;
if icr ! = 0 {
// clear it
ixgbe [ IXGBE_EICR ] . write ( icr ) ;
if icr & ( 1 < < 20 ) ! = 0 {
// link status change
let status = ixgbe [ IXGBE_LINKS ] . read ( ) ;
if status & ( 1 < < 7 ) ! = 0 {
// link up
info! ( " ixgbe: interface {} link up " , & self . ifname ) ;
} else {
// link down
info! ( " ixgbe: interface {} link down " , & self . ifname ) ;
}
}
if icr & ( 1 < < 0 ) ! = 0 {
// rx interrupt
( true , true )
} else {
( true , false )
}
} else {
( false , false )
}
self . driver . 0. try_handle_interrupt ( )
} ;
if rx {
if handled {
let timestamp = Instant ::from_millis ( crate ::trap ::uptime_msec ( ) as i64 ) ;
let mut sockets = SOCKETS . lock ( ) ;
match self . iface . lock ( ) . poll ( & mut sockets , timestamp ) {
@ -247,29 +112,6 @@ impl Driver for IXGBEInterface {
}
}
}
#[ repr(C) ]
#[ derive(Copy, Clone, Debug) ]
struct IXGBESendDesc {
addr : u64 ,
len : u16 ,
cso : u8 ,
cmd : u8 ,
status : u8 ,
css : u8 ,
vlan : u16 ,
}
#[ repr(C) ]
#[ derive(Copy, Clone, Debug) ]
struct IXGBERecvDesc {
addr : u64 ,
len : u16 ,
frag_chksum : u16 ,
status_error : u16 ,
vlan_tag : u16 ,
}
pub struct IXGBERxToken ( Vec < u8 > ) ;
pub struct IXGBETxToken ( IXGBEDriver ) ;
@ -278,83 +120,19 @@ impl<'a> phy::Device<'a> for IXGBEDriver {
type TxToken = IXGBETxToken ;
fn receive ( & ' a mut self ) -> Option < ( Self ::RxToken , Self ::TxToken ) > {
let driver = self . 0. lock ( ) ;
if let None = active_table ( ) . get_entry ( driver . header ) {
let mut current_addr = driver . header ;
while current_addr < driver . header + driver . size {
active_table ( ) . map_if_not_exists ( current_addr , current_addr ) ;
current_addr = current_addr + PAGE_SIZE ;
if self . 0. can_send ( ) {
if let Some ( data ) = self . 0. recv ( ) {
Some ( ( IXGBERxToken ( data ) , IXGBETxToken ( self . clone ( ) ) ) )
} else {
None
}
}
let ixgbe = unsafe {
slice ::from_raw_parts_mut ( driver . header as * mut Volatile < u32 > , driver . size / 4 )
} ;
let send_queue_size = PAGE_SIZE / size_of ::< IXGBESendDesc > ( ) ;
let send_queue = unsafe {
slice ::from_raw_parts_mut ( driver . send_page as * mut IXGBESendDesc , send_queue_size )
} ;
let tdt = ixgbe [ IXGBE_TDT ] . read ( ) ;
let index = ( tdt as usize ) % send_queue_size ;
let send_desc = & mut send_queue [ index ] ;
let recv_queue_size = PAGE_SIZE / size_of ::< IXGBERecvDesc > ( ) ;
let mut recv_queue = unsafe {
slice ::from_raw_parts_mut ( driver . recv_page as * mut IXGBERecvDesc , recv_queue_size )
} ;
let mut rdt = ixgbe [ IXGBE_RDT ] . read ( ) ;
let index = ( rdt as usize + 1 ) % recv_queue_size ;
let recv_desc = & mut recv_queue [ index ] ;
let transmit_avail = driver . first_trans | | ( * send_desc ) . status & 1 ! = 0 ;
// Ignore packet spanning multiple descriptor
let receive_avail = ( * recv_desc ) . status_error & 1 ! = 0 ;
if transmit_avail & & receive_avail {
let buffer = unsafe {
slice ::from_raw_parts (
driver . recv_buffers [ index ] as * const u8 ,
recv_desc . len as usize ,
)
} ;
recv_desc . status_error = recv_desc . status_error & ! 1 ;
rdt = ( rdt + 1 ) % recv_queue_size as u32 ;
ixgbe [ IXGBE_RDT ] . write ( rdt ) ;
Some ( ( IXGBERxToken ( buffer . to_vec ( ) ) , IXGBETxToken ( self . clone ( ) ) ) )
} else {
None
}
}
fn transmit ( & ' a mut self ) -> Option < Self ::TxToken > {
let driver = self . 0. lock ( ) ;
if let None = active_table ( ) . get_entry ( driver . header ) {
let mut current_addr = driver . header ;
while current_addr < driver . header + driver . size {
active_table ( ) . map_if_not_exists ( current_addr , current_addr ) ;
current_addr = current_addr + PAGE_SIZE ;
}
}
let ixgbe = unsafe {
slice ::from_raw_parts_mut ( driver . header as * mut Volatile < u32 > , driver . size / 4 )
} ;
let send_queue_size = PAGE_SIZE / size_of ::< IXGBESendDesc > ( ) ;
let send_queue = unsafe {
slice ::from_raw_parts_mut ( driver . send_page as * mut IXGBESendDesc , send_queue_size )
} ;
let tdt = ixgbe [ IXGBE_TDT ] . read ( ) ;
let index = ( tdt as usize ) % send_queue_size ;
let send_desc = & mut send_queue [ index ] ;
let transmit_avail = driver . first_trans | | ( * send_desc ) . status & 1 ! = 0 ;
if transmit_avail {
if self . 0. can_send ( ) {
Some ( IXGBETxToken ( self . clone ( ) ) )
} else {
None
@ -363,7 +141,7 @@ impl<'a> phy::Device<'a> for IXGBEDriver {
fn capabilities ( & self ) -> DeviceCapabilities {
let mut caps = DeviceCapabilities ::default ( ) ;
caps . max_transmission_unit = IXGBE_MTU ; // max MTU
caps . max_transmission_unit = ixgbe ::IXGBEDriver ::get_mtu ( ) ; // max MTU
caps . max_burst_size = Some ( 256 ) ;
// IP Rx checksum is offloaded with RXCSUM
caps . checksum . ipv4 = Checksum ::Tx ;
@ -385,402 +163,26 @@ impl phy::TxToken for IXGBETxToken {
where
F : FnOnce ( & mut [ u8 ] ) -> Result < R > ,
{
let mut buffer = [ 0 u8 ; IXGBE_BUFFER_SIZE ] ;
let mut buffer = [ 0 u8 ; ixgbe ::IXGBEDriver ::get_mtu ( ) ] ;
let result = f ( & mut buffer [ .. len ] ) ;
let mut driver = ( self . 0 ) . 0. lock ( ) ;
let ixgbe = unsafe {
slice ::from_raw_parts_mut ( driver . header as * mut Volatile < u32 > , driver . size / 4 )
} ;
let send_queue_size = PAGE_SIZE / size_of ::< IXGBESendDesc > ( ) ;
let mut send_queue = unsafe {
slice ::from_raw_parts_mut ( driver . send_page as * mut IXGBESendDesc , send_queue_size )
} ;
let mut tdt = ixgbe [ IXGBE_TDT ] . read ( ) ;
let index = ( tdt as usize ) % send_queue_size ;
let send_desc = & mut send_queue [ index ] ;
assert! ( driver . first_trans | | send_desc . status & 1 ! = 0 ) ;
let target =
unsafe { slice ::from_raw_parts_mut ( driver . send_buffers [ index ] as * mut u8 , len ) } ;
target . copy_from_slice ( & buffer [ .. len ] ) ;
send_desc . len = len as u16 ;
// RS | IFCS | EOP
send_desc . cmd = ( 1 < < 3 ) | ( 1 < < 1 ) | ( 1 < < 0 ) ;
send_desc . status = 0 ;
fence ( Ordering ::SeqCst ) ;
tdt = ( tdt + 1 ) % send_queue_size as u32 ;
ixgbe [ IXGBE_TDT ] . write ( tdt ) ;
fence ( Ordering ::SeqCst ) ;
// round
if tdt = = 0 {
driver . first_trans = false ;
if result . is_ok ( ) {
( self . 0 ) . 0. send ( & buffer [ .. len ] ) ;
}
result
}
}
bitflags! {
struct IXGBEStatus : u32 {
// if LANID1 is clear, this is LAN0
// if LANID1 is set, this is LAN1
const LANID1 = 1 < < 2 ;
const LINK_UP = 1 < < 7 ;
const NUM_VFS1 = 1 < < 10 ;
const NUM_VFS2 = 1 < < 11 ;
const NUM_VFS4 = 1 < < 12 ;
const NUM_VFS8 = 1 < < 13 ;
const NUM_VFS16 = 1 < < 14 ;
const NUM_VFS32 = 1 < < 15 ;
const NUM_VFS64 = 1 < < 16 ;
const IOV = 1 < < 18 ;
const PCIE_MASTER_ENABLE = 1 < < 19 ;
}
}
pub fn ixgbe_init ( name : String , irq : Option < u32 > , header : usize , size : usize ) -> Arc < IXGBEInterface > {
assert_eq! ( size_of ::< IXGBESendDesc > ( ) , 16 ) ;
assert_eq! ( size_of ::< IXGBERecvDesc > ( ) , 16 ) ;
let send_page = unsafe {
HEAP_ALLOCATOR . alloc_zeroed ( Layout ::from_size_align ( PAGE_SIZE , PAGE_SIZE ) . unwrap ( ) )
} as usize ;
let recv_page = unsafe {
HEAP_ALLOCATOR . alloc_zeroed ( Layout ::from_size_align ( PAGE_SIZE , PAGE_SIZE ) . unwrap ( ) )
} as usize ;
let send_page_pa = active_table ( ) . get_entry ( send_page ) . unwrap ( ) . target ( ) ;
let recv_page_pa = active_table ( ) . get_entry ( recv_page ) . unwrap ( ) . target ( ) ;
let send_queue_size = PAGE_SIZE / size_of ::< IXGBESendDesc > ( ) ;
let recv_queue_size = PAGE_SIZE / size_of ::< IXGBERecvDesc > ( ) ;
let mut send_queue =
unsafe { slice ::from_raw_parts_mut ( send_page as * mut IXGBESendDesc , send_queue_size ) } ;
let mut recv_queue =
unsafe { slice ::from_raw_parts_mut ( recv_page as * mut IXGBERecvDesc , recv_queue_size ) } ;
let mut current_addr = header ;
while current_addr < header + size {
active_table ( ) . map_if_not_exists ( current_addr , current_addr ) ;
current_addr = current_addr + PAGE_SIZE ;
}
let ixgbe = unsafe { slice ::from_raw_parts_mut ( header as * mut Volatile < u32 > , size / 4 ) } ;
debug! (
" status before setup: {:#?} " ,
IXGBEStatus ::from_bits_truncate ( ixgbe [ IXGBE_STATUS ] . read ( ) )
) ;
// 4.6.3 Initialization Sequence
// 4.6.3.1 Interrupts During Initialization
// 1. Disable interrupts.
// mask all interrupts
ixgbe [ IXGBE_EIMC ] . write ( ! 0 ) ;
ixgbe [ IXGBE_EIMC1 ] . write ( ! 0 ) ;
ixgbe [ IXGBE_EIMC2 ] . write ( ! 0 ) ;
// 2. Issue a global reset.
// reset: LRST | RST
ixgbe [ IXGBE_CTRL ] . write ( 1 < < 3 | 1 < < 26 ) ;
while ixgbe [ IXGBE_CTRL ] . read ( ) & ( 1 < < 3 | 1 < < 26 ) ! = 0 { }
// 3. Disable interrupts (again).
// mask all interrupts
ixgbe [ IXGBE_EIMC ] . write ( ! 0 ) ;
ixgbe [ IXGBE_EIMC1 ] . write ( ! 0 ) ;
ixgbe [ IXGBE_EIMC2 ] . write ( ! 0 ) ;
// 4.6.3.2 Global Reset and General Configuration
// no flow control
for reg in ( IXGBE_FCTTV .. IXGBE_FCTTV_END ) . step_by ( 4 ) {
ixgbe [ reg ] . write ( 0 ) ;
}
for reg in ( IXGBE_FCRTL .. IXGBE_FCRTL_END ) . step_by ( 4 ) {
ixgbe [ reg ] . write ( 0 ) ;
}
for reg in ( IXGBE_FCRTH .. IXGBE_FCRTH_END ) . step_by ( 4 ) {
ixgbe [ reg ] . write ( 0 ) ;
}
ixgbe [ IXGBE_FCRTV ] . write ( 0 ) ;
ixgbe [ IXGBE_FCCFG ] . write ( 0 ) ;
// Auto-Read Done
while ixgbe [ IXGBE_EEC ] . read ( ) & ( 1 < < 9 ) = = 0 { }
// DMA Init Done
while ixgbe [ IXGBE_RDRXCTL ] . read ( ) & ( 1 < < 3 ) = = 0 { }
// BAM, Accept Broadcast packets
ixgbe [ IXGBE_FCTRL ] . write ( ixgbe [ IXGBE_FCTRL ] . read ( ) | ( 1 < < 10 ) ) ;
// 4.6.7 Receive Initialization
// Receive Address (RAL[n] and RAH[n]) for used addresses.
// Read MAC Address
let ral = ixgbe [ IXGBE_RAL ] . read ( ) ;
let rah = ixgbe [ IXGBE_RAH ] . read ( ) ;
let mac : [ u8 ; 6 ] = [
ral as u8 ,
( ral > > 8 ) as u8 ,
( ral > > 16 ) as u8 ,
( ral > > 24 ) as u8 ,
rah as u8 ,
( rah > > 8 ) as u8 ,
] ;
debug! ( " mac {:x?} " , mac ) ;
let mut driver = IXGBE {
header ,
size ,
mac : EthernetAddress ::from_bytes ( & mac ) ,
send_page ,
send_buffers : Vec ::with_capacity ( send_queue_size ) ,
recv_page ,
recv_buffers : Vec ::with_capacity ( recv_queue_size ) ,
first_trans : true ,
} ;
// Unicast Table Array (PFUTA).
for i in IXGBE_PFUTA .. IXGBE_PFUTA_END {
ixgbe [ i ] . write ( 0 ) ;
}
// VLAN Filter Table Array (VFTA[n]).
for i in IXGBE_VFTA .. IXGBE_VFTA_END {
ixgbe [ i ] . write ( 0 ) ;
}
// VLAN Pool Filter (PFVLVF[n]).
for i in IXGBE_PFVLVF .. IXGBE_PFVLVF_END {
ixgbe [ i ] . write ( 0 ) ;
}
// MAC Pool Select Array (MPSAR[n]).
for i in IXGBE_MPSAR .. IXGBE_MPSAR_END {
ixgbe [ i ] . write ( 0 ) ;
}
// VLAN Pool Filter Bitmap (PFVLVFB[n]).
for i in IXGBE_PFVLVFB .. IXGBE_PFVLVFB_END {
ixgbe [ i ] . write ( 0 ) ;
}
// Set up the Multicast Table Array (MTA) registers. This entire table should be zeroed and only the desired multicast addresses should be permitted (by writing 0x1 to the corresponding bit location).
for i in IXGBE_MTA .. IXGBE_MTA_END {
ixgbe [ i ] . write ( 0 ) ;
}
// Program the different Rx filters and Rx offloads via registers FCTRL, VLNCTRL, MCSTCTRL, RXCSUM, RQTC, RFCTL, MPSAR, RSSRK, RETA, SAQF, DAQF, SDPQF, FTQF, SYNQF, ETQF, ETQS, RDRXCTL, RSCDBU.
// IPPCSE IP Payload Checksum Enable
ixgbe [ IXGBE_RXCSUM ] . write ( ixgbe [ IXGBE_RXCSUM ] . read ( ) | ( 1 < < 12 ) ) ;
// Note that RDRXCTL.CRCStrip and HLREG0.RXCRCSTRP must be set to the same value. At the same time the RDRXCTL.RSCFRSTSIZE should be set to 0x0 as opposed to its hardware default.
// CRCStrip | RSCACKC | FCOE_WRFIX
ixgbe [ IXGBE_RDRXCTL ] . write ( ixgbe [ IXGBE_RDRXCTL ] . read ( ) | ( 1 < < 0 ) | ( 1 < < 25 ) | ( 1 < < 26 ) ) ;
/* Not completed part
// Program RXPBSIZE, MRQC, PFQDE, RTRUP2TC, MFLCN.RPFCE, and MFLCN.RFCE according to the DCB and virtualization modes (see Section 4.6.11.3).
// 4.6.11.3.4 DCB-Off, VT-Off
// RXPBSIZE[0].SIZE=0x200, RXPBSIZE[1-7].SIZE=0x0
ixgbe [ IXGBE_RXPBSIZE ] . write ( 0x200 < < 10 ) ;
for i in ( IXGBE_RXPBSIZE + 1 ) .. IXGBE_RXPBSIZE_END {
ixgbe [ i ] . write ( 0 ) ;
}
// TXPBSIZE[0].SIZE=0xA0, TXPBSIZE[1-7].SIZE=0x0
ixgbe [ IXGBE_TXPBSIZE ] . write ( 0xA0 < < 10 ) ;
for i in ( IXGBE_TXPBSIZE + 1 ) .. IXGBE_TXPBSIZE_END {
ixgbe [ i ] . write ( 0 ) ;
}
// TXPBTHRESH.THRESH[0]=0xA0 — Maximum expected Tx packet length in this TC TXPBTHRESH.THRESH[1-7]=0x0
ixgbe [ IXGBE_TXPBTHRESH ] . write ( 0xA0 ) ;
for i in ( IXGBE_TXPBTHRESH + 1 ) .. IXGBE_TXPBTHRESH_END {
ixgbe [ i ] . write ( 0 ) ;
}
// Disbale Arbiter
// ARBDIS = 1
ixgbe [ IXGBE_RTTDCS ] . write ( ixgbe [ IXGBE_RTTDCS ] . read ( ) | ( 1 < < 6 ) ) ;
// Set MRQE to 0xxxb, with the three least significant bits set according to the RSS mode
ixgbe [ IXGBE_MRQC ] . write ( ixgbe [ IXGBE_MRQC ] . read ( ) & ! ( 0xf ) ) ;
// Clear both RT_Ena and VT_Ena bits in the MTQC register.
// Set MTQC.NUM_TC_OR_Q to 00b.
ixgbe [ IXGBE_MTQC ] . write ( ixgbe [ IXGBE_MTQC ] . read ( ) & ! ( 0xf ) ) ;
// Enable Arbiter
// ARBDIS = 0
ixgbe [ IXGBE_RTTDCS ] . write ( ixgbe [ IXGBE_RTTDCS ] . read ( ) & ! ( 1 < < 6 ) ) ;
// Clear PFVTCTL.VT_Ena (as the MRQC.VT_Ena)
ixgbe [ IXGBE_PFVTCTL ] . write ( ixgbe [ IXGBE_PFVTCTL ] . read ( ) & ! ( 1 < < 0 ) ) ;
// Rx UP to TC (RTRUP2TC), UPnMAP=0b, n=0,...,7
ixgbe [ IXGBE_RTRUP2TC ] . write ( 0 ) ;
// Tx UP to TC (RTTUP2TC), UPnMAP=0b, n=0,...,7
ixgbe [ IXGBE_RTTUP2TC ] . write ( 0 ) ;
// DMA TX TCP Maximum Allowed Size Requests (DTXMXSZRQ) — set Max_byte_num_req = 0xFFF = 1 MB
ixgbe [ IXGBE_DTXMXSZRQ ] . write ( 0xfff ) ;
// PFQDE: The QDE bit should be set to 0b in the PFQDE register for all queues enabling per queue policy by the SRRCTL[n] setting.
ixgbe [ IXGBE_PFQDE ] . write ( ixgbe [ IXGBE_PFQDE ] . read ( ) & ! ( 1 < < 0 ) ) ;
// Disable PFC and enable legacy flow control:
ixgbe [ IXGBE_MFLCN ] . write ( ixgbe [ IXGBE_MFLCN ] . read ( ) & ! ( 1 < < 2 ) & ! ( 0xff < < 4 ) ) ;
// Clear RTTDT1C register, per each queue, via setting RTTDQSEL first
for i in 0 .. 128 {
ixgbe [ IXGBE_RTTDQSEL ] . write ( i ) ;
ixgbe [ IXGBE_RTTDT1C ] . write ( 0 ) ;
}
// Clear RTTDT2C[0-7] registers
for i in IXGBE_RTTDT2C .. IXGBE_RTTDT2C_END {
ixgbe [ i ] . write ( 0 ) ;
}
// Clear RTTPT2C[0-7] registers
for i in IXGBE_RTTPT2C .. IXGBE_RTTPT2C_END {
ixgbe [ i ] . write ( 0 ) ;
}
// Clear RTRPT4C[0-7] registers
for i in IXGBE_RTRPT4C .. IXGBE_RTRPT4C_END {
ixgbe [ i ] . write ( 0 ) ;
}
// Tx Descriptor Plane Control and Status (RTTDCS), bits:
// TDPAC=0b, VMPAC=0b, TDRM=0b, BDPM=1b, BPBFSM=1b
ixgbe [ IXGBE_RTTDCS ] . write ( ixgbe [ IXGBE_RTTDCS ] . read ( ) & ! ( 1 < < 0 ) & ! ( 1 < < 1 ) & ! ( 1 < < 4 ) ) ;
ixgbe [ IXGBE_RTTDCS ] . write ( ixgbe [ IXGBE_RTTDCS ] . read ( ) | ( 1 < < 22 ) | ( 1 < < 23 ) ) ;
// Tx Packet Plane Control and Status (RTTPCS): TPPAC=0b, TPRM=0b, ARBD=0x224
ixgbe [ IXGBE_RTTPCS ] . write ( ixgbe [ IXGBE_RTTPCS ] . read ( ) & ! ( 1 < < 5 ) & ! ( 1 < < 8 ) & ! ( 0x3ff < < 22 ) ) ;
ixgbe [ IXGBE_RTTPCS ] . write ( ixgbe [ IXGBE_RTTPCS ] . read ( ) | ( 0x224 < < 22 ) ) ;
// Rx Packet Plane Control and Status (RTRPCS): RAC=0b, RRM=0b
ixgbe [ IXGBE_RTRPCS ] . write ( ixgbe [ IXGBE_RTRPCS ] . read ( ) & ! ( 1 < < 1 ) & ! ( 1 < < 2 ) ) ;
* /
// Setup receive queue 0
// The following steps should be done once per transmit queue:
// 2. Receive buffers of appropriate size should be allocated and pointers to these buffers should be stored in the descriptor ring.
for i in 0 .. recv_queue_size {
let buffer_page = unsafe {
HEAP_ALLOCATOR . alloc_zeroed ( Layout ::from_size_align ( IXGBE_BUFFER_SIZE , PAGE_SIZE ) . unwrap ( ) )
} as usize ;
let buffer_page_pa = active_table ( ) . get_entry ( buffer_page ) . unwrap ( ) . target ( ) ;
recv_queue [ i ] . addr = buffer_page_pa as u64 ;
driver . recv_buffers . push ( buffer_page ) ;
}
// 3. Program the descriptor base address with the address of the region (registers RDBAL, RDBAH).
ixgbe [ IXGBE_RDBAL ] . write ( recv_page_pa as u32 ) ; // RDBAL
ixgbe [ IXGBE_RDBAH ] . write ( ( recv_page_pa > > 32 ) as u32 ) ; // RDBAH
// 4. Set the length register to the size of the descriptor ring (register RDLEN).
ixgbe [ IXGBE_RDLEN ] . write ( PAGE_SIZE as u32 ) ; // RDLEN
// 5. Program SRRCTL associated with this queue according to the size of the buffers and the required header control.
// Legacy descriptor, 8K buffer size
ixgbe [ IXGBE_SRRCTL ] . write ( ( ixgbe [ IXGBE_SRRCTL ] . read ( ) & ! 0xf ) | ( 8 < < 0 ) ) ;
ixgbe [ IXGBE_RDH ] . write ( 0 ) ; // RDH
// 8. Program RXDCTL with appropriate values including the queue Enable bit. Note that packets directed to a disabled queue are dropped.
ixgbe [ IXGBE_RXDCTL ] . write ( ixgbe [ IXGBE_RXDCTL ] . read ( ) | ( 1 < < 25 ) ) ; // enable queue
// 9. Poll the RXDCTL register until the Enable bit is set. The tail should not be bumped before this bit was read as 1b.
while ixgbe [ IXGBE_RXDCTL ] . read ( ) | ( 1 < < 25 ) = = 0 { } // wait for it
// 10. Bump the tail pointer (RDT) to enable descriptors fetching by setting it to the ring length minus one.
ixgbe [ IXGBE_RDT ] . write ( ( recv_queue_size - 1 ) as u32 ) ; // RDT
// all queues are setup
// 11. Enable the receive path by setting RXCTRL.RXEN. This should be done only after all other settings are done following the steps below.
// Halt the receive data path by setting SECRXCTRL.RX_DIS bit.
ixgbe [ IXGBE_SECRXCTRL ] . write ( ixgbe [ IXGBE_SECRXCTRL ] . read ( ) | ( 1 < < 1 ) ) ;
// Wait for the data paths to be emptied by HW. Poll the SECRXSTAT.SECRX_RDY bit until it is asserted by HW.
while ixgbe [ IXGBE_SECRXSTAT ] . read ( ) & ( 1 < < 0 ) = = 0 { } // poll
// Set RXCTRL.RXEN
// enable the queue
ixgbe [ IXGBE_RXCTRL ] . write ( ixgbe [ IXGBE_RXCTRL ] . read ( ) | ( 1 < < 0 ) ) ;
// Clear the SECRXCTRL.SECRX_DIS bits to enable receive data path
ixgbe [ IXGBE_SECRXCTRL ] . write ( ixgbe [ IXGBE_SECRXCTRL ] . read ( ) & ! ( 1 < < 1 ) ) ;
// Set bit 16 of the CTRL_EXT register and clear bit 12 of the DCA_RXCTRL[n] register[n].
ixgbe [ IXGBE_CTRL_EXT ] . write ( ixgbe [ IXGBE_CTRL_EXT ] . read ( ) | ( 1 < < 16 ) ) ;
ixgbe [ IXGBE_DCA_RXCTRL ] . write ( ixgbe [ IXGBE_DCA_RXCTRL ] . read ( ) & ! ( 1 < < 12 ) ) ;
// 4.6.8 Transmit Initialization
// Program the HLREG0 register according to the required MAC behavior.
// TXCRCEN | RXCRCSTRP | JUMBOEN | TXPADEN | RXLNGTHERREN
ixgbe [ IXGBE_HLREG0 ]
. write ( ixgbe [ IXGBE_HLREG0 ] . read ( ) | ( 1 < < 0 ) | ( 1 < < 1 ) | ( 1 < < 2 ) | ( 1 < < 10 ) | ( 1 < < 27 ) ) ;
// Set max MTU
ixgbe [ IXGBE_MAXFRS ] . write ( ( IXGBE_MTU as u32 ) < < 16 ) ;
// The following steps should be done once per transmit queue:
// 1. Allocate a region of memory for the transmit descriptor list.
for i in 0 .. send_queue_size {
let buffer_page = unsafe {
HEAP_ALLOCATOR . alloc_zeroed ( Layout ::from_size_align ( IXGBE_BUFFER_SIZE , PAGE_SIZE ) . unwrap ( ) )
} as usize ;
let buffer_page_pa = active_table ( ) . get_entry ( buffer_page ) . unwrap ( ) . target ( ) ;
send_queue [ i ] . addr = buffer_page_pa as u64 ;
driver . send_buffers . push ( buffer_page ) ;
}
// 2. Program the descriptor base address with the address of the region (TDBAL, TDBAH).
ixgbe [ IXGBE_TDBAL ] . write ( send_page_pa as u32 ) ; // TDBAL
ixgbe [ IXGBE_TDBAH ] . write ( ( send_page_pa > > 32 ) as u32 ) ; // TDBAH
// 3. Set the length register to the size of the descriptor ring (TDLEN).
ixgbe [ IXGBE_TDLEN ] . write ( PAGE_SIZE as u32 ) ; // TDLEN
ixgbe [ IXGBE_TDH ] . write ( 0 ) ; // TDH
ixgbe [ IXGBE_TDT ] . write ( 0 ) ; // TDT
// 6. Enable transmit path by setting DMATXCTL.TE. This step should be executed only for the first enabled transmit queue and does not need to be repeated for any following queues.
ixgbe [ IXGBE_DMATXCTL ] . write ( ixgbe [ IXGBE_DMATXCTL ] . read ( ) | 1 < < 0 ) ;
// 7. Enable the queue using TXDCTL.ENABLE. Poll the TXDCTL register until the Enable bit is set.
ixgbe [ IXGBE_TXDCTL ] . write ( ixgbe [ IXGBE_TXDCTL ] . read ( ) | 1 < < 25 ) ;
while ixgbe [ IXGBE_TXDCTL ] . read ( ) & ( 1 < < 25 ) = = 0 { }
// 4.6.6 Interrupt Initialization
// The software driver associates between Tx and Rx interrupt causes and the EICR register by setting the IVAR[n] registers.
// map Rx0 to interrupt 0
ixgbe [ IXGBE_IVAR ] . write ( 0b00000000_00000000_00000000_10000000 ) ;
// Set the interrupt throttling in EITR[n] and GPIE according to the preferred mode of operation.
// Throttle interrupts
// Seems having good effect on tx bandwidth
// but bad effect on rx bandwidth
// CNT_WDIS | ITR Interval=100us
// if sys_read() spin more times, the interval here should be larger
// Linux use dynamic ETIR based on statistics
ixgbe [ IXGBE_EITR ] . write ( ( ( 100 / 2 ) < < 3 ) | ( 1 < < 31 ) ) ;
// Disable general purpose interrupt
// We don't need them
ixgbe [ IXGBE_GPIE ] . write ( 0 ) ;
// Software clears EICR by writing all ones to clear old interrupt causes.
// clear all interrupt
ixgbe [ IXGBE_EICR ] . write ( ! 0 ) ;
// Software enables the required interrupt causes by setting the EIMS register.
// unmask rx interrupt and link status change
ixgbe [ IXGBE_EIMS ] . write ( ( 1 < < 0 ) | ( 1 < < 20 ) ) ;
debug! (
" status after setup: {:#?} " ,
IXGBEStatus ::from_bits_truncate ( ixgbe [ IXGBE_STATUS ] . read ( ) )
) ;
pub fn ixgbe_init (
name : String ,
irq : Option < u32 > ,
header : usize ,
size : usize ,
) -> Arc < IXGBEInterface > {
let ixgbe = ixgbe ::IXGBEDriver ::init ( Provider ::new ( ) , header , size ) ;
let ethernet_addr = EthernetAddress ::from_bytes ( & ixgbe . get_mac ( ) . as_bytes ( ) ) ;
let net_driver = IXGBEDriver ( Arc::new ( Mutex ::new ( driver ) ) ) ;
let net_driver = IXGBEDriver ( ixgbe ) ;
let ethernet_addr = EthernetAddress ::from_bytes ( & mac ) ;
let ip_addrs = [ IpCidr ::new ( IpAddress ::v4 ( 10 , 0 , 0 , 2 ) , 24 ) ] ;
let neighbor_cache = NeighborCache ::new ( BTreeMap ::new ( ) ) ;
let iface = EthernetInterfaceBuilder ::new ( net_driver . clone ( ) )
@ -797,6 +199,8 @@ pub fn ixgbe_init(name: String, irq: Option<u32>, header: usize, size: usize) ->
ifname : name . clone ( ) ,
id : name ,
irq ,
header ,
size ,
} ;
let driver = Arc ::new ( ixgbe_iface ) ;
@ -804,40 +208,3 @@ pub fn ixgbe_init(name: String, irq: Option<u32>, header: usize, size: usize) ->
NET_DRIVERS . write ( ) . push ( driver . clone ( ) ) ;
driver
}
impl Drop for IXGBE {
fn drop ( & mut self ) {
debug! ( " ixgbe: interface detaching " ) ;
if let None = active_table ( ) . get_entry ( self . header ) {
let mut current_addr = self . header ;
while current_addr < self . header + self . size {
active_table ( ) . map_if_not_exists ( current_addr , current_addr ) ;
current_addr = current_addr + PAGE_SIZE ;
}
}
let ixgbe = unsafe {
slice ::from_raw_parts_mut ( self . header as * mut Volatile < u32 > , self . size / 4 )
} ;
// 1. Disable interrupts.
ixgbe [ IXGBE_EIMC ] . write ( ! 0 ) ;
ixgbe [ IXGBE_EIMC1 ] . write ( ! 0 ) ;
ixgbe [ IXGBE_EIMC2 ] . write ( ! 0 ) ;
// 2. Issue a global reset.
// reset: LRST | RST
ixgbe [ IXGBE_CTRL ] . write ( 1 < < 3 | 1 < < 26 ) ;
while ixgbe [ IXGBE_CTRL ] . read ( ) & ( 1 < < 3 | 1 < < 26 ) ! = 0 { }
unsafe {
HEAP_ALLOCATOR . dealloc ( self . send_page as * mut u8 , Layout ::from_size_align ( PAGE_SIZE , PAGE_SIZE ) . unwrap ( ) ) ;
HEAP_ALLOCATOR . dealloc ( self . recv_page as * mut u8 , Layout ::from_size_align ( PAGE_SIZE , PAGE_SIZE ) . unwrap ( ) ) ;
for send_buffer in self . send_buffers . iter ( ) {
HEAP_ALLOCATOR . dealloc ( * send_buffer as * mut u8 , Layout ::from_size_align ( IXGBE_BUFFER_SIZE , PAGE_SIZE ) . unwrap ( ) ) ;
}
for recv_buffer in self . recv_buffers . iter ( ) {
HEAP_ALLOCATOR . dealloc ( * recv_buffer as * mut u8 , Layout ::from_size_align ( IXGBE_BUFFER_SIZE , PAGE_SIZE ) . unwrap ( ) ) ;
}
}
}
}