cregit-Linux how code gets into the kernel

Release 4.7 drivers/net/caif/caif_virtio.c

Directory: drivers/net/caif
/*
 * Copyright (C) ST-Ericsson AB 2013
 * Authors: Vicram Arv
 *          Dmitry Tarnyagin <dmitry.tarnyagin@lockless.no>
 *          Sjur Brendeland
 * License terms: GNU General Public License (GPL) version 2
 */
#include <linux/module.h>
#include <linux/if_arp.h>
#include <linux/virtio.h>
#include <linux/vringh.h>
#include <linux/debugfs.h>
#include <linux/spinlock.h>
#include <linux/genalloc.h>
#include <linux/interrupt.h>
#include <linux/netdevice.h>
#include <linux/rtnetlink.h>
#include <linux/virtio_ids.h>
#include <linux/virtio_caif.h>
#include <linux/virtio_ring.h>
#include <linux/dma-mapping.h>
#include <net/caif/caif_dev.h>
#include <linux/virtio_config.h>

MODULE_LICENSE("GPL v2");
MODULE_AUTHOR("Vicram Arv");
MODULE_AUTHOR("Sjur Brendeland");
MODULE_DESCRIPTION("Virtio CAIF Driver");

/* NAPI schedule quota */

#define CFV_DEFAULT_QUOTA 32

/* Defaults used if virtio config space is unavailable */

#define CFV_DEF_MTU_SIZE 4096

#define CFV_DEF_HEADROOM 32

#define CFV_DEF_TAILROOM 32

/* Required IP header alignment */

#define IP_HDR_ALIGN 4

/* struct cfv_napi_contxt - NAPI context info
 * @riov: IOV holding data read from the ring. Note that riov may
 *        still hold data when cfv_rx_poll() returns.
 * @head: Last descriptor ID we received from vringh_getdesc_kern.
 *        We use this to put descriptor back on the used ring. USHRT_MAX is
 *        used to indicate invalid head-id.
 */

struct cfv_napi_context {
	
struct vringh_kiov riov;
	
unsigned short head;
};

/* struct cfv_stats - statistics for debugfs
 * @rx_napi_complete:   Number of NAPI completions (RX)
 * @rx_napi_resched:    Number of calls where the full quota was used (RX)
 * @rx_nomem:           Number of SKB alloc failures (RX)
 * @rx_kicks:           Number of RX kicks
 * @tx_full_ring:       Number times TX ring was full
 * @tx_no_mem:          Number of times TX went out of memory
 * @tx_flow_on:         Number of flow on (TX)
 * @tx_kicks:           Number of TX kicks
 */

struct cfv_stats {
	
u32 rx_napi_complete;
	
u32 rx_napi_resched;
	
u32 rx_nomem;
	
u32 rx_kicks;
	
u32 tx_full_ring;
	
u32 tx_no_mem;
	
u32 tx_flow_on;
	
u32 tx_kicks;
};

/* struct cfv_info - Caif Virtio control structure
 * @cfdev:      caif common header
 * @vdev:       Associated virtio device
 * @vr_rx:      rx/downlink host vring
 * @vq_tx:      tx/uplink virtqueue
 * @ndev:       CAIF link layer device
 * @watermark_tx: indicates number of free descriptors we need
 *              to reopen the tx-queues after overload.
 * @tx_lock:    protects vq_tx from concurrent use
 * @tx_release_tasklet: Tasklet for freeing consumed TX buffers
 * @napi:       Napi context used in cfv_rx_poll()
 * @ctx:        Context data used in cfv_rx_poll()
 * @tx_hr:      transmit headroom
 * @rx_hr:      receive headroom
 * @tx_tr:      transmit tail room
 * @rx_tr:      receive tail room
 * @mtu:        transmit max size
 * @mru:        receive max size
 * @allocsz:    size of dma memory reserved for TX buffers
 * @alloc_addr: virtual address to dma memory for TX buffers
 * @alloc_dma:  dma address to dma memory for TX buffers
 * @genpool:    Gen Pool used for allocating TX buffers
 * @reserved_mem: Pointer to memory reserve allocated from genpool
 * @reserved_size: Size of memory reserve allocated from genpool
 * @stats:       Statistics exposed in sysfs
 * @debugfs:    Debugfs dentry for statistic counters
 */

