routing.c 31.8 KB
Newer Older
1
/* Copyright (C) 2007-2015 B.A.T.M.A.N. contributors:
2 3 4 5 6 7 8 9 10 11 12 13 14
 *
 * Marek Lindner, Simon Wunderlich
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of version 2 of the GNU General Public
 * License as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
 * General Public License for more details.
 *
 * 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 "routing.h"
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
#include "main.h"

#include <linux/atomic.h>
#include <linux/byteorder/generic.h>
#include <linux/compiler.h>
#include <linux/errno.h>
#include <linux/etherdevice.h>
#include <linux/if_ether.h>
#include <linux/jiffies.h>
#include <linux/netdevice.h>
#include <linux/printk.h>
#include <linux/rculist.h>
#include <linux/rcupdate.h>
#include <linux/skbuff.h>
#include <linux/spinlock.h>
#include <linux/stddef.h>

#include "bitarray.h"
37
#include "bridge_loop_avoidance.h"
38
#include "distributed-arp-table.h"
39
#include "fragmentation.h"
40 41 42 43 44 45 46 47
#include "hard-interface.h"
#include "icmp_socket.h"
#include "network-coding.h"
#include "originator.h"
#include "packet.h"
#include "send.h"
#include "soft-interface.h"
#include "translation-table.h"
48

49
static int batadv_route_unicast_packet(struct sk_buff *skb,
50
				       struct batadv_hard_iface *recv_if);
51

52 53 54 55 56 57 58 59 60
/**
 * _batadv_update_route - set the router for this originator
 * @bat_priv: the bat priv with all the soft interface information
 * @orig_node: orig node which is to be configured
 * @recv_if: the receive interface for which this route is set
 * @neigh_node: neighbor which should be the next router
 *
 * This function does not perform any error checks
 */
61 62
static void _batadv_update_route(struct batadv_priv *bat_priv,
				 struct batadv_orig_node *orig_node,
63
				 struct batadv_hard_iface *recv_if,
64
				 struct batadv_neigh_node *neigh_node)
65
{
66
	struct batadv_orig_ifinfo *orig_ifinfo;
67
	struct batadv_neigh_node *curr_router;
68

69 70 71 72 73 74 75 76 77
	orig_ifinfo = batadv_orig_ifinfo_get(orig_node, recv_if);
	if (!orig_ifinfo)
		return;

	rcu_read_lock();
	curr_router = rcu_dereference(orig_ifinfo->router);
	if (curr_router && !atomic_inc_not_zero(&curr_router->refcount))
		curr_router = NULL;
	rcu_read_unlock();
78

79
	/* route deleted */
80
	if ((curr_router) && (!neigh_node)) {
81 82
		batadv_dbg(BATADV_DBG_ROUTES, bat_priv,
			   "Deleting route towards: %pM\n", orig_node->orig);
83
		batadv_tt_global_del_orig(bat_priv, orig_node, -1,
84
					  "Deleted route towards originator");
85

86 87
	/* route added */
	} else if ((!curr_router) && (neigh_node)) {
88
		batadv_dbg(BATADV_DBG_ROUTES, bat_priv,
89 90
			   "Adding route towards: %pM (via %pM)\n",
			   orig_node->orig, neigh_node->addr);
91
	/* route changed */
92
	} else if (neigh_node && curr_router) {
93
		batadv_dbg(BATADV_DBG_ROUTES, bat_priv,
94 95 96
			   "Changing route towards: %pM (now via %pM - was via %pM)\n",
			   orig_node->orig, neigh_node->addr,
			   curr_router->addr);
97 98
	}

99
	if (curr_router)
100
		batadv_neigh_node_free_ref(curr_router);
101 102

	/* increase refcount of new best neighbor */
103 104
	if (neigh_node && !atomic_inc_not_zero(&neigh_node->refcount))
		neigh_node = NULL;
105 106

	spin_lock_bh(&orig_node->neigh_list_lock);
107
	rcu_assign_pointer(orig_ifinfo->router, neigh_node);
108
	spin_unlock_bh(&orig_node->neigh_list_lock);
109
	batadv_orig_ifinfo_free_ref(orig_ifinfo);
110 111 112

	/* decrease refcount of previous best neighbor */
	if (curr_router)
113
		batadv_neigh_node_free_ref(curr_router);
114 115
}

116 117 118 119 120 121 122
/**
 * batadv_update_route - set the router for this originator
 * @bat_priv: the bat priv with all the soft interface information
 * @orig_node: orig node which is to be configured
 * @recv_if: the receive interface for which this route is set
 * @neigh_node: neighbor which should be the next router
 */
123 124
void batadv_update_route(struct batadv_priv *bat_priv,
			 struct batadv_orig_node *orig_node,
125
			 struct batadv_hard_iface *recv_if,
126
			 struct batadv_neigh_node *neigh_node)
127
{
128
	struct batadv_neigh_node *router = NULL;
129 130

	if (!orig_node)
131 132
		goto out;

133
	router = batadv_orig_router_get(orig_node, recv_if);
134

135
	if (router != neigh_node)
136
		_batadv_update_route(bat_priv, orig_node, recv_if, neigh_node);
137 138 139

out:
	if (router)
140
		batadv_neigh_node_free_ref(router);
141 142 143 144 145 146 147
}

/* checks whether the host restarted and is in the protection time.
 * returns:
 *  0 if the packet is to be accepted
 *  1 if the packet is to be ignored.
 */
148
int batadv_window_protected(struct batadv_priv *bat_priv, int32_t seq_num_diff,
149
			    unsigned long *last_reset)
150
{
151 152 153 154
	if (seq_num_diff <= -BATADV_TQ_LOCAL_WINDOW_SIZE ||
	    seq_num_diff >= BATADV_EXPECTED_SEQNO_RANGE) {
		if (!batadv_has_timed_out(*last_reset,
					  BATADV_RESET_PROTECTION_MS))
155
			return 1;
156 157

		*last_reset = jiffies;
158
		batadv_dbg(BATADV_DBG_BATMAN, bat_priv,
159
			   "old packet received, start protection\n");
160
	}
161

162 163 164
	return 0;
}

165
bool batadv_check_management_packet(struct sk_buff *skb,
166
				    struct batadv_hard_iface *hard_iface,
167
				    int header_len)
168 169 170 171
{
	struct ethhdr *ethhdr;

	/* drop packet if it has not necessary minimum size */
172 173
	if (unlikely(!pskb_may_pull(skb, header_len)))
		return false;
174

175
	ethhdr = eth_hdr(skb);
176 177 178

	/* packet with broadcast indication but unicast recipient */
	if (!is_broadcast_ether_addr(ethhdr->h_dest))
179
		return false;
180 181 182

	/* packet with broadcast sender address */
	if (is_broadcast_ether_addr(ethhdr->h_source))
183
		return false;
184 185 186

	/* create a copy of the skb, if needed, to modify it. */
	if (skb_cow(skb, 0) < 0)
187
		return false;
188 189 190

	/* keep skb linear */
	if (skb_linearize(skb) < 0)
191
		return false;
192

193
	return true;
194 195
}

196 197 198 199 200 201 202 203
/**
 * batadv_recv_my_icmp_packet - receive an icmp packet locally
 * @bat_priv: the bat priv with all the soft interface information
 * @skb: icmp packet to process
 *
 * Returns NET_RX_SUCCESS if the packet has been consumed or NET_RX_DROP
 * otherwise.
 */
204
static int batadv_recv_my_icmp_packet(struct batadv_priv *bat_priv,
205
				      struct sk_buff *skb)
206
{
207 208
	struct batadv_hard_iface *primary_if = NULL;
	struct batadv_orig_node *orig_node = NULL;
209 210
	struct batadv_icmp_header *icmph;
	int res, ret = NET_RX_DROP;
211

212
	icmph = (struct batadv_icmp_header *)skb->data;
213

214 215 216 217 218 219 220
	switch (icmph->msg_type) {
	case BATADV_ECHO_REPLY:
	case BATADV_DESTINATION_UNREACHABLE:
	case BATADV_TTL_EXCEEDED:
		/* receive the packet */
		if (skb_linearize(skb) < 0)
			break;
221

222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240
		batadv_socket_receive_packet(icmph, skb->len);
		break;
	case BATADV_ECHO_REQUEST:
		/* answer echo request (ping) */
		primary_if = batadv_primary_if_get_selected(bat_priv);
		if (!primary_if)
			goto out;

		/* get routing information */
		orig_node = batadv_orig_hash_find(bat_priv, icmph->orig);
		if (!orig_node)
			goto out;

		/* create a copy of the skb, if needed, to modify it. */
		if (skb_cow(skb, ETH_HLEN) < 0)
			goto out;

		icmph = (struct batadv_icmp_header *)skb->data;

241 242
		ether_addr_copy(icmph->dst, icmph->orig);
		ether_addr_copy(icmph->orig, primary_if->net_dev->dev_addr);
243
		icmph->msg_type = BATADV_ECHO_REPLY;
244
		icmph->ttl = BATADV_TTL;
245 246 247 248 249 250 251 252

		res = batadv_send_skb_to_orig(skb, orig_node, NULL);
		if (res != NET_XMIT_DROP)
			ret = NET_RX_SUCCESS;

		break;
	default:
		/* drop unknown type */
253
		goto out;
254
	}
255
out:
256
	if (primary_if)
257
		batadv_hardif_free_ref(primary_if);
258
	if (orig_node)
259
		batadv_orig_node_free_ref(orig_node);
260 261 262
	return ret;
}

263
static int batadv_recv_icmp_ttl_exceeded(struct batadv_priv *bat_priv,
264
					 struct sk_buff *skb)
265
{
266 267
	struct batadv_hard_iface *primary_if = NULL;
	struct batadv_orig_node *orig_node = NULL;
268
	struct batadv_icmp_packet *icmp_packet;
269
	int ret = NET_RX_DROP;
270

271
	icmp_packet = (struct batadv_icmp_packet *)skb->data;
272 273

	/* send TTL exceeded if packet is an echo request (traceroute) */
274
	if (icmp_packet->msg_type != BATADV_ECHO_REQUEST) {
275
		pr_debug("Warning - can't forward icmp packet from %pM to %pM: ttl exceeded\n",
276
			 icmp_packet->orig, icmp_packet->dst);
277
		goto out;
278 279
	}

280
	primary_if = batadv_primary_if_get_selected(bat_priv);
281
	if (!primary_if)
282
		goto out;
283 284

	/* get routing information */
285
	orig_node = batadv_orig_hash_find(bat_priv, icmp_packet->orig);
286
	if (!orig_node)
287
		goto out;
288

289
	/* create a copy of the skb, if needed, to modify it. */
290
	if (skb_cow(skb, ETH_HLEN) < 0)
291
		goto out;
292

293
	icmp_packet = (struct batadv_icmp_packet *)skb->data;
294

295 296
	ether_addr_copy(icmp_packet->dst, icmp_packet->orig);
	ether_addr_copy(icmp_packet->orig, primary_if->net_dev->dev_addr);
297 298
	icmp_packet->msg_type = BATADV_TTL_EXCEEDED;
	icmp_packet->ttl = BATADV_TTL;
299

300
	if (batadv_send_skb_to_orig(skb, orig_node, NULL) != NET_XMIT_DROP)
301
		ret = NET_RX_SUCCESS;
302

303
out:
304
	if (primary_if)
305
		batadv_hardif_free_ref(primary_if);
306
	if (orig_node)
307
		batadv_orig_node_free_ref(orig_node);
308 309 310
	return ret;
}

