2 * Copyright (c) 2014 Nicira, Inc.
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License
6 * as published by the Free Software Foundation; either version
7 * 2 of the License, or (at your option) any later version.
10 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
14 #include <linux/net.h>
15 #include <linux/rculist.h>
16 #include <linux/udp.h>
17 #include <linux/if_vlan.h>
18 #include <linux/module.h>
20 #include <net/geneve.h>
23 #include <net/route.h>
29 #include "vport-netdev.h"
31 static struct vport_ops ovs_geneve_vport_ops
;
33 * struct geneve_port - Keeps track of open UDP ports
34 * @dst_port: destination port.
40 static inline struct geneve_port
*geneve_vport(const struct vport
*vport
)
42 return vport_priv(vport
);
45 static int geneve_get_options(const struct vport
*vport
,
48 struct geneve_port
*geneve_port
= geneve_vport(vport
);
50 if (nla_put_u16(skb
, OVS_TUNNEL_ATTR_DST_PORT
, geneve_port
->port_no
))
55 static int geneve_get_egress_tun_info(struct vport
*vport
, struct sk_buff
*skb
,
56 struct dp_upcall_info
*upcall
)
58 struct geneve_port
*geneve_port
= geneve_vport(vport
);
59 struct net
*net
= ovs_dp_get_net(vport
->dp
);
60 __be16 dport
= htons(geneve_port
->port_no
);
61 __be16 sport
= udp_flow_src_port(net
, skb
, 1, USHRT_MAX
, true);
63 return ovs_tunnel_get_egress_info(upcall
, ovs_dp_get_net(vport
->dp
),
64 skb
, IPPROTO_UDP
, sport
, dport
);
67 static struct vport
*geneve_tnl_create(const struct vport_parms
*parms
)
69 struct net
*net
= ovs_dp_get_net(parms
->dp
);
70 struct nlattr
*options
= parms
->options
;
71 struct geneve_port
*geneve_port
;
72 struct net_device
*dev
;
83 a
= nla_find_nested(options
, OVS_TUNNEL_ATTR_DST_PORT
);
84 if (a
&& nla_len(a
) == sizeof(u16
)) {
85 dst_port
= nla_get_u16(a
);
87 /* Require destination port from userspace. */
92 vport
= ovs_vport_alloc(sizeof(struct geneve_port
),
93 &ovs_geneve_vport_ops
, parms
);
97 geneve_port
= geneve_vport(vport
);
98 geneve_port
->port_no
= dst_port
;
101 dev
= geneve_dev_create_fb(net
, parms
->name
, NET_NAME_USER
, dst_port
);
104 ovs_vport_free(vport
);
105 return ERR_CAST(dev
);
108 dev_change_flags(dev
, dev
->flags
| IFF_UP
);
115 static struct vport
*geneve_create(const struct vport_parms
*parms
)
119 vport
= geneve_tnl_create(parms
);
123 return ovs_netdev_link(vport
, parms
->name
);
126 static struct vport_ops ovs_geneve_vport_ops
= {
127 .type
= OVS_VPORT_TYPE_GENEVE
,
128 .create
= geneve_create
,
129 .destroy
= ovs_netdev_tunnel_destroy
,
130 .get_options
= geneve_get_options
,
131 .send
= ovs_netdev_send
,
132 .owner
= THIS_MODULE
,
133 .get_egress_tun_info
= geneve_get_egress_tun_info
,
136 static int __init
ovs_geneve_tnl_init(void)
138 return ovs_vport_ops_register(&ovs_geneve_vport_ops
);
141 static void __exit
ovs_geneve_tnl_exit(void)
143 ovs_vport_ops_unregister(&ovs_geneve_vport_ops
);
146 module_init(ovs_geneve_tnl_init
);
147 module_exit(ovs_geneve_tnl_exit
);
149 MODULE_DESCRIPTION("OVS: Geneve swiching port");
150 MODULE_LICENSE("GPL");
151 MODULE_ALIAS("vport-type-5");