struct cfv_info {
	
struct caif_dev_common cfdev;
	
struct virtio_device *vdev;
	
struct vringh *vr_rx;
	
struct virtqueue *vq_tx;
	
struct net_device *ndev;
	
unsigned int watermark_tx;
	/* Protect access to vq_tx */
	
spinlock_t tx_lock;
	
struct tasklet_struct tx_release_tasklet;
	
struct napi_struct napi;
	
struct cfv_napi_context ctx;
	
u16 tx_hr;
	
u16 rx_hr;
	
u16 tx_tr;
	
u16 rx_tr;
	
u32 mtu;
	
u32 mru;
	
size_t allocsz;
	
void *alloc_addr;
	
dma_addr_t alloc_dma;
	
struct gen_pool *genpool;
	
unsigned long reserved_mem;
	
size_t reserved_size;
	
struct cfv_stats stats;
	
struct dentry *debugfs;
};

/* struct buf_info - maintains transmit buffer data handle
 * @size:       size of transmit buffer
 * @dma_handle: handle to allocated dma device memory area
 * @vaddr:      virtual address mapping to allocated memory area
 */

struct buf_info {
	
size_t size;
	
u8 *vaddr;
};

/* Called from virtio device, in IRQ context */

static void cfv_release_cb(struct virtqueue *vq_tx) { struct cfv_info *cfv = vq_tx->vdev->priv; ++cfv->stats.tx_kicks; tasklet_schedule(&cfv->tx_release_tasklet); }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin37100.00%1100.00%
Total37100.00%1100.00%


static void free_buf_info(struct cfv_info *cfv, struct buf_info *buf_info) { if (!buf_info) return; gen_pool_free(cfv->genpool, (unsigned long) buf_info->vaddr, buf_info->size); kfree(buf_info); }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin46100.00%1100.00%
Total46100.00%1100.00%

/* This is invoked whenever the remote processor completed processing * a TX msg we just sent, and the buffer is put back to the used ring. */
static void cfv_release_used_buf(struct virtqueue *vq_tx) { struct cfv_info *cfv = vq_tx->vdev->priv; unsigned long flags; BUG_ON(vq_tx != cfv->vq_tx); for (;;) { unsigned int len; struct buf_info *buf_info; /* Get used buffer from used ring to recycle used descriptors */ spin_lock_irqsave(&cfv->tx_lock, flags); buf_info = virtqueue_get_buf(vq_tx, &len); spin_unlock_irqrestore(&cfv->tx_lock, flags); /* Stop looping if there are no more buffers to free */ if (!buf_info) break; free_buf_info(cfv, buf_info); /* watermark_tx indicates if we previously stopped the tx * queues. If we have enough free stots in the virtio ring, * re-establish memory reserved and open up tx queues. */ if (cfv->vq_tx->num_free <= cfv->watermark_tx) continue; /* Re-establish memory reserve */ if (cfv->reserved_mem == 0 && cfv->genpool) cfv->reserved_mem = gen_pool_alloc(cfv->genpool, cfv->reserved_size); /* Open up the tx queues */ if (cfv->reserved_mem) { cfv->watermark_tx = virtqueue_get_vring_size(cfv->vq_tx); netif_tx_wake_all_queues(cfv->ndev); /* Buffers are recycled in cfv_netdev_tx, so * disable notifications when queues are opened. */ virtqueue_disable_cb(cfv->vq_tx); ++cfv->stats.tx_flow_on; } else { /* if no memory reserve, wait for more free slots */ WARN_ON(cfv->watermark_tx > virtqueue_get_vring_size(cfv->vq_tx)); cfv->watermark_tx += virtqueue_get_vring_size(cfv->vq_tx) / 4; } } }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin211100.00%1100.00%
Total211100.00%1100.00%

