1 // SPDX-License-Identifier: GPL-2.0-only
3 * vxcan.c - Virtual CAN Tunnel for cross namespace communication
5 * This code is derived from drivers/net/can/vcan.c for the virtual CAN
6 * specific parts and from drivers/net/veth.c to implement the netlink API
7 * for network interface pairs in a common and established way.
9 * Copyright (c) 2017 Oliver Hartkopp <socketcan@hartkopp.net>
12 #include <linux/module.h>
13 #include <linux/init.h>
14 #include <linux/netdevice.h>
15 #include <linux/if_arp.h>
16 #include <linux/if_ether.h>
17 #include <linux/can.h>
18 #include <linux/can/dev.h>
19 #include <linux/can/skb.h>
20 #include <linux/can/vxcan.h>
21 #include <linux/can/can-ml.h>
22 #include <linux/slab.h>
23 #include <net/rtnetlink.h>
25 #define DRV_NAME "vxcan"
27 MODULE_DESCRIPTION("Virtual CAN Tunnel");
28 MODULE_LICENSE("GPL");
29 MODULE_AUTHOR("Oliver Hartkopp <socketcan@hartkopp.net>");
30 MODULE_ALIAS_RTNL_LINK(DRV_NAME
);
33 struct net_device __rcu
*peer
;
36 static netdev_tx_t
vxcan_xmit(struct sk_buff
*skb
, struct net_device
*dev
)
38 struct vxcan_priv
*priv
= netdev_priv(dev
);
39 struct net_device
*peer
;
40 struct canfd_frame
*cfd
= (struct canfd_frame
*)skb
->data
;
41 struct net_device_stats
*peerstats
, *srcstats
= &dev
->stats
;
43 if (can_dropped_invalid_skb(dev
, skb
))
47 peer
= rcu_dereference(priv
->peer
);
48 if (unlikely(!peer
)) {
50 dev
->stats
.tx_dropped
++;
54 skb
= can_create_echo_skb(skb
);
58 /* reset CAN GW hop counter */
60 skb
->pkt_type
= PACKET_BROADCAST
;
62 skb
->ip_summed
= CHECKSUM_UNNECESSARY
;
64 if (netif_rx_ni(skb
) == NET_RX_SUCCESS
) {
65 srcstats
->tx_packets
++;
66 srcstats
->tx_bytes
+= cfd
->len
;
67 peerstats
= &peer
->stats
;
68 peerstats
->rx_packets
++;
69 peerstats
->rx_bytes
+= cfd
->len
;
78 static int vxcan_open(struct net_device
*dev
)
80 struct vxcan_priv
*priv
= netdev_priv(dev
);
81 struct net_device
*peer
= rtnl_dereference(priv
->peer
);
86 if (peer
->flags
& IFF_UP
) {
87 netif_carrier_on(dev
);
88 netif_carrier_on(peer
);
93 static int vxcan_close(struct net_device
*dev
)
95 struct vxcan_priv
*priv
= netdev_priv(dev
);
96 struct net_device
*peer
= rtnl_dereference(priv
->peer
);
98 netif_carrier_off(dev
);
100 netif_carrier_off(peer
);
105 static int vxcan_get_iflink(const struct net_device
*dev
)
107 struct vxcan_priv
*priv
= netdev_priv(dev
);
108 struct net_device
*peer
;
112 peer
= rcu_dereference(priv
->peer
);
113 iflink
= peer
? peer
->ifindex
: 0;
119 static int vxcan_change_mtu(struct net_device
*dev
, int new_mtu
)
121 /* Do not allow changing the MTU while running */
122 if (dev
->flags
& IFF_UP
)
125 if (new_mtu
!= CAN_MTU
&& new_mtu
!= CANFD_MTU
)
132 static const struct net_device_ops vxcan_netdev_ops
= {
133 .ndo_open
= vxcan_open
,
134 .ndo_stop
= vxcan_close
,
135 .ndo_start_xmit
= vxcan_xmit
,
136 .ndo_get_iflink
= vxcan_get_iflink
,
137 .ndo_change_mtu
= vxcan_change_mtu
,
140 static void vxcan_setup(struct net_device
*dev
)
142 dev
->type
= ARPHRD_CAN
;
143 dev
->mtu
= CANFD_MTU
;
144 dev
->hard_header_len
= 0;
146 dev
->tx_queue_len
= 0;
147 dev
->flags
= (IFF_NOARP
|IFF_ECHO
);
148 dev
->netdev_ops
= &vxcan_netdev_ops
;
149 dev
->needs_free_netdev
= true;
150 dev
->ml_priv
= netdev_priv(dev
) + ALIGN(sizeof(struct vxcan_priv
), NETDEV_ALIGN
);
153 /* forward declaration for rtnl_create_link() */
154 static struct rtnl_link_ops vxcan_link_ops
;
156 static int vxcan_newlink(struct net
*net
, struct net_device
*dev
,
157 struct nlattr
*tb
[], struct nlattr
*data
[],
158 struct netlink_ext_ack
*extack
)
160 struct vxcan_priv
*priv
;
161 struct net_device
*peer
;
162 struct net
*peer_net
;
164 struct nlattr
*peer_tb
[IFLA_MAX
+ 1], **tbp
= tb
;
165 char ifname
[IFNAMSIZ
];
166 unsigned char name_assign_type
;
167 struct ifinfomsg
*ifmp
= NULL
;
170 /* register peer device */
171 if (data
&& data
[VXCAN_INFO_PEER
]) {
172 struct nlattr
*nla_peer
;
174 nla_peer
= data
[VXCAN_INFO_PEER
];
175 ifmp
= nla_data(nla_peer
);
176 err
= rtnl_nla_parse_ifla(peer_tb
,
178 sizeof(struct ifinfomsg
),
180 sizeof(struct ifinfomsg
),
188 if (ifmp
&& tbp
[IFLA_IFNAME
]) {
189 nla_strlcpy(ifname
, tbp
[IFLA_IFNAME
], IFNAMSIZ
);
190 name_assign_type
= NET_NAME_USER
;
192 snprintf(ifname
, IFNAMSIZ
, DRV_NAME
"%%d");
193 name_assign_type
= NET_NAME_ENUM
;
196 peer_net
= rtnl_link_get_net(net
, tbp
);
197 if (IS_ERR(peer_net
))
198 return PTR_ERR(peer_net
);
200 peer
= rtnl_create_link(peer_net
, ifname
, name_assign_type
,
201 &vxcan_link_ops
, tbp
, extack
);
204 return PTR_ERR(peer
);
207 if (ifmp
&& dev
->ifindex
)
208 peer
->ifindex
= ifmp
->ifi_index
;
210 err
= register_netdevice(peer
);
218 netif_carrier_off(peer
);
220 err
= rtnl_configure_link(peer
, ifmp
);
222 goto unregister_network_device
;
224 /* register first device */
226 nla_strlcpy(dev
->name
, tb
[IFLA_IFNAME
], IFNAMSIZ
);
228 snprintf(dev
->name
, IFNAMSIZ
, DRV_NAME
"%%d");
230 err
= register_netdevice(dev
);
232 goto unregister_network_device
;
234 netif_carrier_off(dev
);
236 /* cross link the device pair */
237 priv
= netdev_priv(dev
);
238 rcu_assign_pointer(priv
->peer
, peer
);
240 priv
= netdev_priv(peer
);
241 rcu_assign_pointer(priv
->peer
, dev
);
245 unregister_network_device
:
246 unregister_netdevice(peer
);
250 static void vxcan_dellink(struct net_device
*dev
, struct list_head
*head
)
252 struct vxcan_priv
*priv
;
253 struct net_device
*peer
;
255 priv
= netdev_priv(dev
);
256 peer
= rtnl_dereference(priv
->peer
);
258 /* Note : dellink() is called from default_device_exit_batch(),
259 * before a rcu_synchronize() point. The devices are guaranteed
260 * not being freed before one RCU grace period.
262 RCU_INIT_POINTER(priv
->peer
, NULL
);
263 unregister_netdevice_queue(dev
, head
);
266 priv
= netdev_priv(peer
);
267 RCU_INIT_POINTER(priv
->peer
, NULL
);
268 unregister_netdevice_queue(peer
, head
);
272 static const struct nla_policy vxcan_policy
[VXCAN_INFO_MAX
+ 1] = {
273 [VXCAN_INFO_PEER
] = { .len
= sizeof(struct ifinfomsg
) },
276 static struct net
*vxcan_get_link_net(const struct net_device
*dev
)
278 struct vxcan_priv
*priv
= netdev_priv(dev
);
279 struct net_device
*peer
= rtnl_dereference(priv
->peer
);
281 return peer
? dev_net(peer
) : dev_net(dev
);
284 static struct rtnl_link_ops vxcan_link_ops
= {
286 .priv_size
= ALIGN(sizeof(struct vxcan_priv
), NETDEV_ALIGN
) + sizeof(struct can_ml_priv
),
287 .setup
= vxcan_setup
,
288 .newlink
= vxcan_newlink
,
289 .dellink
= vxcan_dellink
,
290 .policy
= vxcan_policy
,
291 .maxtype
= VXCAN_INFO_MAX
,
292 .get_link_net
= vxcan_get_link_net
,
295 static __init
int vxcan_init(void)
297 pr_info("vxcan: Virtual CAN Tunnel driver\n");
299 return rtnl_link_register(&vxcan_link_ops
);
302 static __exit
void vxcan_exit(void)
304 rtnl_link_unregister(&vxcan_link_ops
);
307 module_init(vxcan_init
);
308 module_exit(vxcan_exit
);