Commit d95fcdf4 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag 'for_linus' of git://git.kernel.org/pub/scm/linux/kernel/git/mst/vhost

Pull virtio updates from Michael Tsirkin:

 - Per vq sizes in vdpa

 - Info query for block devices support in vdpa

 - DMA sync callbacks in vduse

 - Fixes, cleanups

* tag 'for_linus' of git://git.kernel.org/pub/scm/linux/kernel/git/mst/vhost: (35 commits)
  virtio_net: rename free_old_xmit_skbs to free_old_xmit
  virtio_net: unify the code for recycling the xmit ptr
  virtio-net: add cond_resched() to the command waiting loop
  virtio-net: convert rx mode setting to use workqueue
  virtio: packed: fix unmap leak for indirect desc table
  vDPA: report virtio-blk flush info to user space
  vDPA: report virtio-block read-only info to user space
  vDPA: report virtio-block write zeroes configuration to user space
  vDPA: report virtio-block discarding configuration to user space
  vDPA: report virtio-block topology info to user space
  vDPA: report virtio-block MQ info to user space
  vDPA: report virtio-block max segments in a request to user space
  vDPA: report virtio-block block-size to user space
  vDPA: report virtio-block max segment size to user space
  vDPA: report virtio-block capacity to user space
  virtio: make virtio_bus const
  vdpa: make vdpa_bus const
  vDPA/ifcvf: implement vdpa_config_ops.get_vq_num_min
  vDPA/ifcvf: get_max_vq_size to return max size
  virtio_vdpa: create vqs with the actual size
  ...
