return NULL;
 }
 
-static struct vxlan_dev *vxlan_vs_find_vni(struct vxlan_sock *vs, int ifindex,
-                                          __be32 vni)
+static struct vxlan_dev *vxlan_vs_find_vni(struct vxlan_sock *vs,
+                                          int ifindex, __be32 vni,
+                                          struct vxlan_vni_node **vninode)
 {
+       struct vxlan_vni_node *vnode;
        struct vxlan_dev_node *node;
 
        /* For flow based devices, map all packets to VNI 0 */
        hlist_for_each_entry_rcu(node, vni_head(vs, vni), hlist) {
                if (!node->vxlan)
                        continue;
+               vnode = NULL;
                if (node->vxlan->cfg.flags & VXLAN_F_VNIFILTER) {
-                       if (!vxlan_vnifilter_lookup(node->vxlan, vni))
+                       vnode = vxlan_vnifilter_lookup(node->vxlan, vni);
+                       if (!vnode)
                                continue;
                } else if (node->vxlan->default_dst.remote_vni != vni) {
                        continue;
                                continue;
                }
 
+               if (vninode)
+                       *vninode = vnode;
                return node->vxlan;
        }
 
        if (!vs)
                return NULL;
 
-       return vxlan_vs_find_vni(vs, ifindex, vni);
+       return vxlan_vs_find_vni(vs, ifindex, vni, NULL);
 }
 
 /* Fill in neighbour message in skbuff. */
 /* Callback from net/ipv4/udp.c to receive packets */
 static int vxlan_rcv(struct sock *sk, struct sk_buff *skb)
 {
+       struct vxlan_vni_node *vninode = NULL;
        struct vxlan_dev *vxlan;
        struct vxlan_sock *vs;
        struct vxlanhdr unparsed;
 
        vni = vxlan_vni(vxlan_hdr(skb)->vx_vni);
 
-       vxlan = vxlan_vs_find_vni(vs, skb->dev->ifindex, vni);
+       vxlan = vxlan_vs_find_vni(vs, skb->dev->ifindex, vni, &vninode);
        if (!vxlan)
                goto drop;
 
        if (!vxlan_ecn_decapsulate(vs, oiph, skb)) {
                ++vxlan->dev->stats.rx_frame_errors;
                ++vxlan->dev->stats.rx_errors;
+               vxlan_vnifilter_count(vxlan, vni, vninode,
+                                     VXLAN_VNI_STATS_RX_ERRORS, 0);
                goto drop;
        }
 
        if (unlikely(!(vxlan->dev->flags & IFF_UP))) {
                rcu_read_unlock();
                atomic_long_inc(&vxlan->dev->rx_dropped);
+               vxlan_vnifilter_count(vxlan, vni, vninode,
+                                     VXLAN_VNI_STATS_RX_DROPS, 0);
                goto drop;
        }
 
        dev_sw_netstats_rx_add(vxlan->dev, skb->len);
+       vxlan_vnifilter_count(vxlan, vni, vninode, VXLAN_VNI_STATS_RX, skb->len);
        gro_cells_receive(&vxlan->gro_cells, skb);
 
        rcu_read_unlock();
                return -ENOENT;
 
        vni = vxlan_vni(hdr->vx_vni);
-       vxlan = vxlan_vs_find_vni(vs, skb->dev->ifindex, vni);
+       vxlan = vxlan_vs_find_vni(vs, skb->dev->ifindex, vni, NULL);
        if (!vxlan)
                return -ENOENT;
 
                reply->ip_summed = CHECKSUM_UNNECESSARY;
                reply->pkt_type = PACKET_HOST;
 
-               if (netif_rx_ni(reply) == NET_RX_DROP)
+               if (netif_rx_ni(reply) == NET_RX_DROP) {
                        dev->stats.rx_dropped++;
+                       vxlan_vnifilter_count(vxlan, vni, NULL,
+                                             VXLAN_VNI_STATS_RX_DROPS, 0);
+               }
+
        } else if (vxlan->cfg.flags & VXLAN_F_L3MISS) {
                union vxlan_addr ipa = {
                        .sin.sin_addr.s_addr = tip,
                if (reply == NULL)
                        goto out;
 
-               if (netif_rx_ni(reply) == NET_RX_DROP)
+               if (netif_rx_ni(reply) == NET_RX_DROP) {
                        dev->stats.rx_dropped++;
-
+                       vxlan_vnifilter_count(vxlan, vni, NULL,
+                                             VXLAN_VNI_STATS_RX_DROPS, 0);
+               }
        } else if (vxlan->cfg.flags & VXLAN_F_L3MISS) {
                union vxlan_addr ipa = {
                        .sin6.sin6_addr = msg->target,
        tx_stats->tx_packets++;
        tx_stats->tx_bytes += len;
        u64_stats_update_end(&tx_stats->syncp);
+       vxlan_vnifilter_count(src_vxlan, vni, NULL, VXLAN_VNI_STATS_TX, len);
 
        if (__netif_rx(skb) == NET_RX_SUCCESS) {
                u64_stats_update_begin(&rx_stats->syncp);
                rx_stats->rx_packets++;
                rx_stats->rx_bytes += len;
                u64_stats_update_end(&rx_stats->syncp);
+               vxlan_vnifilter_count(dst_vxlan, vni, NULL, VXLAN_VNI_STATS_RX,
+                                     len);
        } else {
 drop:
                dev->stats.rx_dropped++;
+               vxlan_vnifilter_count(dst_vxlan, vni, NULL,
+                                     VXLAN_VNI_STATS_RX_DROPS, 0);
        }
        rcu_read_unlock();
 }
                                           vxlan->cfg.flags);
                if (!dst_vxlan) {
                        dev->stats.tx_errors++;
+                       vxlan_vnifilter_count(vxlan, vni, NULL,
+                                             VXLAN_VNI_STATS_TX_ERRORS, 0);
                        kfree_skb(skb);
 
                        return -ENOENT;
        union vxlan_addr remote_ip, local_ip;
        struct vxlan_metadata _md;
        struct vxlan_metadata *md = &_md;
+       unsigned int pkt_len = skb->len;
        __be16 src_port = 0, dst_port;
        struct dst_entry *ndst = NULL;
        __u8 tos, ttl;
                                     label, src_port, dst_port, !udp_sum);
 #endif
        }
