2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2017-04-21 10:10:44 +01:00
/*
* Tap functions for AF_VSOCK sockets .
*
* Code based on net / netlink / af_netlink . c tap functions .
*/
# include <linux/module.h>
# include <net/sock.h>
# include <net/af_vsock.h>
# include <linux/if_arp.h>
static DEFINE_SPINLOCK ( vsock_tap_lock ) ;
static struct list_head vsock_tap_all __read_mostly =
LIST_HEAD_INIT ( vsock_tap_all ) ;
int vsock_add_tap ( struct vsock_tap * vt )
{
if ( unlikely ( vt - > dev - > type ! = ARPHRD_VSOCKMON ) )
return - EINVAL ;
__module_get ( vt - > module ) ;
spin_lock ( & vsock_tap_lock ) ;
list_add_rcu ( & vt - > list , & vsock_tap_all ) ;
spin_unlock ( & vsock_tap_lock ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( vsock_add_tap ) ;
int vsock_remove_tap ( struct vsock_tap * vt )
{
struct vsock_tap * tmp ;
bool found = false ;
spin_lock ( & vsock_tap_lock ) ;
list_for_each_entry ( tmp , & vsock_tap_all , list ) {
if ( vt = = tmp ) {
list_del_rcu ( & vt - > list ) ;
found = true ;
goto out ;
}
}
pr_warn ( " vsock_remove_tap: %p not found \n " , vt ) ;
out :
spin_unlock ( & vsock_tap_lock ) ;
synchronize_net ( ) ;
if ( found )
module_put ( vt - > module ) ;
return found ? 0 : - ENODEV ;
}
EXPORT_SYMBOL_GPL ( vsock_remove_tap ) ;
static int __vsock_deliver_tap_skb ( struct sk_buff * skb ,
struct net_device * dev )
{
int ret = 0 ;
struct sk_buff * nskb = skb_clone ( skb , GFP_ATOMIC ) ;
if ( nskb ) {
dev_hold ( dev ) ;
nskb - > dev = dev ;
ret = dev_queue_xmit ( nskb ) ;
if ( unlikely ( ret > 0 ) )
ret = net_xmit_errno ( ret ) ;
dev_put ( dev ) ;
}
return ret ;
}
static void __vsock_deliver_tap ( struct sk_buff * skb )
{
int ret ;
struct vsock_tap * tmp ;
list_for_each_entry_rcu ( tmp , & vsock_tap_all , list ) {
ret = __vsock_deliver_tap_skb ( skb , tmp - > dev ) ;
if ( unlikely ( ret ) )
break ;
}
}
void vsock_deliver_tap ( struct sk_buff * build_skb ( void * opaque ) , void * opaque )
{
struct sk_buff * skb ;
rcu_read_lock ( ) ;
if ( likely ( list_empty ( & vsock_tap_all ) ) )
goto out ;
skb = build_skb ( opaque ) ;
if ( skb ) {
__vsock_deliver_tap ( skb ) ;
consume_skb ( skb ) ;
}
out :
rcu_read_unlock ( ) ;
}
EXPORT_SYMBOL_GPL ( vsock_deliver_tap ) ;