2007-09-25 16:14:46 -07:00
/*
* drivers / net / veth . c
*
* Copyright ( C ) 2007 OpenVZ http : //openvz.org, SWsoft Inc
*
* Author : Pavel Emelianov < xemul @ openvz . org >
* Ethtool interface from : Eric W . Biederman < ebiederm @ xmission . com >
*
*/
# include <linux/netdevice.h>
# include <linux/ethtool.h>
# include <linux/etherdevice.h>
# include <net/dst.h>
# include <net/xfrm.h>
2007-12-25 17:23:59 -08:00
# include <linux/veth.h>
2007-09-25 16:14:46 -07:00
# define DRV_NAME "veth"
# define DRV_VERSION "1.0"
2009-03-03 23:36:04 -08:00
# define MIN_MTU 68 /* Min L3 MTU */
# define MAX_MTU 65535 /* Max L3 MTU (arbitrary) */
# define MTU_PAD (ETH_HLEN + 4) /* Max difference between L2 and L3 size MTU */
2007-09-25 16:14:46 -07:00
struct veth_net_stats {
unsigned long rx_packets ;
unsigned long tx_packets ;
unsigned long rx_bytes ;
unsigned long tx_bytes ;
unsigned long tx_dropped ;
2009-03-03 23:36:04 -08:00
unsigned long rx_dropped ;
2007-09-25 16:14:46 -07:00
} ;
struct veth_priv {
struct net_device * peer ;
struct veth_net_stats * stats ;
unsigned ip_summed ;
} ;
/*
* ethtool interface
*/
static struct {
const char string [ ETH_GSTRING_LEN ] ;
} ethtool_stats_keys [ ] = {
{ " peer_ifindex " } ,
} ;
static int veth_get_settings ( struct net_device * dev , struct ethtool_cmd * cmd )
{
cmd - > supported = 0 ;
cmd - > advertising = 0 ;
cmd - > speed = SPEED_10000 ;
cmd - > duplex = DUPLEX_FULL ;
cmd - > port = PORT_TP ;
cmd - > phy_address = 0 ;
cmd - > transceiver = XCVR_INTERNAL ;
cmd - > autoneg = AUTONEG_DISABLE ;
cmd - > maxtxpkt = 0 ;
cmd - > maxrxpkt = 0 ;
return 0 ;
}
static void veth_get_drvinfo ( struct net_device * dev , struct ethtool_drvinfo * info )
{
strcpy ( info - > driver , DRV_NAME ) ;
strcpy ( info - > version , DRV_VERSION ) ;
strcpy ( info - > fw_version , " N/A " ) ;
}
static void veth_get_strings ( struct net_device * dev , u32 stringset , u8 * buf )
{
switch ( stringset ) {
case ETH_SS_STATS :
memcpy ( buf , & ethtool_stats_keys , sizeof ( ethtool_stats_keys ) ) ;
break ;
}
}
2007-10-03 18:07:32 -07:00
static int veth_get_sset_count ( struct net_device * dev , int sset )
2007-09-25 16:14:46 -07:00
{
2007-10-03 18:07:32 -07:00
switch ( sset ) {
case ETH_SS_STATS :
return ARRAY_SIZE ( ethtool_stats_keys ) ;
default :
return - EOPNOTSUPP ;
}
2007-09-25 16:14:46 -07:00
}
static void veth_get_ethtool_stats ( struct net_device * dev ,
struct ethtool_stats * stats , u64 * data )
{
struct veth_priv * priv ;
priv = netdev_priv ( dev ) ;
data [ 0 ] = priv - > peer - > ifindex ;
}
static u32 veth_get_rx_csum ( struct net_device * dev )
{
struct veth_priv * priv ;
priv = netdev_priv ( dev ) ;
return priv - > ip_summed = = CHECKSUM_UNNECESSARY ;
}
static int veth_set_rx_csum ( struct net_device * dev , u32 data )
{
struct veth_priv * priv ;
priv = netdev_priv ( dev ) ;
priv - > ip_summed = data ? CHECKSUM_UNNECESSARY : CHECKSUM_NONE ;
return 0 ;
}
static u32 veth_get_tx_csum ( struct net_device * dev )
{
return ( dev - > features & NETIF_F_NO_CSUM ) ! = 0 ;
}
static int veth_set_tx_csum ( struct net_device * dev , u32 data )
{
if ( data )
dev - > features | = NETIF_F_NO_CSUM ;
else
dev - > features & = ~ NETIF_F_NO_CSUM ;
return 0 ;
}
static struct ethtool_ops veth_ethtool_ops = {
. get_settings = veth_get_settings ,
. get_drvinfo = veth_get_drvinfo ,
. get_link = ethtool_op_get_link ,
. get_rx_csum = veth_get_rx_csum ,
. set_rx_csum = veth_set_rx_csum ,
. get_tx_csum = veth_get_tx_csum ,
. set_tx_csum = veth_set_tx_csum ,
. get_sg = ethtool_op_get_sg ,
. set_sg = ethtool_op_set_sg ,
. get_strings = veth_get_strings ,
2007-10-03 18:07:32 -07:00
. get_sset_count = veth_get_sset_count ,
2007-09-25 16:14:46 -07:00
. get_ethtool_stats = veth_get_ethtool_stats ,
} ;
/*
* xmit
*/
static int veth_xmit ( struct sk_buff * skb , struct net_device * dev )
{
struct net_device * rcv = NULL ;
struct veth_priv * priv , * rcv_priv ;
2009-03-03 23:36:04 -08:00
struct veth_net_stats * stats , * rcv_stats ;
2007-09-25 16:14:46 -07:00
int length , cpu ;
skb_orphan ( skb ) ;
priv = netdev_priv ( dev ) ;
rcv = priv - > peer ;
rcv_priv = netdev_priv ( rcv ) ;
cpu = smp_processor_id ( ) ;
stats = per_cpu_ptr ( priv - > stats , cpu ) ;
2009-03-03 23:36:04 -08:00
rcv_stats = per_cpu_ptr ( rcv_priv - > stats , cpu ) ;
2007-09-25 16:14:46 -07:00
if ( ! ( rcv - > flags & IFF_UP ) )
2009-03-03 23:36:04 -08:00
goto tx_drop ;
if ( skb - > len > ( rcv - > mtu + MTU_PAD ) )
goto rx_drop ;
2007-09-25 16:14:46 -07:00
skb - > pkt_type = PACKET_HOST ;
skb - > protocol = eth_type_trans ( skb , rcv ) ;
if ( dev - > features & NETIF_F_NO_CSUM )
skb - > ip_summed = rcv_priv - > ip_summed ;
skb - > mark = 0 ;
secpath_reset ( skb ) ;
nf_reset ( skb ) ;
length = skb - > len ;
stats - > tx_bytes + = length ;
stats - > tx_packets + + ;
2009-03-03 23:36:04 -08:00
rcv_stats - > rx_bytes + = length ;
rcv_stats - > rx_packets + + ;
2007-09-25 16:14:46 -07:00
netif_rx ( skb ) ;
2009-06-23 06:03:08 +00:00
return NETDEV_TX_OK ;
2007-09-25 16:14:46 -07:00
2009-03-03 23:36:04 -08:00
tx_drop :
2007-09-25 16:14:46 -07:00
kfree_skb ( skb ) ;
stats - > tx_dropped + + ;
2009-06-23 06:03:08 +00:00
return NETDEV_TX_OK ;
2009-03-03 23:36:04 -08:00
rx_drop :
kfree_skb ( skb ) ;
rcv_stats - > rx_dropped + + ;
2009-06-23 06:03:08 +00:00
return NETDEV_TX_OK ;
2007-09-25 16:14:46 -07:00
}
/*
* general routines
*/
static struct net_device_stats * veth_get_stats ( struct net_device * dev )
{
2009-06-25 02:45:42 -07:00
struct veth_priv * priv ;
struct net_device_stats * dev_stats ;
int cpu ;
2007-09-25 16:14:46 -07:00
struct veth_net_stats * stats ;
2009-06-25 02:45:42 -07:00
priv = netdev_priv ( dev ) ;
dev_stats = & dev - > stats ;
2007-09-25 16:14:46 -07:00
dev_stats - > rx_packets = 0 ;
dev_stats - > tx_packets = 0 ;
dev_stats - > rx_bytes = 0 ;
dev_stats - > tx_bytes = 0 ;
dev_stats - > tx_dropped = 0 ;
2009-03-03 23:36:04 -08:00
dev_stats - > rx_dropped = 0 ;
2007-09-25 16:14:46 -07:00
2009-06-25 02:45:42 -07:00
for_each_online_cpu ( cpu ) {
stats = per_cpu_ptr ( priv - > stats , cpu ) ;
2007-09-25 16:14:46 -07:00
2009-06-25 02:45:42 -07:00
dev_stats - > rx_packets + = stats - > rx_packets ;
dev_stats - > tx_packets + = stats - > tx_packets ;
dev_stats - > rx_bytes + = stats - > rx_bytes ;
dev_stats - > tx_bytes + = stats - > tx_bytes ;
dev_stats - > tx_dropped + = stats - > tx_dropped ;
dev_stats - > rx_dropped + = stats - > rx_dropped ;
}
2007-09-25 16:14:46 -07:00
return dev_stats ;
}
static int veth_open ( struct net_device * dev )
{
struct veth_priv * priv ;
priv = netdev_priv ( dev ) ;
if ( priv - > peer = = NULL )
return - ENOTCONN ;
if ( priv - > peer - > flags & IFF_UP ) {
netif_carrier_on ( dev ) ;
netif_carrier_on ( priv - > peer ) ;
}
return 0 ;
}
2009-02-25 19:47:29 +00:00
static int veth_close ( struct net_device * dev )
{
struct veth_priv * priv = netdev_priv ( dev ) ;
netif_carrier_off ( dev ) ;
netif_carrier_off ( priv - > peer ) ;
return 0 ;
}
2009-03-03 23:36:04 -08:00
static int is_valid_veth_mtu ( int new_mtu )
{
return ( new_mtu > = MIN_MTU & & new_mtu < = MAX_MTU ) ;
}
static int veth_change_mtu ( struct net_device * dev , int new_mtu )
{
if ( ! is_valid_veth_mtu ( new_mtu ) )
return - EINVAL ;
dev - > mtu = new_mtu ;
return 0 ;
}
2007-09-25 16:14:46 -07:00
static int veth_dev_init ( struct net_device * dev )
{
struct veth_net_stats * stats ;
struct veth_priv * priv ;
stats = alloc_percpu ( struct veth_net_stats ) ;
if ( stats = = NULL )
return - ENOMEM ;
priv = netdev_priv ( dev ) ;
priv - > stats = stats ;
return 0 ;
}
2009-06-25 02:45:42 -07:00
static void veth_dev_free ( struct net_device * dev )
{
struct veth_priv * priv ;
priv = netdev_priv ( dev ) ;
free_percpu ( priv - > stats ) ;
free_netdev ( dev ) ;
}
2008-11-19 21:50:10 -08:00
static const struct net_device_ops veth_netdev_ops = {
2009-02-22 00:04:45 -08:00
. ndo_init = veth_dev_init ,
. ndo_open = veth_open ,
2009-02-25 19:47:29 +00:00
. ndo_stop = veth_close ,
2009-02-22 00:04:45 -08:00
. ndo_start_xmit = veth_xmit ,
2009-03-03 23:36:04 -08:00
. ndo_change_mtu = veth_change_mtu ,
2009-02-22 00:04:45 -08:00
. ndo_get_stats = veth_get_stats ,
. ndo_set_mac_address = eth_mac_addr ,
2008-11-19 21:50:10 -08:00
} ;
2007-09-25 16:14:46 -07:00
static void veth_setup ( struct net_device * dev )
{
ether_setup ( dev ) ;
2008-11-19 21:50:10 -08:00
dev - > netdev_ops = & veth_netdev_ops ;
2007-09-25 16:14:46 -07:00
dev - > ethtool_ops = & veth_ethtool_ops ;
dev - > features | = NETIF_F_LLTX ;
2009-06-25 02:45:42 -07:00
dev - > destructor = veth_dev_free ;
2007-09-25 16:14:46 -07:00
}
/*
* netlink interface
*/
static int veth_validate ( struct nlattr * tb [ ] , struct nlattr * data [ ] )
{
if ( tb [ IFLA_ADDRESS ] ) {
if ( nla_len ( tb [ IFLA_ADDRESS ] ) ! = ETH_ALEN )
return - EINVAL ;
if ( ! is_valid_ether_addr ( nla_data ( tb [ IFLA_ADDRESS ] ) ) )
return - EADDRNOTAVAIL ;
}
2009-03-03 23:36:04 -08:00
if ( tb [ IFLA_MTU ] ) {
if ( ! is_valid_veth_mtu ( nla_get_u32 ( tb [ IFLA_MTU ] ) ) )
return - EINVAL ;
}
2007-09-25 16:14:46 -07:00
return 0 ;
}
static struct rtnl_link_ops veth_link_ops ;
static int veth_newlink ( struct net_device * dev ,
struct nlattr * tb [ ] , struct nlattr * data [ ] )
{
int err ;
struct net_device * peer ;
struct veth_priv * priv ;
char ifname [ IFNAMSIZ ] ;
struct nlattr * peer_tb [ IFLA_MAX + 1 ] , * * tbp ;
/*
* create and register peer first
*
* struct ifinfomsg is at the head of VETH_INFO_PEER , but we
* skip it since no info from it is useful yet
*/
if ( data ! = NULL & & data [ VETH_INFO_PEER ] ! = NULL ) {
struct nlattr * nla_peer ;
nla_peer = data [ VETH_INFO_PEER ] ;
err = nla_parse ( peer_tb , IFLA_MAX ,
nla_data ( nla_peer ) + sizeof ( struct ifinfomsg ) ,
nla_len ( nla_peer ) - sizeof ( struct ifinfomsg ) ,
ifla_policy ) ;
if ( err < 0 )
return err ;
err = veth_validate ( peer_tb , NULL ) ;
if ( err < 0 )
return err ;
tbp = peer_tb ;
} else
tbp = tb ;
if ( tbp [ IFLA_IFNAME ] )
nla_strlcpy ( ifname , tbp [ IFLA_IFNAME ] , IFNAMSIZ ) ;
else
snprintf ( ifname , IFNAMSIZ , DRV_NAME " %%d " ) ;
2008-03-25 21:47:49 +09:00
peer = rtnl_create_link ( dev_net ( dev ) , ifname , & veth_link_ops , tbp ) ;
2007-09-25 16:14:46 -07:00
if ( IS_ERR ( peer ) )
return PTR_ERR ( peer ) ;
if ( tbp [ IFLA_ADDRESS ] = = NULL )
random_ether_addr ( peer - > dev_addr ) ;
err = register_netdevice ( peer ) ;
if ( err < 0 )
goto err_register_peer ;
netif_carrier_off ( peer ) ;
/*
* register dev last
*
* note , that since we ' ve registered new device the dev ' s name
* should be re - allocated
*/
if ( tb [ IFLA_ADDRESS ] = = NULL )
random_ether_addr ( dev - > dev_addr ) ;
if ( tb [ IFLA_IFNAME ] )
nla_strlcpy ( dev - > name , tb [ IFLA_IFNAME ] , IFNAMSIZ ) ;
else
snprintf ( dev - > name , IFNAMSIZ , DRV_NAME " %%d " ) ;
if ( strchr ( dev - > name , ' % ' ) ) {
err = dev_alloc_name ( dev , dev - > name ) ;
if ( err < 0 )
goto err_alloc_name ;
}
err = register_netdevice ( dev ) ;
if ( err < 0 )
goto err_register_dev ;
netif_carrier_off ( dev ) ;
/*
* tie the deviced together
*/
priv = netdev_priv ( dev ) ;
priv - > peer = peer ;
priv = netdev_priv ( peer ) ;
priv - > peer = dev ;
return 0 ;
err_register_dev :
/* nothing to do */
err_alloc_name :
unregister_netdevice ( peer ) ;
return err ;
err_register_peer :
free_netdev ( peer ) ;
return err ;
}
static void veth_dellink ( struct net_device * dev )
{
struct veth_priv * priv ;
struct net_device * peer ;
priv = netdev_priv ( dev ) ;
peer = priv - > peer ;
unregister_netdevice ( dev ) ;
unregister_netdevice ( peer ) ;
}
static const struct nla_policy veth_policy [ VETH_INFO_MAX + 1 ] ;
static struct rtnl_link_ops veth_link_ops = {
. kind = DRV_NAME ,
. priv_size = sizeof ( struct veth_priv ) ,
. setup = veth_setup ,
. validate = veth_validate ,
. newlink = veth_newlink ,
. dellink = veth_dellink ,
. policy = veth_policy ,
. maxtype = VETH_INFO_MAX ,
} ;
/*
* init / fini
*/
static __init int veth_init ( void )
{
return rtnl_link_register ( & veth_link_ops ) ;
}
static __exit void veth_exit ( void )
{
2008-01-20 17:25:14 -08:00
rtnl_link_unregister ( & veth_link_ops ) ;
2007-09-25 16:14:46 -07:00
}
module_init ( veth_init ) ;
module_exit ( veth_exit ) ;
MODULE_DESCRIPTION ( " Virtual Ethernet Tunnel " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_ALIAS_RTNL_LINK ( DRV_NAME ) ;