311 312
int batadv_recv_icmp_packet(struct sk_buff *skb,
			    struct batadv_hard_iface *recv_if)
313
{
314
	struct batadv_priv *bat_priv = netdev_priv(recv_if->soft_iface);
315 316
	struct batadv_icmp_header *icmph;
	struct batadv_icmp_packet_rr *icmp_packet_rr;
317
	struct ethhdr *ethhdr;
318
	struct batadv_orig_node *orig_node = NULL;
319
	int hdr_size = sizeof(struct batadv_icmp_header);
320
	int ret = NET_RX_DROP;
321 322 323

	/* drop packet if it has not necessary minimum size */
	if (unlikely(!pskb_may_pull(skb, hdr_size)))
324
		goto out;
325

326
	ethhdr = eth_hdr(skb);
327 328 329

	/* packet with unicast indication but broadcast recipient */
	if (is_broadcast_ether_addr(ethhdr->h_dest))
330
		goto out;
331 332 333

	/* packet with broadcast sender address */
	if (is_broadcast_ether_addr(ethhdr->h_source))
334
		goto out;
335 336

	/* not for me */
337
	if (!batadv_is_my_mac(bat_priv, ethhdr->h_dest))
338
		goto out;
339

340
	icmph = (struct batadv_icmp_header *)skb->data;
341 342

	/* add record route information if not full */
343 344 345 346 347 348 349 350 351 352 353 354 355 356 357
	if ((icmph->msg_type == BATADV_ECHO_REPLY ||
	     icmph->msg_type == BATADV_ECHO_REQUEST) &&
	    (skb->len >= sizeof(struct batadv_icmp_packet_rr))) {
		if (skb_linearize(skb) < 0)
			goto out;

		/* create a copy of the skb, if needed, to modify it. */
		if (skb_cow(skb, ETH_HLEN) < 0)
			goto out;

		icmph = (struct batadv_icmp_header *)skb->data;
		icmp_packet_rr = (struct batadv_icmp_packet_rr *)icmph;
		if (icmp_packet_rr->rr_cur >= BATADV_RR_LEN)
			goto out;

358 359
		ether_addr_copy(icmp_packet_rr->rr[icmp_packet_rr->rr_cur],
				ethhdr->h_dest);
360
		icmp_packet_rr->rr_cur++;
361 362 363
	}

	/* packet for me */
364 365
	if (batadv_is_my_mac(bat_priv, icmph->dst))
		return batadv_recv_my_icmp_packet(bat_priv, skb);
366 367

	/* TTL exceeded */
368
	if (icmph->ttl < 2)
369
		return batadv_recv_icmp_ttl_exceeded(bat_priv, skb);
370 371

	/* get routing information */
372
	orig_node = batadv_orig_hash_find(bat_priv, icmph->dst);
373
	if (!orig_node)
374
		goto out;
375

376
	/* create a copy of the skb, if needed, to modify it. */
377
	if (skb_cow(skb, ETH_HLEN) < 0)
378
		goto out;
379

380
	icmph = (struct batadv_icmp_header *)skb->data;
381

382
	/* decrement ttl */
383
	icmph->ttl--;
384 385

	/* route it */
386
	if (batadv_send_skb_to_orig(skb, orig_node, recv_if) != NET_XMIT_DROP)
387
		ret = NET_RX_SUCCESS;
388

389 390
out:
	if (orig_node)
391
		batadv_orig_node_free_ref(orig_node);
392 393 394
	return ret;
}

395 396
/**
 * batadv_check_unicast_packet - Check for malformed unicast packets
397
 * @bat_priv: the bat priv with all the soft interface information
398 399 400 401 402 403 404 405
 * @skb: packet to check
 * @hdr_size: size of header to pull
 *
 * Check for short header and bad addresses in given packet. Returns negative
 * value when check fails and 0 otherwise. The negative value depends on the
 * reason: -ENODATA for bad header, -EBADR for broadcast destination or source,
 * and -EREMOTE for non-local (other host) destination.
 */
406 407
static int batadv_check_unicast_packet(struct batadv_priv *bat_priv,
				       struct sk_buff *skb, int hdr_size)
408 409 410 411 412
{
	struct ethhdr *ethhdr;

	/* drop packet if it has not necessary minimum size */
	if (unlikely(!pskb_may_pull(skb, hdr_size)))
413
		return -ENODATA;
414

415
	ethhdr = eth_hdr(skb);
416 417 418

	/* packet with unicast indication but broadcast recipient */
	if (is_broadcast_ether_addr(ethhdr->h_dest))
419
		return -EBADR;
420 421 422

	/* packet with broadcast sender address */
	if (is_broadcast_ether_addr(ethhdr->h_source))
423
		return -EBADR;
424 425

	/* not for me */
426
	if (!batadv_is_my_mac(bat_priv, ethhdr->h_dest))
427
		return -EREMOTE;
428 429 430 431

	return 0;
}