+       vxlan_vnifilter_count(vxlan, vni, NULL, VXLAN_VNI_STATS_TX, pkt_len);
 out_unlock:
        rcu_read_unlock();
        return;
 
 drop:
        dev->stats.tx_dropped++;
+       vxlan_vnifilter_count(vxlan, vni, NULL, VXLAN_VNI_STATS_TX_DROPS, 0);
        dev_kfree_skb(skb);
        return;
 
                dev->stats.tx_carrier_errors++;
        dst_release(ndst);
        dev->stats.tx_errors++;
+       vxlan_vnifilter_count(vxlan, vni, NULL, VXLAN_VNI_STATS_TX_ERRORS, 0);
        kfree_skb(skb);
 }
 
 
 drop:
        dev->stats.tx_dropped++;
+       vxlan_vnifilter_count(netdev_priv(dev), vni, NULL,
+                             VXLAN_VNI_STATS_TX_DROPS, 0);
        dev_kfree_skb(skb);
 }
 
                                vxlan_fdb_miss(vxlan, eth->h_dest);
 
                        dev->stats.tx_dropped++;
+                       vxlan_vnifilter_count(vxlan, vni, NULL,
+                                             VXLAN_VNI_STATS_TX_DROPS, 0);
                        kfree_skb(skb);
                        return NETDEV_TX_OK;
                }
 
        spin_unlock(&vn->sock_lock);
 }
 
+static void vxlan_vnifilter_stats_add(struct vxlan_vni_node *vninode,
+                                     int type, unsigned int len)
+{
+       struct vxlan_vni_stats_pcpu *pstats = this_cpu_ptr(vninode->stats);
+
+       u64_stats_update_begin(&pstats->syncp);
+       switch (type) {
+       case VXLAN_VNI_STATS_RX:
+               pstats->stats.rx_bytes += len;
+               pstats->stats.rx_packets++;
+               break;
+       case VXLAN_VNI_STATS_RX_DROPS:
+               pstats->stats.rx_drops++;
+               break;
+       case VXLAN_VNI_STATS_RX_ERRORS:
+               pstats->stats.rx_errors++;
+               break;
+       case VXLAN_VNI_STATS_TX:
+               pstats->stats.tx_bytes += len;
+               pstats->stats.tx_packets++;
+               break;
+       case VXLAN_VNI_STATS_TX_DROPS:
+               pstats->stats.tx_drops++;
+               break;
+       case VXLAN_VNI_STATS_TX_ERRORS:
+               pstats->stats.tx_errors++;
+               break;
+       }
+       u64_stats_update_end(&pstats->syncp);
+}
+
+void vxlan_vnifilter_count(struct vxlan_dev *vxlan, __be32 vni,
+                          struct vxlan_vni_node *vninode,
+                          int type, unsigned int len)
+{
+       struct vxlan_vni_node *vnode;
+
+       if (!(vxlan->cfg.flags & VXLAN_F_VNIFILTER))
+               return;
+
+       if (vninode) {
+               vnode = vninode;
+       } else {
+               vnode = vxlan_vnifilter_lookup(vxlan, vni);
+               if (!vnode)
+                       return;
+       }
+
+       vxlan_vnifilter_stats_add(vnode, type, len);
+}
+
 static u32 vnirange(struct vxlan_vni_node *vbegin,
                    struct vxlan_vni_node *vend)
 {
        vninode = kzalloc(sizeof(*vninode), GFP_ATOMIC);
        if (!vninode)
                return NULL;
+       vninode->stats = netdev_alloc_pcpu_stats(struct vxlan_vni_stats_pcpu);
+       if (!vninode->stats) {
+               kfree(vninode);
+               return NULL;
+       }
        vninode->vni = vni;
        vninode->hlist4.vxlan = vxlan;
 #if IS_ENABLED(CONFIG_IPV6)
        struct vxlan_vni_node *v;
 
        v = container_of(rcu, struct vxlan_vni_node, rcu);
+       free_percpu(v->stats);
        kfree(v);
 }