struct virtio_net_hdr *gso = &hdr->gso;
        struct bpf_prog *xdp_prog;
        struct sk_buff *skb = NULL;
+       struct sk_buff_head *queue;
        u32 rxhash = 0, act;
        int buflen = hdr->buflen;
-       int err = 0;
+       int ret = 0;
        bool skb_xdp = false;
        struct page *page;
 
                xdp_set_data_meta_invalid(xdp);
 
                act = bpf_prog_run_xdp(xdp_prog, xdp);
-               err = tun_xdp_act(tun, xdp_prog, xdp, act);
-               if (err < 0) {
+               ret = tun_xdp_act(tun, xdp_prog, xdp, act);
+               if (ret < 0) {
                        put_page(virt_to_head_page(xdp->data));
-                       return err;
+                       return ret;
                }
 
-               switch (err) {
+               switch (ret) {
                case XDP_REDIRECT:
                        *flush = true;
                        fallthrough;
 build:
        skb = build_skb(xdp->data_hard_start, buflen);
        if (!skb) {
-               err = -ENOMEM;
+               ret = -ENOMEM;
                goto out;
        }
 
        if (virtio_net_hdr_to_skb(skb, gso, tun_is_little_endian(tun))) {
                atomic_long_inc(&tun->rx_frame_errors);
                kfree_skb(skb);
-               err = -EINVAL;
+               ret = -EINVAL;
                goto out;
        }
 
        skb_record_rx_queue(skb, tfile->queue_index);
 
        if (skb_xdp) {
-               err = do_xdp_generic(xdp_prog, skb);
-               if (err != XDP_PASS)
+               ret = do_xdp_generic(xdp_prog, skb);
+               if (ret != XDP_PASS) {
+                       ret = 0;
                        goto out;
+               }
        }
 
        if (!rcu_dereference(tun->steering_prog) && tun->numqueues > 1 &&
            !tfile->detached)
                rxhash = __skb_get_hash_symmetric(skb);
 
-       netif_receive_skb(skb);
+       if (tfile->napi_enabled) {
+               queue = &tfile->sk.sk_write_queue;
+               spin_lock(&queue->lock);
+               __skb_queue_tail(queue, skb);
+               spin_unlock(&queue->lock);
+               ret = 1;
+       } else {
+               netif_receive_skb(skb);
+               ret = 0;
+       }
 
        /* No need to disable preemption here since this function is
         * always called with bh disabled
                tun_flow_update(tun, rxhash, tfile);
 
 out:
-       return err;
+       return ret;
 }
 
 static int tun_sendmsg(struct socket *sock, struct msghdr *m, size_t total_len)
        if (ctl && (ctl->type == TUN_MSG_PTR)) {
                struct tun_page tpage;
                int n = ctl->num;
-               int flush = 0;
+               int flush = 0, queued = 0;
 
                memset(&tpage, 0, sizeof(tpage));
 
 
                for (i = 0; i < n; i++) {
                        xdp = &((struct xdp_buff *)ctl->ptr)[i];
-                       tun_xdp_one(tun, tfile, xdp, &flush, &tpage);
+                       ret = tun_xdp_one(tun, tfile, xdp, &flush, &tpage);
+                       if (ret > 0)
+                               queued += ret;
                }
 
                if (flush)
                        xdp_do_flush();
 
+               if (tfile->napi_enabled && queued > 0)
+                       napi_schedule(&tfile->napi);
+
                rcu_read_unlock();
                local_bh_enable();