49da7e64f3
1. Provides infrastructure for vector IO using recvmmsg/sendmmsg. 1.1. Multi-message read. 1.2. Multi-message write. 1.3. Optimized queue support for multi-packet enqueue/dequeue. 1.4. BQL/DQL support. 2. Implements transports for several transports as well support for direct wiring of PWEs to NIC. Allows direct connection of VMs to host, other VMs and network devices with no switch in use. 2.1. Raw socket >4 times higher PPS and 10 times higher tcp RX than existing pcap based transport (> 4Gbit) 2.2. New tap transport using socket RX and tap xmit. Similar performance improvements (>4Gbit) 2.3. GRE transport - direct wiring to GRE PWE 2.4. L2TPv3 transport - direct wiring to L2TPv3 PWE 3. Tuning, performance and offload related setting support via ethtool. 4. Initial BPF support - used in tap/raw to avoid software looping 5. Scatter Gather support. 6. VNET and checksum offload support for raw socket transport. 7. TSO/GSO support where applicable or available 8. Migrates all error messages to netdevice_*() and rate limits them where needed. Signed-off-by: Anton Ivanov <anton.ivanov@cambridgegreys.com> Signed-off-by: Richard Weinberger <richard@nod.at>
459 lines
11 KiB
C
459 lines
11 KiB
C
/*
|
|
* Copyright (C) 2017 - Cambridge Greys Limited
|
|
* Copyright (C) 2011 - 2014 Cisco Systems Inc
|
|
* Licensed under the GPL.
|
|
*/
|
|
|
|
#include <linux/etherdevice.h>
|
|
#include <linux/netdevice.h>
|
|
#include <linux/skbuff.h>
|
|
#include <linux/slab.h>
|
|
#include <asm/byteorder.h>
|
|
#include <uapi/linux/ip.h>
|
|
#include <uapi/linux/virtio_net.h>
|
|
#include <linux/virtio_net.h>
|
|
#include <linux/virtio_byteorder.h>
|
|
#include <linux/netdev_features.h>
|
|
#include "vector_user.h"
|
|
#include "vector_kern.h"
|
|
|
|
#define GOOD_LINEAR 512
|
|
#define GSO_ERROR "Incoming GSO frames and GRO disabled on the interface"
|
|
|
|
struct gre_minimal_header {
|
|
uint16_t header;
|
|
uint16_t arptype;
|
|
};
|
|
|
|
|
|
struct uml_gre_data {
|
|
uint32_t rx_key;
|
|
uint32_t tx_key;
|
|
uint32_t sequence;
|
|
|
|
bool ipv6;
|
|
bool has_sequence;
|
|
bool pin_sequence;
|
|
bool checksum;
|
|
bool key;
|
|
struct gre_minimal_header expected_header;
|
|
|
|
uint32_t checksum_offset;
|
|
uint32_t key_offset;
|
|
uint32_t sequence_offset;
|
|
|
|
};
|
|
|
|
struct uml_l2tpv3_data {
|
|
uint64_t rx_cookie;
|
|
uint64_t tx_cookie;
|
|
uint64_t rx_session;
|
|
uint64_t tx_session;
|
|
uint32_t counter;
|
|
|
|
bool udp;
|
|
bool ipv6;
|
|
bool has_counter;
|
|
bool pin_counter;
|
|
bool cookie;
|
|
bool cookie_is_64;
|
|
|
|
uint32_t cookie_offset;
|
|
uint32_t session_offset;
|
|
uint32_t counter_offset;
|
|
};
|
|
|
|
static int l2tpv3_form_header(uint8_t *header,
|
|
struct sk_buff *skb, struct vector_private *vp)
|
|
{
|
|
struct uml_l2tpv3_data *td = vp->transport_data;
|
|
uint32_t *counter;
|
|
|
|
if (td->udp)
|
|
*(uint32_t *) header = cpu_to_be32(L2TPV3_DATA_PACKET);
|
|
(*(uint32_t *) (header + td->session_offset)) = td->tx_session;
|
|
|
|
if (td->cookie) {
|
|
if (td->cookie_is_64)
|
|
(*(uint64_t *)(header + td->cookie_offset)) =
|
|
td->tx_cookie;
|
|
else
|
|
(*(uint32_t *)(header + td->cookie_offset)) =
|
|
td->tx_cookie;
|
|
}
|
|
if (td->has_counter) {
|
|
counter = (uint32_t *)(header + td->counter_offset);
|
|
if (td->pin_counter) {
|
|
*counter = 0;
|
|
} else {
|
|
td->counter++;
|
|
*counter = cpu_to_be32(td->counter);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int gre_form_header(uint8_t *header,
|
|
struct sk_buff *skb, struct vector_private *vp)
|
|
{
|
|
struct uml_gre_data *td = vp->transport_data;
|
|
uint32_t *sequence;
|
|
*((uint32_t *) header) = *((uint32_t *) &td->expected_header);
|
|
if (td->key)
|
|
(*(uint32_t *) (header + td->key_offset)) = td->tx_key;
|
|
if (td->has_sequence) {
|
|
sequence = (uint32_t *)(header + td->sequence_offset);
|
|
if (td->pin_sequence)
|
|
*sequence = 0;
|
|
else
|
|
*sequence = cpu_to_be32(++td->sequence);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int raw_form_header(uint8_t *header,
|
|
struct sk_buff *skb, struct vector_private *vp)
|
|
{
|
|
struct virtio_net_hdr *vheader = (struct virtio_net_hdr *) header;
|
|
|
|
virtio_net_hdr_from_skb(
|
|
skb,
|
|
vheader,
|
|
virtio_legacy_is_little_endian(),
|
|
false
|
|
);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int l2tpv3_verify_header(
|
|
uint8_t *header, struct sk_buff *skb, struct vector_private *vp)
|
|
{
|
|
struct uml_l2tpv3_data *td = vp->transport_data;
|
|
uint32_t *session;
|
|
uint64_t cookie;
|
|
|
|
if ((!td->udp) && (!td->ipv6))
|
|
header += sizeof(struct iphdr) /* fix for ipv4 raw */;
|
|
|
|
/* we do not do a strict check for "data" packets as per
|
|
* the RFC spec because the pure IP spec does not have
|
|
* that anyway.
|
|
*/
|
|
|
|
if (td->cookie) {
|
|
if (td->cookie_is_64)
|
|
cookie = *(uint64_t *)(header + td->cookie_offset);
|
|
else
|
|
cookie = *(uint32_t *)(header + td->cookie_offset);
|
|
if (cookie != td->rx_cookie) {
|
|
if (net_ratelimit())
|
|
netdev_err(vp->dev, "uml_l2tpv3: unknown cookie id");
|
|
return -1;
|
|
}
|
|
}
|
|
session = (uint32_t *) (header + td->session_offset);
|
|
if (*session != td->rx_session) {
|
|
if (net_ratelimit())
|
|
netdev_err(vp->dev, "uml_l2tpv3: session mismatch");
|
|
return -1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int gre_verify_header(
|
|
uint8_t *header, struct sk_buff *skb, struct vector_private *vp)
|
|
{
|
|
|
|
uint32_t key;
|
|
struct uml_gre_data *td = vp->transport_data;
|
|
|
|
if (!td->ipv6)
|
|
header += sizeof(struct iphdr) /* fix for ipv4 raw */;
|
|
|
|
if (*((uint32_t *) header) != *((uint32_t *) &td->expected_header)) {
|
|
if (net_ratelimit())
|
|
netdev_err(vp->dev, "header type disagreement, expecting %0x, got %0x",
|
|
*((uint32_t *) &td->expected_header),
|
|
*((uint32_t *) header)
|
|
);
|
|
return -1;
|
|
}
|
|
|
|
if (td->key) {
|
|
key = (*(uint32_t *)(header + td->key_offset));
|
|
if (key != td->rx_key) {
|
|
if (net_ratelimit())
|
|
netdev_err(vp->dev, "unknown key id %0x, expecting %0x",
|
|
key, td->rx_key);
|
|
return -1;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int raw_verify_header(
|
|
uint8_t *header, struct sk_buff *skb, struct vector_private *vp)
|
|
{
|
|
struct virtio_net_hdr *vheader = (struct virtio_net_hdr *) header;
|
|
|
|
if ((vheader->gso_type != VIRTIO_NET_HDR_GSO_NONE) &&
|
|
(vp->req_size != 65536)) {
|
|
if (net_ratelimit())
|
|
netdev_err(
|
|
vp->dev,
|
|
GSO_ERROR
|
|
);
|
|
}
|
|
if ((vheader->flags & VIRTIO_NET_HDR_F_DATA_VALID) > 0)
|
|
return 1;
|
|
|
|
virtio_net_hdr_to_skb(skb, vheader, virtio_legacy_is_little_endian());
|
|
return 0;
|
|
}
|
|
|
|
static bool get_uint_param(
|
|
struct arglist *def, char *param, unsigned int *result)
|
|
{
|
|
char *arg = uml_vector_fetch_arg(def, param);
|
|
|
|
if (arg != NULL) {
|
|
if (kstrtoint(arg, 0, result) == 0)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static bool get_ulong_param(
|
|
struct arglist *def, char *param, unsigned long *result)
|
|
{
|
|
char *arg = uml_vector_fetch_arg(def, param);
|
|
|
|
if (arg != NULL) {
|
|
if (kstrtoul(arg, 0, result) == 0)
|
|
return true;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
static int build_gre_transport_data(struct vector_private *vp)
|
|
{
|
|
struct uml_gre_data *td;
|
|
int temp_int;
|
|
int temp_rx;
|
|
int temp_tx;
|
|
|
|
vp->transport_data = kmalloc(sizeof(struct uml_gre_data), GFP_KERNEL);
|
|
if (vp->transport_data == NULL)
|
|
return -ENOMEM;
|
|
td = vp->transport_data;
|
|
td->sequence = 0;
|
|
|
|
td->expected_header.arptype = GRE_IRB;
|
|
td->expected_header.header = 0;
|
|
|
|
vp->form_header = &gre_form_header;
|
|
vp->verify_header = &gre_verify_header;
|
|
vp->header_size = 4;
|
|
td->key_offset = 4;
|
|
td->sequence_offset = 4;
|
|
td->checksum_offset = 4;
|
|
|
|
td->ipv6 = false;
|
|
if (get_uint_param(vp->parsed, "v6", &temp_int)) {
|
|
if (temp_int > 0)
|
|
td->ipv6 = true;
|
|
}
|
|
td->key = false;
|
|
if (get_uint_param(vp->parsed, "rx_key", &temp_rx)) {
|
|
if (get_uint_param(vp->parsed, "tx_key", &temp_tx)) {
|
|
td->key = true;
|
|
td->expected_header.header |= GRE_MODE_KEY;
|
|
td->rx_key = cpu_to_be32(temp_rx);
|
|
td->tx_key = cpu_to_be32(temp_tx);
|
|
vp->header_size += 4;
|
|
td->sequence_offset += 4;
|
|
} else {
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
td->sequence = false;
|
|
if (get_uint_param(vp->parsed, "sequence", &temp_int)) {
|
|
if (temp_int > 0) {
|
|
vp->header_size += 4;
|
|
td->has_sequence = true;
|
|
td->expected_header.header |= GRE_MODE_SEQUENCE;
|
|
if (get_uint_param(
|
|
vp->parsed, "pin_sequence", &temp_int)) {
|
|
if (temp_int > 0)
|
|
td->pin_sequence = true;
|
|
}
|
|
}
|
|
}
|
|
vp->rx_header_size = vp->header_size;
|
|
if (!td->ipv6)
|
|
vp->rx_header_size += sizeof(struct iphdr);
|
|
return 0;
|
|
}
|
|
|
|
static int build_l2tpv3_transport_data(struct vector_private *vp)
|
|
{
|
|
|
|
struct uml_l2tpv3_data *td;
|
|
int temp_int, temp_rxs, temp_txs;
|
|
unsigned long temp_rx;
|
|
unsigned long temp_tx;
|
|
|
|
vp->transport_data = kmalloc(
|
|
sizeof(struct uml_l2tpv3_data), GFP_KERNEL);
|
|
|
|
if (vp->transport_data == NULL)
|
|
return -ENOMEM;
|
|
|
|
td = vp->transport_data;
|
|
|
|
vp->form_header = &l2tpv3_form_header;
|
|
vp->verify_header = &l2tpv3_verify_header;
|
|
td->counter = 0;
|
|
|
|
vp->header_size = 4;
|
|
td->session_offset = 0;
|
|
td->cookie_offset = 4;
|
|
td->counter_offset = 4;
|
|
|
|
|
|
td->ipv6 = false;
|
|
if (get_uint_param(vp->parsed, "v6", &temp_int)) {
|
|
if (temp_int > 0)
|
|
td->ipv6 = true;
|
|
}
|
|
|
|
if (get_uint_param(vp->parsed, "rx_session", &temp_rxs)) {
|
|
if (get_uint_param(vp->parsed, "tx_session", &temp_txs)) {
|
|
td->tx_session = cpu_to_be32(temp_txs);
|
|
td->rx_session = cpu_to_be32(temp_rxs);
|
|
} else {
|
|
return -EINVAL;
|
|
}
|
|
} else {
|
|
return -EINVAL;
|
|
}
|
|
|
|
td->cookie_is_64 = false;
|
|
if (get_uint_param(vp->parsed, "cookie64", &temp_int)) {
|
|
if (temp_int > 0)
|
|
td->cookie_is_64 = true;
|
|
}
|
|
td->cookie = false;
|
|
if (get_ulong_param(vp->parsed, "rx_cookie", &temp_rx)) {
|
|
if (get_ulong_param(vp->parsed, "tx_cookie", &temp_tx)) {
|
|
td->cookie = true;
|
|
if (td->cookie_is_64) {
|
|
td->rx_cookie = cpu_to_be64(temp_rx);
|
|
td->tx_cookie = cpu_to_be64(temp_tx);
|
|
vp->header_size += 8;
|
|
td->counter_offset += 8;
|
|
} else {
|
|
td->rx_cookie = cpu_to_be32(temp_rx);
|
|
td->tx_cookie = cpu_to_be32(temp_tx);
|
|
vp->header_size += 4;
|
|
td->counter_offset += 4;
|
|
}
|
|
} else {
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
td->has_counter = false;
|
|
if (get_uint_param(vp->parsed, "counter", &temp_int)) {
|
|
if (temp_int > 0) {
|
|
td->has_counter = true;
|
|
vp->header_size += 4;
|
|
if (get_uint_param(
|
|
vp->parsed, "pin_counter", &temp_int)) {
|
|
if (temp_int > 0)
|
|
td->pin_counter = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (get_uint_param(vp->parsed, "udp", &temp_int)) {
|
|
if (temp_int > 0) {
|
|
td->udp = true;
|
|
vp->header_size += 4;
|
|
td->counter_offset += 4;
|
|
td->session_offset += 4;
|
|
td->cookie_offset += 4;
|
|
}
|
|
}
|
|
|
|
vp->rx_header_size = vp->header_size;
|
|
if ((!td->ipv6) && (!td->udp))
|
|
vp->rx_header_size += sizeof(struct iphdr);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int build_raw_transport_data(struct vector_private *vp)
|
|
{
|
|
if (uml_raw_enable_vnet_headers(vp->fds->rx_fd)) {
|
|
if (!uml_raw_enable_vnet_headers(vp->fds->tx_fd))
|
|
return -1;
|
|
vp->form_header = &raw_form_header;
|
|
vp->verify_header = &raw_verify_header;
|
|
vp->header_size = sizeof(struct virtio_net_hdr);
|
|
vp->rx_header_size = sizeof(struct virtio_net_hdr);
|
|
vp->dev->hw_features |= (NETIF_F_TSO | NETIF_F_GRO);
|
|
vp->dev->features |=
|
|
(NETIF_F_RXCSUM | NETIF_F_HW_CSUM |
|
|
NETIF_F_TSO | NETIF_F_GRO);
|
|
netdev_info(
|
|
vp->dev,
|
|
"raw: using vnet headers for tso and tx/rx checksum"
|
|
);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int build_tap_transport_data(struct vector_private *vp)
|
|
{
|
|
if (uml_raw_enable_vnet_headers(vp->fds->rx_fd)) {
|
|
vp->form_header = &raw_form_header;
|
|
vp->verify_header = &raw_verify_header;
|
|
vp->header_size = sizeof(struct virtio_net_hdr);
|
|
vp->rx_header_size = sizeof(struct virtio_net_hdr);
|
|
vp->dev->hw_features |=
|
|
(NETIF_F_TSO | NETIF_F_GSO | NETIF_F_GRO);
|
|
vp->dev->features |=
|
|
(NETIF_F_RXCSUM | NETIF_F_HW_CSUM |
|
|
NETIF_F_TSO | NETIF_F_GSO | NETIF_F_GRO);
|
|
netdev_info(
|
|
vp->dev,
|
|
"tap/raw: using vnet headers for tso and tx/rx checksum"
|
|
);
|
|
} else {
|
|
return 0; /* do not try to enable tap too if raw failed */
|
|
}
|
|
if (uml_tap_enable_vnet_headers(vp->fds->tx_fd))
|
|
return 0;
|
|
return -1;
|
|
}
|
|
|
|
int build_transport_data(struct vector_private *vp)
|
|
{
|
|
char *transport = uml_vector_fetch_arg(vp->parsed, "transport");
|
|
|
|
if (strncmp(transport, TRANS_GRE, TRANS_GRE_LEN) == 0)
|
|
return build_gre_transport_data(vp);
|
|
if (strncmp(transport, TRANS_L2TPV3, TRANS_L2TPV3_LEN) == 0)
|
|
return build_l2tpv3_transport_data(vp);
|
|
if (strncmp(transport, TRANS_RAW, TRANS_RAW_LEN) == 0)
|
|
return build_raw_transport_data(vp);
|
|
if (strncmp(transport, TRANS_TAP, TRANS_TAP_LEN) == 0)
|
|
return build_tap_transport_data(vp);
|
|
return 0;
|
|
}
|
|
|