parents 0815d5cc 5da7137d
...@@ -80,6 +80,11 @@ struct virtnet_stat_desc { ...@@ -80,6 +80,11 @@ struct virtnet_stat_desc {
size_t offset; size_t offset;
}; };
struct virtnet_sq_free_stats {
u64 packets;
u64 bytes;
};
struct virtnet_sq_stats { struct virtnet_sq_stats {
struct u64_stats_sync syncp; struct u64_stats_sync syncp;
u64_stats_t packets; u64_stats_t packets;
...@@ -304,6 +309,12 @@ struct virtnet_info { ...@@ -304,6 +309,12 @@ struct virtnet_info {
/* Work struct for config space updates */ /* Work struct for config space updates */
struct work_struct config_work; struct work_struct config_work;
/* Work struct for setting rx mode */
struct work_struct rx_mode_work;
/* OK to queue work setting RX mode? */
bool rx_mode_work_enabled;
/* Does the affinity hint is set for virtqueues? */ /* Does the affinity hint is set for virtqueues? */
bool affinity_hint_set; bool affinity_hint_set;
...@@ -366,6 +377,31 @@ static struct xdp_frame *ptr_to_xdp(void *ptr) ...@@ -366,6 +377,31 @@ static struct xdp_frame *ptr_to_xdp(void *ptr)
return (struct xdp_frame *)((unsigned long)ptr & ~VIRTIO_XDP_FLAG); return (struct xdp_frame *)((unsigned long)ptr & ~VIRTIO_XDP_FLAG);
} }
static void __free_old_xmit(struct send_queue *sq, bool in_napi,
struct virtnet_sq_free_stats *stats)
{
unsigned int len;
void *ptr;
while ((ptr = virtqueue_get_buf(sq->vq, &len)) != NULL) {
++stats->packets;
if (!is_xdp_frame(ptr)) {
struct sk_buff *skb = ptr;
pr_debug("Sent skb %p\n", skb);
stats->bytes += skb->len;
napi_consume_skb(skb, in_napi);
} else {
struct xdp_frame *frame = ptr_to_xdp(ptr);
stats->bytes += xdp_get_frame_len(frame);
xdp_return_frame(frame);
}
}
}
/* Converting between virtqueue no. and kernel tx/rx queue no. /* Converting between virtqueue no. and kernel tx/rx queue no.
* 0:rx0 1:tx0 2:rx1 3:tx1 ... 2N:rxN 2N+1:txN 2N+2:cvq * 0:rx0 1:tx0 2:rx1 3:tx1 ... 2N:rxN 2N+1:txN 2N+2:cvq
*/ */
...@@ -447,6 +483,20 @@ static void disable_delayed_refill(struct virtnet_info *vi) ...@@ -447,6 +483,20 @@ static void disable_delayed_refill(struct virtnet_info *vi)
spin_unlock_bh(&vi->refill_lock); spin_unlock_bh(&vi->refill_lock);
} }
static void enable_rx_mode_work(struct virtnet_info *vi)
{
rtnl_lock();
vi->rx_mode_work_enabled = true;
rtnl_unlock();
}
static void disable_rx_mode_work(struct virtnet_info *vi)
{
rtnl_lock();
vi->rx_mode_work_enabled = false;
rtnl_unlock();
}
static void virtqueue_napi_schedule(struct napi_struct *napi, static void virtqueue_napi_schedule(struct napi_struct *napi,
struct virtqueue *vq) struct virtqueue *vq)
{ {
...@@ -776,39 +826,21 @@ static void virtnet_rq_unmap_free_buf(struct virtqueue *vq, void *buf) ...@@ -776,39 +826,21 @@ static void virtnet_rq_unmap_free_buf(struct virtqueue *vq, void *buf)
virtnet_rq_free_buf(vi, rq, buf); virtnet_rq_free_buf(vi, rq, buf);
} }
static void free_old_xmit_skbs(struct send_queue *sq, bool in_napi) static void free_old_xmit(struct send_queue *sq, bool in_napi)
{ {
unsigned int len; struct virtnet_sq_free_stats stats = {0};
unsigned int packets = 0;
unsigned int bytes = 0;
void *ptr;
while ((ptr = virtqueue_get_buf(sq->vq, &len)) != NULL) {
if (likely(!is_xdp_frame(ptr))) {
struct sk_buff *skb = ptr;
pr_debug("Sent skb %p\n", skb);
bytes += skb->len;
napi_consume_skb(skb, in_napi);
} else {
struct xdp_frame *frame = ptr_to_xdp(ptr);
bytes += xdp_get_frame_len(frame); __free_old_xmit(sq, in_napi, &stats);
xdp_return_frame(frame);
}
packets++;
}
/* Avoid overhead when no packets have been processed /* Avoid overhead when no packets have been processed
* happens when called speculatively from start_xmit. * happens when called speculatively from start_xmit.
*/ */
if (!packets) if (!stats.packets)
return; return;
u64_stats_update_begin(&sq->stats.syncp); u64_stats_update_begin(&sq->stats.syncp);
u64_stats_add(&sq->stats.bytes, bytes); u64_stats_add(&sq->stats.bytes, stats.bytes);
u64_stats_add(&sq->stats.packets, packets); u64_stats_add(&sq->stats.packets, stats.packets);
u64_stats_update_end(&sq->stats.syncp); u64_stats_update_end(&sq->stats.syncp);
} }
...@@ -848,7 +880,7 @@ static void check_sq_full_and_disable(struct virtnet_info *vi, ...@@ -848,7 +880,7 @@ static void check_sq_full_and_disable(struct virtnet_info *vi,
virtqueue_napi_schedule(&sq->napi, sq->vq); virtqueue_napi_schedule(&sq->napi, sq->vq);
} else if (unlikely(!virtqueue_enable_cb_delayed(sq->vq))) { } else if (unlikely(!virtqueue_enable_cb_delayed(sq->vq))) {
/* More just got used, free them then recheck. */ /* More just got used, free them then recheck. */
free_old_xmit_skbs(sq, false); free_old_xmit(sq, false);
if (sq->vq->num_free >= 2+MAX_SKB_FRAGS) { if (sq->vq->num_free >= 2+MAX_SKB_FRAGS) {
netif_start_subqueue(dev, qnum); netif_start_subqueue(dev, qnum);
virtqueue_disable_cb(sq->vq); virtqueue_disable_cb(sq->vq);
...@@ -947,15 +979,12 @@ static int virtnet_xdp_xmit(struct net_device *dev, ...@@ -947,15 +979,12 @@ static int virtnet_xdp_xmit(struct net_device *dev,
int n, struct xdp_frame **frames, u32 flags) int n, struct xdp_frame **frames, u32 flags)
{ {
struct virtnet_info *vi = netdev_priv(dev); struct virtnet_info *vi = netdev_priv(dev);
struct virtnet_sq_free_stats stats = {0};
struct receive_queue *rq = vi->rq; struct receive_queue *rq = vi->rq;
struct bpf_prog *xdp_prog; struct bpf_prog *xdp_prog;
struct send_queue *sq; struct send_queue *sq;
unsigned int len;
int packets = 0;
int bytes = 0;
int nxmit = 0; int nxmit = 0;
int kicks = 0; int kicks = 0;
void *ptr;
int ret; int ret;
int i; int i;
...@@ -974,20 +1003,7 @@ static int virtnet_xdp_xmit(struct net_device *dev, ...@@ -974,20 +1003,7 @@ static int virtnet_xdp_xmit(struct net_device *dev,
} }
/* Free up any pending old buffers before queueing new ones. */ /* Free up any pending old buffers before queueing new ones. */
while ((ptr = virtqueue_get_buf(sq->vq, &len)) != NULL) { __free_old_xmit(sq, false, &stats);
if (likely(is_xdp_frame(ptr))) {
struct xdp_frame *frame = ptr_to_xdp(ptr);
bytes += xdp_get_frame_len(frame);
xdp_return_frame(frame);
} else {
struct sk_buff *skb = ptr;
bytes += skb->len;
napi_consume_skb(skb, false);
}
packets++;
}
for (i = 0; i < n; i++) { for (i = 0; i < n; i++) {
struct xdp_frame *xdpf = frames[i]; struct xdp_frame *xdpf = frames[i];
...@@ -1007,8 +1023,8 @@ static int virtnet_xdp_xmit(struct net_device *dev, ...@@ -1007,8 +1023,8 @@ static int virtnet_xdp_xmit(struct net_device *dev,
} }
out: out:
u64_stats_update_begin(&sq->stats.syncp); u64_stats_update_begin(&sq->stats.syncp);
u64_stats_add(&sq->stats.bytes, bytes); u64_stats_add(&sq->stats.bytes, stats.bytes);
u64_stats_add(&sq->stats.packets, packets); u64_stats_add(&sq->stats.packets, stats.packets);
u64_stats_add(&sq->stats.xdp_tx, n); u64_stats_add(&sq->stats.xdp_tx, n);
u64_stats_add(&sq->stats.xdp_tx_drops, n - nxmit); u64_stats_add(&sq->stats.xdp_tx_drops, n - nxmit);
u64_stats_add(&sq->stats.kicks, kicks); u64_stats_add(&sq->stats.kicks, kicks);
...@@ -2160,7 +2176,7 @@ static void virtnet_poll_cleantx(struct receive_queue *rq) ...@@ -2160,7 +2176,7 @@ static void virtnet_poll_cleantx(struct receive_queue *rq)
do { do {
virtqueue_disable_cb(sq->vq); virtqueue_disable_cb(sq->vq);
free_old_xmit_skbs(sq, true); free_old_xmit(sq, true);
} while (unlikely(!virtqueue_enable_cb_delayed(sq->vq))); } while (unlikely(!virtqueue_enable_cb_delayed(sq->vq)));
if (sq->vq->num_free >= 2 + MAX_SKB_FRAGS) if (sq->vq->num_free >= 2 + MAX_SKB_FRAGS)
...@@ -2308,7 +2324,7 @@ static int virtnet_poll_tx(struct napi_struct *napi, int budget) ...@@ -2308,7 +2324,7 @@ static int virtnet_poll_tx(struct napi_struct *napi, int budget)
txq = netdev_get_tx_queue(vi->dev, index); txq = netdev_get_tx_queue(vi->dev, index);
__netif_tx_lock(txq, raw_smp_processor_id()); __netif_tx_lock(txq, raw_smp_processor_id());
virtqueue_disable_cb(sq->vq); virtqueue_disable_cb(sq->vq);
free_old_xmit_skbs(sq, true); free_old_xmit(sq, true);
if (sq->vq->num_free >= 2 + MAX_SKB_FRAGS) if (sq->vq->num_free >= 2 + MAX_SKB_FRAGS)
netif_tx_wake_queue(txq); netif_tx_wake_queue(txq);
...@@ -2398,7 +2414,7 @@ static netdev_tx_t start_xmit(struct sk_buff *skb, struct net_device *dev) ...@@ -2398,7 +2414,7 @@ static netdev_tx_t start_xmit(struct sk_buff *skb, struct net_device *dev)
if (use_napi) if (use_napi)
virtqueue_disable_cb(sq->vq); virtqueue_disable_cb(sq->vq);
free_old_xmit_skbs(sq, false); free_old_xmit(sq, false);
} while (use_napi && kick && } while (use_napi && kick &&
unlikely(!virtqueue_enable_cb_delayed(sq->vq))); unlikely(!virtqueue_enable_cb_delayed(sq->vq)));
...@@ -2550,8 +2566,10 @@ static bool virtnet_send_command(struct virtnet_info *vi, u8 class, u8 cmd, ...@@ -2550,8 +2566,10 @@ static bool virtnet_send_command(struct virtnet_info *vi, u8 class, u8 cmd,
* into the hypervisor, so the request should be handled immediately. * into the hypervisor, so the request should be handled immediately.
*/ */
while (!virtqueue_get_buf(vi->cvq, &tmp) && while (!virtqueue_get_buf(vi->cvq, &tmp) &&
!virtqueue_is_broken(vi->cvq)) !virtqueue_is_broken(vi->cvq)) {
cond_resched();
cpu_relax(); cpu_relax();
}
return vi->ctrl->status == VIRTIO_NET_OK; return vi->ctrl->status == VIRTIO_NET_OK;
} }
...@@ -2706,9 +2724,11 @@ static int virtnet_close(struct net_device *dev) ...@@ -2706,9 +2724,11 @@ static int virtnet_close(struct net_device *dev)
return 0; return 0;
} }
static void virtnet_set_rx_mode(struct net_device *dev) static void virtnet_rx_mode_work(struct work_struct *work)
{ {
struct virtnet_info *vi = netdev_priv(dev); struct virtnet_info *vi =
container_of(work, struct virtnet_info, rx_mode_work);
struct net_device *dev = vi->dev;
struct scatterlist sg[2]; struct scatterlist sg[2];
struct virtio_net_ctrl_mac *mac_data; struct virtio_net_ctrl_mac *mac_data;
struct netdev_hw_addr *ha; struct netdev_hw_addr *ha;
...@@ -2721,6 +2741,8 @@ static void virtnet_set_rx_mode(struct net_device *dev) ...@@ -2721,6 +2741,8 @@ static void virtnet_set_rx_mode(struct net_device *dev)
if (!virtio_has_feature(vi->vdev, VIRTIO_NET_F_CTRL_RX)) if (!virtio_has_feature(vi->vdev, VIRTIO_NET_F_CTRL_RX))
return; return;
rtnl_lock();
vi->ctrl->promisc = ((dev->flags & IFF_PROMISC) != 0); vi->ctrl->promisc = ((dev->flags & IFF_PROMISC) != 0);
vi->ctrl->allmulti = ((dev->flags & IFF_ALLMULTI) != 0); vi->ctrl->allmulti = ((dev->flags & IFF_ALLMULTI) != 0);
...@@ -2738,14 +2760,19 @@ static void virtnet_set_rx_mode(struct net_device *dev) ...@@ -2738,14 +2760,19 @@ static void virtnet_set_rx_mode(struct net_device *dev)
dev_warn(&dev->dev, "Failed to %sable allmulti mode.\n", dev_warn(&dev->dev, "Failed to %sable allmulti mode.\n",
vi->ctrl->allmulti ? "en" : "dis"); vi->ctrl->allmulti ? "en" : "dis");
netif_addr_lock_bh(dev);
uc_count = netdev_uc_count(dev); uc_count = netdev_uc_count(dev);
mc_count = netdev_mc_count(dev); mc_count = netdev_mc_count(dev);
/* MAC filter - use one buffer for both lists */ /* MAC filter - use one buffer for both lists */
buf = kzalloc(((uc_count + mc_count) * ETH_ALEN) + buf = kzalloc(((uc_count + mc_count) * ETH_ALEN) +
(2 * sizeof(mac_data->entries)), GFP_ATOMIC); (2 * sizeof(mac_data->entries)), GFP_ATOMIC);
mac_data = buf; mac_data = buf;
if (!buf) if (!buf) {
netif_addr_unlock_bh(dev);
rtnl_unlock();
return; return;
}
sg_init_table(sg, 2); sg_init_table(sg, 2);
...@@ -2766,6 +2793,8 @@ static void virtnet_set_rx_mode(struct net_device *dev) ...@@ -2766,6 +2793,8 @@ static void virtnet_set_rx_mode(struct net_device *dev)
netdev_for_each_mc_addr(ha, dev) netdev_for_each_mc_addr(ha, dev)
memcpy(&mac_data->macs[i++][0], ha->addr, ETH_ALEN); memcpy(&mac_data->macs[i++][0], ha->addr, ETH_ALEN);
netif_addr_unlock_bh(dev);
sg_set_buf(&sg[1], mac_data, sg_set_buf(&sg[1], mac_data,
sizeof(mac_data->entries) + (mc_count * ETH_ALEN)); sizeof(mac_data->entries) + (mc_count * ETH_ALEN));
...@@ -2773,9 +2802,19 @@ static void virtnet_set_rx_mode(struct net_device *dev) ...@@ -2773,9 +2802,19 @@ static void virtnet_set_rx_mode(struct net_device *dev)
VIRTIO_NET_CTRL_MAC_TABLE_SET, sg)) VIRTIO_NET_CTRL_MAC_TABLE_SET, sg))
dev_warn(&dev->dev, "Failed to set MAC filter table.\n"); dev_warn(&dev->dev, "Failed to set MAC filter table.\n");
rtnl_unlock();
kfree(buf); kfree(buf);
} }
static void virtnet_set_rx_mode(struct net_device *dev)
{
struct virtnet_info *vi = netdev_priv(dev);
if (vi->rx_mode_work_enabled)
schedule_work(&vi->rx_mode_work);
}
static int virtnet_vlan_rx_add_vid(struct net_device *dev, static int virtnet_vlan_rx_add_vid(struct net_device *dev,
__be16 proto, u16 vid) __be16 proto, u16 vid)
{ {
...@@ -3856,6 +3895,8 @@ static void virtnet_freeze_down(struct virtio_device *vdev) ...@@ -3856,6 +3895,8 @@ static void virtnet_freeze_down(struct virtio_device *vdev)
/* Make sure no work handler is accessing the device */ /* Make sure no work handler is accessing the device */
flush_work(&vi->config_work); flush_work(&vi->config_work);
disable_rx_mode_work(vi);
flush_work(&vi->rx_mode_work);
netif_tx_lock_bh(vi->dev); netif_tx_lock_bh(vi->dev);
netif_device_detach(vi->dev); netif_device_detach(vi->dev);
...@@ -3878,6 +3919,7 @@ static int virtnet_restore_up(struct virtio_device *vdev) ...@@ -3878,6 +3919,7 @@ static int virtnet_restore_up(struct virtio_device *vdev)
virtio_device_ready(vdev); virtio_device_ready(vdev);
enable_delayed_refill(vi); enable_delayed_refill(vi);
enable_rx_mode_work(vi);
if (netif_running(vi->dev)) { if (netif_running(vi->dev)) {
err = virtnet_open(vi->dev); err = virtnet_open(vi->dev);
...@@ -4676,6 +4718,7 @@ static int virtnet_probe(struct virtio_device *vdev) ...@@ -4676,6 +4718,7 @@ static int virtnet_probe(struct virtio_device *vdev)
vdev->priv = vi; vdev->priv = vi;
INIT_WORK(&vi->config_work, virtnet_config_changed_work); INIT_WORK(&vi->config_work, virtnet_config_changed_work);
INIT_WORK(&vi->rx_mode_work, virtnet_rx_mode_work);
spin_lock_init(&vi->refill_lock); spin_lock_init(&vi->refill_lock);
if (virtio_has_feature(vdev, VIRTIO_NET_F_MRG_RXBUF)) { if (virtio_has_feature(vdev, VIRTIO_NET_F_MRG_RXBUF)) {
...@@ -4798,6 +4841,8 @@ static int virtnet_probe(struct virtio_device *vdev) ...@@ -4798,6 +4841,8 @@ static int virtnet_probe(struct virtio_device *vdev)
if (vi->has_rss || vi->has_rss_hash_report) if (vi->has_rss || vi->has_rss_hash_report)
virtnet_init_default_rss(vi); virtnet_init_default_rss(vi);
enable_rx_mode_work(vi);
/* serialize netdev register + virtio_device_ready() with ndo_open() */ /* serialize netdev register + virtio_device_ready() with ndo_open() */
rtnl_lock(); rtnl_lock();
...@@ -4895,6 +4940,8 @@ static void virtnet_remove(struct virtio_device *vdev) ...@@ -4895,6 +4940,8 @@ static void virtnet_remove(struct virtio_device *vdev)
/* Make sure no work handler is accessing the device. */ /* Make sure no work handler is accessing the device. */
flush_work(&vi->config_work); flush_work(&vi->config_work);
disable_rx_mode_work(vi);
flush_work(&vi->rx_mode_work);
unregister_netdev(vi->dev); unregister_netdev(vi->dev);
......
...@@ -254,6 +254,13 @@ static u16 eni_vdpa_get_vq_num_min(struct vdpa_device *vdpa) ...@@ -254,6 +254,13 @@ static u16 eni_vdpa_get_vq_num_min(struct vdpa_device *vdpa)
return vp_legacy_get_queue_size(ldev, 0); return vp_legacy_get_queue_size(ldev, 0);
} }
static u16 eni_vdpa_get_vq_size(struct vdpa_device *vdpa, u16 qid)
{
struct virtio_pci_legacy_device *ldev = vdpa_to_ldev(vdpa);
return vp_legacy_get_queue_size(ldev, qid);
}
static int eni_vdpa_get_vq_state(struct vdpa_device *vdpa, u16 qid, static int eni_vdpa_get_vq_state(struct vdpa_device *vdpa, u16 qid,
struct vdpa_vq_state *state) struct vdpa_vq_state *state)
{ {
...@@ -416,6 +423,7 @@ static const struct vdpa_config_ops eni_vdpa_ops = { ...@@ -416,6 +423,7 @@ static const struct vdpa_config_ops eni_vdpa_ops = {
.reset = eni_vdpa_reset, .reset = eni_vdpa_reset,
.get_vq_num_max = eni_vdpa_get_vq_num_max, .get_vq_num_max = eni_vdpa_get_vq_num_max,
.get_vq_num_min = eni_vdpa_get_vq_num_min, .get_vq_num_min = eni_vdpa_get_vq_num_min,
.get_vq_size = eni_vdpa_get_vq_size,
.get_vq_state = eni_vdpa_get_vq_state, .get_vq_state = eni_vdpa_get_vq_state,
.set_vq_state = eni_vdpa_set_vq_state, .set_vq_state = eni_vdpa_set_vq_state,
.set_vq_cb = eni_vdpa_set_vq_cb, .set_vq_cb = eni_vdpa_set_vq_cb,
......
...@@ -69,20 +69,19 @@ static int ifcvf_read_config_range(struct pci_dev *dev, ...@@ -69,20 +69,19 @@ static int ifcvf_read_config_range(struct pci_dev *dev,
return 0; return 0;
} }
static u16 ifcvf_get_vq_size(struct ifcvf_hw *hw, u16 qid) u16 ifcvf_get_vq_size(struct ifcvf_hw *hw, u16 qid)
{ {
u16 queue_size; u16 queue_size;
if (qid >= hw->nr_vring)
return 0;
vp_iowrite16(qid, &hw->common_cfg->queue_select); vp_iowrite16(qid, &hw->common_cfg->queue_select);
queue_size = vp_ioread16(&hw->common_cfg->queue_size); queue_size = vp_ioread16(&hw->common_cfg->queue_size);
return queue_size; return queue_size;
} }
/* This function returns the max allowed safe size for
* all virtqueues. It is the minimal size that can be
* suppprted by all virtqueues.
*/
u16 ifcvf_get_max_vq_size(struct ifcvf_hw *hw) u16 ifcvf_get_max_vq_size(struct ifcvf_hw *hw)
{ {
u16 queue_size, max_size, qid; u16 queue_size, max_size, qid;
...@@ -94,7 +93,7 @@ u16 ifcvf_get_max_vq_size(struct ifcvf_hw *hw) ...@@ -94,7 +93,7 @@ u16 ifcvf_get_max_vq_size(struct ifcvf_hw *hw)
if (!queue_size) if (!queue_size)
continue; continue;
max_size = min(queue_size, max_size); max_size = max(queue_size, max_size);
} }
return max_size; return max_size;
......
...@@ -28,6 +28,7 @@ ...@@ -28,6 +28,7 @@
#define IFCVF_PCI_MAX_RESOURCE 6 #define IFCVF_PCI_MAX_RESOURCE 6
#define IFCVF_LM_BAR 4 #define IFCVF_LM_BAR 4
#define IFCVF_MIN_VQ_SIZE 64
#define IFCVF_ERR(pdev, fmt, ...) dev_err(&pdev->dev, fmt, ##__VA_ARGS__) #define IFCVF_ERR(pdev, fmt, ...) dev_err(&pdev->dev, fmt, ##__VA_ARGS__)
#define IFCVF_DBG(pdev, fmt, ...) dev_dbg(&pdev->dev, fmt, ##__VA_ARGS__) #define IFCVF_DBG(pdev, fmt, ...) dev_dbg(&pdev->dev, fmt, ##__VA_ARGS__)
...@@ -131,4 +132,5 @@ void ifcvf_set_vq_ready(struct ifcvf_hw *hw, u16 qid, bool ready); ...@@ -131,4 +132,5 @@ void ifcvf_set_vq_ready(struct ifcvf_hw *hw, u16 qid, bool ready);
void ifcvf_set_driver_features(struct ifcvf_hw *hw, u64 features); void ifcvf_set_driver_features(struct ifcvf_hw *hw, u64 features);
u64 ifcvf_get_driver_features(struct ifcvf_hw *hw); u64 ifcvf_get_driver_features(struct ifcvf_hw *hw);
u16 ifcvf_get_max_vq_size(struct ifcvf_hw *hw); u16 ifcvf_get_max_vq_size(struct ifcvf_hw *hw);
u16 ifcvf_get_vq_size(struct ifcvf_hw *hw, u16 qid);
#endif /* _IFCVF_H_ */ #endif /* _IFCVF_H_ */
...@@ -456,6 +456,11 @@ static u16 ifcvf_vdpa_get_vq_num_max(struct vdpa_device *vdpa_dev) ...@@ -456,6 +456,11 @@ static u16 ifcvf_vdpa_get_vq_num_max(struct vdpa_device *vdpa_dev)
return ifcvf_get_max_vq_size(vf); return ifcvf_get_max_vq_size(vf);
} }
static u16 ifcvf_vdpa_get_vq_num_min(struct vdpa_device *vdpa_dev)
{
return IFCVF_MIN_VQ_SIZE;
}
static int ifcvf_vdpa_get_vq_state(struct vdpa_device *vdpa_dev, u16 qid, static int ifcvf_vdpa_get_vq_state(struct vdpa_device *vdpa_dev, u16 qid,
struct vdpa_vq_state *state) struct vdpa_vq_state *state)
{ {
...@@ -597,6 +602,14 @@ static int ifcvf_vdpa_get_vq_irq(struct vdpa_device *vdpa_dev, ...@@ -597,6 +602,14 @@ static int ifcvf_vdpa_get_vq_irq(struct vdpa_device *vdpa_dev,
return -EINVAL; return -EINVAL;
} }
static u16 ifcvf_vdpa_get_vq_size(struct vdpa_device *vdpa_dev,
u16 qid)
{
struct ifcvf_hw *vf = vdpa_to_vf(vdpa_dev);
return ifcvf_get_vq_size(vf, qid);
}
static struct vdpa_notification_area ifcvf_get_vq_notification(struct vdpa_device *vdpa_dev, static struct vdpa_notification_area ifcvf_get_vq_notification(struct vdpa_device *vdpa_dev,
u16 idx) u16 idx)
{ {
...@@ -624,6 +637,7 @@ static const struct vdpa_config_ops ifc_vdpa_ops = { ...@@ -624,6 +637,7 @@ static const struct vdpa_config_ops ifc_vdpa_ops = {
.set_status = ifcvf_vdpa_set_status, .set_status = ifcvf_vdpa_set_status,
.reset = ifcvf_vdpa_reset, .reset = ifcvf_vdpa_reset,
.get_vq_num_max = ifcvf_vdpa_get_vq_num_max, .get_vq_num_max = ifcvf_vdpa_get_vq_num_max,
.get_vq_num_min = ifcvf_vdpa_get_vq_num_min,
.get_vq_state = ifcvf_vdpa_get_vq_state, .get_vq_state = ifcvf_vdpa_get_vq_state,
.set_vq_state = ifcvf_vdpa_set_vq_state, .set_vq_state = ifcvf_vdpa_set_vq_state,
.set_vq_cb = ifcvf_vdpa_set_vq_cb, .set_vq_cb = ifcvf_vdpa_set_vq_cb,
...@@ -632,6 +646,7 @@ static const struct vdpa_config_ops ifc_vdpa_ops = { ...@@ -632,6 +646,7 @@ static const struct vdpa_config_ops ifc_vdpa_ops = {
.set_vq_num = ifcvf_vdpa_set_vq_num, .set_vq_num = ifcvf_vdpa_set_vq_num,
.set_vq_address = ifcvf_vdpa_set_vq_address, .set_vq_address = ifcvf_vdpa_set_vq_address,
.get_vq_irq = ifcvf_vdpa_get_vq_irq, .get_vq_irq = ifcvf_vdpa_get_vq_irq,
.get_vq_size = ifcvf_vdpa_get_vq_size,
.kick_vq = ifcvf_vdpa_kick_vq, .kick_vq = ifcvf_vdpa_kick_vq,
.get_generation = ifcvf_vdpa_get_generation, .get_generation = ifcvf_vdpa_get_generation,
.get_device_id = ifcvf_vdpa_get_device_id, .get_device_id = ifcvf_vdpa_get_device_id,
......
...@@ -151,8 +151,6 @@ static void teardown_driver(struct mlx5_vdpa_net *ndev); ...@@ -151,8 +151,6 @@ static void teardown_driver(struct mlx5_vdpa_net *ndev);
static bool mlx5_vdpa_debug; static bool mlx5_vdpa_debug;
#define MLX5_CVQ_MAX_ENT 16
#define MLX5_LOG_VIO_FLAG(_feature) \ #define MLX5_LOG_VIO_FLAG(_feature) \
do { \ do { \
if (features & BIT_ULL(_feature)) \ if (features & BIT_ULL(_feature)) \
...@@ -2276,9 +2274,16 @@ static void mlx5_vdpa_set_vq_num(struct vdpa_device *vdev, u16 idx, u32 num) ...@@ -2276,9 +2274,16 @@ static void mlx5_vdpa_set_vq_num(struct vdpa_device *vdev, u16 idx, u32 num)
struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev); struct mlx5_vdpa_net *ndev = to_mlx5_vdpa_ndev(mvdev);
struct mlx5_vdpa_virtqueue *mvq; struct mlx5_vdpa_virtqueue *mvq;
if (!is_index_valid(mvdev, idx) || is_ctrl_vq_idx(mvdev, idx)) if (!is_index_valid(mvdev, idx))
return; return;
if (is_ctrl_vq_idx(mvdev, idx)) {
struct mlx5_control_vq *cvq = &mvdev->cvq;
cvq->vring.vring.num = num;
return;
}
mvq = &ndev->vqs[idx]; mvq = &ndev->vqs[idx];
mvq->num_ent = num; mvq->num_ent = num;
} }
...@@ -2963,7 +2968,7 @@ static int setup_cvq_vring(struct mlx5_vdpa_dev *mvdev) ...@@ -2963,7 +2968,7 @@ static int setup_cvq_vring(struct mlx5_vdpa_dev *mvdev)
u16 idx = cvq->vring.last_avail_idx; u16 idx = cvq->vring.last_avail_idx;
err = vringh_init_iotlb(&cvq->vring, mvdev->actual_features, err = vringh_init_iotlb(&cvq->vring, mvdev->actual_features,
MLX5_CVQ_MAX_ENT, false, cvq->vring.vring.num, false,
(struct vring_desc *)(uintptr_t)cvq->desc_addr, (struct vring_desc *)(uintptr_t)cvq->desc_addr,
(struct vring_avail *)(uintptr_t)cvq->driver_addr, (struct vring_avail *)(uintptr_t)cvq->driver_addr,
(struct vring_used *)(uintptr_t)cvq->device_addr); (struct vring_used *)(uintptr_t)cvq->device_addr);
......
...@@ -93,8 +93,8 @@ static void pds_vdpa_remove(struct auxiliary_device *aux_dev) ...@@ -93,8 +93,8 @@ static void pds_vdpa_remove(struct auxiliary_device *aux_dev)
struct device *dev = &aux_dev->dev; struct device *dev = &aux_dev->dev;
vdpa_mgmtdev_unregister(&vdpa_aux->vdpa_mdev); vdpa_mgmtdev_unregister(&vdpa_aux->vdpa_mdev);
pds_vdpa_release_irqs(vdpa_aux->pdsv);
vp_modern_remove(&vdpa_aux->vd_mdev); vp_modern_remove(&vdpa_aux->vd_mdev);
pci_free_irq_vectors(vdpa_aux->padev->vf_pdev);
pds_vdpa_debugfs_del_vdpadev(vdpa_aux); pds_vdpa_debugfs_del_vdpadev(vdpa_aux);
kfree(vdpa_aux); kfree(vdpa_aux);
......
...@@ -426,12 +426,18 @@ static int pds_vdpa_request_irqs(struct pds_vdpa_device *pdsv) ...@@ -426,12 +426,18 @@ static int pds_vdpa_request_irqs(struct pds_vdpa_device *pdsv)
return err; return err;
} }
static void pds_vdpa_release_irqs(struct pds_vdpa_device *pdsv) void pds_vdpa_release_irqs(struct pds_vdpa_device *pdsv)
{ {
struct pci_dev *pdev = pdsv->vdpa_aux->padev->vf_pdev; struct pds_vdpa_aux *vdpa_aux;
struct pds_vdpa_aux *vdpa_aux = pdsv->vdpa_aux; struct pci_dev *pdev;
int qid; int qid;
if (!pdsv)
return;
pdev = pdsv->vdpa_aux->padev->vf_pdev;
vdpa_aux = pdsv->vdpa_aux;
if (!vdpa_aux->nintrs) if (!vdpa_aux->nintrs)
return; return;
...@@ -612,6 +618,7 @@ static int pds_vdpa_dev_add(struct vdpa_mgmt_dev *mdev, const char *name, ...@@ -612,6 +618,7 @@ static int pds_vdpa_dev_add(struct vdpa_mgmt_dev *mdev, const char *name,
struct device *dma_dev; struct device *dma_dev;
struct pci_dev *pdev; struct pci_dev *pdev;
struct device *dev; struct device *dev;
u8 status;
int err; int err;
int i; int i;
...@@ -638,6 +645,13 @@ static int pds_vdpa_dev_add(struct vdpa_mgmt_dev *mdev, const char *name, ...@@ -638,6 +645,13 @@ static int pds_vdpa_dev_add(struct vdpa_mgmt_dev *mdev, const char *name,
dma_dev = &pdev->dev; dma_dev = &pdev->dev;
pdsv->vdpa_dev.dma_dev = dma_dev; pdsv->vdpa_dev.dma_dev = dma_dev;
status = pds_vdpa_get_status(&pdsv->vdpa_dev);
if (status == 0xff) {
dev_err(dev, "Broken PCI - status %#x\n", status);
err = -ENXIO;
goto err_unmap;
}
pdsv->supported_features = mgmt->supported_features; pdsv->supported_features = mgmt->supported_features;
if (add_config->mask & BIT_ULL(VDPA_ATTR_DEV_FEATURES)) { if (add_config->mask & BIT_ULL(VDPA_ATTR_DEV_FEATURES)) {
......
...@@ -46,5 +46,6 @@ struct pds_vdpa_device { ...@@ -46,5 +46,6 @@ struct pds_vdpa_device {
#define PDS_VDPA_PACKED_INVERT_IDX 0x8000 #define PDS_VDPA_PACKED_INVERT_IDX 0x8000
void pds_vdpa_release_irqs(struct pds_vdpa_device *pdsv);
int pds_vdpa_get_mgmt_info(struct pds_vdpa_aux *vdpa_aux); int pds_vdpa_get_mgmt_info(struct pds_vdpa_aux *vdpa_aux);
#endif /* _VDPA_DEV_H_ */ #endif /* _VDPA_DEV_H_ */
...@@ -115,7 +115,7 @@ static const struct attribute_group vdpa_dev_group = { ...@@ -115,7 +115,7 @@ static const struct attribute_group vdpa_dev_group = {
}; };
__ATTRIBUTE_GROUPS(vdpa_dev); __ATTRIBUTE_GROUPS(vdpa_dev);
static struct bus_type vdpa_bus = { static const struct bus_type vdpa_bus = {
.name = "vdpa", .name = "vdpa",
.dev_groups = vdpa_dev_groups, .dev_groups = vdpa_dev_groups,
.match = vdpa_dev_match, .match = vdpa_dev_match,
...@@ -944,6 +944,215 @@ static int vdpa_dev_net_config_fill(struct vdpa_device *vdev, struct sk_buff *ms ...@@ -944,6 +944,215 @@ static int vdpa_dev_net_config_fill(struct vdpa_device *vdev, struct sk_buff *ms
return vdpa_dev_net_mq_config_fill(msg, features_device, &config); return vdpa_dev_net_mq_config_fill(msg, features_device, &config);
} }
static int
vdpa_dev_blk_capacity_config_fill(struct sk_buff *msg,
const struct virtio_blk_config *config)
{
u64 val_u64;
val_u64 = __virtio64_to_cpu(true, config->capacity);
return nla_put_u64_64bit(msg, VDPA_ATTR_DEV_BLK_CFG_CAPACITY,
val_u64, VDPA_ATTR_PAD);
}
static int
vdpa_dev_blk_seg_size_config_fill(struct sk_buff *msg, u64 features,
const struct virtio_blk_config *config)
{
u32 val_u32;
if ((features & BIT_ULL(VIRTIO_BLK_F_SIZE_MAX)) == 0)
return 0;
val_u32 = __virtio32_to_cpu(true, config->size_max);
return nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_SEG_SIZE, val_u32);
}
/* fill the block size*/
static int
vdpa_dev_blk_block_size_config_fill(struct sk_buff *msg, u64 features,
const struct virtio_blk_config *config)
{
u32 val_u32;
if ((features & BIT_ULL(VIRTIO_BLK_F_BLK_SIZE)) == 0)
return 0;
val_u32 = __virtio32_to_cpu(true, config->blk_size);
return nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_BLK_SIZE, val_u32);
}
static int
vdpa_dev_blk_seg_max_config_fill(struct sk_buff *msg, u64 features,
const struct virtio_blk_config *config)
{
u32 val_u32;
if ((features & BIT_ULL(VIRTIO_BLK_F_SEG_MAX)) == 0)
return 0;
val_u32 = __virtio32_to_cpu(true, config->seg_max);
return nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_SEG_MAX, val_u32);
}
static int vdpa_dev_blk_mq_config_fill(struct sk_buff *msg, u64 features,
const struct virtio_blk_config *config)
{
u16 val_u16;
if ((features & BIT_ULL(VIRTIO_BLK_F_MQ)) == 0)
return 0;
val_u16 = __virtio16_to_cpu(true, config->num_queues);
return nla_put_u16(msg, VDPA_ATTR_DEV_BLK_CFG_NUM_QUEUES, val_u16);
}
static int vdpa_dev_blk_topology_config_fill(struct sk_buff *msg, u64 features,
const struct virtio_blk_config *config)
{
u16 min_io_size;
u32 opt_io_size;
if ((features & BIT_ULL(VIRTIO_BLK_F_TOPOLOGY)) == 0)
return 0;
min_io_size = __virtio16_to_cpu(true, config->min_io_size);
opt_io_size = __virtio32_to_cpu(true, config->opt_io_size);
if (nla_put_u8(msg, VDPA_ATTR_DEV_BLK_CFG_PHY_BLK_EXP,
config->physical_block_exp))
return -EMSGSIZE;
if (nla_put_u8(msg, VDPA_ATTR_DEV_BLK_CFG_ALIGN_OFFSET,
config->alignment_offset))
return -EMSGSIZE;
if (nla_put_u16(msg, VDPA_ATTR_DEV_BLK_CFG_MIN_IO_SIZE, min_io_size))
return -EMSGSIZE;
if (nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_OPT_IO_SIZE, opt_io_size))
return -EMSGSIZE;
return 0;
}
static int vdpa_dev_blk_discard_config_fill(struct sk_buff *msg, u64 features,
const struct virtio_blk_config *config)
{
u32 val_u32;
if ((features & BIT_ULL(VIRTIO_BLK_F_DISCARD)) == 0)
return 0;
val_u32 = __virtio32_to_cpu(true, config->max_discard_sectors);
if (nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_MAX_DISCARD_SEC, val_u32))
return -EMSGSIZE;
val_u32 = __virtio32_to_cpu(true, config->max_discard_seg);
if (nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_MAX_DISCARD_SEG, val_u32))
return -EMSGSIZE;
val_u32 = __virtio32_to_cpu(true, config->discard_sector_alignment);
if (nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_DISCARD_SEC_ALIGN, val_u32))
return -EMSGSIZE;
return 0;
}
static int
vdpa_dev_blk_write_zeroes_config_fill(struct sk_buff *msg, u64 features,
const struct virtio_blk_config *config)
{
u32 val_u32;
if ((features & BIT_ULL(VIRTIO_BLK_F_WRITE_ZEROES)) == 0)
return 0;
val_u32 = __virtio32_to_cpu(true, config->max_write_zeroes_sectors);
if (nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_MAX_WRITE_ZEROES_SEC, val_u32))
return -EMSGSIZE;
val_u32 = __virtio32_to_cpu(true, config->max_write_zeroes_seg);
if (nla_put_u32(msg, VDPA_ATTR_DEV_BLK_CFG_MAX_WRITE_ZEROES_SEG, val_u32))
return -EMSGSIZE;
return 0;
}
static int vdpa_dev_blk_ro_config_fill(struct sk_buff *msg, u64 features)
{
u8 ro;
ro = ((features & BIT_ULL(VIRTIO_BLK_F_RO)) == 0) ? 0 : 1;
if (nla_put_u8(msg, VDPA_ATTR_DEV_BLK_CFG_READ_ONLY, ro))
return -EMSGSIZE;
return 0;
}
static int vdpa_dev_blk_flush_config_fill(struct sk_buff *msg, u64 features)
{
u8 flush;
flush = ((features & BIT_ULL(VIRTIO_BLK_F_FLUSH)) == 0) ? 0 : 1;
if (nla_put_u8(msg, VDPA_ATTR_DEV_BLK_CFG_FLUSH, flush))
return -EMSGSIZE;
return 0;
}
static int vdpa_dev_blk_config_fill(struct vdpa_device *vdev,
struct sk_buff *msg)
{
struct virtio_blk_config config = {};
u64 features_device;
vdev->config->get_config(vdev, 0, &config, sizeof(config));
features_device = vdev->config->get_device_features(vdev);
if (nla_put_u64_64bit(msg, VDPA_ATTR_DEV_FEATURES, features_device,
VDPA_ATTR_PAD))
return -EMSGSIZE;
if (vdpa_dev_blk_capacity_config_fill(msg, &config))
return -EMSGSIZE;
if (vdpa_dev_blk_seg_size_config_fill(msg, features_device, &config))
return -EMSGSIZE;
if (vdpa_dev_blk_block_size_config_fill(msg, features_device, &config))
return -EMSGSIZE;
if (vdpa_dev_blk_seg_max_config_fill(msg, features_device, &config))
return -EMSGSIZE;
if (vdpa_dev_blk_mq_config_fill(msg, features_device, &config))
return -EMSGSIZE;
if (vdpa_dev_blk_topology_config_fill(msg, features_device, &config))
return -EMSGSIZE;
if (vdpa_dev_blk_discard_config_fill(msg, features_device, &config))
return -EMSGSIZE;
if (vdpa_dev_blk_write_zeroes_config_fill(msg, features_device, &config))
return -EMSGSIZE;
if (vdpa_dev_blk_ro_config_fill(msg, features_device))
return -EMSGSIZE;
if (vdpa_dev_blk_flush_config_fill(msg, features_device))
return -EMSGSIZE;
return 0;
}
static int static int
vdpa_dev_config_fill(struct vdpa_device *vdev, struct sk_buff *msg, u32 portid, u32 seq, vdpa_dev_config_fill(struct vdpa_device *vdev, struct sk_buff *msg, u32 portid, u32 seq,
int flags, struct netlink_ext_ack *extack) int flags, struct netlink_ext_ack *extack)
...@@ -988,6 +1197,9 @@ vdpa_dev_config_fill(struct vdpa_device *vdev, struct sk_buff *msg, u32 portid, ...@@ -988,6 +1197,9 @@ vdpa_dev_config_fill(struct vdpa_device *vdev, struct sk_buff *msg, u32 portid,
case VIRTIO_ID_NET: case VIRTIO_ID_NET:
err = vdpa_dev_net_config_fill(vdev, msg); err = vdpa_dev_net_config_fill(vdev, msg);
break; break;
case VIRTIO_ID_BLOCK:
err = vdpa_dev_blk_config_fill(vdev, msg);
break;
default: default:
err = -EOPNOTSUPP; err = -EOPNOTSUPP;
break; break;
......
...@@ -160,7 +160,7 @@ static void vdpasim_do_reset(struct vdpasim *vdpasim, u32 flags) ...@@ -160,7 +160,7 @@ static void vdpasim_do_reset(struct vdpasim *vdpasim, u32 flags)
} }
} }
vdpasim->running = true; vdpasim->running = false;
spin_unlock(&vdpasim->iommu_lock); spin_unlock(&vdpasim->iommu_lock);
vdpasim->features = 0; vdpasim->features = 0;
...@@ -311,6 +311,17 @@ static void vdpasim_set_vq_num(struct vdpa_device *vdpa, u16 idx, u32 num) ...@@ -311,6 +311,17 @@ static void vdpasim_set_vq_num(struct vdpa_device *vdpa, u16 idx, u32 num)
vq->num = num; vq->num = num;
} }
static u16 vdpasim_get_vq_size(struct vdpa_device *vdpa, u16 idx)
{
struct vdpasim *vdpasim = vdpa_to_sim(vdpa);
struct vdpasim_virtqueue *vq = &vdpasim->vqs[idx];
if (vdpasim->status & VIRTIO_CONFIG_S_DRIVER_OK)
return vq->num;
else
return VDPASIM_QUEUE_MAX;
}
static void vdpasim_kick_vq(struct vdpa_device *vdpa, u16 idx) static void vdpasim_kick_vq(struct vdpa_device *vdpa, u16 idx)
{ {
struct vdpasim *vdpasim = vdpa_to_sim(vdpa); struct vdpasim *vdpasim = vdpa_to_sim(vdpa);
...@@ -483,6 +494,7 @@ static void vdpasim_set_status(struct vdpa_device *vdpa, u8 status) ...@@ -483,6 +494,7 @@ static void vdpasim_set_status(struct vdpa_device *vdpa, u8 status)
mutex_lock(&vdpasim->mutex); mutex_lock(&vdpasim->mutex);
vdpasim->status = status; vdpasim->status = status;
vdpasim->running = (status & VIRTIO_CONFIG_S_DRIVER_OK) != 0;
mutex_unlock(&vdpasim->mutex); mutex_unlock(&vdpasim->mutex);
} }
...@@ -774,6 +786,7 @@ static const struct vdpa_config_ops vdpasim_config_ops = { ...@@ -774,6 +786,7 @@ static const struct vdpa_config_ops vdpasim_config_ops = {
.get_driver_features = vdpasim_get_driver_features, .get_driver_features = vdpasim_get_driver_features,
.set_config_cb = vdpasim_set_config_cb, .set_config_cb = vdpasim_set_config_cb,
.get_vq_num_max = vdpasim_get_vq_num_max, .get_vq_num_max = vdpasim_get_vq_num_max,
.get_vq_size = vdpasim_get_vq_size,
.get_device_id = vdpasim_get_device_id, .get_device_id = vdpasim_get_device_id,
.get_vendor_id = vdpasim_get_vendor_id, .get_vendor_id = vdpasim_get_vendor_id,
.get_status = vdpasim_get_status, .get_status = vdpasim_get_status,
......
...@@ -373,6 +373,26 @@ static void vduse_domain_free_iova(struct iova_domain *iovad, ...@@ -373,6 +373,26 @@ static void vduse_domain_free_iova(struct iova_domain *iovad,
free_iova_fast(iovad, iova >> shift, iova_len); free_iova_fast(iovad, iova >> shift, iova_len);
} }
void vduse_domain_sync_single_for_device(struct vduse_iova_domain *domain,
dma_addr_t dma_addr, size_t size,
enum dma_data_direction dir)
{
read_lock(&domain->bounce_lock);
if (dir == DMA_TO_DEVICE || dir == DMA_BIDIRECTIONAL)
vduse_domain_bounce(domain, dma_addr, size, DMA_TO_DEVICE);
read_unlock(&domain->bounce_lock);
}
void vduse_domain_sync_single_for_cpu(struct vduse_iova_domain *domain,
dma_addr_t dma_addr, size_t size,
enum dma_data_direction dir)
{
read_lock(&domain->bounce_lock);
if (dir == DMA_FROM_DEVICE || dir == DMA_BIDIRECTIONAL)
vduse_domain_bounce(domain, dma_addr, size, DMA_FROM_DEVICE);
read_unlock(&domain->bounce_lock);
}
dma_addr_t vduse_domain_map_page(struct vduse_iova_domain *domain, dma_addr_t vduse_domain_map_page(struct vduse_iova_domain *domain,
struct page *page, unsigned long offset, struct page *page, unsigned long offset,
size_t size, enum dma_data_direction dir, size_t size, enum dma_data_direction dir,
...@@ -393,7 +413,8 @@ dma_addr_t vduse_domain_map_page(struct vduse_iova_domain *domain, ...@@ -393,7 +413,8 @@ dma_addr_t vduse_domain_map_page(struct vduse_iova_domain *domain,
if (vduse_domain_map_bounce_page(domain, (u64)iova, (u64)size, pa)) if (vduse_domain_map_bounce_page(domain, (u64)iova, (u64)size, pa))
goto err_unlock; goto err_unlock;
if (dir == DMA_TO_DEVICE || dir == DMA_BIDIRECTIONAL) if (!(attrs & DMA_ATTR_SKIP_CPU_SYNC) &&
(dir == DMA_TO_DEVICE || dir == DMA_BIDIRECTIONAL))
vduse_domain_bounce(domain, iova, size, DMA_TO_DEVICE); vduse_domain_bounce(domain, iova, size, DMA_TO_DEVICE);
read_unlock(&domain->bounce_lock); read_unlock(&domain->bounce_lock);
...@@ -411,9 +432,9 @@ void vduse_domain_unmap_page(struct vduse_iova_domain *domain, ...@@ -411,9 +432,9 @@ void vduse_domain_unmap_page(struct vduse_iova_domain *domain,
enum dma_data_direction dir, unsigned long attrs) enum dma_data_direction dir, unsigned long attrs)
{ {
struct iova_domain *iovad = &domain->stream_iovad; struct iova_domain *iovad = &domain->stream_iovad;
read_lock(&domain->bounce_lock); read_lock(&domain->bounce_lock);
if (dir == DMA_FROM_DEVICE || dir == DMA_BIDIRECTIONAL) if (!(attrs & DMA_ATTR_SKIP_CPU_SYNC) &&
(dir == DMA_FROM_DEVICE || dir == DMA_BIDIRECTIONAL))
vduse_domain_bounce(domain, dma_addr, size, DMA_FROM_DEVICE); vduse_domain_bounce(domain, dma_addr, size, DMA_FROM_DEVICE);
vduse_domain_unmap_bounce_page(domain, (u64)dma_addr, (u64)size); vduse_domain_unmap_bounce_page(domain, (u64)dma_addr, (u64)size);
......
...@@ -44,6 +44,14 @@ int vduse_domain_set_map(struct vduse_iova_domain *domain, ...@@ -44,6 +44,14 @@ int vduse_domain_set_map(struct vduse_iova_domain *domain,
void vduse_domain_clear_map(struct vduse_iova_domain *domain, void vduse_domain_clear_map(struct vduse_iova_domain *domain,
struct vhost_iotlb *iotlb); struct vhost_iotlb *iotlb);
void vduse_domain_sync_single_for_device(struct vduse_iova_domain *domain,
dma_addr_t dma_addr, size_t size,
enum dma_data_direction dir);
void vduse_domain_sync_single_for_cpu(struct vduse_iova_domain *domain,
dma_addr_t dma_addr, size_t size,
enum dma_data_direction dir);
dma_addr_t vduse_domain_map_page(struct vduse_iova_domain *domain, dma_addr_t vduse_domain_map_page(struct vduse_iova_domain *domain,
struct page *page, unsigned long offset, struct page *page, unsigned long offset,
size_t size, enum dma_data_direction dir, size_t size, enum dma_data_direction dir,
......
...@@ -541,6 +541,17 @@ static void vduse_vdpa_set_vq_num(struct vdpa_device *vdpa, u16 idx, u32 num) ...@@ -541,6 +541,17 @@ static void vduse_vdpa_set_vq_num(struct vdpa_device *vdpa, u16 idx, u32 num)
vq->num = num; vq->num = num;
} }
static u16 vduse_vdpa_get_vq_size(struct vdpa_device *vdpa, u16 idx)
{
struct vduse_dev *dev = vdpa_to_vduse(vdpa);
struct vduse_virtqueue *vq = dev->vqs[idx];
if (vq->num)
return vq->num;
else
return vq->num_max;
}
static void vduse_vdpa_set_vq_ready(struct vdpa_device *vdpa, static void vduse_vdpa_set_vq_ready(struct vdpa_device *vdpa,
u16 idx, bool ready) u16 idx, bool ready)
{ {
...@@ -773,6 +784,7 @@ static const struct vdpa_config_ops vduse_vdpa_config_ops = { ...@@ -773,6 +784,7 @@ static const struct vdpa_config_ops vduse_vdpa_config_ops = {
.kick_vq = vduse_vdpa_kick_vq, .kick_vq = vduse_vdpa_kick_vq,
.set_vq_cb = vduse_vdpa_set_vq_cb, .set_vq_cb = vduse_vdpa_set_vq_cb,
.set_vq_num = vduse_vdpa_set_vq_num, .set_vq_num = vduse_vdpa_set_vq_num,
.get_vq_size = vduse_vdpa_get_vq_size,
.set_vq_ready = vduse_vdpa_set_vq_ready, .set_vq_ready = vduse_vdpa_set_vq_ready,
.get_vq_ready = vduse_vdpa_get_vq_ready, .get_vq_ready = vduse_vdpa_get_vq_ready,
.set_vq_state = vduse_vdpa_set_vq_state, .set_vq_state = vduse_vdpa_set_vq_state,
...@@ -798,6 +810,26 @@ static const struct vdpa_config_ops vduse_vdpa_config_ops = { ...@@ -798,6 +810,26 @@ static const struct vdpa_config_ops vduse_vdpa_config_ops = {
.free = vduse_vdpa_free, .free = vduse_vdpa_free,
}; };
static void vduse_dev_sync_single_for_device(struct device *dev,
dma_addr_t dma_addr, size_t size,
enum dma_data_direction dir)
{
struct vduse_dev *vdev = dev_to_vduse(dev);
struct vduse_iova_domain *domain = vdev->domain;
vduse_domain_sync_single_for_device(domain, dma_addr, size, dir);
}
static void vduse_dev_sync_single_for_cpu(struct device *dev,
dma_addr_t dma_addr, size_t size,
enum dma_data_direction dir)
{
struct vduse_dev *vdev = dev_to_vduse(dev);
struct vduse_iova_domain *domain = vdev->domain;
vduse_domain_sync_single_for_cpu(domain, dma_addr, size, dir);
}
static dma_addr_t vduse_dev_map_page(struct device *dev, struct page *page, static dma_addr_t vduse_dev_map_page(struct device *dev, struct page *page,
unsigned long offset, size_t size, unsigned long offset, size_t size,
enum dma_data_direction dir, enum dma_data_direction dir,
...@@ -858,6 +890,8 @@ static size_t vduse_dev_max_mapping_size(struct device *dev) ...@@ -858,6 +890,8 @@ static size_t vduse_dev_max_mapping_size(struct device *dev)
} }
static const struct dma_map_ops vduse_dev_dma_ops = { static const struct dma_map_ops vduse_dev_dma_ops = {
.sync_single_for_device = vduse_dev_sync_single_for_device,
.sync_single_for_cpu = vduse_dev_sync_single_for_cpu,
.map_page = vduse_dev_map_page, .map_page = vduse_dev_map_page,
.unmap_page = vduse_dev_unmap_page, .unmap_page = vduse_dev_unmap_page,
.alloc = vduse_dev_alloc_coherent, .alloc = vduse_dev_alloc_coherent,
......
...@@ -328,6 +328,13 @@ static void vp_vdpa_set_vq_num(struct vdpa_device *vdpa, u16 qid, ...@@ -328,6 +328,13 @@ static void vp_vdpa_set_vq_num(struct vdpa_device *vdpa, u16 qid,
vp_modern_set_queue_size(mdev, qid, num); vp_modern_set_queue_size(mdev, qid, num);
} }
static u16 vp_vdpa_get_vq_size(struct vdpa_device *vdpa, u16 qid)
{
struct virtio_pci_modern_device *mdev = vdpa_to_mdev(vdpa);
return vp_modern_get_queue_size(mdev, qid);
}
static int vp_vdpa_set_vq_address(struct vdpa_device *vdpa, u16 qid, static int vp_vdpa_set_vq_address(struct vdpa_device *vdpa, u16 qid,
u64 desc_area, u64 driver_area, u64 desc_area, u64 driver_area,
u64 device_area) u64 device_area)
...@@ -449,6 +456,7 @@ static const struct vdpa_config_ops vp_vdpa_ops = { ...@@ -449,6 +456,7 @@ static const struct vdpa_config_ops vp_vdpa_ops = {
.set_vq_ready = vp_vdpa_set_vq_ready, .set_vq_ready = vp_vdpa_set_vq_ready,
.get_vq_ready = vp_vdpa_get_vq_ready, .get_vq_ready = vp_vdpa_get_vq_ready,
.set_vq_num = vp_vdpa_set_vq_num, .set_vq_num = vp_vdpa_set_vq_num,
.get_vq_size = vp_vdpa_get_vq_size,
.set_vq_address = vp_vdpa_set_vq_address, .set_vq_address = vp_vdpa_set_vq_address,
.kick_vq = vp_vdpa_kick_vq, .kick_vq = vp_vdpa_kick_vq,
.get_generation = vp_vdpa_get_generation, .get_generation = vp_vdpa_get_generation,
......
...@@ -697,6 +697,9 @@ static int vhost_net_build_xdp(struct vhost_net_virtqueue *nvq, ...@@ -697,6 +697,9 @@ static int vhost_net_build_xdp(struct vhost_net_virtqueue *nvq,
hdr = buf; hdr = buf;
gso = &hdr->gso; gso = &hdr->gso;
if (!sock_hlen)
memset(buf, 0, pad);
if ((gso->flags & VIRTIO_NET_HDR_F_NEEDS_CSUM) && if ((gso->flags & VIRTIO_NET_HDR_F_NEEDS_CSUM) &&
vhost16_to_cpu(vq, gso->csum_start) + vhost16_to_cpu(vq, gso->csum_start) +
vhost16_to_cpu(vq, gso->csum_offset) + 2 > vhost16_to_cpu(vq, gso->csum_offset) + 2 >
......
...@@ -595,6 +595,9 @@ static long vhost_vdpa_suspend(struct vhost_vdpa *v) ...@@ -595,6 +595,9 @@ static long vhost_vdpa_suspend(struct vhost_vdpa *v)
const struct vdpa_config_ops *ops = vdpa->config; const struct vdpa_config_ops *ops = vdpa->config;
int ret; int ret;
if (!(ops->get_status(vdpa) & VIRTIO_CONFIG_S_DRIVER_OK))
return 0;
if (!ops->suspend) if (!ops->suspend)
return -EOPNOTSUPP; return -EOPNOTSUPP;
...@@ -615,6 +618,9 @@ static long vhost_vdpa_resume(struct vhost_vdpa *v) ...@@ -615,6 +618,9 @@ static long vhost_vdpa_resume(struct vhost_vdpa *v)
const struct vdpa_config_ops *ops = vdpa->config; const struct vdpa_config_ops *ops = vdpa->config;
int ret; int ret;
if (!(ops->get_status(vdpa) & VIRTIO_CONFIG_S_DRIVER_OK))
return 0;
if (!ops->resume) if (!ops->resume)
return -EOPNOTSUPP; return -EOPNOTSUPP;
...@@ -681,6 +687,14 @@ static long vhost_vdpa_vring_ioctl(struct vhost_vdpa *v, unsigned int cmd, ...@@ -681,6 +687,14 @@ static long vhost_vdpa_vring_ioctl(struct vhost_vdpa *v, unsigned int cmd,
if (!ops->set_group_asid) if (!ops->set_group_asid)
return -EOPNOTSUPP; return -EOPNOTSUPP;
return ops->set_group_asid(vdpa, idx, s.num); return ops->set_group_asid(vdpa, idx, s.num);
case VHOST_VDPA_GET_VRING_SIZE:
if (!ops->get_vq_size)
return -EOPNOTSUPP;
s.index = idx;
s.num = ops->get_vq_size(vdpa, idx);
if (copy_to_user(argp, &s, sizeof(s)))
return -EFAULT;
return 0;
case VHOST_GET_VRING_BASE: case VHOST_GET_VRING_BASE:
r = ops->get_vq_state(v->vdpa, idx, &vq_state); r = ops->get_vq_state(v->vdpa, idx, &vq_state);
if (r) if (r)
......
...@@ -353,7 +353,7 @@ static void virtio_dev_remove(struct device *_d) ...@@ -353,7 +353,7 @@ static void virtio_dev_remove(struct device *_d)
of_node_put(dev->dev.of_node); of_node_put(dev->dev.of_node);
} }
static struct bus_type virtio_bus = { static const struct bus_type virtio_bus = {
.name = "virtio", .name = "virtio",
.match = virtio_dev_match, .match = virtio_dev_match,
.dev_groups = virtio_dev_groups, .dev_groups = virtio_dev_groups,
...@@ -510,9 +510,11 @@ int virtio_device_freeze(struct virtio_device *dev) ...@@ -510,9 +510,11 @@ int virtio_device_freeze(struct virtio_device *dev)
if (drv && drv->freeze) { if (drv && drv->freeze) {
ret = drv->freeze(dev); ret = drv->freeze(dev);
if (ret) if (ret) {
virtio_config_enable(dev);
return ret; return ret;
} }
}
if (dev->config->destroy_avq) if (dev->config->destroy_avq)
dev->config->destroy_avq(dev); dev->config->destroy_avq(dev);
......
...@@ -1340,7 +1340,7 @@ static int virtqueue_add_indirect_packed(struct vring_virtqueue *vq, ...@@ -1340,7 +1340,7 @@ static int virtqueue_add_indirect_packed(struct vring_virtqueue *vq,
sizeof(struct vring_packed_desc)); sizeof(struct vring_packed_desc));
vq->packed.vring.desc[head].id = cpu_to_le16(id); vq->packed.vring.desc[head].id = cpu_to_le16(id);
if (vq->do_unmap) { if (vq->use_dma_api) {
vq->packed.desc_extra[id].addr = addr; vq->packed.desc_extra[id].addr = addr;
vq->packed.desc_extra[id].len = total_sg * vq->packed.desc_extra[id].len = total_sg *
sizeof(struct vring_packed_desc); sizeof(struct vring_packed_desc);
...@@ -1481,7 +1481,7 @@ static inline int virtqueue_add_packed(struct virtqueue *_vq, ...@@ -1481,7 +1481,7 @@ static inline int virtqueue_add_packed(struct virtqueue *_vq,
desc[i].len = cpu_to_le32(sg->length); desc[i].len = cpu_to_le32(sg->length);
desc[i].id = cpu_to_le16(id); desc[i].id = cpu_to_le16(id);
if (unlikely(vq->do_unmap)) { if (unlikely(vq->use_dma_api)) {
vq->packed.desc_extra[curr].addr = addr; vq->packed.desc_extra[curr].addr = addr;
vq->packed.desc_extra[curr].len = sg->length; vq->packed.desc_extra[curr].len = sg->length;
vq->packed.desc_extra[curr].flags = vq->packed.desc_extra[curr].flags =
...@@ -1615,7 +1615,7 @@ static void detach_buf_packed(struct vring_virtqueue *vq, ...@@ -1615,7 +1615,7 @@ static void detach_buf_packed(struct vring_virtqueue *vq,
vq->free_head = id; vq->free_head = id;
vq->vq.num_free += state->num; vq->vq.num_free += state->num;
if (unlikely(vq->do_unmap)) { if (unlikely(vq->use_dma_api)) {
curr = id; curr = id;
for (i = 0; i < state->num; i++) { for (i = 0; i < state->num; i++) {
vring_unmap_extra_packed(vq, vring_unmap_extra_packed(vq,
......
...@@ -183,8 +183,11 @@ virtio_vdpa_setup_vq(struct virtio_device *vdev, unsigned int index, ...@@ -183,8 +183,11 @@ virtio_vdpa_setup_vq(struct virtio_device *vdev, unsigned int index,
info = kmalloc(sizeof(*info), GFP_KERNEL); info = kmalloc(sizeof(*info), GFP_KERNEL);
if (!info) if (!info)
return ERR_PTR(-ENOMEM); return ERR_PTR(-ENOMEM);
if (ops->get_vq_size)
max_num = ops->get_vq_size(vdpa, index);
else
max_num = ops->get_vq_num_max(vdpa); max_num = ops->get_vq_num_max(vdpa);
if (max_num == 0) { if (max_num == 0) {
err = -ENOENT; err = -ENOENT;
goto error_new_virtqueue; goto error_new_virtqueue;
......
...@@ -7,6 +7,7 @@ ...@@ -7,6 +7,7 @@
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/vhost_iotlb.h> #include <linux/vhost_iotlb.h>
#include <linux/virtio_net.h> #include <linux/virtio_net.h>
#include <linux/virtio_blk.h>
#include <linux/if_ether.h> #include <linux/if_ether.h>
/** /**
...@@ -195,6 +196,10 @@ struct vdpa_map_file { ...@@ -195,6 +196,10 @@ struct vdpa_map_file {
* @idx: virtqueue index * @idx: virtqueue index
* Returns int: irq number of a virtqueue, * Returns int: irq number of a virtqueue,
* negative number if no irq assigned. * negative number if no irq assigned.
* @get_vq_size: Get the size of a specific virtqueue (optional)
* @vdev: vdpa device
* @idx: virtqueue index
* Return u16: the size of the virtqueue
* @get_vq_align: Get the virtqueue align requirement * @get_vq_align: Get the virtqueue align requirement
* for the device * for the device
* @vdev: vdpa device * @vdev: vdpa device
...@@ -386,6 +391,7 @@ struct vdpa_config_ops { ...@@ -386,6 +391,7 @@ struct vdpa_config_ops {
(*get_vq_notification)(struct vdpa_device *vdev, u16 idx); (*get_vq_notification)(struct vdpa_device *vdev, u16 idx);
/* vq irq is not expected to be changed once DRIVER_OK is set */ /* vq irq is not expected to be changed once DRIVER_OK is set */
int (*get_vq_irq)(struct vdpa_device *vdev, u16 idx); int (*get_vq_irq)(struct vdpa_device *vdev, u16 idx);
u16 (*get_vq_size)(struct vdpa_device *vdev, u16 idx);
/* Device ops */ /* Device ops */
u32 (*get_vq_align)(struct vdpa_device *vdev); u32 (*get_vq_align)(struct vdpa_device *vdev);
......
...@@ -56,6 +56,23 @@ enum vdpa_attr { ...@@ -56,6 +56,23 @@ enum vdpa_attr {
/* virtio features that are provisioned to the vDPA device */ /* virtio features that are provisioned to the vDPA device */
VDPA_ATTR_DEV_FEATURES, /* u64 */ VDPA_ATTR_DEV_FEATURES, /* u64 */
VDPA_ATTR_DEV_BLK_CFG_CAPACITY, /* u64 */
VDPA_ATTR_DEV_BLK_CFG_SEG_SIZE, /* u32 */
VDPA_ATTR_DEV_BLK_CFG_BLK_SIZE, /* u32 */
VDPA_ATTR_DEV_BLK_CFG_SEG_MAX, /* u32 */
VDPA_ATTR_DEV_BLK_CFG_NUM_QUEUES, /* u16 */
VDPA_ATTR_DEV_BLK_CFG_PHY_BLK_EXP, /* u8 */
VDPA_ATTR_DEV_BLK_CFG_ALIGN_OFFSET, /* u8 */
VDPA_ATTR_DEV_BLK_CFG_MIN_IO_SIZE, /* u16 */
VDPA_ATTR_DEV_BLK_CFG_OPT_IO_SIZE, /* u32 */
VDPA_ATTR_DEV_BLK_CFG_MAX_DISCARD_SEC, /* u32 */
VDPA_ATTR_DEV_BLK_CFG_MAX_DISCARD_SEG, /* u32 */
VDPA_ATTR_DEV_BLK_CFG_DISCARD_SEC_ALIGN,/* u32 */
VDPA_ATTR_DEV_BLK_CFG_MAX_WRITE_ZEROES_SEC, /* u32 */
VDPA_ATTR_DEV_BLK_CFG_MAX_WRITE_ZEROES_SEG, /* u32 */
VDPA_ATTR_DEV_BLK_CFG_READ_ONLY, /* u8 */
VDPA_ATTR_DEV_BLK_CFG_FLUSH, /* u8 */
/* new attributes must be added above here */ /* new attributes must be added above here */
VDPA_ATTR_MAX, VDPA_ATTR_MAX,
}; };
......
...@@ -227,4 +227,11 @@ ...@@ -227,4 +227,11 @@
*/ */
#define VHOST_VDPA_GET_VRING_DESC_GROUP _IOWR(VHOST_VIRTIO, 0x7F, \ #define VHOST_VDPA_GET_VRING_DESC_GROUP _IOWR(VHOST_VIRTIO, 0x7F, \
struct vhost_vring_state) struct vhost_vring_state)
/* Get the queue size of a specific virtqueue.
* userspace set the vring index in vhost_vring_state.index
* kernel set the queue size in vhost_vring_state.num
*/
#define VHOST_VDPA_GET_VRING_SIZE _IOWR(VHOST_VIRTIO, 0x80, \
struct vhost_vring_state)
#endif #endif
...@@ -240,7 +240,7 @@ struct virtio_pci_cfg_cap { ...@@ -240,7 +240,7 @@ struct virtio_pci_cfg_cap {
#define VIRTIO_ADMIN_CMD_LEGACY_DEV_CFG_READ 0x5 #define VIRTIO_ADMIN_CMD_LEGACY_DEV_CFG_READ 0x5
#define VIRTIO_ADMIN_CMD_LEGACY_NOTIFY_INFO 0x6 #define VIRTIO_ADMIN_CMD_LEGACY_NOTIFY_INFO 0x6
struct __packed virtio_admin_cmd_hdr { struct virtio_admin_cmd_hdr {
__le16 opcode; __le16 opcode;
/* /*
* 1 - SR-IOV * 1 - SR-IOV
...@@ -252,20 +252,20 @@ struct __packed virtio_admin_cmd_hdr { ...@@ -252,20 +252,20 @@ struct __packed virtio_admin_cmd_hdr {
__le64 group_member_id; __le64 group_member_id;
}; };
struct __packed virtio_admin_cmd_status { struct virtio_admin_cmd_status {
__le16 status; __le16 status;
__le16 status_qualifier; __le16 status_qualifier;
/* Unused, reserved for future extensions. */ /* Unused, reserved for future extensions. */
__u8 reserved2[4]; __u8 reserved2[4];
}; };
struct __packed virtio_admin_cmd_legacy_wr_data { struct virtio_admin_cmd_legacy_wr_data {
__u8 offset; /* Starting offset of the register(s) to write. */ __u8 offset; /* Starting offset of the register(s) to write. */
__u8 reserved[7]; __u8 reserved[7];
__u8 registers[]; __u8 registers[];
}; };
struct __packed virtio_admin_cmd_legacy_rd_data { struct virtio_admin_cmd_legacy_rd_data {
__u8 offset; /* Starting offset of the register(s) to read. */ __u8 offset; /* Starting offset of the register(s) to read. */
}; };
...@@ -275,7 +275,7 @@ struct __packed virtio_admin_cmd_legacy_rd_data { ...@@ -275,7 +275,7 @@ struct __packed virtio_admin_cmd_legacy_rd_data {
#define VIRTIO_ADMIN_CMD_MAX_NOTIFY_INFO 4 #define VIRTIO_ADMIN_CMD_MAX_NOTIFY_INFO 4
struct __packed virtio_admin_cmd_notify_info_data { struct virtio_admin_cmd_notify_info_data {
__u8 flags; /* 0 = end of list, 1 = owner device, 2 = member device */ __u8 flags; /* 0 = end of list, 1 = owner device, 2 = member device */
__u8 bar; /* BAR of the member or the owner device */ __u8 bar; /* BAR of the member or the owner device */
__u8 padding[6]; __u8 padding[6];
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment