• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1  /* Copyright (C) 2007-2017  B.A.T.M.A.N. contributors:
2   *
3   * Marek Lindner, Simon Wunderlich
4   *
5   * This program is free software; you can redistribute it and/or
6   * modify it under the terms of version 2 of the GNU General Public
7   * License as published by the Free Software Foundation.
8   *
9   * This program is distributed in the hope that it will be useful, but
10   * WITHOUT ANY WARRANTY; without even the implied warranty of
11   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12   * General Public License for more details.
13   *
14   * You should have received a copy of the GNU General Public License
15   * along with this program; if not, see <http://www.gnu.org/licenses/>.
16   */
17  
18  #include "main.h"
19  
20  #include <linux/atomic.h>
21  #include <linux/bug.h>
22  #include <linux/byteorder/generic.h>
23  #include <linux/crc32c.h>
24  #include <linux/errno.h>
25  #include <linux/fs.h>
26  #include <linux/genetlink.h>
27  #include <linux/if_ether.h>
28  #include <linux/if_vlan.h>
29  #include <linux/init.h>
30  #include <linux/ip.h>
31  #include <linux/ipv6.h>
32  #include <linux/kernel.h>
33  #include <linux/kref.h>
34  #include <linux/list.h>
35  #include <linux/module.h>
36  #include <linux/netdevice.h>
37  #include <linux/printk.h>
38  #include <linux/rculist.h>
39  #include <linux/rcupdate.h>
40  #include <linux/seq_file.h>
41  #include <linux/skbuff.h>
42  #include <linux/spinlock.h>
43  #include <linux/stddef.h>
44  #include <linux/string.h>
45  #include <linux/workqueue.h>
46  #include <net/dsfield.h>
47  #include <net/rtnetlink.h>
48  #include <uapi/linux/batman_adv.h>
49  
50  #include "bat_algo.h"
51  #include "bat_iv_ogm.h"
52  #include "bat_v.h"
53  #include "bridge_loop_avoidance.h"
54  #include "debugfs.h"
55  #include "distributed-arp-table.h"
56  #include "gateway_client.h"
57  #include "gateway_common.h"
58  #include "hard-interface.h"
59  #include "icmp_socket.h"
60  #include "log.h"
61  #include "multicast.h"
62  #include "netlink.h"
63  #include "network-coding.h"
64  #include "originator.h"
65  #include "packet.h"
66  #include "routing.h"
67  #include "send.h"
68  #include "soft-interface.h"
69  #include "tp_meter.h"
70  #include "translation-table.h"
71  
72  /* List manipulations on hardif_list have to be rtnl_lock()'ed,
73   * list traversals just rcu-locked
74   */
75  struct list_head batadv_hardif_list;
76  static int (*batadv_rx_handler[256])(struct sk_buff *,
77  				     struct batadv_hard_iface *);
78  
79  unsigned char batadv_broadcast_addr[] = {0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
80  
81  struct workqueue_struct *batadv_event_workqueue;
82  
83  static void batadv_recv_handler_init(void);
84  
batadv_init(void)85  static int __init batadv_init(void)
86  {
87  	int ret;
88  
89  	ret = batadv_tt_cache_init();
90  	if (ret < 0)
91  		return ret;
92  
93  	INIT_LIST_HEAD(&batadv_hardif_list);
94  	batadv_algo_init();
95  
96  	batadv_recv_handler_init();
97  
98  	batadv_v_init();
99  	batadv_iv_init();
100  	batadv_nc_init();
101  	batadv_tp_meter_init();
102  
103  	batadv_event_workqueue = create_singlethread_workqueue("bat_events");
104  	if (!batadv_event_workqueue)
105  		goto err_create_wq;
106  
107  	batadv_socket_init();
108  	batadv_debugfs_init();
109  
110  	register_netdevice_notifier(&batadv_hard_if_notifier);
111  	rtnl_link_register(&batadv_link_ops);
112  	batadv_netlink_register();
113  
114  	pr_info("B.A.T.M.A.N. advanced %s (compatibility version %i) loaded\n",
115  		BATADV_SOURCE_VERSION, BATADV_COMPAT_VERSION);
116  
117  	return 0;
118  
119  err_create_wq:
120  	batadv_tt_cache_destroy();
121  
122  	return -ENOMEM;
123  }
124  
batadv_exit(void)125  static void __exit batadv_exit(void)
126  {
127  	batadv_debugfs_destroy();
128  	batadv_netlink_unregister();
129  	rtnl_link_unregister(&batadv_link_ops);
130  	unregister_netdevice_notifier(&batadv_hard_if_notifier);
131  	batadv_hardif_remove_interfaces();
132  
133  	flush_workqueue(batadv_event_workqueue);
134  	destroy_workqueue(batadv_event_workqueue);
135  	batadv_event_workqueue = NULL;
136  
137  	rcu_barrier();
138  
139  	batadv_tt_cache_destroy();
140  }
141  
batadv_mesh_init(struct net_device * soft_iface)142  int batadv_mesh_init(struct net_device *soft_iface)
143  {
144  	struct batadv_priv *bat_priv = netdev_priv(soft_iface);
145  	int ret;
146  
147  	spin_lock_init(&bat_priv->forw_bat_list_lock);
148  	spin_lock_init(&bat_priv->forw_bcast_list_lock);
149  	spin_lock_init(&bat_priv->tt.changes_list_lock);
150  	spin_lock_init(&bat_priv->tt.req_list_lock);
151  	spin_lock_init(&bat_priv->tt.roam_list_lock);
152  	spin_lock_init(&bat_priv->tt.last_changeset_lock);
153  	spin_lock_init(&bat_priv->tt.commit_lock);
154  	spin_lock_init(&bat_priv->gw.list_lock);
155  #ifdef CONFIG_BATMAN_ADV_MCAST
156  	spin_lock_init(&bat_priv->mcast.mla_lock);
157  	spin_lock_init(&bat_priv->mcast.want_lists_lock);
158  #endif
159  	spin_lock_init(&bat_priv->tvlv.container_list_lock);
160  	spin_lock_init(&bat_priv->tvlv.handler_list_lock);
161  	spin_lock_init(&bat_priv->softif_vlan_list_lock);
162  	spin_lock_init(&bat_priv->tp_list_lock);
163  
164  	INIT_HLIST_HEAD(&bat_priv->forw_bat_list);
165  	INIT_HLIST_HEAD(&bat_priv->forw_bcast_list);
166  	INIT_HLIST_HEAD(&bat_priv->gw.gateway_list);
167  #ifdef CONFIG_BATMAN_ADV_MCAST
168  	INIT_HLIST_HEAD(&bat_priv->mcast.want_all_unsnoopables_list);
169  	INIT_HLIST_HEAD(&bat_priv->mcast.want_all_ipv4_list);
170  	INIT_HLIST_HEAD(&bat_priv->mcast.want_all_ipv6_list);
171  #endif
172  	INIT_LIST_HEAD(&bat_priv->tt.changes_list);
173  	INIT_HLIST_HEAD(&bat_priv->tt.req_list);
174  	INIT_LIST_HEAD(&bat_priv->tt.roam_list);
175  #ifdef CONFIG_BATMAN_ADV_MCAST
176  	INIT_HLIST_HEAD(&bat_priv->mcast.mla_list);
177  #endif
178  	INIT_HLIST_HEAD(&bat_priv->tvlv.container_list);
179  	INIT_HLIST_HEAD(&bat_priv->tvlv.handler_list);
180  	INIT_HLIST_HEAD(&bat_priv->softif_vlan_list);
181  	INIT_HLIST_HEAD(&bat_priv->tp_list);
182  
183  	ret = batadv_v_mesh_init(bat_priv);
184  	if (ret < 0)
185  		goto err;
186  
187  	ret = batadv_originator_init(bat_priv);
188  	if (ret < 0)
189  		goto err;
190  
191  	ret = batadv_tt_init(bat_priv);
192  	if (ret < 0)
193  		goto err;
194  
195  	ret = batadv_bla_init(bat_priv);
196  	if (ret < 0)
197  		goto err;
198  
199  	ret = batadv_dat_init(bat_priv);
200  	if (ret < 0)
201  		goto err;
202  
203  	ret = batadv_nc_mesh_init(bat_priv);
204  	if (ret < 0)
205  		goto err;
206  
207  	batadv_gw_init(bat_priv);
208  	batadv_mcast_init(bat_priv);
209  
210  	atomic_set(&bat_priv->gw.reselect, 0);
211  	atomic_set(&bat_priv->mesh_state, BATADV_MESH_ACTIVE);
212  
213  	return 0;
214  
215  err:
216  	batadv_mesh_free(soft_iface);
217  	return ret;
218  }
219  
batadv_mesh_free(struct net_device * soft_iface)220  void batadv_mesh_free(struct net_device *soft_iface)
221  {
222  	struct batadv_priv *bat_priv = netdev_priv(soft_iface);
223  
224  	atomic_set(&bat_priv->mesh_state, BATADV_MESH_DEACTIVATING);
225  
226  	batadv_purge_outstanding_packets(bat_priv, NULL);
227  
228  	batadv_gw_node_free(bat_priv);
229  
230  	batadv_v_mesh_free(bat_priv);
231  	batadv_nc_mesh_free(bat_priv);
232  	batadv_dat_free(bat_priv);
233  	batadv_bla_free(bat_priv);
234  
235  	batadv_mcast_free(bat_priv);
236  
237  	/* Free the TT and the originator tables only after having terminated
238  	 * all the other depending components which may use these structures for
239  	 * their purposes.
240  	 */
241  	batadv_tt_free(bat_priv);
242  
243  	/* Since the originator table clean up routine is accessing the TT
244  	 * tables as well, it has to be invoked after the TT tables have been
245  	 * freed and marked as empty. This ensures that no cleanup RCU callbacks
246  	 * accessing the TT data are scheduled for later execution.
247  	 */
248  	batadv_originator_free(bat_priv);
249  
250  	batadv_gw_free(bat_priv);
251  
252  	free_percpu(bat_priv->bat_counters);
253  	bat_priv->bat_counters = NULL;
254  
255  	atomic_set(&bat_priv->mesh_state, BATADV_MESH_INACTIVE);
256  }
257  
258  /**
259   * batadv_is_my_mac - check if the given mac address belongs to any of the real
260   * interfaces in the current mesh
261   * @bat_priv: the bat priv with all the soft interface information
262   * @addr: the address to check
263   *
264   * Return: 'true' if the mac address was found, false otherwise.
265   */
batadv_is_my_mac(struct batadv_priv * bat_priv,const u8 * addr)266  bool batadv_is_my_mac(struct batadv_priv *bat_priv, const u8 *addr)
267  {
268  	const struct batadv_hard_iface *hard_iface;
269  	bool is_my_mac = false;
270  
271  	rcu_read_lock();
272  	list_for_each_entry_rcu(hard_iface, &batadv_hardif_list, list) {
273  		if (hard_iface->if_status != BATADV_IF_ACTIVE)
274  			continue;
275  
276  		if (hard_iface->soft_iface != bat_priv->soft_iface)
277  			continue;
278  
279  		if (batadv_compare_eth(hard_iface->net_dev->dev_addr, addr)) {
280  			is_my_mac = true;
281  			break;
282  		}
283  	}
284  	rcu_read_unlock();
285  	return is_my_mac;
286  }
287  
288  #ifdef CONFIG_BATMAN_ADV_DEBUGFS
289  /**
290   * batadv_seq_print_text_primary_if_get - called from debugfs table printing
291   *  function that requires the primary interface
292   * @seq: debugfs table seq_file struct
293   *
294   * Return: primary interface if found or NULL otherwise.
295   */
296  struct batadv_hard_iface *
batadv_seq_print_text_primary_if_get(struct seq_file * seq)297  batadv_seq_print_text_primary_if_get(struct seq_file *seq)
298  {
299  	struct net_device *net_dev = (struct net_device *)seq->private;
300  	struct batadv_priv *bat_priv = netdev_priv(net_dev);
301  	struct batadv_hard_iface *primary_if;
302  
303  	primary_if = batadv_primary_if_get_selected(bat_priv);
304  
305  	if (!primary_if) {
306  		seq_printf(seq,
307  			   "BATMAN mesh %s disabled - please specify interfaces to enable it\n",
308  			   net_dev->name);
309  		goto out;
310  	}
311  
312  	if (primary_if->if_status == BATADV_IF_ACTIVE)
313  		goto out;
314  
315  	seq_printf(seq,
316  		   "BATMAN mesh %s disabled - primary interface not active\n",
317  		   net_dev->name);
318  	batadv_hardif_put(primary_if);
319  	primary_if = NULL;
320  
321  out:
322  	return primary_if;
323  }
324  #endif
325  
326  /**
327   * batadv_max_header_len - calculate maximum encapsulation overhead for a
328   *  payload packet
329   *
330   * Return: the maximum encapsulation overhead in bytes.
331   */
batadv_max_header_len(void)332  int batadv_max_header_len(void)
333  {
334  	int header_len = 0;
335  
336  	header_len = max_t(int, header_len,
337  			   sizeof(struct batadv_unicast_packet));
338  	header_len = max_t(int, header_len,
339  			   sizeof(struct batadv_unicast_4addr_packet));
340  	header_len = max_t(int, header_len,
341  			   sizeof(struct batadv_bcast_packet));
342  
343  #ifdef CONFIG_BATMAN_ADV_NC
344  	header_len = max_t(int, header_len,
345  			   sizeof(struct batadv_coded_packet));
346  #endif
347  
348  	return header_len + ETH_HLEN;
349  }
350  
351  /**
352   * batadv_skb_set_priority - sets skb priority according to packet content
353   * @skb: the packet to be sent
354   * @offset: offset to the packet content
355   *
356   * This function sets a value between 256 and 263 (802.1d priority), which
357   * can be interpreted by the cfg80211 or other drivers.
358   */
batadv_skb_set_priority(struct sk_buff * skb,int offset)359  void batadv_skb_set_priority(struct sk_buff *skb, int offset)
360  {
361  	struct iphdr ip_hdr_tmp, *ip_hdr;
362  	struct ipv6hdr ip6_hdr_tmp, *ip6_hdr;
363  	struct ethhdr ethhdr_tmp, *ethhdr;
364  	struct vlan_ethhdr *vhdr, vhdr_tmp;
365  	u32 prio;
366  
367  	/* already set, do nothing */
368  	if (skb->priority >= 256 && skb->priority <= 263)
369  		return;
370  
371  	ethhdr = skb_header_pointer(skb, offset, sizeof(*ethhdr), &ethhdr_tmp);
372  	if (!ethhdr)
373  		return;
374  
375  	switch (ethhdr->h_proto) {
376  	case htons(ETH_P_8021Q):
377  		vhdr = skb_header_pointer(skb, offset + sizeof(*vhdr),
378  					  sizeof(*vhdr), &vhdr_tmp);
379  		if (!vhdr)
380  			return;
381  		prio = ntohs(vhdr->h_vlan_TCI) & VLAN_PRIO_MASK;
382  		prio = prio >> VLAN_PRIO_SHIFT;
383  		break;
384  	case htons(ETH_P_IP):
385  		ip_hdr = skb_header_pointer(skb, offset + sizeof(*ethhdr),
386  					    sizeof(*ip_hdr), &ip_hdr_tmp);
387  		if (!ip_hdr)
388  			return;
389  		prio = (ipv4_get_dsfield(ip_hdr) & 0xfc) >> 5;
390  		break;
391  	case htons(ETH_P_IPV6):
392  		ip6_hdr = skb_header_pointer(skb, offset + sizeof(*ethhdr),
393  					     sizeof(*ip6_hdr), &ip6_hdr_tmp);
394  		if (!ip6_hdr)
395  			return;
396  		prio = (ipv6_get_dsfield(ip6_hdr) & 0xfc) >> 5;
397  		break;
398  	default:
399  		return;
400  	}
401  
402  	skb->priority = prio + 256;
403  }
404  
batadv_recv_unhandled_packet(struct sk_buff * skb,struct batadv_hard_iface * recv_if)405  static int batadv_recv_unhandled_packet(struct sk_buff *skb,
406  					struct batadv_hard_iface *recv_if)
407  {
408  	kfree_skb(skb);
409  
410  	return NET_RX_DROP;
411  }
412  
413  /* incoming packets with the batman ethertype received on any active hard
414   * interface
415   */
batadv_batman_skb_recv(struct sk_buff * skb,struct net_device * dev,struct packet_type * ptype,struct net_device * orig_dev)416  int batadv_batman_skb_recv(struct sk_buff *skb, struct net_device *dev,
417  			   struct packet_type *ptype,
418  			   struct net_device *orig_dev)
419  {
420  	struct batadv_priv *bat_priv;
421  	struct batadv_ogm_packet *batadv_ogm_packet;
422  	struct batadv_hard_iface *hard_iface;
423  	u8 idx;
424  
425  	hard_iface = container_of(ptype, struct batadv_hard_iface,
426  				  batman_adv_ptype);
427  
428  	/* Prevent processing a packet received on an interface which is getting
429  	 * shut down otherwise the packet may trigger de-reference errors
430  	 * further down in the receive path.
431  	 */
432  	if (!kref_get_unless_zero(&hard_iface->refcount))
433  		goto err_out;
434  
435  	skb = skb_share_check(skb, GFP_ATOMIC);
436  
437  	/* skb was released by skb_share_check() */
438  	if (!skb)
439  		goto err_put;
440  
441  	/* packet should hold at least type and version */
442  	if (unlikely(!pskb_may_pull(skb, 2)))
443  		goto err_free;
444  
445  	/* expect a valid ethernet header here. */
446  	if (unlikely(skb->mac_len != ETH_HLEN || !skb_mac_header(skb)))
447  		goto err_free;
448  
449  	if (!hard_iface->soft_iface)
450  		goto err_free;
451  
452  	bat_priv = netdev_priv(hard_iface->soft_iface);
453  
454  	if (atomic_read(&bat_priv->mesh_state) != BATADV_MESH_ACTIVE)
455  		goto err_free;
456  
457  	/* discard frames on not active interfaces */
458  	if (hard_iface->if_status != BATADV_IF_ACTIVE)
459  		goto err_free;
460  
461  	batadv_ogm_packet = (struct batadv_ogm_packet *)skb->data;
462  
463  	if (batadv_ogm_packet->version != BATADV_COMPAT_VERSION) {
464  		batadv_dbg(BATADV_DBG_BATMAN, bat_priv,
465  			   "Drop packet: incompatible batman version (%i)\n",
466  			   batadv_ogm_packet->version);
467  		goto err_free;
468  	}
469  
470  	/* reset control block to avoid left overs from previous users */
471  	memset(skb->cb, 0, sizeof(struct batadv_skb_cb));
472  
473  	idx = batadv_ogm_packet->packet_type;
474  	(*batadv_rx_handler[idx])(skb, hard_iface);
475  
476  	batadv_hardif_put(hard_iface);
477  
478  	/* return NET_RX_SUCCESS in any case as we
479  	 * most probably dropped the packet for
480  	 * routing-logical reasons.
481  	 */
482  	return NET_RX_SUCCESS;
483  
484  err_free:
485  	kfree_skb(skb);
486  err_put:
487  	batadv_hardif_put(hard_iface);
488  err_out:
489  	return NET_RX_DROP;
490  }
491  
batadv_recv_handler_init(void)492  static void batadv_recv_handler_init(void)
493  {
494  	int i;
495  
496  	for (i = 0; i < ARRAY_SIZE(batadv_rx_handler); i++)
497  		batadv_rx_handler[i] = batadv_recv_unhandled_packet;
498  
499  	for (i = BATADV_UNICAST_MIN; i <= BATADV_UNICAST_MAX; i++)
500  		batadv_rx_handler[i] = batadv_recv_unhandled_unicast_packet;
501  
502  	/* compile time checks for sizes */
503  	BUILD_BUG_ON(sizeof(struct batadv_bla_claim_dst) != 6);
504  	BUILD_BUG_ON(sizeof(struct batadv_ogm_packet) != 24);
505  	BUILD_BUG_ON(sizeof(struct batadv_icmp_header) != 20);
506  	BUILD_BUG_ON(sizeof(struct batadv_icmp_packet) != 20);
507  	BUILD_BUG_ON(sizeof(struct batadv_icmp_packet_rr) != 116);
508  	BUILD_BUG_ON(sizeof(struct batadv_unicast_packet) != 10);
509  	BUILD_BUG_ON(sizeof(struct batadv_unicast_4addr_packet) != 18);
510  	BUILD_BUG_ON(sizeof(struct batadv_frag_packet) != 20);
511  	BUILD_BUG_ON(sizeof(struct batadv_bcast_packet) != 14);
512  	BUILD_BUG_ON(sizeof(struct batadv_coded_packet) != 46);
513  	BUILD_BUG_ON(sizeof(struct batadv_unicast_tvlv_packet) != 20);
514  	BUILD_BUG_ON(sizeof(struct batadv_tvlv_hdr) != 4);
515  	BUILD_BUG_ON(sizeof(struct batadv_tvlv_gateway_data) != 8);
516  	BUILD_BUG_ON(sizeof(struct batadv_tvlv_tt_vlan_data) != 8);
517  	BUILD_BUG_ON(sizeof(struct batadv_tvlv_tt_change) != 12);
518  	BUILD_BUG_ON(sizeof(struct batadv_tvlv_roam_adv) != 8);
519  
520  	i = FIELD_SIZEOF(struct sk_buff, cb);
521  	BUILD_BUG_ON(sizeof(struct batadv_skb_cb) > i);
522  
523  	/* broadcast packet */
524  	batadv_rx_handler[BATADV_BCAST] = batadv_recv_bcast_packet;
525  
526  	/* unicast packets ... */
527  	/* unicast with 4 addresses packet */
528  	batadv_rx_handler[BATADV_UNICAST_4ADDR] = batadv_recv_unicast_packet;
529  	/* unicast packet */
530  	batadv_rx_handler[BATADV_UNICAST] = batadv_recv_unicast_packet;
531  	/* unicast tvlv packet */
532  	batadv_rx_handler[BATADV_UNICAST_TVLV] = batadv_recv_unicast_tvlv;
533  	/* batman icmp packet */
534  	batadv_rx_handler[BATADV_ICMP] = batadv_recv_icmp_packet;
535  	/* Fragmented packets */
536  	batadv_rx_handler[BATADV_UNICAST_FRAG] = batadv_recv_frag_packet;
537  }
538  
539  int
batadv_recv_handler_register(u8 packet_type,int (* recv_handler)(struct sk_buff *,struct batadv_hard_iface *))540  batadv_recv_handler_register(u8 packet_type,
541  			     int (*recv_handler)(struct sk_buff *,
542  						 struct batadv_hard_iface *))
543  {
544  	int (*curr)(struct sk_buff *,
545  		    struct batadv_hard_iface *);
546  	curr = batadv_rx_handler[packet_type];
547  
548  	if ((curr != batadv_recv_unhandled_packet) &&
549  	    (curr != batadv_recv_unhandled_unicast_packet))
550  		return -EBUSY;
551  
552  	batadv_rx_handler[packet_type] = recv_handler;
553  	return 0;
554  }
555  
batadv_recv_handler_unregister(u8 packet_type)556  void batadv_recv_handler_unregister(u8 packet_type)
557  {
558  	batadv_rx_handler[packet_type] = batadv_recv_unhandled_packet;
559  }
560  
561  /**
562   * batadv_skb_crc32 - calculate CRC32 of the whole packet and skip bytes in
563   *  the header
564   * @skb: skb pointing to fragmented socket buffers
565   * @payload_ptr: Pointer to position inside the head buffer of the skb
566   *  marking the start of the data to be CRC'ed
567   *
568   * payload_ptr must always point to an address in the skb head buffer and not to
569   * a fragment.
570   *
571   * Return: big endian crc32c of the checksummed data
572   */
batadv_skb_crc32(struct sk_buff * skb,u8 * payload_ptr)573  __be32 batadv_skb_crc32(struct sk_buff *skb, u8 *payload_ptr)
574  {
575  	u32 crc = 0;
576  	unsigned int from;
577  	unsigned int to = skb->len;
578  	struct skb_seq_state st;
579  	const u8 *data;
580  	unsigned int len;
581  	unsigned int consumed = 0;
582  
583  	from = (unsigned int)(payload_ptr - skb->data);
584  
585  	skb_prepare_seq_read(skb, from, to, &st);
586  	while ((len = skb_seq_read(consumed, &data, &st)) != 0) {
587  		crc = crc32c(crc, data, len);
588  		consumed += len;
589  	}
590  
591  	return htonl(crc);
592  }
593  
594  /**
595   * batadv_get_vid - extract the VLAN identifier from skb if any
596   * @skb: the buffer containing the packet
597   * @header_len: length of the batman header preceding the ethernet header
598   *
599   * Return: VID with the BATADV_VLAN_HAS_TAG flag when the packet embedded in the
600   * skb is vlan tagged. Otherwise BATADV_NO_FLAGS.
601   */
batadv_get_vid(struct sk_buff * skb,size_t header_len)602  unsigned short batadv_get_vid(struct sk_buff *skb, size_t header_len)
603  {
604  	struct ethhdr *ethhdr = (struct ethhdr *)(skb->data + header_len);
605  	struct vlan_ethhdr *vhdr;
606  	unsigned short vid;
607  
608  	if (ethhdr->h_proto != htons(ETH_P_8021Q))
609  		return BATADV_NO_FLAGS;
610  
611  	if (!pskb_may_pull(skb, header_len + VLAN_ETH_HLEN))
612  		return BATADV_NO_FLAGS;
613  
614  	vhdr = (struct vlan_ethhdr *)(skb->data + header_len);
615  	vid = ntohs(vhdr->h_vlan_TCI) & VLAN_VID_MASK;
616  	vid |= BATADV_VLAN_HAS_TAG;
617  
618  	return vid;
619  }
620  
621  /**
622   * batadv_vlan_ap_isola_get - return the AP isolation status for the given vlan
623   * @bat_priv: the bat priv with all the soft interface information
624   * @vid: the VLAN identifier for which the AP isolation attributed as to be
625   *  looked up
626   *
627   * Return: true if AP isolation is on for the VLAN idenfied by vid, false
628   * otherwise
629   */
batadv_vlan_ap_isola_get(struct batadv_priv * bat_priv,unsigned short vid)630  bool batadv_vlan_ap_isola_get(struct batadv_priv *bat_priv, unsigned short vid)
631  {
632  	bool ap_isolation_enabled = false;
633  	struct batadv_softif_vlan *vlan;
634  
635  	/* if the AP isolation is requested on a VLAN, then check for its
636  	 * setting in the proper VLAN private data structure
637  	 */
638  	vlan = batadv_softif_vlan_get(bat_priv, vid);
639  	if (vlan) {
640  		ap_isolation_enabled = atomic_read(&vlan->ap_isolation);
641  		batadv_softif_vlan_put(vlan);
642  	}
643  
644  	return ap_isolation_enabled;
645  }
646  
647  module_init(batadv_init);
648  module_exit(batadv_exit);
649  
650  MODULE_LICENSE("GPL");
651  
652  MODULE_AUTHOR(BATADV_DRIVER_AUTHOR);
653  MODULE_DESCRIPTION(BATADV_DRIVER_DESC);
654  MODULE_SUPPORTED_DEVICE(BATADV_DRIVER_DEVICE);
655  MODULE_VERSION(BATADV_SOURCE_VERSION);
656  MODULE_ALIAS_RTNL_LINK("batadv");
657  MODULE_ALIAS_GENL_FAMILY(BATADV_NL_NAME);
658