diff options
Diffstat (limited to 'drivers/net/tun.c')
| -rw-r--r-- | drivers/net/tun.c | 29 | 
1 files changed, 23 insertions, 6 deletions
| diff --git a/drivers/net/tun.c b/drivers/net/tun.c index 88bb8cc3555b..2c9e45f50edb 100644 --- a/drivers/net/tun.c +++ b/drivers/net/tun.c @@ -187,6 +187,7 @@ struct tun_struct {  #define TUN_USER_FEATURES (NETIF_F_HW_CSUM|NETIF_F_TSO_ECN|NETIF_F_TSO| \  			  NETIF_F_TSO6|NETIF_F_UFO) +	int			align;  	int			vnet_hdr_sz;  	int			sndbuf;  	struct tap_filter	txflt; @@ -621,7 +622,8 @@ static int tun_attach(struct tun_struct *tun, struct file *file, bool skip_filte  	/* Re-attach the filter to persist device */  	if (!skip_filter && (tun->filter_attached == true)) { -		err = sk_attach_filter(&tun->fprog, tfile->socket.sk); +		err = __sk_attach_filter(&tun->fprog, tfile->socket.sk, +					 lockdep_rtnl_is_held());  		if (!err)  			goto out;  	} @@ -934,6 +936,17 @@ static void tun_poll_controller(struct net_device *dev)  	return;  }  #endif + +static void tun_set_headroom(struct net_device *dev, int new_hr) +{ +	struct tun_struct *tun = netdev_priv(dev); + +	if (new_hr < NET_SKB_PAD) +		new_hr = NET_SKB_PAD; + +	tun->align = new_hr; +} +  static const struct net_device_ops tun_netdev_ops = {  	.ndo_uninit		= tun_net_uninit,  	.ndo_open		= tun_net_open, @@ -945,6 +958,7 @@ static const struct net_device_ops tun_netdev_ops = {  #ifdef CONFIG_NET_POLL_CONTROLLER  	.ndo_poll_controller	= tun_poll_controller,  #endif +	.ndo_set_rx_headroom	= tun_set_headroom,  };  static const struct net_device_ops tap_netdev_ops = { @@ -962,6 +976,7 @@ static const struct net_device_ops tap_netdev_ops = {  	.ndo_poll_controller	= tun_poll_controller,  #endif  	.ndo_features_check	= passthru_features_check, +	.ndo_set_rx_headroom	= tun_set_headroom,  };  static void tun_flow_init(struct tun_struct *tun) @@ -1000,7 +1015,6 @@ static void tun_net_init(struct net_device *dev)  		/* Zero header length */  		dev->type = ARPHRD_NONE;  		dev->flags = IFF_POINTOPOINT | IFF_NOARP | IFF_MULTICAST; -		dev->tx_queue_len = TUN_READQ_SIZE;  /* We prefer our own queue length */  		break;  	case IFF_TAP: @@ -1012,7 +1026,6 @@ static void tun_net_init(struct net_device *dev)  		eth_hw_addr_random(dev); -		dev->tx_queue_len = TUN_READQ_SIZE;  /* We prefer our own queue length */  		break;  	}  } @@ -1086,7 +1099,7 @@ static ssize_t tun_get_user(struct tun_struct *tun, struct tun_file *tfile,  	struct tun_pi pi = { 0, cpu_to_be16(ETH_P_IP) };  	struct sk_buff *skb;  	size_t total_len = iov_iter_count(from); -	size_t len = total_len, align = NET_SKB_PAD, linear; +	size_t len = total_len, align = tun->align, linear;  	struct virtio_net_hdr gso = { 0 };  	int good_linear;  	int copylen; @@ -1466,6 +1479,8 @@ static void tun_setup(struct net_device *dev)  	dev->ethtool_ops = &tun_ethtool_ops;  	dev->destructor = tun_free_netdev; +	/* We prefer our own queue length */ +	dev->tx_queue_len = TUN_READQ_SIZE;  }  /* Trivial set of netlink ops to allow deleting tun or tap @@ -1694,6 +1709,7 @@ static int tun_set_iff(struct net *net, struct file *file, struct ifreq *ifr)  		tun->txflt.count = 0;  		tun->vnet_hdr_sz = sizeof(struct virtio_net_hdr); +		tun->align = NET_SKB_PAD;  		tun->filter_attached = false;  		tun->sndbuf = tfile->socket.sk->sk_sndbuf; @@ -1807,7 +1823,7 @@ static void tun_detach_filter(struct tun_struct *tun, int n)  	for (i = 0; i < n; i++) {  		tfile = rtnl_dereference(tun->tfiles[i]); -		sk_detach_filter(tfile->socket.sk); +		__sk_detach_filter(tfile->socket.sk, lockdep_rtnl_is_held());  	}  	tun->filter_attached = false; @@ -1820,7 +1836,8 @@ static int tun_attach_filter(struct tun_struct *tun)  	for (i = 0; i < tun->numqueues; i++) {  		tfile = rtnl_dereference(tun->tfiles[i]); -		ret = sk_attach_filter(&tun->fprog, tfile->socket.sk); +		ret = __sk_attach_filter(&tun->fprog, tfile->socket.sk, +					 lockdep_rtnl_is_held());  		if (ret) {  			tun_detach_filter(tun, i);  			return ret; | 