432 433 434 435 436 437 438 439
/**
 * batadv_find_router - find a suitable router for this originator
 * @bat_priv: the bat priv with all the soft interface information
 * @orig_node: the destination node
 * @recv_if: pointer to interface this packet was received on
 *
 * Returns the router which should be used for this orig_node on
 * this interface, or NULL if not available.
440
 */
441 442 443
struct batadv_neigh_node *
batadv_find_router(struct batadv_priv *bat_priv,
		   struct batadv_orig_node *orig_node,
S
Simon Wunderlich 已提交
444
		   struct batadv_hard_iface *recv_if)
445
{
S
Simon Wunderlich 已提交
446 447 448 449 450 451 452 453 454
	struct batadv_algo_ops *bao = bat_priv->bat_algo_ops;
	struct batadv_neigh_node *first_candidate_router = NULL;
	struct batadv_neigh_node *next_candidate_router = NULL;
	struct batadv_neigh_node *router, *cand_router = NULL;
	struct batadv_neigh_node *last_cand_router = NULL;
	struct batadv_orig_ifinfo *cand, *first_candidate = NULL;
	struct batadv_orig_ifinfo *next_candidate = NULL;
	struct batadv_orig_ifinfo *last_candidate;
	bool last_candidate_found = false;
455 456 457 458

	if (!orig_node)
		return NULL;

459
	router = batadv_orig_router_get(orig_node, recv_if);
460

461 462 463
	if (!router)
		return router;

S
Simon Wunderlich 已提交
464 465 466
	/* only consider bonding for recv_if == BATADV_IF_DEFAULT (first hop)
	 * and if activated.
	 */
467
	if (!(recv_if == BATADV_IF_DEFAULT && atomic_read(&bat_priv->bonding)))
S
Simon Wunderlich 已提交
468 469 470 471 472 473 474
		return router;

	/* bonding: loop through the list of possible routers found
	 * for the various outgoing interfaces and find a candidate after
	 * the last chosen bonding candidate (next_candidate). If no such
	 * router is found, use the first candidate found (the previously
	 * chosen bonding candidate might have been the last one in the list).
475
	 * If this can't be found either, return the previously chosen
S
Simon Wunderlich 已提交
476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566
	 * router - obviously there are no other candidates.
	 */
	rcu_read_lock();
	last_candidate = orig_node->last_bonding_candidate;
	if (last_candidate)
		last_cand_router = rcu_dereference(last_candidate->router);

	hlist_for_each_entry_rcu(cand, &orig_node->ifinfo_list, list) {
		/* acquire some structures and references ... */
		if (!atomic_inc_not_zero(&cand->refcount))
			continue;

		cand_router = rcu_dereference(cand->router);
		if (!cand_router)
			goto next;

		if (!atomic_inc_not_zero(&cand_router->refcount)) {
			cand_router = NULL;
			goto next;
		}

		/* alternative candidate should be good enough to be
		 * considered
		 */
		if (!bao->bat_neigh_is_equiv_or_better(cand_router,
						       cand->if_outgoing,
						       router, recv_if))
			goto next;

		/* don't use the same router twice */
		if (last_cand_router == cand_router)
			goto next;

		/* mark the first possible candidate */
		if (!first_candidate) {
			atomic_inc(&cand_router->refcount);
			atomic_inc(&cand->refcount);
			first_candidate = cand;
			first_candidate_router = cand_router;
		}

		/* check if the loop has already passed the previously selected
		 * candidate ... this function should select the next candidate
		 * AFTER the previously used bonding candidate.
		 */
		if (!last_candidate || last_candidate_found) {
			next_candidate = cand;
			next_candidate_router = cand_router;
			break;
		}

		if (last_candidate == cand)
			last_candidate_found = true;
next:
		/* free references */
		if (cand_router) {
			batadv_neigh_node_free_ref(cand_router);
			cand_router = NULL;
		}
		batadv_orig_ifinfo_free_ref(cand);
	}
	rcu_read_unlock();

	/* last_bonding_candidate is reset below, remove the old reference. */
	if (orig_node->last_bonding_candidate)
		batadv_orig_ifinfo_free_ref(orig_node->last_bonding_candidate);

	/* After finding candidates, handle the three cases:
	 * 1) there is a next candidate, use that
	 * 2) there is no next candidate, use the first of the list
	 * 3) there is no candidate at all, return the default router
	 */
	if (next_candidate) {
		batadv_neigh_node_free_ref(router);

		/* remove references to first candidate, we don't need it. */
		if (first_candidate) {
			batadv_neigh_node_free_ref(first_candidate_router);
			batadv_orig_ifinfo_free_ref(first_candidate);
		}
		router = next_candidate_router;
		orig_node->last_bonding_candidate = next_candidate;
	} else if (first_candidate) {
		batadv_neigh_node_free_ref(router);

		/* refcounting has already been done in the loop above. */
		router = first_candidate_router;
		orig_node->last_bonding_candidate = first_candidate;
	} else {
		orig_node->last_bonding_candidate = NULL;
	}
567

568 569 570
	return router;
}

571
static int batadv_route_unicast_packet(struct sk_buff *skb,
572
				       struct batadv_hard_iface *recv_if)
