2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2014-10-03 15:35:33 -07:00
/*
* Copyright ( c ) 2014 Nicira , Inc .
*/
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
# include <linux/in.h>
# include <linux/ip.h>
# include <linux/net.h>
# include <linux/rculist.h>
# include <linux/udp.h>
# include <linux/if_vlan.h>
2014-10-22 17:29:06 +02:00
# include <linux/module.h>
2014-10-03 15:35:33 -07:00
# include <net/geneve.h>
# include <net/icmp.h>
# include <net/ip.h>
# include <net/route.h>
# include <net/udp.h>
# include <net/xfrm.h>
# include "datapath.h"
# include "vport.h"
2015-08-26 23:46:53 -07:00
# include "vport-netdev.h"
2014-10-03 15:35:33 -07:00
2014-10-22 17:29:06 +02:00
static struct vport_ops ovs_geneve_vport_ops ;
2014-10-03 15:35:33 -07:00
/**
* struct geneve_port - Keeps track of open UDP ports
2015-08-26 23:46:53 -07:00
* @ dst_port : destination port .
2014-10-03 15:35:33 -07:00
*/
struct geneve_port {
2016-01-09 16:07:10 -07:00
u16 dst_port ;
2014-10-03 15:35:33 -07:00
} ;
static inline struct geneve_port * geneve_vport ( const struct vport * vport )
{
return vport_priv ( vport ) ;
}
static int geneve_get_options ( const struct vport * vport ,
struct sk_buff * skb )
{
struct geneve_port * geneve_port = geneve_vport ( vport ) ;
2016-01-09 16:07:10 -07:00
if ( nla_put_u16 ( skb , OVS_TUNNEL_ATTR_DST_PORT , geneve_port - > dst_port ) )
2014-10-03 15:35:33 -07:00
return - EMSGSIZE ;
return 0 ;
}
static struct vport * geneve_tnl_create ( const struct vport_parms * parms )
{
struct net * net = ovs_dp_get_net ( parms - > dp ) ;
struct nlattr * options = parms - > options ;
struct geneve_port * geneve_port ;
2015-08-26 23:46:53 -07:00
struct net_device * dev ;
2014-10-03 15:35:33 -07:00
struct vport * vport ;
struct nlattr * a ;
u16 dst_port ;
2015-08-26 23:46:53 -07:00
int err ;
2014-10-03 15:35:33 -07:00
if ( ! options ) {
err = - EINVAL ;
goto error ;
}
a = nla_find_nested ( options , OVS_TUNNEL_ATTR_DST_PORT ) ;
if ( a & & nla_len ( a ) = = sizeof ( u16 ) ) {
dst_port = nla_get_u16 ( a ) ;
} else {
/* Require destination port from userspace. */
err = - EINVAL ;
goto error ;
}
vport = ovs_vport_alloc ( sizeof ( struct geneve_port ) ,
& ovs_geneve_vport_ops , parms ) ;
if ( IS_ERR ( vport ) )
return vport ;
geneve_port = geneve_vport ( vport ) ;
2016-01-09 16:07:10 -07:00
geneve_port - > dst_port = dst_port ;
2014-10-03 15:35:33 -07:00
2015-08-26 23:46:53 -07:00
rtnl_lock ( ) ;
dev = geneve_dev_create_fb ( net , parms - > name , NET_NAME_USER , dst_port ) ;
if ( IS_ERR ( dev ) ) {
rtnl_unlock ( ) ;
2014-10-03 15:35:33 -07:00
ovs_vport_free ( vport ) ;
2015-08-26 23:46:53 -07:00
return ERR_CAST ( dev ) ;
2014-10-03 15:35:33 -07:00
}
2018-12-06 17:05:42 +00:00
err = dev_change_flags ( dev , dev - > flags | IFF_UP , NULL ) ;
2016-08-09 16:24:50 +01:00
if ( err < 0 ) {
2022-10-28 04:42:24 -04:00
rtnl_delete_link ( dev , 0 , NULL ) ;
2016-08-09 16:24:50 +01:00
rtnl_unlock ( ) ;
ovs_vport_free ( vport ) ;
goto error ;
}
2015-08-26 23:46:53 -07:00
rtnl_unlock ( ) ;
2014-10-03 15:35:33 -07:00
return vport ;
error :
return ERR_PTR ( err ) ;
}
2015-08-26 23:46:53 -07:00
static struct vport * geneve_create ( const struct vport_parms * parms )
2014-10-03 15:35:33 -07:00
{
2015-08-26 23:46:53 -07:00
struct vport * vport ;
2014-10-03 15:35:33 -07:00
2015-08-26 23:46:53 -07:00
vport = geneve_tnl_create ( parms ) ;
if ( IS_ERR ( vport ) )
return vport ;
2014-11-06 06:51:24 -08:00
2015-08-26 23:46:53 -07:00
return ovs_netdev_link ( vport , parms - > name ) ;
2014-11-06 06:51:24 -08:00
}
2014-10-22 17:29:06 +02:00
static struct vport_ops ovs_geneve_vport_ops = {
2014-10-03 15:35:33 -07:00
. type = OVS_VPORT_TYPE_GENEVE ,
2015-08-26 23:46:53 -07:00
. create = geneve_create ,
. destroy = ovs_netdev_tunnel_destroy ,
2014-10-03 15:35:33 -07:00
. get_options = geneve_get_options ,
2015-10-20 23:00:10 -07:00
. send = dev_queue_xmit ,
2014-10-03 15:35:33 -07:00
} ;
2014-10-22 17:29:06 +02:00
static int __init ovs_geneve_tnl_init ( void )
{
return ovs_vport_ops_register ( & ovs_geneve_vport_ops ) ;
}
static void __exit ovs_geneve_tnl_exit ( void )
{
ovs_vport_ops_unregister ( & ovs_geneve_vport_ops ) ;
}
module_init ( ovs_geneve_tnl_init ) ;
module_exit ( ovs_geneve_tnl_exit ) ;
2015-12-13 15:26:11 +09:00
MODULE_DESCRIPTION ( " OVS: Geneve switching port " ) ;
2014-10-22 17:29:06 +02:00
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " vport-type-5 " ) ;