Re: [PATCH v3 2/4] drivers/net/virtio_net: Added basic RSS support.
On Fri, Feb 18, 2022 at 08:43:20AM -0700, Willem de Bruijn wrote: > On Thu, Feb 17, 2022 at 12:05 PM Andrew Melnichenko wrote: > > > > Hi all, > > > > On Mon, Feb 14, 2022 at 12:09 AM Willem de Bruijn > > wrote: > > > > > > > > > @@ -3113,13 +3270,14 @@ static int virtnet_probe(struct > > > > > > virtio_device *vdev) > > > > > > u16 max_queue_pairs; > > > > > > int mtu; > > > > > > > > > > > > - /* Find if host supports multiqueue virtio_net device */ > > > > > > - err = virtio_cread_feature(vdev, VIRTIO_NET_F_MQ, > > > > > > - struct virtio_net_config, > > > > > > - max_virtqueue_pairs, > > > > > > _queue_pairs); > > > > > > + /* Find if host supports multiqueue/rss virtio_net device */ > > > > > > + max_queue_pairs = 1; > > > > > > + if (virtio_has_feature(vdev, VIRTIO_NET_F_MQ) || > > > > > > virtio_has_feature(vdev, VIRTIO_NET_F_RSS)) > > > > > > + max_queue_pairs = > > > > > > +virtio_cread16(vdev, offsetof(struct > > > > > > virtio_net_config, max_virtqueue_pairs)); > > > > > > > > > > Instead of testing either feature and treating them as somewhat equal, > > > > > shouldn't RSS be dependent on MQ? > > > > > > > > No, RSS is dependent on CTRL_VQ. Technically RSS and MQ are similar > > > > features. > > > > > > RSS depends on having multiple queues. > > > > > > What would enabling VIRTIO_NET_F_RSS without VIRTIO_NET_F_MQ do? > > > > RSS would work. > > What does that mean, exactly? RSS is load balancing, does that not > require multi-queue? It does, but VIRTIO_NET_F_MQ is a misnomer. \item[VIRTIO_NET_F_MQ(22)] Device supports multiqueue with automatic receive steering. VIRTIO_NET_F_RSS implies multi queue and does not depend on VIRTIO_NET_F_MQ. -- MST ___ Virtualization mailing list Virtualization@lists.linux-foundation.org https://lists.linuxfoundation.org/mailman/listinfo/virtualization
Re: [PATCH v3 2/4] drivers/net/virtio_net: Added basic RSS support.
On Thu, Feb 17, 2022 at 12:05 PM Andrew Melnichenko wrote: > > Hi all, > > On Mon, Feb 14, 2022 at 12:09 AM Willem de Bruijn > wrote: > > > > > > > @@ -3113,13 +3270,14 @@ static int virtnet_probe(struct virtio_device > > > > > *vdev) > > > > > u16 max_queue_pairs; > > > > > int mtu; > > > > > > > > > > - /* Find if host supports multiqueue virtio_net device */ > > > > > - err = virtio_cread_feature(vdev, VIRTIO_NET_F_MQ, > > > > > - struct virtio_net_config, > > > > > - max_virtqueue_pairs, > > > > > _queue_pairs); > > > > > + /* Find if host supports multiqueue/rss virtio_net device */ > > > > > + max_queue_pairs = 1; > > > > > + if (virtio_has_feature(vdev, VIRTIO_NET_F_MQ) || > > > > > virtio_has_feature(vdev, VIRTIO_NET_F_RSS)) > > > > > + max_queue_pairs = > > > > > +virtio_cread16(vdev, offsetof(struct > > > > > virtio_net_config, max_virtqueue_pairs)); > > > > > > > > Instead of testing either feature and treating them as somewhat equal, > > > > shouldn't RSS be dependent on MQ? > > > > > > No, RSS is dependent on CTRL_VQ. Technically RSS and MQ are similar > > > features. > > > > RSS depends on having multiple queues. > > > > What would enabling VIRTIO_NET_F_RSS without VIRTIO_NET_F_MQ do? > > RSS would work. What does that mean, exactly? RSS is load balancing, does that not require multi-queue? ___ Virtualization mailing list Virtualization@lists.linux-foundation.org https://lists.linuxfoundation.org/mailman/listinfo/virtualization
Re: [PATCH v3 2/4] drivers/net/virtio_net: Added basic RSS support.
Hi all, On Mon, Feb 14, 2022 at 12:09 AM Willem de Bruijn wrote: > > > > > @@ -3113,13 +3270,14 @@ static int virtnet_probe(struct virtio_device > > > > *vdev) > > > > u16 max_queue_pairs; > > > > int mtu; > > > > > > > > - /* Find if host supports multiqueue virtio_net device */ > > > > - err = virtio_cread_feature(vdev, VIRTIO_NET_F_MQ, > > > > - struct virtio_net_config, > > > > - max_virtqueue_pairs, > > > > _queue_pairs); > > > > + /* Find if host supports multiqueue/rss virtio_net device */ > > > > + max_queue_pairs = 1; > > > > + if (virtio_has_feature(vdev, VIRTIO_NET_F_MQ) || > > > > virtio_has_feature(vdev, VIRTIO_NET_F_RSS)) > > > > + max_queue_pairs = > > > > +virtio_cread16(vdev, offsetof(struct > > > > virtio_net_config, max_virtqueue_pairs)); > > > > > > Instead of testing either feature and treating them as somewhat equal, > > > shouldn't RSS be dependent on MQ? > > > > No, RSS is dependent on CTRL_VQ. Technically RSS and MQ are similar > > features. > > RSS depends on having multiple queues. > > What would enabling VIRTIO_NET_F_RSS without VIRTIO_NET_F_MQ do? RSS would work. According to virtio spec article 5.1.6.5.5: > A device MAY support one of these features or both. The driver MAY negotiate > any set of these features > that the device supports. Also, in 5.1.3.1: > VIRTIO_NET_F_RSS Requires VIRTIO_NET_F_CTRL_VQ. > > > > > > > > > > > > /* We need at least 2 queue's */ > > > > - if (err || max_queue_pairs < VIRTIO_NET_CTRL_MQ_VQ_PAIRS_MIN || > > > > + if (max_queue_pairs < VIRTIO_NET_CTRL_MQ_VQ_PAIRS_MIN || > > > > max_queue_pairs > VIRTIO_NET_CTRL_MQ_VQ_PAIRS_MAX || > > > > !virtio_has_feature(vdev, VIRTIO_NET_F_CTRL_VQ)) > > > > max_queue_pairs = 1; > > > > @@ -3207,6 +3365,23 @@ static int virtnet_probe(struct virtio_device > > > > *vdev) > > > > if (virtio_has_feature(vdev, VIRTIO_NET_F_MRG_RXBUF)) > > > > vi->mergeable_rx_bufs = true; > > > > > > > > + if (virtio_has_feature(vdev, VIRTIO_NET_F_RSS)) { > > > > + vi->has_rss = true; > > > > + vi->rss_indir_table_size = > > > > + virtio_cread16(vdev, offsetof(struct > > > > virtio_net_config, > > > > + rss_max_indirection_table_length)); > > > > + vi->rss_key_size = > > > > + virtio_cread8(vdev, offsetof(struct > > > > virtio_net_config, rss_max_key_size)); > > > > + > > > > + vi->rss_hash_types_supported = > > > > + virtio_cread32(vdev, offsetof(struct > > > > virtio_net_config, supported_hash_types)); > > > > + vi->rss_hash_types_supported &= > > > > + ~(VIRTIO_NET_RSS_HASH_TYPE_IP_EX | > > > > + VIRTIO_NET_RSS_HASH_TYPE_TCP_EX | > > > > + VIRTIO_NET_RSS_HASH_TYPE_UDP_EX); > > > > + > > > > + dev->hw_features |= NETIF_F_RXHASH; > > > > > > Only make the feature visible when the hash is actually reported in > > > the skb, patch 3. > > > > VirtioNET has two features: RSS(steering only) and hash(hash report in > > vnet header) > > Both features may be enabled/disabled separately: > > 1. rss on and hash off - packets steered to the corresponding vqs > > 2. rss off and hash on - packets steered by tap(like mq) but headers > > have properly calculated hash. > > 3. rss on and hash on - packets steered to corresponding vqs and hash > > is present in the header. > > > > RXHASH feature allows the user to enable/disable the rss/hash(any > > combination). > > I find that confusing, but.. I see that there is prior art where some > drivers enable/disable entire RSS load balancing based on this flag. > So ok. > > > I think it's a good idea to leave RXHASH in patch 2/4 to give the user > > ability to manipulate the rss only feature. > > But, if you think that it requires to move it to the 3/4, I'll do it. > > > > > > > > Also, clearly separate the feature patches (2) rss, (3) rxhash, (4) > > > rxhash config. > > > > Currently: > > Patch 2/4 - adds VirtioNet rss feature. > > Patch 3/4 - adds VirtioNet hash report feature. > > Patch 4/4 - adds the ability to manipulate supported hash types. > > > > Can you provide more detailed suggestions on how to move hunks? > > I gave one in the follow-on patch, to which you responded. That's probably it. I'll add zero size table check and move hunk for padded header length from 3/4 to 1/4. ___ Virtualization mailing list Virtualization@lists.linux-foundation.org https://lists.linuxfoundation.org/mailman/listinfo/virtualization
Re: [PATCH v3 2/4] drivers/net/virtio_net: Added basic RSS support.
> > > @@ -3113,13 +3270,14 @@ static int virtnet_probe(struct virtio_device > > > *vdev) > > > u16 max_queue_pairs; > > > int mtu; > > > > > > - /* Find if host supports multiqueue virtio_net device */ > > > - err = virtio_cread_feature(vdev, VIRTIO_NET_F_MQ, > > > - struct virtio_net_config, > > > - max_virtqueue_pairs, _queue_pairs); > > > + /* Find if host supports multiqueue/rss virtio_net device */ > > > + max_queue_pairs = 1; > > > + if (virtio_has_feature(vdev, VIRTIO_NET_F_MQ) || > > > virtio_has_feature(vdev, VIRTIO_NET_F_RSS)) > > > + max_queue_pairs = > > > +virtio_cread16(vdev, offsetof(struct > > > virtio_net_config, max_virtqueue_pairs)); > > > > Instead of testing either feature and treating them as somewhat equal, > > shouldn't RSS be dependent on MQ? > > No, RSS is dependent on CTRL_VQ. Technically RSS and MQ are similar features. RSS depends on having multiple queues. What would enabling VIRTIO_NET_F_RSS without VIRTIO_NET_F_MQ do? > > > > > > > > /* We need at least 2 queue's */ > > > - if (err || max_queue_pairs < VIRTIO_NET_CTRL_MQ_VQ_PAIRS_MIN || > > > + if (max_queue_pairs < VIRTIO_NET_CTRL_MQ_VQ_PAIRS_MIN || > > > max_queue_pairs > VIRTIO_NET_CTRL_MQ_VQ_PAIRS_MAX || > > > !virtio_has_feature(vdev, VIRTIO_NET_F_CTRL_VQ)) > > > max_queue_pairs = 1; > > > @@ -3207,6 +3365,23 @@ static int virtnet_probe(struct virtio_device > > > *vdev) > > > if (virtio_has_feature(vdev, VIRTIO_NET_F_MRG_RXBUF)) > > > vi->mergeable_rx_bufs = true; > > > > > > + if (virtio_has_feature(vdev, VIRTIO_NET_F_RSS)) { > > > + vi->has_rss = true; > > > + vi->rss_indir_table_size = > > > + virtio_cread16(vdev, offsetof(struct > > > virtio_net_config, > > > + rss_max_indirection_table_length)); > > > + vi->rss_key_size = > > > + virtio_cread8(vdev, offsetof(struct > > > virtio_net_config, rss_max_key_size)); > > > + > > > + vi->rss_hash_types_supported = > > > + virtio_cread32(vdev, offsetof(struct > > > virtio_net_config, supported_hash_types)); > > > + vi->rss_hash_types_supported &= > > > + ~(VIRTIO_NET_RSS_HASH_TYPE_IP_EX | > > > + VIRTIO_NET_RSS_HASH_TYPE_TCP_EX | > > > + VIRTIO_NET_RSS_HASH_TYPE_UDP_EX); > > > + > > > + dev->hw_features |= NETIF_F_RXHASH; > > > > Only make the feature visible when the hash is actually reported in > > the skb, patch 3. > > VirtioNET has two features: RSS(steering only) and hash(hash report in > vnet header) > Both features may be enabled/disabled separately: > 1. rss on and hash off - packets steered to the corresponding vqs > 2. rss off and hash on - packets steered by tap(like mq) but headers > have properly calculated hash. > 3. rss on and hash on - packets steered to corresponding vqs and hash > is present in the header. > > RXHASH feature allows the user to enable/disable the rss/hash(any > combination). I find that confusing, but.. I see that there is prior art where some drivers enable/disable entire RSS load balancing based on this flag. So ok. > I think it's a good idea to leave RXHASH in patch 2/4 to give the user > ability to manipulate the rss only feature. > But, if you think that it requires to move it to the 3/4, I'll do it. > > > > > Also, clearly separate the feature patches (2) rss, (3) rxhash, (4) > > rxhash config. > > Currently: > Patch 2/4 - adds VirtioNet rss feature. > Patch 3/4 - adds VirtioNet hash report feature. > Patch 4/4 - adds the ability to manipulate supported hash types. > > Can you provide more detailed suggestions on how to move hunks? I gave one in the follow-on patch, to which you responded. That's probably it. ___ Virtualization mailing list Virtualization@lists.linux-foundation.org https://lists.linuxfoundation.org/mailman/listinfo/virtualization
Re: [PATCH v3 2/4] drivers/net/virtio_net: Added basic RSS support.
Hi all, On Tue, Feb 8, 2022 at 10:37 PM Willem de Bruijn wrote: > > On Tue, Feb 8, 2022 at 1:19 PM Andrew Melnychenko wrote: > > > > Added features for RSS. > > Added initialization, RXHASH feature and ethtool ops. > > By default RSS/RXHASH is disabled. > > Virtio RSS "IPv6 extensions" hashes disabled. > > Added ethtools ops to set key and indirection table. > > > > Signed-off-by: Andrew Melnychenko > > --- > > drivers/net/virtio_net.c | 191 +-- > > 1 file changed, 185 insertions(+), 6 deletions(-) > > > > diff --git a/drivers/net/virtio_net.c b/drivers/net/virtio_net.c > > index 1404e683a2fd..495aed524e33 100644 > > --- a/drivers/net/virtio_net.c > > +++ b/drivers/net/virtio_net.c > > @@ -169,6 +169,24 @@ struct receive_queue { > > struct xdp_rxq_info xdp_rxq; > > }; > > > > +/* This structure can contain rss message with maximum settings for > > indirection table and keysize > > + * Note, that default structure that describes RSS configuration > > virtio_net_rss_config > > + * contains same info but can't handle table values. > > + * In any case, structure would be passed to virtio hw through sg_buf > > split by parts > > + * because table sizes may be differ according to the device configuration. > > + */ > > +#define VIRTIO_NET_RSS_MAX_KEY_SIZE 40 > > Future proof, may want to support larger sizes. > > netdevice.h defines NETDEV_RSS_KEY_LEN at 52. > > tools/testing/selftests/net/toeplitz.c supports up to 60 According to virtio specification, the length of the key is 40bytes(and an indirection table is 128 entries max). So for now, we support a maximum of the spec regardless of what the kernel is capable of. > > > +#define VIRTIO_NET_RSS_MAX_TABLE_LEN128 > > +struct virtio_net_ctrl_rss { > > + u32 hash_types; > > conversely, u32 is a bit extreme? No, the structure virtio_net_ctrl_rss is specified by the specification. > > > + u16 indirection_table_mask; > > + u16 unclassified_queue; > > + u16 indirection_table[VIRTIO_NET_RSS_MAX_TABLE_LEN]; > > + u16 max_tx_vq; > > + u8 hash_key_length; > > + u8 key[VIRTIO_NET_RSS_MAX_KEY_SIZE]; > > +}; > > + > > /* Control VQ buffers: protected by the rtnl lock */ > > struct control_buf { > > struct virtio_net_ctrl_hdr hdr; > > @@ -178,6 +196,7 @@ struct control_buf { > > u8 allmulti; > > __virtio16 vid; > > __virtio64 offloads; > > + struct virtio_net_ctrl_rss rss; > > }; > > > > struct virtnet_info { > > @@ -206,6 +225,12 @@ struct virtnet_info { > > /* Host will merge rx buffers for big packets (shake it! shake it!) > > */ > > bool mergeable_rx_bufs; > > > > + /* Host supports rss and/or hash report */ > > + bool has_rss; > > + u8 rss_key_size; > > + u16 rss_indir_table_size; > > + u32 rss_hash_types_supported; > > + > > /* Has control virtqueue */ > > bool has_cvq; > > > > @@ -2184,6 +2209,56 @@ static void virtnet_get_ringparam(struct net_device > > *dev, > > ring->tx_pending = ring->tx_max_pending; > > } > > > > +static bool virtnet_commit_rss_command(struct virtnet_info *vi) > > +{ > > + struct net_device *dev = vi->dev; > > + struct scatterlist sgs[4]; > > + unsigned int sg_buf_size; > > + > > + /* prepare sgs */ > > + sg_init_table(sgs, 4); > > + > > + sg_buf_size = offsetof(struct virtio_net_ctrl_rss, > > indirection_table); > > + sg_set_buf([0], >ctrl->rss, sg_buf_size); > > + > > + sg_buf_size = sizeof(uint16_t) * vi->rss_indir_table_size; > > + sg_set_buf([1], vi->ctrl->rss.indirection_table, sg_buf_size); > > + > > + sg_buf_size = offsetof(struct virtio_net_ctrl_rss, key) > > + - offsetof(struct virtio_net_ctrl_rss, max_tx_vq); > > + sg_set_buf([2], >ctrl->rss.max_tx_vq, sg_buf_size); > > + > > + sg_buf_size = vi->rss_key_size; > > + sg_set_buf([3], vi->ctrl->rss.key, sg_buf_size); > > + > > + if (!virtnet_send_command(vi, VIRTIO_NET_CTRL_MQ, > > + VIRTIO_NET_CTRL_MQ_RSS_CONFIG, sgs)) { > > + dev_warn(>dev, "VIRTIONET issue with committing RSS > > sgs\n"); > > + return false; > > + } > > + return true; > > +} > > + > > +static void virtnet_init_default_rss(struct virtnet_info *vi) > > +{ > > + u32 indir_val = 0; > > + int i = 0; > > + > > + vi->ctrl->rss.hash_types = vi->rss_hash_types_supported; > > + vi->ctrl->rss.indirection_table_mask = vi->rss_indir_table_size - 1; > > Is table size always a power of two? Yes, it should be. > > > + vi->ctrl->rss.unclassified_queue = 0; > > + > > + for (; i < vi->rss_indir_table_size; ++i) { > > + indir_val = ethtool_rxfh_indir_default(i, > > vi->curr_queue_pairs); > > + vi->ctrl->rss.indirection_table[i] = indir_val; > > + } > > + > > +
Re: [PATCH v3 2/4] drivers/net/virtio_net: Added basic RSS support.
On Tue, Feb 8, 2022 at 1:19 PM Andrew Melnychenko wrote: > > Added features for RSS. > Added initialization, RXHASH feature and ethtool ops. > By default RSS/RXHASH is disabled. > Virtio RSS "IPv6 extensions" hashes disabled. > Added ethtools ops to set key and indirection table. > > Signed-off-by: Andrew Melnychenko > --- > drivers/net/virtio_net.c | 191 +-- > 1 file changed, 185 insertions(+), 6 deletions(-) > > diff --git a/drivers/net/virtio_net.c b/drivers/net/virtio_net.c > index 1404e683a2fd..495aed524e33 100644 > --- a/drivers/net/virtio_net.c > +++ b/drivers/net/virtio_net.c > @@ -169,6 +169,24 @@ struct receive_queue { > struct xdp_rxq_info xdp_rxq; > }; > > +/* This structure can contain rss message with maximum settings for > indirection table and keysize > + * Note, that default structure that describes RSS configuration > virtio_net_rss_config > + * contains same info but can't handle table values. > + * In any case, structure would be passed to virtio hw through sg_buf split > by parts > + * because table sizes may be differ according to the device configuration. > + */ > +#define VIRTIO_NET_RSS_MAX_KEY_SIZE 40 Future proof, may want to support larger sizes. netdevice.h defines NETDEV_RSS_KEY_LEN at 52. tools/testing/selftests/net/toeplitz.c supports up to 60 > +#define VIRTIO_NET_RSS_MAX_TABLE_LEN128 > +struct virtio_net_ctrl_rss { > + u32 hash_types; conversely, u32 is a bit extreme? > + u16 indirection_table_mask; > + u16 unclassified_queue; > + u16 indirection_table[VIRTIO_NET_RSS_MAX_TABLE_LEN]; > + u16 max_tx_vq; > + u8 hash_key_length; > + u8 key[VIRTIO_NET_RSS_MAX_KEY_SIZE]; > +}; > + > /* Control VQ buffers: protected by the rtnl lock */ > struct control_buf { > struct virtio_net_ctrl_hdr hdr; > @@ -178,6 +196,7 @@ struct control_buf { > u8 allmulti; > __virtio16 vid; > __virtio64 offloads; > + struct virtio_net_ctrl_rss rss; > }; > > struct virtnet_info { > @@ -206,6 +225,12 @@ struct virtnet_info { > /* Host will merge rx buffers for big packets (shake it! shake it!) */ > bool mergeable_rx_bufs; > > + /* Host supports rss and/or hash report */ > + bool has_rss; > + u8 rss_key_size; > + u16 rss_indir_table_size; > + u32 rss_hash_types_supported; > + > /* Has control virtqueue */ > bool has_cvq; > > @@ -2184,6 +2209,56 @@ static void virtnet_get_ringparam(struct net_device > *dev, > ring->tx_pending = ring->tx_max_pending; > } > > +static bool virtnet_commit_rss_command(struct virtnet_info *vi) > +{ > + struct net_device *dev = vi->dev; > + struct scatterlist sgs[4]; > + unsigned int sg_buf_size; > + > + /* prepare sgs */ > + sg_init_table(sgs, 4); > + > + sg_buf_size = offsetof(struct virtio_net_ctrl_rss, indirection_table); > + sg_set_buf([0], >ctrl->rss, sg_buf_size); > + > + sg_buf_size = sizeof(uint16_t) * vi->rss_indir_table_size; > + sg_set_buf([1], vi->ctrl->rss.indirection_table, sg_buf_size); > + > + sg_buf_size = offsetof(struct virtio_net_ctrl_rss, key) > + - offsetof(struct virtio_net_ctrl_rss, max_tx_vq); > + sg_set_buf([2], >ctrl->rss.max_tx_vq, sg_buf_size); > + > + sg_buf_size = vi->rss_key_size; > + sg_set_buf([3], vi->ctrl->rss.key, sg_buf_size); > + > + if (!virtnet_send_command(vi, VIRTIO_NET_CTRL_MQ, > + VIRTIO_NET_CTRL_MQ_RSS_CONFIG, sgs)) { > + dev_warn(>dev, "VIRTIONET issue with committing RSS > sgs\n"); > + return false; > + } > + return true; > +} > + > +static void virtnet_init_default_rss(struct virtnet_info *vi) > +{ > + u32 indir_val = 0; > + int i = 0; > + > + vi->ctrl->rss.hash_types = vi->rss_hash_types_supported; > + vi->ctrl->rss.indirection_table_mask = vi->rss_indir_table_size - 1; Is table size always a power of two? > + vi->ctrl->rss.unclassified_queue = 0; > + > + for (; i < vi->rss_indir_table_size; ++i) { > + indir_val = ethtool_rxfh_indir_default(i, > vi->curr_queue_pairs); > + vi->ctrl->rss.indirection_table[i] = indir_val; > + } > + > + vi->ctrl->rss.max_tx_vq = vi->curr_queue_pairs; > + vi->ctrl->rss.hash_key_length = vi->rss_key_size; > + > + netdev_rss_key_fill(vi->ctrl->rss.key, vi->rss_key_size); > +} > + > > static void virtnet_get_drvinfo(struct net_device *dev, > struct ethtool_drvinfo *info) > @@ -2412,6 +2487,71 @@ static void virtnet_update_settings(struct > virtnet_info *vi) > vi->duplex = duplex; > } > > +static u32 virtnet_get_rxfh_key_size(struct net_device *dev) > +{ > + return ((struct virtnet_info *)netdev_priv(dev))->rss_key_size; > +} > + > +static u32
[PATCH v3 2/4] drivers/net/virtio_net: Added basic RSS support.
Added features for RSS. Added initialization, RXHASH feature and ethtool ops. By default RSS/RXHASH is disabled. Virtio RSS "IPv6 extensions" hashes disabled. Added ethtools ops to set key and indirection table. Signed-off-by: Andrew Melnychenko --- drivers/net/virtio_net.c | 191 +-- 1 file changed, 185 insertions(+), 6 deletions(-) diff --git a/drivers/net/virtio_net.c b/drivers/net/virtio_net.c index 1404e683a2fd..495aed524e33 100644 --- a/drivers/net/virtio_net.c +++ b/drivers/net/virtio_net.c @@ -169,6 +169,24 @@ struct receive_queue { struct xdp_rxq_info xdp_rxq; }; +/* This structure can contain rss message with maximum settings for indirection table and keysize + * Note, that default structure that describes RSS configuration virtio_net_rss_config + * contains same info but can't handle table values. + * In any case, structure would be passed to virtio hw through sg_buf split by parts + * because table sizes may be differ according to the device configuration. + */ +#define VIRTIO_NET_RSS_MAX_KEY_SIZE 40 +#define VIRTIO_NET_RSS_MAX_TABLE_LEN128 +struct virtio_net_ctrl_rss { + u32 hash_types; + u16 indirection_table_mask; + u16 unclassified_queue; + u16 indirection_table[VIRTIO_NET_RSS_MAX_TABLE_LEN]; + u16 max_tx_vq; + u8 hash_key_length; + u8 key[VIRTIO_NET_RSS_MAX_KEY_SIZE]; +}; + /* Control VQ buffers: protected by the rtnl lock */ struct control_buf { struct virtio_net_ctrl_hdr hdr; @@ -178,6 +196,7 @@ struct control_buf { u8 allmulti; __virtio16 vid; __virtio64 offloads; + struct virtio_net_ctrl_rss rss; }; struct virtnet_info { @@ -206,6 +225,12 @@ struct virtnet_info { /* Host will merge rx buffers for big packets (shake it! shake it!) */ bool mergeable_rx_bufs; + /* Host supports rss and/or hash report */ + bool has_rss; + u8 rss_key_size; + u16 rss_indir_table_size; + u32 rss_hash_types_supported; + /* Has control virtqueue */ bool has_cvq; @@ -2184,6 +2209,56 @@ static void virtnet_get_ringparam(struct net_device *dev, ring->tx_pending = ring->tx_max_pending; } +static bool virtnet_commit_rss_command(struct virtnet_info *vi) +{ + struct net_device *dev = vi->dev; + struct scatterlist sgs[4]; + unsigned int sg_buf_size; + + /* prepare sgs */ + sg_init_table(sgs, 4); + + sg_buf_size = offsetof(struct virtio_net_ctrl_rss, indirection_table); + sg_set_buf([0], >ctrl->rss, sg_buf_size); + + sg_buf_size = sizeof(uint16_t) * vi->rss_indir_table_size; + sg_set_buf([1], vi->ctrl->rss.indirection_table, sg_buf_size); + + sg_buf_size = offsetof(struct virtio_net_ctrl_rss, key) + - offsetof(struct virtio_net_ctrl_rss, max_tx_vq); + sg_set_buf([2], >ctrl->rss.max_tx_vq, sg_buf_size); + + sg_buf_size = vi->rss_key_size; + sg_set_buf([3], vi->ctrl->rss.key, sg_buf_size); + + if (!virtnet_send_command(vi, VIRTIO_NET_CTRL_MQ, + VIRTIO_NET_CTRL_MQ_RSS_CONFIG, sgs)) { + dev_warn(>dev, "VIRTIONET issue with committing RSS sgs\n"); + return false; + } + return true; +} + +static void virtnet_init_default_rss(struct virtnet_info *vi) +{ + u32 indir_val = 0; + int i = 0; + + vi->ctrl->rss.hash_types = vi->rss_hash_types_supported; + vi->ctrl->rss.indirection_table_mask = vi->rss_indir_table_size - 1; + vi->ctrl->rss.unclassified_queue = 0; + + for (; i < vi->rss_indir_table_size; ++i) { + indir_val = ethtool_rxfh_indir_default(i, vi->curr_queue_pairs); + vi->ctrl->rss.indirection_table[i] = indir_val; + } + + vi->ctrl->rss.max_tx_vq = vi->curr_queue_pairs; + vi->ctrl->rss.hash_key_length = vi->rss_key_size; + + netdev_rss_key_fill(vi->ctrl->rss.key, vi->rss_key_size); +} + static void virtnet_get_drvinfo(struct net_device *dev, struct ethtool_drvinfo *info) @@ -2412,6 +2487,71 @@ static void virtnet_update_settings(struct virtnet_info *vi) vi->duplex = duplex; } +static u32 virtnet_get_rxfh_key_size(struct net_device *dev) +{ + return ((struct virtnet_info *)netdev_priv(dev))->rss_key_size; +} + +static u32 virtnet_get_rxfh_indir_size(struct net_device *dev) +{ + return ((struct virtnet_info *)netdev_priv(dev))->rss_indir_table_size; +} + +static int virtnet_get_rxfh(struct net_device *dev, u32 *indir, u8 *key, u8 *hfunc) +{ + struct virtnet_info *vi = netdev_priv(dev); + int i; + + if (indir) { + for (i = 0; i < vi->rss_indir_table_size; ++i) + indir[i] = vi->ctrl->rss.indirection_table[i]; + } + + if (key) + memcpy(key, vi->ctrl->rss.key, vi->rss_key_size); + + if (hfunc) +