573
{
574 575
	struct batadv_priv *bat_priv = netdev_priv(recv_if->soft_iface);
	struct batadv_orig_node *orig_node = NULL;
576
	struct batadv_unicast_packet *unicast_packet;
577
	struct ethhdr *ethhdr = eth_hdr(skb);
578
	int res, hdr_len, ret = NET_RX_DROP;
579

580
	unicast_packet = (struct batadv_unicast_packet *)skb->data;
581 582

	/* TTL exceeded */
583
	if (unicast_packet->ttl < 2) {
584 585
		pr_debug("Warning - can't forward unicast packet from %pM to %pM: ttl exceeded\n",
			 ethhdr->h_source, unicast_packet->dest);
586
		goto out;
587 588 589
	}

	/* get routing information */
590
	orig_node = batadv_orig_hash_find(bat_priv, unicast_packet->dest);
591

592
	if (!orig_node)
593
		goto out;
594 595

	/* create a copy of the skb, if needed, to modify it. */
596
	if (skb_cow(skb, ETH_HLEN) < 0)
597
		goto out;
598 599

	/* decrement ttl */
600
	unicast_packet = (struct batadv_unicast_packet *)skb->data;
601
	unicast_packet->ttl--;
602

603
	switch (unicast_packet->packet_type) {
604 605 606 607 608 609 610 611 612 613 614 615 616 617 618
	case BATADV_UNICAST_4ADDR:
		hdr_len = sizeof(struct batadv_unicast_4addr_packet);
		break;
	case BATADV_UNICAST:
		hdr_len = sizeof(struct batadv_unicast_packet);
		break;
	default:
		/* other packet types not supported - yet */
		hdr_len = -1;
		break;
	}

	if (hdr_len > 0)
		batadv_skb_set_priority(skb, hdr_len);

619
	res = batadv_send_skb_to_orig(skb, orig_node, recv_if);
620

621 622 623
	/* translate transmit result into receive result */
	if (res == NET_XMIT_SUCCESS) {
		/* skb was transmitted and consumed */
624 625 626
		batadv_inc_counter(bat_priv, BATADV_CNT_FORWARD);
		batadv_add_counter(bat_priv, BATADV_CNT_FORWARD_BYTES,
				   skb->len + ETH_HLEN);
627 628 629 630 631

		ret = NET_RX_SUCCESS;
	} else if (res == NET_XMIT_POLICED) {
		/* skb was buffered and consumed */
		ret = NET_RX_SUCCESS;
632
	}
633

634 635
out:
	if (orig_node)
636
		batadv_orig_node_free_ref(orig_node);
637
	return ret;
638 639
}

640 641 642 643 644
/**
 * batadv_reroute_unicast_packet - update the unicast header for re-routing
 * @bat_priv: the bat priv with all the soft interface information
 * @unicast_packet: the unicast header to be updated
 * @dst_addr: the payload destination
645
 * @vid: VLAN identifier
646 647 648 649 650 651 652 653 654 655
 *
 * Search the translation table for dst_addr and update the unicast header with
 * the new corresponding information (originator address where the destination
 * client currently is and its known TTVN)
 *
 * Returns true if the packet header has been updated, false otherwise
 */
static bool
batadv_reroute_unicast_packet(struct batadv_priv *bat_priv,
			      struct batadv_unicast_packet *unicast_packet,
656
			      uint8_t *dst_addr, unsigned short vid)
657 658 659 660 661 662
{
	struct batadv_orig_node *orig_node = NULL;
	struct batadv_hard_iface *primary_if = NULL;
	bool ret = false;
	uint8_t *orig_addr, orig_ttvn;

663
	if (batadv_is_my_client(bat_priv, dst_addr, vid)) {
664 665 666 667 668 669
		primary_if = batadv_primary_if_get_selected(bat_priv);
		if (!primary_if)
			goto out;
		orig_addr = primary_if->net_dev->dev_addr;
		orig_ttvn = (uint8_t)atomic_read(&bat_priv->tt.vn);
	} else {
670 671
		orig_node = batadv_transtable_search(bat_priv, NULL, dst_addr,
						     vid);
672 673 674 675 676 677 678 679 680 681 682
		if (!orig_node)
			goto out;

		if (batadv_compare_eth(orig_node->orig, unicast_packet->dest))
			goto out;

		orig_addr = orig_node->orig;
		orig_ttvn = (uint8_t)atomic_read(&orig_node->last_ttvn);
	}

	/* update the packet header */
683
	ether_addr_copy(unicast_packet->dest, orig_addr);
684 685 686 687 688 689 690 691 692 693 694 695
	unicast_packet->ttvn = orig_ttvn;

	ret = true;
out:
	if (primary_if)
		batadv_hardif_free_ref(primary_if);
	if (orig_node)
		batadv_orig_node_free_ref(orig_node);

	return ret;
}

