mirror of
https://github.com/cloud-hypervisor/cloud-hypervisor.git
synced 2025-01-05 12:25:19 +00:00
vm-virtio: net: Implement counters for network traffic
Add counters for RX/TX bytes and RX/TX frames. These are collected on a per queue basis and then accumulated into an atomic shared value across the different threads for the device as a whole. Collecting and accumulating these counters makes minimal difference in the iperf results. Any difference seen is within what is observed as natural variation in this test. e.g. With counter updates: $ iperf3 -c 192.168.249.2 Connecting to host 192.168.249.2, port 5201 [ 5] local 192.168.249.1 port 52706 connected to 192.168.249.2 port 5201 [ ID] Interval Transfer Bitrate Retr Cwnd [ 5] 0.00-1.00 sec 6.19 GBytes 53.2 Gbits/sec 0 3.01 MBytes [ 5] 1.00-2.00 sec 6.31 GBytes 54.2 Gbits/sec 0 3.01 MBytes [ 5] 2.00-3.00 sec 6.29 GBytes 54.0 Gbits/sec 0 3.01 MBytes [ 5] 3.00-4.00 sec 6.22 GBytes 53.4 Gbits/sec 0 3.01 MBytes [ 5] 4.00-5.00 sec 6.14 GBytes 52.8 Gbits/sec 0 3.01 MBytes [ 5] 5.00-6.00 sec 6.13 GBytes 52.7 Gbits/sec 0 3.01 MBytes [ 5] 6.00-7.00 sec 6.20 GBytes 53.3 Gbits/sec 0 3.01 MBytes [ 5] 7.00-8.00 sec 6.16 GBytes 52.9 Gbits/sec 0 3.01 MBytes [ 5] 8.00-9.00 sec 6.13 GBytes 52.6 Gbits/sec 0 3.01 MBytes [ 5] 9.00-10.00 sec 6.15 GBytes 52.8 Gbits/sec 0 3.01 MBytes - - - - - - - - - - - - - - - - - - - - - - - - - [ ID] Interval Transfer Bitrate Retr [ 5] 0.00-10.00 sec 61.9 GBytes 53.2 Gbits/sec 0 sender [ 5] 0.00-10.00 sec 61.9 GBytes 53.2 Gbits/sec receiver iperf Done. Without counter updates: $ iperf3 -c 192.168.249.2 Connecting to host 192.168.249.2, port 5201 [ 5] local 192.168.249.1 port 52716 connected to 192.168.249.2 port 5201 [ ID] Interval Transfer Bitrate Retr Cwnd [ 5] 0.00-1.00 sec 6.29 GBytes 54.1 Gbits/sec 0 3.03 MBytes [ 5] 1.00-2.00 sec 6.18 GBytes 53.1 Gbits/sec 0 3.03 MBytes [ 5] 2.00-3.00 sec 6.26 GBytes 53.8 Gbits/sec 0 3.03 MBytes [ 5] 3.00-4.00 sec 6.24 GBytes 53.6 Gbits/sec 0 3.03 MBytes [ 5] 4.00-5.00 sec 6.27 GBytes 53.9 Gbits/sec 1 3.03 MBytes [ 5] 5.00-6.00 sec 6.31 GBytes 54.2 Gbits/sec 0 3.03 MBytes [ 5] 6.00-7.00 sec 6.29 GBytes 54.1 Gbits/sec 0 3.03 MBytes [ 5] 7.00-8.00 sec 6.16 GBytes 52.9 Gbits/sec 0 3.03 MBytes [ 5] 8.00-9.00 sec 6.13 GBytes 52.6 Gbits/sec 0 3.03 MBytes [ 5] 9.00-10.00 sec 6.25 GBytes 53.7 Gbits/sec 0 3.03 MBytes - - - - - - - - - - - - - - - - - - - - - - - - - [ ID] Interval Transfer Bitrate Retr [ 5] 0.00-10.00 sec 62.4 GBytes 53.6 Gbits/sec 1 sender [ 5] 0.00-10.00 sec 62.4 GBytes 53.6 Gbits/sec receiver iperf Done. Signed-off-by: Rob Bradford <robert.bradford@intel.com>
This commit is contained in:
parent
0583ce921b
commit
2b4a0358de
@ -30,7 +30,7 @@ use virtio_bindings::bindings::virtio_net::*;
|
|||||||
use virtio_bindings::bindings::virtio_ring::VIRTIO_RING_F_EVENT_IDX;
|
use virtio_bindings::bindings::virtio_ring::VIRTIO_RING_F_EVENT_IDX;
|
||||||
use vm_memory::{GuestMemoryAtomic, GuestMemoryMmap};
|
use vm_memory::{GuestMemoryAtomic, GuestMemoryMmap};
|
||||||
use vm_virtio::net_util::{open_tap, RxVirtio, TxVirtio};
|
use vm_virtio::net_util::{open_tap, RxVirtio, TxVirtio};
|
||||||
use vm_virtio::NetQueuePair;
|
use vm_virtio::{NetCounters, NetQueuePair};
|
||||||
use vmm::config::{OptionParser, OptionParserError};
|
use vmm::config::{OptionParser, OptionParserError};
|
||||||
use vmm_sys_util::eventfd::EventFd;
|
use vmm_sys_util::eventfd::EventFd;
|
||||||
|
|
||||||
@ -111,6 +111,7 @@ impl VhostUserNetThread {
|
|||||||
tx: TxVirtio::new(),
|
tx: TxVirtio::new(),
|
||||||
rx_tap_listening: false,
|
rx_tap_listening: false,
|
||||||
epoll_fd: None,
|
epoll_fd: None,
|
||||||
|
counters: NetCounters::default(),
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
@ -24,9 +24,10 @@ use std::fs::File;
|
|||||||
use std::io::Read;
|
use std::io::Read;
|
||||||
use std::io::{self, Write};
|
use std::io::{self, Write};
|
||||||
use std::net::Ipv4Addr;
|
use std::net::Ipv4Addr;
|
||||||
|
use std::num::Wrapping;
|
||||||
use std::os::unix::io::{AsRawFd, FromRawFd, RawFd};
|
use std::os::unix::io::{AsRawFd, FromRawFd, RawFd};
|
||||||
use std::result;
|
use std::result;
|
||||||
use std::sync::atomic::{AtomicBool, Ordering};
|
use std::sync::atomic::{AtomicBool, AtomicU64, Ordering};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::thread;
|
use std::thread;
|
||||||
use std::vec::Vec;
|
use std::vec::Vec;
|
||||||
@ -54,6 +55,7 @@ pub struct NetQueuePair {
|
|||||||
pub tx: TxVirtio,
|
pub tx: TxVirtio,
|
||||||
pub epoll_fd: Option<RawFd>,
|
pub epoll_fd: Option<RawFd>,
|
||||||
pub rx_tap_listening: bool,
|
pub rx_tap_listening: bool,
|
||||||
|
pub counters: NetCounters,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl NetQueuePair {
|
impl NetQueuePair {
|
||||||
@ -112,6 +114,19 @@ impl NetQueuePair {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Consume the counters from the Rx/Tx queues and accumulate into
|
||||||
|
// the counters for the device as whole. This consumption is needed
|
||||||
|
// to handle MQ.
|
||||||
|
self.counters
|
||||||
|
.rx_bytes
|
||||||
|
.fetch_add(self.rx.counter_bytes.0, Ordering::AcqRel);
|
||||||
|
self.counters
|
||||||
|
.rx_frames
|
||||||
|
.fetch_add(self.rx.counter_frames.0, Ordering::AcqRel);
|
||||||
|
self.rx.counter_bytes = Wrapping(0);
|
||||||
|
self.rx.counter_frames = Wrapping(0);
|
||||||
|
|
||||||
if self.rx.deferred_irqs {
|
if self.rx.deferred_irqs {
|
||||||
self.rx.deferred_irqs = false;
|
self.rx.deferred_irqs = false;
|
||||||
let mem = self
|
let mem = self
|
||||||
@ -166,6 +181,16 @@ impl NetQueuePair {
|
|||||||
.ok_or(DeviceError::NoMemoryConfigured)
|
.ok_or(DeviceError::NoMemoryConfigured)
|
||||||
.map(|m| m.memory())?;
|
.map(|m| m.memory())?;
|
||||||
self.tx.process_desc_chain(&mem, &mut self.tap, &mut queue);
|
self.tx.process_desc_chain(&mem, &mut self.tap, &mut queue);
|
||||||
|
|
||||||
|
self.counters
|
||||||
|
.tx_bytes
|
||||||
|
.fetch_add(self.tx.counter_bytes.0, Ordering::AcqRel);
|
||||||
|
self.counters
|
||||||
|
.tx_frames
|
||||||
|
.fetch_add(self.tx.counter_frames.0, Ordering::AcqRel);
|
||||||
|
self.tx.counter_bytes = Wrapping(0);
|
||||||
|
self.tx.counter_frames = Wrapping(0);
|
||||||
|
|
||||||
Ok(queue.needs_notification(&mem, queue.next_used))
|
Ok(queue.needs_notification(&mem, queue.next_used))
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -390,6 +415,14 @@ impl NetEpollHandler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Default, Clone)]
|
||||||
|
pub struct NetCounters {
|
||||||
|
tx_bytes: Arc<AtomicU64>,
|
||||||
|
tx_frames: Arc<AtomicU64>,
|
||||||
|
rx_bytes: Arc<AtomicU64>,
|
||||||
|
rx_frames: Arc<AtomicU64>,
|
||||||
|
}
|
||||||
|
|
||||||
pub struct Net {
|
pub struct Net {
|
||||||
id: String,
|
id: String,
|
||||||
kill_evt: Option<EventFd>,
|
kill_evt: Option<EventFd>,
|
||||||
@ -404,6 +437,7 @@ pub struct Net {
|
|||||||
ctrl_queue_epoll_thread: Option<thread::JoinHandle<result::Result<(), DeviceError>>>,
|
ctrl_queue_epoll_thread: Option<thread::JoinHandle<result::Result<(), DeviceError>>>,
|
||||||
paused: Arc<AtomicBool>,
|
paused: Arc<AtomicBool>,
|
||||||
queue_size: Vec<u16>,
|
queue_size: Vec<u16>,
|
||||||
|
counters: NetCounters,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
#[derive(Serialize, Deserialize)]
|
||||||
@ -461,6 +495,7 @@ impl Net {
|
|||||||
ctrl_queue_epoll_thread: None,
|
ctrl_queue_epoll_thread: None,
|
||||||
paused: Arc::new(AtomicBool::new(false)),
|
paused: Arc::new(AtomicBool::new(false)),
|
||||||
queue_size: vec![queue_size; queue_num],
|
queue_size: vec![queue_size; queue_num],
|
||||||
|
counters: NetCounters::default(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -661,6 +696,7 @@ impl VirtioDevice for Net {
|
|||||||
tx,
|
tx,
|
||||||
epoll_fd: None,
|
epoll_fd: None,
|
||||||
rx_tap_listening,
|
rx_tap_listening,
|
||||||
|
counters: self.counters.clone(),
|
||||||
},
|
},
|
||||||
interrupt_cb: interrupt_cb.clone(),
|
interrupt_cb: interrupt_cb.clone(),
|
||||||
kill_evt: kill_evt.try_clone().unwrap(),
|
kill_evt: kill_evt.try_clone().unwrap(),
|
||||||
|
@ -12,6 +12,7 @@ use std::fs::File;
|
|||||||
use std::io::{self, Write};
|
use std::io::{self, Write};
|
||||||
use std::mem;
|
use std::mem;
|
||||||
use std::net::Ipv4Addr;
|
use std::net::Ipv4Addr;
|
||||||
|
use std::num::Wrapping;
|
||||||
use std::os::unix::io::{AsRawFd, FromRawFd, RawFd};
|
use std::os::unix::io::{AsRawFd, FromRawFd, RawFd};
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::sync::atomic::{AtomicBool, Ordering};
|
use std::sync::atomic::{AtomicBool, Ordering};
|
||||||
@ -340,6 +341,8 @@ impl NetCtrlEpollHandler {
|
|||||||
pub struct TxVirtio {
|
pub struct TxVirtio {
|
||||||
pub iovec: Vec<(GuestAddress, usize)>,
|
pub iovec: Vec<(GuestAddress, usize)>,
|
||||||
pub frame_buf: [u8; MAX_BUFFER_SIZE],
|
pub frame_buf: [u8; MAX_BUFFER_SIZE],
|
||||||
|
pub counter_bytes: Wrapping<u64>,
|
||||||
|
pub counter_frames: Wrapping<u64>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for TxVirtio {
|
impl Default for TxVirtio {
|
||||||
@ -353,6 +356,8 @@ impl TxVirtio {
|
|||||||
TxVirtio {
|
TxVirtio {
|
||||||
iovec: Vec::new(),
|
iovec: Vec::new(),
|
||||||
frame_buf: [0u8; MAX_BUFFER_SIZE],
|
frame_buf: [0u8; MAX_BUFFER_SIZE],
|
||||||
|
counter_bytes: Wrapping(0),
|
||||||
|
counter_frames: Wrapping(0),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -400,6 +405,10 @@ impl TxVirtio {
|
|||||||
println!("net: tx: error failed to write to tap: {}", e);
|
println!("net: tx: error failed to write to tap: {}", e);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
self.counter_bytes += Wrapping((read_count - vnet_hdr_len()) as u64);
|
||||||
|
self.counter_frames += Wrapping(1);
|
||||||
|
|
||||||
queue.add_used(&mem, head_index, 0);
|
queue.add_used(&mem, head_index, 0);
|
||||||
queue.update_avail_event(&mem);
|
queue.update_avail_event(&mem);
|
||||||
}
|
}
|
||||||
@ -412,6 +421,8 @@ pub struct RxVirtio {
|
|||||||
pub deferred_irqs: bool,
|
pub deferred_irqs: bool,
|
||||||
pub bytes_read: usize,
|
pub bytes_read: usize,
|
||||||
pub frame_buf: [u8; MAX_BUFFER_SIZE],
|
pub frame_buf: [u8; MAX_BUFFER_SIZE],
|
||||||
|
pub counter_bytes: Wrapping<u64>,
|
||||||
|
pub counter_frames: Wrapping<u64>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for RxVirtio {
|
impl Default for RxVirtio {
|
||||||
@ -427,6 +438,8 @@ impl RxVirtio {
|
|||||||
deferred_irqs: false,
|
deferred_irqs: false,
|
||||||
bytes_read: 0,
|
bytes_read: 0,
|
||||||
frame_buf: [0u8; MAX_BUFFER_SIZE],
|
frame_buf: [0u8; MAX_BUFFER_SIZE],
|
||||||
|
counter_bytes: Wrapping(0),
|
||||||
|
counter_frames: Wrapping(0),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -472,6 +485,9 @@ impl RxVirtio {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
self.counter_bytes += Wrapping((write_count - vnet_hdr_len()) as u64);
|
||||||
|
self.counter_frames += Wrapping(1);
|
||||||
|
|
||||||
queue.add_used(&mem, head_index, write_count as u32);
|
queue.add_used(&mem, head_index, write_count as u32);
|
||||||
queue.update_avail_event(&mem);
|
queue.update_avail_event(&mem);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user