/* Allocate a SKB and copy packet data to it */
static struct sk_buff *cfv_alloc_and_copy_skb(int *err, struct cfv_info *cfv, u8 *frm, u32 frm_len) { struct sk_buff *skb; u32 cfpkt_len, pad_len; *err = 0; /* Verify that packet size with down-link header and mtu size */ if (frm_len > cfv->mru || frm_len <= cfv->rx_hr + cfv->rx_tr) { netdev_err(cfv->ndev, "Invalid frmlen:%u mtu:%u hr:%d tr:%d\n", frm_len, cfv->mru, cfv->rx_hr, cfv->rx_tr); *err = -EPROTO; return NULL; } cfpkt_len = frm_len - (cfv->rx_hr + cfv->rx_tr); pad_len = (unsigned long)(frm + cfv->rx_hr) & (IP_HDR_ALIGN - 1); skb = netdev_alloc_skb(cfv->ndev, frm_len + pad_len); if (!skb) { *err = -ENOMEM; return NULL; } skb_reserve(skb, cfv->rx_hr + pad_len); memcpy(skb_put(skb, cfpkt_len), frm + cfv->rx_hr, cfpkt_len); return skb; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin187100.00%1100.00%
Total187100.00%1100.00%

/* Get packets from the host vring */
static int cfv_rx_poll(struct napi_struct *napi, int quota) { struct cfv_info *cfv = container_of(napi, struct cfv_info, napi); int rxcnt = 0; int err = 0; void *buf; struct sk_buff *skb; struct vringh_kiov *riov = &cfv->ctx.riov; unsigned int skb_len; do { skb = NULL; /* Put the previous iovec back on the used ring and * fetch a new iovec if we have processed all elements. */ if (riov->i == riov->used) { if (cfv->ctx.head != USHRT_MAX) { vringh_complete_kern(cfv->vr_rx, cfv->ctx.head, 0); cfv->ctx.head = USHRT_MAX; } err = vringh_getdesc_kern( cfv->vr_rx, riov, NULL, &cfv->ctx.head, GFP_ATOMIC); if (err <= 0) goto exit; } buf = phys_to_virt((unsigned long) riov->iov[riov->i].iov_base); /* TODO: Add check on valid buffer address */ skb = cfv_alloc_and_copy_skb(&err, cfv, buf, riov->iov[riov->i].iov_len); if (unlikely(err)) goto exit; /* Push received packet up the stack. */ skb_len = skb->len; skb->protocol = htons(ETH_P_CAIF); skb_reset_mac_header(skb); skb->dev = cfv->ndev; err = netif_receive_skb(skb); if (unlikely(err)) { ++cfv->ndev->stats.rx_dropped; } else { ++cfv->ndev->stats.rx_packets; cfv->ndev->stats.rx_bytes += skb_len; } ++riov->i; ++rxcnt; } while (rxcnt < quota); ++cfv->stats.rx_napi_resched; goto out; exit: switch (err) { case 0: ++cfv->stats.rx_napi_complete; /* Really out of patckets? (stolen from virtio_net)*/ napi_complete(napi); if (unlikely(!vringh_notify_enable_kern(cfv->vr_rx)) && napi_schedule_prep(napi)) { vringh_notify_disable_kern(cfv->vr_rx); __napi_schedule(napi); } break; case -ENOMEM: ++cfv->stats.rx_nomem; dev_kfree_skb(skb); /* Stop NAPI poll on OOM, we hope to be polled later */ napi_complete(napi); vringh_notify_enable_kern(cfv->vr_rx); break; default: /* We're doomed, any modem fault is fatal */ netdev_warn(cfv->ndev, "Bad ring, disable device\n"); cfv->ndev->stats.rx_dropped = riov->used - riov->i; napi_complete(napi); vringh_notify_disable_kern(cfv->vr_rx); netif_carrier_off(cfv->ndev); break; } out: if (rxcnt && vringh_need_notify_kern(cfv->vr_rx) > 0) vringh_notify(cfv->vr_rx); return rxcnt; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin46399.78%150.00%
sjur braendelandsjur braendeland10.22%150.00%
Total464100.00%2100.00%


static void cfv_recv(struct virtio_device *vdev, struct vringh *vr_rx) { struct cfv_info *cfv = vdev->priv; ++cfv->stats.rx_kicks; vringh_notify_disable_kern(cfv->vr_rx); napi_schedule(&cfv->napi); }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin47100.00%1100.00%
Total47100.00%1100.00%


static void cfv_destroy_genpool(struct cfv_info *cfv) { if (cfv->alloc_addr) dma_free_coherent(cfv->vdev->dev.parent->parent, cfv->allocsz, cfv->alloc_addr, cfv->alloc_dma); if (!cfv->genpool) return; gen_pool_free(cfv->genpool, cfv->reserved_mem, cfv->reserved_size); gen_pool_destroy(cfv->genpool); cfv->genpool = NULL; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin78100.00%1100.00%
Total78100.00%1100.00%


static int cfv_create_genpool(struct cfv_info *cfv) { int err; /* dma_alloc can only allocate whole pages, and we need a more * fine graned allocation so we use genpool. We ask for space needed * by IP and a full ring. If the dma allcoation fails we retry with a * smaller allocation size. */ err = -ENOMEM; cfv->allocsz = (virtqueue_get_vring_size(cfv->vq_tx) * (ETH_DATA_LEN + cfv->tx_hr + cfv->tx_tr) * 11)/10; if (cfv->allocsz <= (num_possible_cpus() + 1) * cfv->ndev->mtu) return -EINVAL; for (;;) { if (cfv->allocsz <= num_possible_cpus() * cfv->ndev->mtu) { netdev_info(cfv->ndev, "Not enough device memory\n"); return -ENOMEM; } cfv->alloc_addr = dma_alloc_coherent( cfv->vdev->dev.parent->parent, cfv->allocsz, &cfv->alloc_dma, GFP_ATOMIC); if (cfv->alloc_addr) break; cfv->allocsz = (cfv->allocsz * 3) >> 2; } netdev_dbg(cfv->ndev, "Allocated %zd bytes from dma-memory\n", cfv->allocsz); /* Allocate on 128 bytes boundaries (1 << 7)*/ cfv->genpool = gen_pool_create(7, -1); if (!cfv->genpool) goto err; err = gen_pool_add_virt(cfv->genpool, (unsigned long)cfv->alloc_addr, (phys_addr_t)virt_to_phys(cfv->alloc_addr), cfv->allocsz, -1); if (err) goto err; /* Reserve some memory for low memory situations. If we hit the roof * in the memory pool, we stop TX flow and release the reserve. */ cfv->reserved_size = num_possible_cpus() * cfv->ndev->mtu; cfv->reserved_mem = gen_pool_alloc(cfv->genpool, cfv->reserved_size); if (!cfv->reserved_mem) { err = -ENOMEM; goto err; } cfv->watermark_tx = virtqueue_get_vring_size(cfv->vq_tx); return 0; err: cfv_destroy_genpool(cfv); return err; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin29897.70%150.00%
wei yongjunwei yongjun72.30%150.00%
Total305100.00%2100.00%

/* Enable the CAIF interface and allocate the memory-pool */
static int cfv_netdev_open(struct net_device *netdev) { struct cfv_info *cfv = netdev_priv(netdev); if (cfv_create_genpool(cfv)) return -ENOMEM; netif_carrier_on(netdev); napi_enable(&cfv->napi); /* Schedule NAPI to read any pending packets */ napi_schedule(&cfv->napi); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin57100.00%1100.00%
Total57100.00%1100.00%

/* Disable the CAIF interface and free the memory-pool */
static int cfv_netdev_close(struct net_device *netdev) { struct cfv_info *cfv = netdev_priv(netdev); unsigned long flags; struct buf_info *buf_info; /* Disable interrupts, queues and NAPI polling */ netif_carrier_off(netdev); virtqueue_disable_cb(cfv->vq_tx); vringh_notify_disable_kern(cfv->vr_rx); napi_disable(&cfv->napi); /* Release any TX buffers on both used and avilable rings */ cfv_release_used_buf(cfv->vq_tx); spin_lock_irqsave(&cfv->tx_lock, flags); while ((buf_info = virtqueue_detach_unused_buf(cfv->vq_tx))) free_buf_info(cfv, buf_info); spin_unlock_irqrestore(&cfv->tx_lock, flags); /* Release all dma allocated memory and destroy the pool */ cfv_destroy_genpool(cfv); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin115100.00%1100.00%
Total115100.00%1100.00%

/* Allocate a buffer in dma-memory and copy skb to it */
static struct buf_info *cfv_alloc_and_copy_to_shm(struct cfv_info *cfv, struct sk_buff *skb, struct scatterlist *sg) { struct caif_payload_info *info = (void *)&skb->cb; struct buf_info *buf_info = NULL; u8 pad_len, hdr_ofs; if (!cfv->genpool) goto err; if (unlikely(cfv->tx_hr + skb->len + cfv->tx_tr > cfv->mtu)) { netdev_warn(cfv->ndev, "Invalid packet len (%d > %d)\n", cfv->tx_hr + skb->len + cfv->tx_tr, cfv->mtu); goto err; } buf_info = kmalloc(sizeof(struct buf_info), GFP_ATOMIC); if (unlikely(!buf_info)) goto err; /* Make the IP header aligned in tbe buffer */ hdr_ofs = cfv->tx_hr + info->hdr_len; pad_len = hdr_ofs & (IP_HDR_ALIGN - 1); buf_info->size = cfv->tx_hr + skb->len + cfv->tx_tr + pad_len; /* allocate dma memory buffer */ buf_info->vaddr = (void *)gen_pool_alloc(cfv->genpool, buf_info->size); if (unlikely(!buf_info->vaddr)) goto err; /* copy skbuf contents to send buffer */ skb_copy_bits(skb, 0, buf_info->vaddr + cfv->tx_hr + pad_len, skb->len); sg_init_one(sg, buf_info->vaddr + pad_len, skb->len + cfv->tx_hr + cfv->rx_hr); return buf_info; err: kfree(buf_info); return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin264100.00%1100.00%
Total264100.00%1100.00%

/* Put the CAIF packet on the virtio ring and kick the receiver */
static int cfv_netdev_tx(struct sk_buff *skb, struct net_device *netdev) { struct cfv_info *cfv = netdev_priv(netdev); struct buf_info *buf_info; struct scatterlist sg; unsigned long flags; bool flow_off = false; int ret; /* garbage collect released buffers */ cfv_release_used_buf(cfv->vq_tx); spin_lock_irqsave(&cfv->tx_lock, flags); /* Flow-off check takes into account number of cpus to make sure * virtqueue will not be overfilled in any possible smp conditions. * * Flow-on is triggered when sufficient buffers are freed */ if (unlikely(cfv->vq_tx->num_free <= num_present_cpus())) { flow_off = true; cfv->stats.tx_full_ring++; } /* If we run out of memory, we release the memory reserve and retry * allocation. */ buf_info = cfv_alloc_and_copy_to_shm(cfv, skb, &sg); if (unlikely(!buf_info)) { cfv->stats.tx_no_mem++; flow_off = true; if (cfv->reserved_mem && cfv->genpool) { gen_pool_free(cfv->genpool, cfv->reserved_mem, cfv->reserved_size); cfv->reserved_mem = 0; buf_info = cfv_alloc_and_copy_to_shm(cfv, skb, &sg); } } if (unlikely(flow_off)) { /* Turn flow on when a 1/4 of the descriptors are released */ cfv->watermark_tx = virtqueue_get_vring_size(cfv->vq_tx) / 4; /* Enable notifications of recycled TX buffers */ virtqueue_enable_cb(cfv->vq_tx); netif_tx_stop_all_queues(netdev); } if (unlikely(!buf_info)) { /* If the memory reserve does it's job, this shouldn't happen */ netdev_warn(cfv->ndev, "Out of gen_pool memory\n"); goto err; } ret = virtqueue_add_outbuf(cfv->vq_tx, &sg, 1, buf_info, GFP_ATOMIC); if (unlikely((ret < 0))) { /* If flow control works, this shouldn't happen */ netdev_warn(cfv->ndev, "Failed adding buffer to TX vring:%d\n", ret); goto err; } /* update netdev statistics */ cfv->ndev->stats.tx_packets++; cfv->ndev->stats.tx_bytes += skb->len; spin_unlock_irqrestore(&cfv->tx_lock, flags); /* tell the remote processor it has a pending message to read */ virtqueue_kick(cfv->vq_tx); dev_kfree_skb(skb); return NETDEV_TX_OK; err: spin_unlock_irqrestore(&cfv->tx_lock, flags); cfv->ndev->stats.tx_dropped++; free_buf_info(cfv, buf_info); dev_kfree_skb(skb); return NETDEV_TX_OK; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin36099.72%150.00%
rusty russellrusty russell10.28%150.00%
Total361100.00%2100.00%


static void cfv_tx_release_tasklet(unsigned long drv) { struct cfv_info *cfv = (struct cfv_info *)drv; cfv_release_used_buf(cfv->vq_tx); }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin29100.00%1100.00%
Total29100.00%1100.00%

static const struct net_device_ops cfv_netdev_ops = { .ndo_open = cfv_netdev_open, .ndo_stop = cfv_netdev_close, .ndo_start_xmit = cfv_netdev_tx, };
static void cfv_netdev_setup(struct net_device *netdev) { netdev->netdev_ops = &cfv_netdev_ops; netdev->type = ARPHRD_CAIF; netdev->tx_queue_len = 100; netdev->flags = IFF_POINTOPOINT | IFF_NOARP; netdev->mtu = CFV_DEF_MTU_SIZE; netdev->destructor = free_netdev; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin50100.00%1100.00%
Total50100.00%1100.00%

/* Create debugfs counters for the device */
static inline void debugfs_init(struct cfv_info *cfv) { cfv->debugfs = debugfs_create_dir(netdev_name(cfv->ndev), NULL); if (IS_ERR(cfv->debugfs)) return; debugfs_create_u32("rx-napi-complete", S_IRUSR, cfv->debugfs, &cfv->stats.rx_napi_complete); debugfs_create_u32("rx-napi-resched", S_IRUSR, cfv->debugfs, &cfv->stats.rx_napi_resched); debugfs_create_u32("rx-nomem", S_IRUSR, cfv->debugfs, &cfv->stats.rx_nomem); debugfs_create_u32("rx-kicks", S_IRUSR, cfv->debugfs, &cfv->stats.rx_kicks); debugfs_create_u32("tx-full-ring", S_IRUSR, cfv->debugfs, &cfv->stats.tx_full_ring); debugfs_create_u32("tx-no-mem", S_IRUSR, cfv->debugfs, &cfv->stats.tx_no_mem); debugfs_create_u32("tx-kicks", S_IRUSR, cfv->debugfs, &cfv->stats.tx_kicks); debugfs_create_u32("tx-flow-on", S_IRUSR, cfv->debugfs, &cfv->stats.tx_flow_on); }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin182100.00%1100.00%
Total182100.00%1100.00%

/* Setup CAIF for the a virtio device */
static int cfv_probe(struct virtio_device *vdev) { vq_callback_t *vq_cbs = cfv_release_cb; vrh_callback_t *vrh_cbs = cfv_recv; const char *names = "output"; const char *cfv_netdev_name = "cfvrt"; struct net_device *netdev; struct cfv_info *cfv; int err = -EINVAL; netdev = alloc_netdev(sizeof(struct cfv_info), cfv_netdev_name, NET_NAME_UNKNOWN, cfv_netdev_setup); if (!netdev) return -ENOMEM; cfv = netdev_priv(netdev); cfv->vdev = vdev; cfv->ndev = netdev; spin_lock_init(&cfv->tx_lock); /* Get the RX virtio ring. This is a "host side vring". */ err = -ENODEV; if (!vdev->vringh_config || !vdev->vringh_config->find_vrhs) goto err; err = vdev->vringh_config->find_vrhs(vdev, 1, &cfv->vr_rx, &vrh_cbs); if (err) goto err; /* Get the TX virtio ring. This is a "guest side vring". */ err = vdev->config->find_vqs(vdev, 1, &cfv->vq_tx, &vq_cbs, &names); if (err) goto err; /* Get the CAIF configuration from virtio config space, if available */ if (vdev->config->get) { virtio_cread(vdev, struct virtio_caif_transf_config, headroom, &cfv->tx_hr); virtio_cread(vdev, struct virtio_caif_transf_config, headroom, &cfv->rx_hr); virtio_cread(vdev, struct virtio_caif_transf_config, tailroom, &cfv->tx_tr); virtio_cread(vdev, struct virtio_caif_transf_config, tailroom, &cfv->rx_tr); virtio_cread(vdev, struct virtio_caif_transf_config, mtu, &cfv->mtu); virtio_cread(vdev, struct virtio_caif_transf_config, mtu, &cfv->mru); } else { cfv->tx_hr = CFV_DEF_HEADROOM; cfv->rx_hr = CFV_DEF_HEADROOM; cfv->tx_tr = CFV_DEF_TAILROOM; cfv->rx_tr = CFV_DEF_TAILROOM; cfv->mtu = CFV_DEF_MTU_SIZE; cfv->mru = CFV_DEF_MTU_SIZE; } netdev->needed_headroom = cfv->tx_hr; netdev->needed_tailroom = cfv->tx_tr; /* Disable buffer release interrupts unless we have stopped TX queues */ virtqueue_disable_cb(cfv->vq_tx); netdev->mtu = cfv->mtu - cfv->tx_tr; vdev->priv = cfv; /* Initialize NAPI poll context data */ vringh_kiov_init(&cfv->ctx.riov, NULL, 0); cfv->ctx.head = USHRT_MAX; netif_napi_add(netdev, &cfv->napi, cfv_rx_poll, CFV_DEFAULT_QUOTA); tasklet_init(&cfv->tx_release_tasklet, cfv_tx_release_tasklet, (unsigned long)cfv); /* Carrier is off until netdevice is opened */ netif_carrier_off(netdev); /* register Netdev */ err = register_netdev(netdev); if (err) { dev_err(&vdev->dev, "Unable to register netdev (%d)\n", err); goto err; } debugfs_init(cfv); return 0; err: netdev_warn(cfv->ndev, "CAIF Virtio probe failed:%d\n", err); if (cfv->vr_rx) vdev->vringh_config->del_vrhs(cfv->vdev); if (cfv->vdev) vdev->config->del_vqs(cfv->vdev); free_netdev(netdev); return err; }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin45687.36%125.00%
rusty russellrusty russell428.05%125.00%
sjur braendelandsjur braendeland224.21%125.00%
tom gundersentom gundersen20.38%125.00%
Total522100.00%4100.00%


static void cfv_remove(struct virtio_device *vdev) { struct cfv_info *cfv = vdev->priv; rtnl_lock(); dev_close(cfv->ndev); rtnl_unlock(); tasklet_kill(&cfv->tx_release_tasklet); debugfs_remove_recursive(cfv->debugfs); vringh_kiov_cleanup(&cfv->ctx.riov); vdev->config->reset(vdev); vdev->vringh_config->del_vrhs(cfv->vdev); cfv->vr_rx = NULL; vdev->config->del_vqs(cfv->vdev); unregister_netdev(cfv->ndev); }

Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin102100.00%1100.00%
Total102100.00%1100.00%

static struct virtio_device_id id_table[] = { { VIRTIO_ID_CAIF, VIRTIO_DEV_ANY_ID }, { 0 }, }; static unsigned int features[] = { }; static struct virtio_driver caif_virtio_driver = { .feature_table = features, .feature_table_size = ARRAY_SIZE(features), .driver.name = KBUILD_MODNAME, .driver.owner = THIS_MODULE, .id_table = id_table, .probe = cfv_probe, .remove = cfv_remove, }; module_virtio_driver(caif_virtio_driver); MODULE_DEVICE_TABLE(virtio, id_table);

Overall Contributors

PersonTokensPropCommitsCommitProp
erwan yvinerwan yvin334797.72%112.50%
rusty russellrusty russell431.26%225.00%
sjur braendelandsjur braendeland260.76%337.50%
wei yongjunwei yongjun70.20%112.50%
tom gundersentom gundersen20.06%112.50%
Total3425100.00%8100.00%
Directory: drivers/net/caif
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
{% endraw %}