696
static int batadv_check_unicast_ttvn(struct batadv_priv *bat_priv,
697
				     struct sk_buff *skb, int hdr_len) {
698 699
	struct batadv_unicast_packet *unicast_packet;
	struct batadv_hard_iface *primary_if;
700
	struct batadv_orig_node *orig_node;
701
	uint8_t curr_ttvn, old_ttvn;
702
	struct ethhdr *ethhdr;
703
	unsigned short vid;
704
	int is_old_ttvn;
705

706
	/* check if there is enough data before accessing it */
707
	if (!pskb_may_pull(skb, hdr_len + ETH_HLEN))
708 709 710 711
		return 0;

	/* create a copy of the skb (in case of for re-routing) to modify it. */
	if (skb_cow(skb, sizeof(*unicast_packet)) < 0)
712 713
		return 0;

714
	unicast_packet = (struct batadv_unicast_packet *)skb->data;
715
	vid = batadv_get_vid(skb, hdr_len);
716
	ethhdr = (struct ethhdr *)(skb->data + hdr_len);
717

718 719 720 721 722
	/* check if the destination client was served by this node and it is now
	 * roaming. In this case, it means that the node has got a ROAM_ADV
	 * message and that it knows the new destination in the mesh to re-route
	 * the packet to
	 */
723
	if (batadv_tt_local_client_is_roaming(bat_priv, ethhdr->h_dest, vid)) {
724
		if (batadv_reroute_unicast_packet(bat_priv, unicast_packet,
725
						  ethhdr->h_dest, vid))
726 727 728 729 730
			batadv_dbg_ratelimited(BATADV_DBG_TT,
					       bat_priv,
					       "Rerouting unicast packet to %pM (dst=%pM): Local Roaming\n",
					       unicast_packet->dest,
					       ethhdr->h_dest);
731 732 733 734 735 736 737 738 739 740 741 742 743
		/* at this point the mesh destination should have been
		 * substituted with the originator address found in the global
		 * table. If not, let the packet go untouched anyway because
		 * there is nothing the node can do
		 */
		return 1;
	}

	/* retrieve the TTVN known by this node for the packet destination. This
	 * value is used later to check if the node which sent (or re-routed
	 * last time) the packet had an updated information or not
	 */
	curr_ttvn = (uint8_t)atomic_read(&bat_priv->tt.vn);
744
	if (!batadv_is_my_mac(bat_priv, unicast_packet->dest)) {
745 746
		orig_node = batadv_orig_hash_find(bat_priv,
						  unicast_packet->dest);
747 748 749 750
		/* if it is not possible to find the orig_node representing the
		 * destination, the packet can immediately be dropped as it will
		 * not be possible to deliver it
		 */
751 752 753 754
		if (!orig_node)
			return 0;

		curr_ttvn = (uint8_t)atomic_read(&orig_node->last_ttvn);
755
		batadv_orig_node_free_ref(orig_node);
756 757
	}

758 759 760
	/* check if the TTVN contained in the packet is fresher than what the
	 * node knows
	 */
761
	is_old_ttvn = batadv_seq_before(unicast_packet->ttvn, curr_ttvn);
762 763
	if (!is_old_ttvn)
		return 1;
764

765 766 767 768 769 770
	old_ttvn = unicast_packet->ttvn;
	/* the packet was forged based on outdated network information. Its
	 * destination can possibly be updated and forwarded towards the new
	 * target host
	 */
	if (batadv_reroute_unicast_packet(bat_priv, unicast_packet,
771
					  ethhdr->h_dest, vid)) {
772 773 774 775
		batadv_dbg_ratelimited(BATADV_DBG_TT, bat_priv,
				       "Rerouting unicast packet to %pM (dst=%pM): TTVN mismatch old_ttvn=%u new_ttvn=%u\n",
				       unicast_packet->dest, ethhdr->h_dest,
				       old_ttvn, curr_ttvn);
776 777
		return 1;
	}
778

779 780 781 782
	/* the packet has not been re-routed: either the destination is
	 * currently served by this node or there is no destination at all and
	 * it is possible to drop the packet
	 */
783
	if (!batadv_is_my_client(bat_priv, ethhdr->h_dest, vid))
784
		return 0;
785

786 787 788 789 790 791
	/* update the header in order to let the packet be delivered to this
	 * node's soft interface
	 */
	primary_if = batadv_primary_if_get_selected(bat_priv);
	if (!primary_if)
		return 0;
792

793
	ether_addr_copy(unicast_packet->dest, primary_if->net_dev->dev_addr);
794 795 796 797

	batadv_hardif_free_ref(primary_if);

	unicast_packet->ttvn = curr_ttvn;
798 799 800 801

	return 1;
}

802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828 829
/**
 * batadv_recv_unhandled_unicast_packet - receive and process packets which
 *	are in the unicast number space but not yet known to the implementation
 * @skb: unicast tvlv packet to process
 * @recv_if: pointer to interface this packet was received on
 *
 * Returns NET_RX_SUCCESS if the packet has been consumed or NET_RX_DROP
 * otherwise.
 */
int batadv_recv_unhandled_unicast_packet(struct sk_buff *skb,
					 struct batadv_hard_iface *recv_if)
{
	struct batadv_unicast_packet *unicast_packet;
	struct batadv_priv *bat_priv = netdev_priv(recv_if->soft_iface);
	int check, hdr_size = sizeof(*unicast_packet);

	check = batadv_check_unicast_packet(bat_priv, skb, hdr_size);
	if (check < 0)
		return NET_RX_DROP;

	/* we don't know about this type, drop it. */
	unicast_packet = (struct batadv_unicast_packet *)skb->data;
	if (batadv_is_my_mac(bat_priv, unicast_packet->dest))
		return NET_RX_DROP;

	return batadv_route_unicast_packet(skb, recv_if);
}

830 831
int batadv_recv_unicast_packet(struct sk_buff *skb,
			       struct batadv_hard_iface *recv_if)
832
{
833
	struct batadv_priv *bat_priv = netdev_priv(recv_if->soft_iface);
834
	struct batadv_unicast_packet *unicast_packet;
835
	struct batadv_unicast_4addr_packet *unicast_4addr_packet;
836 837
	uint8_t *orig_addr;
	struct batadv_orig_node *orig_node = NULL;
838
	int check, hdr_size = sizeof(*unicast_packet);
839
	bool is4addr;
840

841
	unicast_packet = (struct batadv_unicast_packet *)skb->data;
842
	unicast_4addr_packet = (struct batadv_unicast_4addr_packet *)skb->data;
843

844
	is4addr = unicast_packet->packet_type == BATADV_UNICAST_4ADDR;
845
	/* the caller function should have already pulled 2 bytes */
846
	if (is4addr)
847
		hdr_size = sizeof(*unicast_4addr_packet);
848

849
	/* function returns -EREMOTE for promiscuous packets */
850
	check = batadv_check_unicast_packet(bat_priv, skb, hdr_size);
851 852 853 854 855 856 857 858

	/* Even though the packet is not for us, we might save it to use for
	 * decoding a later received coded packet
	 */
	if (check == -EREMOTE)
		batadv_nc_skb_store_sniffed_unicast(bat_priv, skb);

	if (check < 0)
859
		return NET_RX_DROP;
860
	if (!batadv_check_unicast_ttvn(bat_priv, skb, hdr_size))
861 862
		return NET_RX_DROP;

863
	/* packet for me */
864
	if (batadv_is_my_mac(bat_priv, unicast_packet->dest)) {
865
		if (is4addr) {
866 867
			batadv_dat_inc_counter(bat_priv,
					       unicast_4addr_packet->subtype);
868 869 870
			orig_addr = unicast_4addr_packet->src;
			orig_node = batadv_orig_hash_find(bat_priv, orig_addr);
		}
871

872 873 874 875 876 877 878
		if (batadv_dat_snoop_incoming_arp_request(bat_priv, skb,
							  hdr_size))
			goto rx_success;
		if (batadv_dat_snoop_incoming_arp_reply(bat_priv, skb,
							hdr_size))
			goto rx_success;

879
		batadv_interface_rx(recv_if->soft_iface, skb, recv_if, hdr_size,
880
				    orig_node);
881

882
rx_success:
883 884 885
		if (orig_node)
			batadv_orig_node_free_ref(orig_node);

886 887 888
		return NET_RX_SUCCESS;
	}

889
	return batadv_route_unicast_packet(skb, recv_if);
890 891
}

892 893 894 895 896 897 898 899 900 901 902 903 904 905 906 907 908 909 910 911 912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936
/**
 * batadv_recv_unicast_tvlv - receive and process unicast tvlv packets
 * @skb: unicast tvlv packet to process
 * @recv_if: pointer to interface this packet was received on
 * @dst_addr: the payload destination
 *
 * Returns NET_RX_SUCCESS if the packet has been consumed or NET_RX_DROP
 * otherwise.
 */
int batadv_recv_unicast_tvlv(struct sk_buff *skb,
			     struct batadv_hard_iface *recv_if)
{
	struct batadv_priv *bat_priv = netdev_priv(recv_if->soft_iface);
	struct batadv_unicast_tvlv_packet *unicast_tvlv_packet;
	unsigned char *tvlv_buff;
	uint16_t tvlv_buff_len;
	int hdr_size = sizeof(*unicast_tvlv_packet);
	int ret = NET_RX_DROP;

	if (batadv_check_unicast_packet(bat_priv, skb, hdr_size) < 0)
		return NET_RX_DROP;

	/* the header is likely to be modified while forwarding */
	if (skb_cow(skb, hdr_size) < 0)
		return NET_RX_DROP;

	/* packet needs to be linearized to access the tvlv content */
	if (skb_linearize(skb) < 0)
		return NET_RX_DROP;

	unicast_tvlv_packet = (struct batadv_unicast_tvlv_packet *)skb->data;

	tvlv_buff = (unsigned char *)(skb->data + hdr_size);
	tvlv_buff_len = ntohs(unicast_tvlv_packet->tvlv_len);

	if (tvlv_buff_len > skb->len - hdr_size)
		return NET_RX_DROP;

	ret = batadv_tvlv_containers_process(bat_priv, false, NULL,
					     unicast_tvlv_packet->src,
					     unicast_tvlv_packet->dst,
					     tvlv_buff, tvlv_buff_len);

	if (ret != NET_RX_SUCCESS)
		ret = batadv_route_unicast_packet(skb, recv_if);
937 938
	else
		consume_skb(skb);
939 940 941

	return ret;
}
942

943 944 945 946 947 948 949 950 951 952 953 954 955 956 957 958 959 960 961 962 963 964 965 966 967 968 969 970 971 972 973 974 975 976 977 978 979 980 981 982 983 984 985 986 987 988 989 990 991 992 993 994 995 996 997 998 999 1000
/**
 * batadv_recv_frag_packet - process received fragment
 * @skb: the received fragment
 * @recv_if: interface that the skb is received on
 *
 * This function does one of the three following things: 1) Forward fragment, if
 * the assembled packet will exceed our MTU; 2) Buffer fragment, if we till
 * lack further fragments; 3) Merge fragments, if we have all needed parts.
 *
 * Return NET_RX_DROP if the skb is not consumed, NET_RX_SUCCESS otherwise.
 */
int batadv_recv_frag_packet(struct sk_buff *skb,
			    struct batadv_hard_iface *recv_if)
{
	struct batadv_priv *bat_priv = netdev_priv(recv_if->soft_iface);
	struct batadv_orig_node *orig_node_src = NULL;
	struct batadv_frag_packet *frag_packet;
	int ret = NET_RX_DROP;

	if (batadv_check_unicast_packet(bat_priv, skb,
					sizeof(*frag_packet)) < 0)
		goto out;

	frag_packet = (struct batadv_frag_packet *)skb->data;
	orig_node_src = batadv_orig_hash_find(bat_priv, frag_packet->orig);
	if (!orig_node_src)
		goto out;

	/* Route the fragment if it is not for us and too big to be merged. */
	if (!batadv_is_my_mac(bat_priv, frag_packet->dest) &&
	    batadv_frag_skb_fwd(skb, recv_if, orig_node_src)) {
		ret = NET_RX_SUCCESS;
		goto out;
	}

	batadv_inc_counter(bat_priv, BATADV_CNT_FRAG_RX);
	batadv_add_counter(bat_priv, BATADV_CNT_FRAG_RX_BYTES, skb->len);

	/* Add fragment to buffer and merge if possible. */
	if (!batadv_frag_skb_buffer(&skb, orig_node_src))
		goto out;

	/* Deliver merged packet to the appropriate handler, if it was
	 * merged
	 */
	if (skb)
		batadv_batman_skb_recv(skb, recv_if->net_dev,
				       &recv_if->batman_adv_ptype, NULL);

	ret = NET_RX_SUCCESS;

out:
	if (orig_node_src)
		batadv_orig_node_free_ref(orig_node_src);

	return ret;
}

1001 1002
int batadv_recv_bcast_packet(struct sk_buff *skb,
			     struct batadv_hard_iface *recv_if)
1003
{
1004 1005
	struct batadv_priv *bat_priv = netdev_priv(recv_if->soft_iface);
	struct batadv_orig_node *orig_node = NULL;
1006
	struct batadv_bcast_packet *bcast_packet;
1007
	struct ethhdr *ethhdr;
1008
	int hdr_size = sizeof(*bcast_packet);
1009
	int ret = NET_RX_DROP;
1010
	int32_t seq_diff;
1011
	uint32_t seqno;
1012 1013 1014

	/* drop packet if it has not necessary minimum size */
	if (unlikely(!pskb_may_pull(skb, hdr_size)))
1015
		goto out;
1016

1017
	ethhdr = eth_hdr(skb);
1018 1019 1020

	/* packet with broadcast indication but unicast recipient */
	if (!is_broadcast_ether_addr(ethhdr->h_dest))
1021
		goto out;
1022 1023 1024

	/* packet with broadcast sender address */
	if (is_broadcast_ether_addr(ethhdr->h_source))
1025
		goto out;
1026 1027

	/* ignore broadcasts sent by myself */
1028
	if (batadv_is_my_mac(bat_priv, ethhdr->h_source))
1029
		goto out;
1030

1031
	bcast_packet = (struct batadv_bcast_packet *)skb->data;
1032 1033

	/* ignore broadcasts originated by myself */
1034
	if (batadv_is_my_mac(bat_priv, bcast_packet->orig))
1035
		goto out;
1036

1037
	if (bcast_packet->ttl < 2)
1038
		goto out;
1039

1040
	orig_node = batadv_orig_hash_find(bat_priv, bcast_packet->orig);
1041 1042

	if (!orig_node)
1043
		goto out;
1044

1045
	spin_lock_bh(&orig_node->bcast_seqno_lock);
1046

1047
	seqno = ntohl(bcast_packet->seqno);
1048
	/* check whether the packet is a duplicate */
1049
	if (batadv_test_bit(orig_node->bcast_bits, orig_node->last_bcast_seqno,
1050
			    seqno))
1051
		goto spin_unlock;
1052

1053
	seq_diff = seqno - orig_node->last_bcast_seqno;
1054 1055

	/* check whether the packet is old and the host just restarted. */
1056 1057
	if (batadv_window_protected(bat_priv, seq_diff,
				    &orig_node->bcast_seqno_reset))
1058
		goto spin_unlock;
1059 1060

	/* mark broadcast in flood history, update window position
1061 1062
	 * if required.
	 */
1063
	if (batadv_bit_get_packet(bat_priv, orig_node->bcast_bits, seq_diff, 1))
1064
		orig_node->last_bcast_seqno = seqno;
1065

1066 1067
	spin_unlock_bh(&orig_node->bcast_seqno_lock);

1068
	/* check whether this has been sent by another originator before */
1069
	if (batadv_bla_check_bcast_duplist(bat_priv, skb))
1070 1071
		goto out;

1072 1073
	batadv_skb_set_priority(skb, sizeof(struct batadv_bcast_packet));

1074
	/* rebroadcast packet */
1075
	batadv_add_bcast_packet_to_list(bat_priv, skb, 1);
1076

1077 1078 1079
	/* don't hand the broadcast up if it is from an originator
	 * from the same backbone.
	 */
1080
	if (batadv_bla_is_backbone_gw(skb, orig_node, hdr_size))
1081 1082
		goto out;

1083 1084 1085 1086 1087
	if (batadv_dat_snoop_incoming_arp_request(bat_priv, skb, hdr_size))
		goto rx_success;
	if (batadv_dat_snoop_incoming_arp_reply(bat_priv, skb, hdr_size))
		goto rx_success;

1088
	/* broadcast for me */
1089 1090
	batadv_interface_rx(recv_if->soft_iface, skb, recv_if, hdr_size,
			    orig_node);
1091 1092

rx_success:
1093 1094
	ret = NET_RX_SUCCESS;
	goto out;
1095

1096 1097 1098 1099
spin_unlock:
	spin_unlock_bh(&orig_node->bcast_seqno_lock);
out:
	if (orig_node)
1100
		batadv_orig_node_free_ref(orig_node);
1101
	return ret;
1102
}