translation-table.c 114.9 KB
Newer Older
1
/* Copyright (C) 2007-2016  B.A.T.M.A.N. contributors:
2
 *
3
 * Marek Lindner, Simon Wunderlich, Antonio Quartulli
4 5 6 7 8 9 10 11 12 13 14
 *
 * 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 "translation-table.h"
19 20 21
#include "main.h"

#include <linux/atomic.h>
22
#include <linux/bitops.h>
23 24 25 26 27 28 29 30 31 32 33
#include <linux/bug.h>
#include <linux/byteorder/generic.h>
#include <linux/compiler.h>
#include <linux/crc32c.h>
#include <linux/errno.h>
#include <linux/etherdevice.h>
#include <linux/fs.h>
#include <linux/if_ether.h>
#include <linux/jhash.h>
#include <linux/jiffies.h>
#include <linux/kernel.h>
34
#include <linux/kref.h>
35 36 37 38 39 40 41 42 43 44 45 46 47 48
#include <linux/list.h>
#include <linux/lockdep.h>
#include <linux/netdevice.h>
#include <linux/rculist.h>
#include <linux/rcupdate.h>
#include <linux/seq_file.h>
#include <linux/slab.h>
#include <linux/spinlock.h>
#include <linux/stddef.h>
#include <linux/string.h>
#include <linux/workqueue.h>
#include <net/net_namespace.h>

#include "bridge_loop_avoidance.h"
49
#include "hard-interface.h"
50
#include "hash.h"
51
#include "multicast.h"
52 53 54
#include "originator.h"
#include "packet.h"
#include "soft-interface.h"
55

56 57 58 59
/* hash class keys */
static struct lock_class_key batadv_tt_local_hash_lock_class_key;
static struct lock_class_key batadv_tt_global_hash_lock_class_key;

60
static void batadv_send_roam_adv(struct batadv_priv *bat_priv, u8 *client,
61
				 unsigned short vid,
62
				 struct batadv_orig_node *orig_node);
63 64
static void batadv_tt_purge(struct work_struct *work);
static void
65
batadv_tt_global_del_orig_list(struct batadv_tt_global_entry *tt_global_entry);
66 67 68
static void batadv_tt_global_del(struct batadv_priv *bat_priv,
				 struct batadv_orig_node *orig_node,
				 const unsigned char *addr,
69 70
				 unsigned short vid, const char *message,
				 bool roaming);
71

72
/**
73 74 75
 * batadv_compare_tt - check if two TT entries are the same
 * @node: the list element pointer of the first TT entry
 * @data2: pointer to the tt_common_entry of the second TT entry
76
 *
77 78 79
 * Compare the MAC address and the VLAN ID of the two TT entries and check if
 * they are the same TT client.
 * Return: 1 if the two TT clients are the same, 0 otherwise
80
 */
81
static int batadv_compare_tt(const struct hlist_node *node, const void *data2)
82
{
83
	const void *data1 = container_of(node, struct batadv_tt_common_entry,
84
					 hash_entry);
85 86
	const struct batadv_tt_common_entry *tt1 = data1;
	const struct batadv_tt_common_entry *tt2 = data2;
87

88
	return (tt1->vid == tt2->vid) && batadv_compare_eth(data1, data2);
89 90
}

91 92 93 94 95
/**
 * batadv_choose_tt - return the index of the tt entry in the hash table
 * @data: pointer to the tt_common_entry object to map
 * @size: the size of the hash table
 *
96
 * Return: the hash index where the object represented by 'data' should be
97 98
 * stored at.
 */
99
static inline u32 batadv_choose_tt(const void *data, u32 size)
100 101
{
	struct batadv_tt_common_entry *tt;
102
	u32 hash = 0;
103 104

	tt = (struct batadv_tt_common_entry *)data;
105 106
	hash = jhash(&tt->addr, ETH_ALEN, hash);
	hash = jhash(&tt->vid, sizeof(tt->vid), hash);
107 108 109 110 111 112 113 114 115 116

	return hash % size;
}

/**
 * batadv_tt_hash_find - look for a client in the given hash table
 * @hash: the hash table to search
 * @addr: the mac address of the client to look for
 * @vid: VLAN identifier
 *
117
 * Return: a pointer to the tt_common struct belonging to the searched client if
118 119
 * found, NULL otherwise.
 */
120
static struct batadv_tt_common_entry *
121
batadv_tt_hash_find(struct batadv_hashtable *hash, const u8 *addr,
122
		    unsigned short vid)
123 124
{
	struct hlist_head *head;
125
	struct batadv_tt_common_entry to_search, *tt, *tt_tmp = NULL;
126
	u32 index;
127 128 129 130

	if (!hash)
		return NULL;

131
	ether_addr_copy(to_search.addr, addr);
132 133 134
	to_search.vid = vid;

	index = batadv_choose_tt(&to_search, hash->size);
135 136 137
	head = &hash->table[index];

	rcu_read_lock();
138 139 140 141 142
	hlist_for_each_entry_rcu(tt, head, hash_entry) {
		if (!batadv_compare_eth(tt, addr))
			continue;

		if (tt->vid != vid)
143 144
			continue;

145
		if (!kref_get_unless_zero(&tt->refcount))
146 147
			continue;

148
		tt_tmp = tt;
149 150 151 152
		break;
	}
	rcu_read_unlock();

153
	return tt_tmp;
154 155
}

156 157 158 159 160 161
/**
 * batadv_tt_local_hash_find - search the local table for a given client
 * @bat_priv: the bat priv with all the soft interface information
 * @addr: the mac address of the client to look for
 * @vid: VLAN identifier
 *
162
 * Return: a pointer to the corresponding tt_local_entry struct if the client is
163 164
 * found, NULL otherwise.
 */
165
static struct batadv_tt_local_entry *
166
batadv_tt_local_hash_find(struct batadv_priv *bat_priv, const u8 *addr,
167
			  unsigned short vid)
168
{
169 170
	struct batadv_tt_common_entry *tt_common_entry;
	struct batadv_tt_local_entry *tt_local_entry = NULL;
171

172 173
	tt_common_entry = batadv_tt_hash_find(bat_priv->tt.local_hash, addr,
					      vid);
174 175
	if (tt_common_entry)
		tt_local_entry = container_of(tt_common_entry,
176 177
					      struct batadv_tt_local_entry,
					      common);
178 179
	return tt_local_entry;
}
180

181 182 183 184 185 186
/**
 * batadv_tt_global_hash_find - search the global table for a given client
 * @bat_priv: the bat priv with all the soft interface information
 * @addr: the mac address of the client to look for
 * @vid: VLAN identifier
 *
187
 * Return: a pointer to the corresponding tt_global_entry struct if the client
188 189
 * is found, NULL otherwise.
 */
190
static struct batadv_tt_global_entry *
191
batadv_tt_global_hash_find(struct batadv_priv *bat_priv, const u8 *addr,
192
			   unsigned short vid)
193
{
194 195
	struct batadv_tt_common_entry *tt_common_entry;
	struct batadv_tt_global_entry *tt_global_entry = NULL;
196

197 198
	tt_common_entry = batadv_tt_hash_find(bat_priv->tt.global_hash, addr,
					      vid);
199 200
	if (tt_common_entry)
		tt_global_entry = container_of(tt_common_entry,
201 202
					       struct batadv_tt_global_entry,
					       common);
203
	return tt_global_entry;
204 205
}

206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225
/**
 * batadv_tt_local_entry_release - release tt_local_entry from lists and queue
 *  for free after rcu grace period
 * @ref: kref pointer of the nc_node
 */
static void batadv_tt_local_entry_release(struct kref *ref)
{
	struct batadv_tt_local_entry *tt_local_entry;

	tt_local_entry = container_of(ref, struct batadv_tt_local_entry,
				      common.refcount);

	kfree_rcu(tt_local_entry, common.rcu);
}

/**
 * batadv_tt_local_entry_free_ref - decrement the tt_local_entry refcounter and
 *  possibly release it
 * @tt_local_entry: tt_local_entry to be free'd
 */
226
static void
227
batadv_tt_local_entry_free_ref(struct batadv_tt_local_entry *tt_local_entry)
228
{
229 230
	kref_put(&tt_local_entry->common.refcount,
		 batadv_tt_local_entry_release);
231 232
}

233
/**
234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252
 * batadv_tt_global_entry_release - release tt_global_entry from lists and queue
 *  for free after rcu grace period
 * @ref: kref pointer of the nc_node
 */
static void batadv_tt_global_entry_release(struct kref *ref)
{
	struct batadv_tt_global_entry *tt_global_entry;

	tt_global_entry = container_of(ref, struct batadv_tt_global_entry,
				       common.refcount);

	batadv_tt_global_del_orig_list(tt_global_entry);
	kfree_rcu(tt_global_entry, common.rcu);
}

/**
 * batadv_tt_global_entry_free_ref - decrement the tt_global_entry refcounter
 *  and possibly release it
 * @tt_global_entry: tt_global_entry to be free'd
253
 */
254
static void
255
batadv_tt_global_entry_free_ref(struct batadv_tt_global_entry *tt_global_entry)
256
{
257 258
	kref_put(&tt_global_entry->common.refcount,
		 batadv_tt_global_entry_release);
259 260
}

261 262
/**
 * batadv_tt_global_hash_count - count the number of orig entries
263
 * @bat_priv: the bat priv with all the soft interface information
264 265 266
 * @addr: the mac address of the client to count entries for
 * @vid: VLAN identifier
 *
267
 * Return: the number of originators advertising the given address/data
268 269 270
 * (excluding ourself).
 */
int batadv_tt_global_hash_count(struct batadv_priv *bat_priv,
271
				const u8 *addr, unsigned short vid)
272 273 274 275 276 277 278 279 280 281 282 283 284 285
{
	struct batadv_tt_global_entry *tt_global_entry;
	int count;

	tt_global_entry = batadv_tt_global_hash_find(bat_priv, addr, vid);
	if (!tt_global_entry)
		return 0;

	count = atomic_read(&tt_global_entry->orig_list_count);
	batadv_tt_global_entry_free_ref(tt_global_entry);

	return count;
}

286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303
/**
 * batadv_tt_local_size_mod - change the size by v of the local table identified
 *  by vid
 * @bat_priv: the bat priv with all the soft interface information
 * @vid: the VLAN identifier of the sub-table to change
 * @v: the amount to sum to the local table size
 */
static void batadv_tt_local_size_mod(struct batadv_priv *bat_priv,
				     unsigned short vid, int v)
{
	struct batadv_softif_vlan *vlan;

	vlan = batadv_softif_vlan_get(bat_priv, vid);
	if (!vlan)
		return;

	atomic_add(v, &vlan->tt.num_entries);

304
	batadv_softif_vlan_put(vlan);
305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331
}

/**
 * batadv_tt_local_size_inc - increase by one the local table size for the given
 *  vid
 * @bat_priv: the bat priv with all the soft interface information
 * @vid: the VLAN identifier
 */
static void batadv_tt_local_size_inc(struct batadv_priv *bat_priv,
				     unsigned short vid)
{
	batadv_tt_local_size_mod(bat_priv, vid, 1);
}

/**
 * batadv_tt_local_size_dec - decrease by one the local table size for the given
 *  vid
 * @bat_priv: the bat priv with all the soft interface information
 * @vid: the VLAN identifier
 */
static void batadv_tt_local_size_dec(struct batadv_priv *bat_priv,
				     unsigned short vid)
{
	batadv_tt_local_size_mod(bat_priv, vid, -1);
}

/**
332 333 334
 * batadv_tt_global_size_mod - change the size by v of the global table
 *  for orig_node identified by vid
 * @orig_node: the originator for which the table has to be modified
335 336 337 338 339 340 341 342 343 344 345 346 347 348
 * @vid: the VLAN identifier
 * @v: the amount to sum to the global table size
 */
static void batadv_tt_global_size_mod(struct batadv_orig_node *orig_node,
				      unsigned short vid, int v)
{
	struct batadv_orig_node_vlan *vlan;

	vlan = batadv_orig_node_vlan_new(orig_node, vid);
	if (!vlan)
		return;

	if (atomic_add_return(v, &vlan->tt.num_entries) == 0) {
		spin_lock_bh(&orig_node->vlan_list_lock);
349 350 351 352
		if (!hlist_unhashed(&vlan->list)) {
			hlist_del_init_rcu(&vlan->list);
			batadv_orig_node_vlan_free_ref(vlan);
		}
353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382
		spin_unlock_bh(&orig_node->vlan_list_lock);
	}

	batadv_orig_node_vlan_free_ref(vlan);
}

/**
 * batadv_tt_global_size_inc - increase by one the global table size for the
 *  given vid
 * @orig_node: the originator which global table size has to be decreased
 * @vid: the vlan identifier
 */
static void batadv_tt_global_size_inc(struct batadv_orig_node *orig_node,
				      unsigned short vid)
{
	batadv_tt_global_size_mod(orig_node, vid, 1);
}

/**
 * batadv_tt_global_size_dec - decrease by one the global table size for the
 *  given vid
 * @orig_node: the originator which global table size has to be decreased
 * @vid: the vlan identifier
 */
static void batadv_tt_global_size_dec(struct batadv_orig_node *orig_node,
				      unsigned short vid)
{
	batadv_tt_global_size_mod(orig_node, vid, -1);
}

383 384 385
/**
 * batadv_tt_orig_list_entry_release - release tt orig entry from lists and
 *  queue for free after rcu grace period
386
 * @ref: kref pointer of the tt orig entry
387
 */
388
static void batadv_tt_orig_list_entry_release(struct kref *ref)
389
{
390 391 392 393 394
	struct batadv_tt_orig_list_entry *orig_entry;

	orig_entry = container_of(ref, struct batadv_tt_orig_list_entry,
				  refcount);

395
	batadv_orig_node_put(orig_entry->orig_node);
396 397 398
	kfree_rcu(orig_entry, rcu);
}

399 400 401 402 403
/**
 * batadv_tt_orig_list_entry_free_ref - decrement the tt orig entry refcounter
 *  and possibly release it
 * @orig_entry: tt orig entry to be free'd
 */
404
static void
405
batadv_tt_orig_list_entry_free_ref(struct batadv_tt_orig_list_entry *orig_entry)
406
{
407
	kref_put(&orig_entry->refcount, batadv_tt_orig_list_entry_release);
408 409
}

410 411 412 413 414 415
/**
 * batadv_tt_local_event - store a local TT event (ADD/DEL)
 * @bat_priv: the bat priv with all the soft interface information
 * @tt_local_entry: the TT entry involved in the event
 * @event_flags: flags to store in the event structure
 */
416
static void batadv_tt_local_event(struct batadv_priv *bat_priv,
417
				  struct batadv_tt_local_entry *tt_local_entry,
418
				  u8 event_flags)
419
{
420
	struct batadv_tt_change_node *tt_change_node, *entry, *safe;
421
	struct batadv_tt_common_entry *common = &tt_local_entry->common;
422
	u8 flags = common->flags | event_flags;
423 424
	bool event_removed = false;
	bool del_op_requested, del_op_entry;
425 426 427 428 429

	tt_change_node = kmalloc(sizeof(*tt_change_node), GFP_ATOMIC);
	if (!tt_change_node)
		return;

430
	tt_change_node->change.flags = flags;
431 432
	memset(tt_change_node->change.reserved, 0,
	       sizeof(tt_change_node->change.reserved));
433
	ether_addr_copy(tt_change_node->change.addr, common->addr);
434
	tt_change_node->change.vid = htons(common->vid);
435

436
	del_op_requested = flags & BATADV_TT_CLIENT_DEL;
437 438

	/* check for ADD+DEL or DEL+ADD events */
439 440
	spin_lock_bh(&bat_priv->tt.changes_list_lock);
	list_for_each_entry_safe(entry, safe, &bat_priv->tt.changes_list,
441
				 list) {
442
		if (!batadv_compare_eth(entry->change.addr, common->addr))
443 444 445 446 447 448 449 450 451
			continue;

		/* DEL+ADD in the same orig interval have no effect and can be
		 * removed to avoid silly behaviour on the receiver side. The
		 * other way around (ADD+DEL) can happen in case of roaming of
		 * a client still in the NEW state. Roaming of NEW clients is
		 * now possible due to automatically recognition of "temporary"
		 * clients
		 */
452
		del_op_entry = entry->change.flags & BATADV_TT_CLIENT_DEL;
453 454 455 456
		if (!del_op_requested && del_op_entry)
			goto del;
		if (del_op_requested && !del_op_entry)
			goto del;
457 458 459 460 461 462 463

		/* this is a second add in the same originator interval. It
		 * means that flags have been changed: update them!
		 */
		if (!del_op_requested && !del_op_entry)
			entry->change.flags = flags;

464 465 466 467
		continue;
del:
		list_del(&entry->list);
		kfree(entry);
468
		kfree(tt_change_node);
469 470 471 472
		event_removed = true;
		goto unlock;
	}

473
	/* track the change in the OGMinterval list */
474
	list_add_tail(&tt_change_node->list, &bat_priv->tt.changes_list);
475 476

unlock:
477
	spin_unlock_bh(&bat_priv->tt.changes_list_lock);
478

479
	if (event_removed)
480
		atomic_dec(&bat_priv->tt.local_changes);
481
	else
482
		atomic_inc(&bat_priv->tt.local_changes);
483 484
}

485 486 487 488
/**
 * batadv_tt_len - compute length in bytes of given number of tt changes
 * @changes_num: number of tt changes
 *
489
 * Return: computed length in bytes.
490 491
 */
static int batadv_tt_len(int changes_num)
492
{
493
	return changes_num * sizeof(struct batadv_tvlv_tt_change);
494 495
}

496 497 498 499
/**
 * batadv_tt_entries - compute the number of entries fitting in tt_len bytes
 * @tt_len: available space
 *
500
 * Return: the number of entries.
501
 */
502
static u16 batadv_tt_entries(u16 tt_len)
503 504 505 506
{
	return tt_len / batadv_tt_len(1);
}

507 508 509 510 511
/**
 * batadv_tt_local_table_transmit_size - calculates the local translation table
 *  size when transmitted over the air
 * @bat_priv: the bat priv with all the soft interface information
 *
512
 * Return: local translation table size in bytes.
513 514 515
 */
static int batadv_tt_local_table_transmit_size(struct batadv_priv *bat_priv)
{
516 517
	u16 num_vlan = 0;
	u16 tt_local_entries = 0;
518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536
	struct batadv_softif_vlan *vlan;
	int hdr_size;

	rcu_read_lock();
	hlist_for_each_entry_rcu(vlan, &bat_priv->softif_vlan_list, list) {
		num_vlan++;
		tt_local_entries += atomic_read(&vlan->tt.num_entries);
	}
	rcu_read_unlock();

	/* header size of tvlv encapsulated tt response payload */
	hdr_size = sizeof(struct batadv_unicast_tvlv_packet);
	hdr_size += sizeof(struct batadv_tvlv_hdr);
	hdr_size += sizeof(struct batadv_tvlv_tt_data);
	hdr_size += num_vlan * sizeof(struct batadv_tvlv_tt_vlan_data);

	return hdr_size + batadv_tt_len(tt_local_entries);
}

537
static int batadv_tt_local_init(struct batadv_priv *bat_priv)
538
{
539
	if (bat_priv->tt.local_hash)
540
		return 0;
541

542
	bat_priv->tt.local_hash = batadv_hash_new(1024);
543

544
	if (!bat_priv->tt.local_hash)
545
		return -ENOMEM;
546

547 548 549
	batadv_hash_set_lock_class(bat_priv->tt.local_hash,
				   &batadv_tt_local_hash_lock_class_key);

550
	return 0;
551 552
}

553 554 555 556 557
static void batadv_tt_global_free(struct batadv_priv *bat_priv,
				  struct batadv_tt_global_entry *tt_global,
				  const char *message)
{
	batadv_dbg(BATADV_DBG_TT, bat_priv,
558 559 560
		   "Deleting global tt entry %pM (vid: %d): %s\n",
		   tt_global->common.addr,
		   BATADV_PRINT_VID(tt_global->common.vid), message);
561 562

	batadv_hash_remove(bat_priv->tt.global_hash, batadv_compare_tt,
563
			   batadv_choose_tt, &tt_global->common);
564 565 566
	batadv_tt_global_entry_free_ref(tt_global);
}

567 568 569 570 571 572 573 574
/**
 * batadv_tt_local_add - add a new client to the local table or update an
 *  existing client
 * @soft_iface: netdev struct of the mesh interface
 * @addr: the mac address of the client to add
 * @vid: VLAN identifier
 * @ifindex: index of the interface where the client is connected to (useful to
 *  identify wireless clients)
575 576
 * @mark: the value contained in the skb->mark field of the received packet (if
 *  any)
577
 *
578
 * Return: true if the client was successfully added, false otherwise.
579
 */
580 581
bool batadv_tt_local_add(struct net_device *soft_iface, const u8 *addr,
			 unsigned short vid, int ifindex, u32 mark)
582
{
583
	struct batadv_priv *bat_priv = netdev_priv(soft_iface);
584
	struct batadv_tt_local_entry *tt_local;
585
	struct batadv_tt_global_entry *tt_global = NULL;
586
	struct batadv_softif_vlan *vlan;
587
	struct net_device *in_dev = NULL;
588
	struct hlist_head *head;
589
	struct batadv_tt_orig_list_entry *orig_entry;
590
	int hash_added, table_size, packet_size_max;
591 592
	bool ret = false;
	bool roamed_back = false;
593 594
	u8 remote_flags;
	u32 match_mark;
595

596 597 598
	if (ifindex != BATADV_NULL_IFINDEX)
		in_dev = dev_get_by_index(&init_net, ifindex);

599
	tt_local = batadv_tt_local_hash_find(bat_priv, addr, vid);
600 601 602

	if (!is_multicast_ether_addr(addr))
		tt_global = batadv_tt_global_hash_find(bat_priv, addr, vid);
603

604 605
	if (tt_local) {
		tt_local->last_seen = jiffies;
606 607
		if (tt_local->common.flags & BATADV_TT_CLIENT_PENDING) {
			batadv_dbg(BATADV_DBG_TT, bat_priv,
608 609
				   "Re-adding pending client %pM (vid: %d)\n",
				   addr, BATADV_PRINT_VID(vid));
610 611 612 613 614 615 616 617 618 619 620
			/* whatever the reason why the PENDING flag was set,
			 * this is a client which was enqueued to be removed in
			 * this orig_interval. Since it popped up again, the
			 * flag can be reset like it was never enqueued
			 */
			tt_local->common.flags &= ~BATADV_TT_CLIENT_PENDING;
			goto add_event;
		}

		if (tt_local->common.flags & BATADV_TT_CLIENT_ROAM) {
			batadv_dbg(BATADV_DBG_TT, bat_priv,
621 622
				   "Roaming client %pM (vid: %d) came back to its original location\n",
				   addr, BATADV_PRINT_VID(vid));
623 624 625 626 627 628 629 630 631
			/* the ROAM flag is set because this client roamed away
			 * and the node got a roaming_advertisement message. Now
			 * that the client popped up again at its original
			 * location such flag can be unset
			 */
			tt_local->common.flags &= ~BATADV_TT_CLIENT_ROAM;
			roamed_back = true;
		}
		goto check_roaming;
632 633
	}

634 635 636 637 638 639 640 641 642 643 644
	/* Ignore the client if we cannot send it in a full table response. */
	table_size = batadv_tt_local_table_transmit_size(bat_priv);
	table_size += batadv_tt_len(1);
	packet_size_max = atomic_read(&bat_priv->packet_size_max);
	if (table_size > packet_size_max) {
		net_ratelimited_function(batadv_info, soft_iface,
					 "Local translation table size (%i) exceeds maximum packet size (%i); Ignoring new local tt entry: %pM\n",
					 table_size, packet_size_max, addr);
		goto out;
	}

645 646
	tt_local = kmalloc(sizeof(*tt_local), GFP_ATOMIC);
	if (!tt_local)
647
		goto out;
648

649 650
	/* increase the refcounter of the related vlan */
	vlan = batadv_softif_vlan_get(bat_priv, vid);
651
	if (WARN(!vlan, "adding TT local entry %pM to non-existent VLAN %d",
652 653 654
		 addr, BATADV_PRINT_VID(vid))) {
		kfree(tt_local);
		tt_local = NULL;
655
		goto out;
656
	}
657

658
	batadv_dbg(BATADV_DBG_TT, bat_priv,
659 660
		   "Creating new local tt entry: %pM (vid: %d, ttvn: %d)\n",
		   addr, BATADV_PRINT_VID(vid),
661
		   (u8)atomic_read(&bat_priv->tt.vn));
662

663
	ether_addr_copy(tt_local->common.addr, addr);
664 665 666 667 668
	/* The local entry has to be marked as NEW to avoid to send it in
	 * a full table response going out before the next ttvn increment
	 * (consistency check)
	 */
	tt_local->common.flags = BATADV_TT_CLIENT_NEW;
669
	tt_local->common.vid = vid;
670
	if (batadv_is_wifi_netdev(in_dev))
671
		tt_local->common.flags |= BATADV_TT_CLIENT_WIFI;
672 673
	kref_init(&tt_local->common.refcount);
	kref_get(&tt_local->common.refcount);
674 675
	tt_local->last_seen = jiffies;
	tt_local->common.added_at = tt_local->last_seen;
676

677 678 679 680 681
	/* the batman interface mac and multicast addresses should never be
	 * purged
	 */
	if (batadv_compare_eth(addr, soft_iface->dev_addr) ||
	    is_multicast_ether_addr(addr))
682
		tt_local->common.flags |= BATADV_TT_CLIENT_NOPURGE;
683

684
	hash_added = batadv_hash_add(bat_priv->tt.local_hash, batadv_compare_tt,
685
				     batadv_choose_tt, &tt_local->common,
686
				     &tt_local->common.hash_entry);
687 688 689

	if (unlikely(hash_added != 0)) {
		/* remove the reference for the hash */
690
		batadv_tt_local_entry_free_ref(tt_local);
691
		batadv_softif_vlan_put(vlan);
692 693 694
		goto out;
	}

695
add_event:
696
	batadv_tt_local_event(bat_priv, tt_local, BATADV_NO_FLAGS);
697

698 699 700 701 702
check_roaming:
	/* Check whether it is a roaming, but don't do anything if the roaming
	 * process has already been handled
	 */
	if (tt_global && !(tt_global->common.flags & BATADV_TT_CLIENT_ROAM)) {
703
		/* These node are probably going to update their tt table */
704
		head = &tt_global->orig_list;
705
		rcu_read_lock();
706
		hlist_for_each_entry_rcu(orig_entry, head, list) {
707
			batadv_send_roam_adv(bat_priv, tt_global->common.addr,
708
					     tt_global->common.vid,
709
					     orig_entry->orig_node);
710 711
		}
		rcu_read_unlock();
712 713 714 715 716 717 718 719 720 721 722
		if (roamed_back) {
			batadv_tt_global_free(bat_priv, tt_global,
					      "Roaming canceled");
			tt_global = NULL;
		} else {
			/* The global entry has to be marked as ROAMING and
			 * has to be kept for consistency purpose
			 */
			tt_global->common.flags |= BATADV_TT_CLIENT_ROAM;
			tt_global->roam_at = jiffies;
		}
723
	}
724

725 726 727 728 729 730 731 732 733
	/* store the current remote flags before altering them. This helps
	 * understanding is flags are changing or not
	 */
	remote_flags = tt_local->common.flags & BATADV_TT_REMOTE_MASK;

	if (batadv_is_wifi_netdev(in_dev))
		tt_local->common.flags |= BATADV_TT_CLIENT_WIFI;
	else
		tt_local->common.flags &= ~BATADV_TT_CLIENT_WIFI;
734

735 736 737 738 739 740 741 742 743 744 745
	/* check the mark in the skb: if it's equal to the configured
	 * isolation_mark, it means the packet is coming from an isolated
	 * non-mesh client
	 */
	match_mark = (mark & bat_priv->isolation_mark_mask);
	if (bat_priv->isolation_mark_mask &&
	    match_mark == bat_priv->isolation_mark)
		tt_local->common.flags |= BATADV_TT_CLIENT_ISOLA;
	else
		tt_local->common.flags &= ~BATADV_TT_CLIENT_ISOLA;

746 747 748 749 750 751 752
	/* if any "dynamic" flag has been modified, resend an ADD event for this
	 * entry so that all the nodes can get the new flags
	 */
	if (remote_flags ^ (tt_local->common.flags & BATADV_TT_REMOTE_MASK))
		batadv_tt_local_event(bat_priv, tt_local, BATADV_NO_FLAGS);

	ret = true;
753
out:
754 755
	if (in_dev)
		dev_put(in_dev);
756 757 758 759
	if (tt_local)
		batadv_tt_local_entry_free_ref(tt_local);
	if (tt_global)
		batadv_tt_global_entry_free_ref(tt_global);
760
	return ret;
761 762
}

763 764 765 766 767 768 769 770 771 772 773 774 775 776 777
/**
 * batadv_tt_prepare_tvlv_global_data - prepare the TVLV TT header to send
 *  within a TT Response directed to another node
 * @orig_node: originator for which the TT data has to be prepared
 * @tt_data: uninitialised pointer to the address of the TVLV buffer
 * @tt_change: uninitialised pointer to the address of the area where the TT
 *  changed can be stored
 * @tt_len: pointer to the length to reserve to the tt_change. if -1 this
 *  function reserves the amount of space needed to send the entire global TT
 *  table. In case of success the value is updated with the real amount of
 *  reserved bytes
 * Allocate the needed amount of memory for the entire TT TVLV and write its
 * header made up by one tvlv_tt_data object and a series of tvlv_tt_vlan_data
 * objects, one per active VLAN served by the originator node.
 *
778
 * Return: the size of the allocated buffer or 0 in case of failure.
779
 */
780
static u16
781 782 783
batadv_tt_prepare_tvlv_global_data(struct batadv_orig_node *orig_node,
				   struct batadv_tvlv_tt_data **tt_data,
				   struct batadv_tvlv_tt_change **tt_change,
784
				   s32 *tt_len)
785
{
786 787 788 789
	u16 num_vlan = 0;
	u16 num_entries = 0;
	u16 change_offset;
	u16 tvlv_len;
790 791
	struct batadv_tvlv_tt_vlan_data *tt_vlan;
	struct batadv_orig_node_vlan *vlan;
792
	u8 *tt_change_ptr;
793 794

	rcu_read_lock();
795
	hlist_for_each_entry_rcu(vlan, &orig_node->vlan_list, list) {
796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819 820
		num_vlan++;
		num_entries += atomic_read(&vlan->tt.num_entries);
	}

	change_offset = sizeof(**tt_data);
	change_offset += num_vlan * sizeof(*tt_vlan);

	/* if tt_len is negative, allocate the space needed by the full table */
	if (*tt_len < 0)
		*tt_len = batadv_tt_len(num_entries);

	tvlv_len = *tt_len;
	tvlv_len += change_offset;

	*tt_data = kmalloc(tvlv_len, GFP_ATOMIC);
	if (!*tt_data) {
		*tt_len = 0;
		goto out;
	}

	(*tt_data)->flags = BATADV_NO_FLAGS;
	(*tt_data)->ttvn = atomic_read(&orig_node->last_ttvn);
	(*tt_data)->num_vlan = htons(num_vlan);

	tt_vlan = (struct batadv_tvlv_tt_vlan_data *)(*tt_data + 1);
821
	hlist_for_each_entry_rcu(vlan, &orig_node->vlan_list, list) {
822 823 824 825 826 827
		tt_vlan->vid = htons(vlan->vid);
		tt_vlan->crc = htonl(vlan->tt.crc);

		tt_vlan++;
	}

828
	tt_change_ptr = (u8 *)*tt_data + change_offset;
829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851
	*tt_change = (struct batadv_tvlv_tt_change *)tt_change_ptr;

out:
	rcu_read_unlock();
	return tvlv_len;
}

/**
 * batadv_tt_prepare_tvlv_local_data - allocate and prepare the TT TVLV for this
 *  node
 * @bat_priv: the bat priv with all the soft interface information
 * @tt_data: uninitialised pointer to the address of the TVLV buffer
 * @tt_change: uninitialised pointer to the address of the area where the TT
 *  changes can be stored
 * @tt_len: pointer to the length to reserve to the tt_change. if -1 this
 *  function reserves the amount of space needed to send the entire local TT
 *  table. In case of success the value is updated with the real amount of
 *  reserved bytes
 *
 * Allocate the needed amount of memory for the entire TT TVLV and write its
 * header made up by one tvlv_tt_data object and a series of tvlv_tt_vlan_data
 * objects, one per active VLAN.
 *
852
 * Return: the size of the allocated buffer or 0 in case of failure.
853
 */
854
static u16
855 856 857
batadv_tt_prepare_tvlv_local_data(struct batadv_priv *bat_priv,
				  struct batadv_tvlv_tt_data **tt_data,
				  struct batadv_tvlv_tt_change **tt_change,
858
				  s32 *tt_len)
859 860 861
{
	struct batadv_tvlv_tt_vlan_data *tt_vlan;
	struct batadv_softif_vlan *vlan;
862 863 864
	u16 num_vlan = 0;
	u16 num_entries = 0;
	u16 tvlv_len;
865
	u8 *tt_change_ptr;
866 867 868 869 870 871 872 873 874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898 899 900 901
	int change_offset;

	rcu_read_lock();
	hlist_for_each_entry_rcu(vlan, &bat_priv->softif_vlan_list, list) {
		num_vlan++;
		num_entries += atomic_read(&vlan->tt.num_entries);
	}

	change_offset = sizeof(**tt_data);
	change_offset += num_vlan * sizeof(*tt_vlan);

	/* if tt_len is negative, allocate the space needed by the full table */
	if (*tt_len < 0)
		*tt_len = batadv_tt_len(num_entries);

	tvlv_len = *tt_len;
	tvlv_len += change_offset;

	*tt_data = kmalloc(tvlv_len, GFP_ATOMIC);
	if (!*tt_data) {
		tvlv_len = 0;
		goto out;
	}

	(*tt_data)->flags = BATADV_NO_FLAGS;
	(*tt_data)->ttvn = atomic_read(&bat_priv->tt.vn);
	(*tt_data)->num_vlan = htons(num_vlan);

	tt_vlan = (struct batadv_tvlv_tt_vlan_data *)(*tt_data + 1);
	hlist_for_each_entry_rcu(vlan, &bat_priv->softif_vlan_list, list) {
		tt_vlan->vid = htons(vlan->vid);
		tt_vlan->crc = htonl(vlan->tt.crc);

		tt_vlan++;
	}

902
	tt_change_ptr = (u8 *)*tt_data + change_offset;
903 904 905 906 907 908 909
	*tt_change = (struct batadv_tvlv_tt_change *)tt_change_ptr;

out:
	rcu_read_unlock();
	return tvlv_len;
}

910 911 912 913 914 915
/**
 * batadv_tt_tvlv_container_update - update the translation table tvlv container
 *  after local tt changes have been committed
 * @bat_priv: the bat priv with all the soft interface information
 */
static void batadv_tt_tvlv_container_update(struct batadv_priv *bat_priv)
916
{
917 918 919
	struct batadv_tt_change_node *entry, *safe;
	struct batadv_tvlv_tt_data *tt_data;
	struct batadv_tvlv_tt_change *tt_change;
920
	int tt_diff_len, tt_change_len = 0;
921 922
	int tt_diff_entries_num = 0;
	int tt_diff_entries_count = 0;
923
	u16 tvlv_len;
924

925 926
	tt_diff_entries_num = atomic_read(&bat_priv->tt.local_changes);
	tt_diff_len = batadv_tt_len(tt_diff_entries_num);
927 928 929 930

	/* if we have too many changes for one packet don't send any
	 * and wait for the tt table request which will be fragmented
	 */
931 932
	if (tt_diff_len > bat_priv->soft_iface->mtu)
		tt_diff_len = 0;
933

934 935 936
	tvlv_len = batadv_tt_prepare_tvlv_local_data(bat_priv, &tt_data,
						     &tt_change, &tt_diff_len);
	if (!tvlv_len)
937
		return;
938

939
	tt_data->flags = BATADV_TT_OGM_DIFF;
940

941 942
	if (tt_diff_len == 0)
		goto container_register;
943

944 945
	spin_lock_bh(&bat_priv->tt.changes_list_lock);
	atomic_set(&bat_priv->tt.local_changes, 0);
946

947
	list_for_each_entry_safe(entry, safe, &bat_priv->tt.changes_list,
948
				 list) {
949 950 951 952 953
		if (tt_diff_entries_count < tt_diff_entries_num) {
			memcpy(tt_change + tt_diff_entries_count,
			       &entry->change,
			       sizeof(struct batadv_tvlv_tt_change));
			tt_diff_entries_count++;
954
		}
955 956
		list_del(&entry->list);
		kfree(entry);
957
	}
958
	spin_unlock_bh(&bat_priv->tt.changes_list_lock);
959 960

	/* Keep the buffer for possible tt_request */
961 962 963 964
	spin_lock_bh(&bat_priv->tt.last_changeset_lock);
	kfree(bat_priv->tt.last_changeset);
	bat_priv->tt.last_changeset_len = 0;
	bat_priv->tt.last_changeset = NULL;
965
	tt_change_len = batadv_tt_len(tt_diff_entries_count);
966
	/* check whether this new OGM has no changes due to size problems */
967
	if (tt_diff_entries_count > 0) {
968
		/* if kmalloc() fails we will reply with the full table
969 970
		 * instead of providing the diff
		 */
971
		bat_priv->tt.last_changeset = kzalloc(tt_diff_len, GFP_ATOMIC);
972
		if (bat_priv->tt.last_changeset) {
973 974 975
			memcpy(bat_priv->tt.last_changeset,
			       tt_change, tt_change_len);
			bat_priv->tt.last_changeset_len = tt_diff_len;
976 977
		}
	}
978
	spin_unlock_bh(&bat_priv->tt.last_changeset_lock);
979

980 981
container_register:
	batadv_tvlv_container_register(bat_priv, BATADV_TVLV_TT, 1, tt_data,
982
				       tvlv_len);
983
	kfree(tt_data);
984 985
}

986
int batadv_tt_local_seq_print_text(struct seq_file *seq, void *offset)
987 988
{
	struct net_device *net_dev = (struct net_device *)seq->private;
989
	struct batadv_priv *bat_priv = netdev_priv(net_dev);
990
	struct batadv_hashtable *hash = bat_priv->tt.local_hash;
991
	struct batadv_tt_common_entry *tt_common_entry;
992
	struct batadv_tt_local_entry *tt_local;
993
	struct batadv_hard_iface *primary_if;
994
	struct batadv_softif_vlan *vlan;
995
	struct hlist_head *head;
996
	unsigned short vid;
997
	u32 i;
998 999 1000 1001
	int last_seen_secs;
	int last_seen_msecs;
	unsigned long last_seen_jiffies;
	bool no_purge;
1002
	u16 np_flag = BATADV_TT_CLIENT_NOPURGE;
1003

1004 1005
	primary_if = batadv_seq_print_text_primary_if_get(seq);
	if (!primary_if)
1006
		goto out;
1007

1008
	seq_printf(seq,
1009
		   "Locally retrieved addresses (from %s) announced via TT (TTVN: %u):\n",
1010
		   net_dev->name, (u8)atomic_read(&bat_priv->tt.vn));
1011
	seq_printf(seq, "       %-13s  %s %-8s %-9s (%-10s)\n", "Client", "VID",
1012
		   "Flags", "Last seen", "CRC");
1013 1014 1015 1016

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];

1017
		rcu_read_lock();
1018
		hlist_for_each_entry_rcu(tt_common_entry,
1019
					 head, hash_entry) {
1020 1021 1022
			tt_local = container_of(tt_common_entry,
						struct batadv_tt_local_entry,
						common);
1023
			vid = tt_common_entry->vid;
1024 1025 1026 1027 1028 1029 1030
			last_seen_jiffies = jiffies - tt_local->last_seen;
			last_seen_msecs = jiffies_to_msecs(last_seen_jiffies);
			last_seen_secs = last_seen_msecs / 1000;
			last_seen_msecs = last_seen_msecs % 1000;

			no_purge = tt_common_entry->flags & np_flag;

1031 1032 1033 1034 1035 1036 1037 1038
			vlan = batadv_softif_vlan_get(bat_priv, vid);
			if (!vlan) {
				seq_printf(seq, "Cannot retrieve VLAN %d\n",
					   BATADV_PRINT_VID(vid));
				continue;
			}

			seq_printf(seq,
1039
				   " * %pM %4i [%c%c%c%c%c%c] %3u.%03u   (%#.8x)\n",
1040
				   tt_common_entry->addr,
1041
				   BATADV_PRINT_VID(tt_common_entry->vid),
1042 1043
				   ((tt_common_entry->flags &
				     BATADV_TT_CLIENT_ROAM) ? 'R' : '.'),
1044
				   no_purge ? 'P' : '.',
1045 1046 1047 1048 1049 1050 1051 1052
				   ((tt_common_entry->flags &
				     BATADV_TT_CLIENT_NEW) ? 'N' : '.'),
				   ((tt_common_entry->flags &
				     BATADV_TT_CLIENT_PENDING) ? 'X' : '.'),
				   ((tt_common_entry->flags &
				     BATADV_TT_CLIENT_WIFI) ? 'W' : '.'),
				   ((tt_common_entry->flags &
				     BATADV_TT_CLIENT_ISOLA) ? 'I' : '.'),
1053
				   no_purge ? 0 : last_seen_secs,
1054 1055 1056
				   no_purge ? 0 : last_seen_msecs,
				   vlan->tt.crc);

1057
			batadv_softif_vlan_put(vlan);
1058
		}
1059
		rcu_read_unlock();
1060
	}
1061 1062
out:
	if (primary_if)
1063
		batadv_hardif_put(primary_if);
1064
	return 0;
1065 1066
}

1067 1068 1069
static void
batadv_tt_local_set_pending(struct batadv_priv *bat_priv,
			    struct batadv_tt_local_entry *tt_local_entry,
1070
			    u16 flags, const char *message)
1071
{
1072
	batadv_tt_local_event(bat_priv, tt_local_entry, flags);
1073

1074 1075
	/* The local client has to be marked as "pending to be removed" but has
	 * to be kept in the table in order to send it in a full table
1076 1077
	 * response issued before the net ttvn increment (consistency check)
	 */
1078
	tt_local_entry->common.flags |= BATADV_TT_CLIENT_PENDING;
1079

1080
	batadv_dbg(BATADV_DBG_TT, bat_priv,
1081 1082 1083
		   "Local tt entry (%pM, vid: %d) pending to be removed: %s\n",
		   tt_local_entry->common.addr,
		   BATADV_PRINT_VID(tt_local_entry->common.vid), message);
1084 1085
}

1086 1087 1088 1089
/**
 * batadv_tt_local_remove - logically remove an entry from the local table
 * @bat_priv: the bat priv with all the soft interface information
 * @addr: the MAC address of the client to remove
1090
 * @vid: VLAN identifier
1091 1092 1093
 * @message: message to append to the log on deletion
 * @roaming: true if the deletion is due to a roaming event
 *
1094
 * Return: the flags assigned to the local entry before being deleted
1095
 */
1096 1097 1098
u16 batadv_tt_local_remove(struct batadv_priv *bat_priv, const u8 *addr,
			   unsigned short vid, const char *message,
			   bool roaming)
1099
{
1100
	struct batadv_tt_local_entry *tt_local_entry;
1101
	u16 flags, curr_flags = BATADV_NO_FLAGS;
1102
	struct batadv_softif_vlan *vlan;
1103
	void *tt_entry_exists;
1104

1105
	tt_local_entry = batadv_tt_local_hash_find(bat_priv, addr, vid);
1106 1107 1108
	if (!tt_local_entry)
		goto out;

1109 1110
	curr_flags = tt_local_entry->common.flags;

1111
	flags = BATADV_TT_CLIENT_DEL;
1112 1113 1114 1115
	/* if this global entry addition is due to a roaming, the node has to
	 * mark the local entry as "roamed" in order to correctly reroute
	 * packets later
	 */
1116
	if (roaming) {
1117
		flags |= BATADV_TT_CLIENT_ROAM;
1118 1119 1120
		/* mark the local client as ROAMed */
		tt_local_entry->common.flags |= BATADV_TT_CLIENT_ROAM;
	}
1121

1122 1123 1124 1125 1126 1127 1128 1129
	if (!(tt_local_entry->common.flags & BATADV_TT_CLIENT_NEW)) {
		batadv_tt_local_set_pending(bat_priv, tt_local_entry, flags,
					    message);
		goto out;
	}
	/* if this client has been added right now, it is possible to
	 * immediately purge it
	 */
1130
	batadv_tt_local_event(bat_priv, tt_local_entry, BATADV_TT_CLIENT_DEL);
1131 1132 1133 1134 1135 1136 1137 1138 1139

	tt_entry_exists = batadv_hash_remove(bat_priv->tt.local_hash,
					     batadv_compare_tt,
					     batadv_choose_tt,
					     &tt_local_entry->common);
	if (!tt_entry_exists)
		goto out;

	/* extra call to free the local tt entry */
1140
	batadv_tt_local_entry_free_ref(tt_local_entry);
1141

1142 1143
	/* decrease the reference held for this vlan */
	vlan = batadv_softif_vlan_get(bat_priv, vid);
1144 1145 1146
	if (!vlan)
		goto out;

1147 1148
	batadv_softif_vlan_put(vlan);
	batadv_softif_vlan_put(vlan);
1149

1150 1151
out:
	if (tt_local_entry)
1152
		batadv_tt_local_entry_free_ref(tt_local_entry);
1153 1154

	return curr_flags;
1155 1156
}

1157 1158 1159 1160 1161 1162 1163
/**
 * batadv_tt_local_purge_list - purge inactive tt local entries
 * @bat_priv: the bat priv with all the soft interface information
 * @head: pointer to the list containing the local tt entries
 * @timeout: parameter deciding whether a given tt local entry is considered
 *  inactive or not
 */
1164
static void batadv_tt_local_purge_list(struct batadv_priv *bat_priv,
1165 1166
				       struct hlist_head *head,
				       int timeout)
1167
{
1168 1169
	struct batadv_tt_local_entry *tt_local_entry;
	struct batadv_tt_common_entry *tt_common_entry;
1170
	struct hlist_node *node_tmp;
1171

1172
	hlist_for_each_entry_safe(tt_common_entry, node_tmp, head,
1173 1174
				  hash_entry) {
		tt_local_entry = container_of(tt_common_entry,
1175 1176
					      struct batadv_tt_local_entry,
					      common);
1177 1178 1179 1180 1181 1182 1183
		if (tt_local_entry->common.flags & BATADV_TT_CLIENT_NOPURGE)
			continue;

		/* entry already marked for deletion */
		if (tt_local_entry->common.flags & BATADV_TT_CLIENT_PENDING)
			continue;

1184
		if (!batadv_has_timed_out(tt_local_entry->last_seen, timeout))
1185 1186 1187 1188 1189 1190 1191
			continue;

		batadv_tt_local_set_pending(bat_priv, tt_local_entry,
					    BATADV_TT_CLIENT_DEL, "timed out");
	}
}

1192 1193 1194 1195 1196 1197 1198 1199
/**
 * batadv_tt_local_purge - purge inactive tt local entries
 * @bat_priv: the bat priv with all the soft interface information
 * @timeout: parameter deciding whether a given tt local entry is considered
 *  inactive or not
 */
static void batadv_tt_local_purge(struct batadv_priv *bat_priv,
				  int timeout)
1200
{
1201
	struct batadv_hashtable *hash = bat_priv->tt.local_hash;
1202
	struct hlist_head *head;
1203
	spinlock_t *list_lock; /* protects write access to the hash lists */
1204
	u32 i;
1205 1206 1207

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];
1208
		list_lock = &hash->list_locks[i];
1209

1210
		spin_lock_bh(list_lock);
1211
		batadv_tt_local_purge_list(bat_priv, head, timeout);
1212
		spin_unlock_bh(list_lock);
1213 1214 1215
	}
}

1216
static void batadv_tt_local_table_free(struct batadv_priv *bat_priv)
1217
{
1218
	struct batadv_hashtable *hash;
1219
	spinlock_t *list_lock; /* protects write access to the hash lists */
1220 1221
	struct batadv_tt_common_entry *tt_common_entry;
	struct batadv_tt_local_entry *tt_local;
1222
	struct batadv_softif_vlan *vlan;
1223
	struct hlist_node *node_tmp;
1224
	struct hlist_head *head;
1225
	u32 i;
1226

1227
	if (!bat_priv->tt.local_hash)
1228 1229
		return;

1230
	hash = bat_priv->tt.local_hash;
1231 1232 1233 1234 1235 1236

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];
		list_lock = &hash->list_locks[i];

		spin_lock_bh(list_lock);
1237
		hlist_for_each_entry_safe(tt_common_entry, node_tmp,
1238
					  head, hash_entry) {
1239
			hlist_del_rcu(&tt_common_entry->hash_entry);
1240 1241 1242
			tt_local = container_of(tt_common_entry,
						struct batadv_tt_local_entry,
						common);
1243 1244 1245 1246

			/* decrease the reference held for this vlan */
			vlan = batadv_softif_vlan_get(bat_priv,
						      tt_common_entry->vid);
1247
			if (vlan) {
1248 1249
				batadv_softif_vlan_put(vlan);
				batadv_softif_vlan_put(vlan);
1250
			}
1251

1252
			batadv_tt_local_entry_free_ref(tt_local);
1253 1254 1255 1256
		}
		spin_unlock_bh(list_lock);
	}

1257
	batadv_hash_destroy(hash);
1258

1259
	bat_priv->tt.local_hash = NULL;
1260 1261
}

1262
static int batadv_tt_global_init(struct batadv_priv *bat_priv)
1263
{
1264
	if (bat_priv->tt.global_hash)
1265
		return 0;
1266

1267
	bat_priv->tt.global_hash = batadv_hash_new(1024);
1268

1269
	if (!bat_priv->tt.global_hash)
1270
		return -ENOMEM;
1271

1272 1273 1274
	batadv_hash_set_lock_class(bat_priv->tt.global_hash,
				   &batadv_tt_global_hash_lock_class_key);

1275
	return 0;
1276 1277
}

1278
static void batadv_tt_changes_list_free(struct batadv_priv *bat_priv)
1279
{
1280
	struct batadv_tt_change_node *entry, *safe;
1281

1282
	spin_lock_bh(&bat_priv->tt.changes_list_lock);
1283

1284
	list_for_each_entry_safe(entry, safe, &bat_priv->tt.changes_list,
1285 1286 1287 1288
				 list) {
		list_del(&entry->list);
		kfree(entry);
	}
1289

1290 1291
	atomic_set(&bat_priv->tt.local_changes, 0);
	spin_unlock_bh(&bat_priv->tt.changes_list_lock);
1292
}
1293

1294
/**
1295 1296 1297 1298
 * batadv_tt_global_orig_entry_find - find a TT orig_list_entry
 * @entry: the TT global entry where the orig_list_entry has to be
 *  extracted from
 * @orig_node: the originator for which the orig_list_entry has to be found
1299
 *
1300
 * retrieve the orig_tt_list_entry belonging to orig_node from the
1301 1302
 * batadv_tt_global_entry list
 *
1303
 * Return: it with an increased refcounter, NULL if not found
1304
 */
1305 1306 1307
static struct batadv_tt_orig_list_entry *
batadv_tt_global_orig_entry_find(const struct batadv_tt_global_entry *entry,
				 const struct batadv_orig_node *orig_node)
1308
{
1309
	struct batadv_tt_orig_list_entry *tmp_orig_entry, *orig_entry = NULL;
1310 1311 1312 1313
	const struct hlist_head *head;

	rcu_read_lock();
	head = &entry->orig_list;
1314
	hlist_for_each_entry_rcu(tmp_orig_entry, head, list) {
1315 1316
		if (tmp_orig_entry->orig_node != orig_node)
			continue;
1317
		if (!kref_get_unless_zero(&tmp_orig_entry->refcount))
1318 1319 1320 1321
			continue;

		orig_entry = tmp_orig_entry;
		break;
1322 1323
	}
	rcu_read_unlock();
1324 1325 1326 1327

	return orig_entry;
}

1328
/**
1329 1330 1331 1332
 * batadv_tt_global_entry_has_orig - check if a TT global entry is also handled
 *  by a given originator
 * @entry: the TT global entry to check
 * @orig_node: the originator to search in the list
1333 1334 1335 1336
 *
 * find out if an orig_node is already in the list of a tt_global_entry.
 *
 * Return: true if found, false otherwise
1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350
 */
static bool
batadv_tt_global_entry_has_orig(const struct batadv_tt_global_entry *entry,
				const struct batadv_orig_node *orig_node)
{
	struct batadv_tt_orig_list_entry *orig_entry;
	bool found = false;

	orig_entry = batadv_tt_global_orig_entry_find(entry, orig_node);
	if (orig_entry) {
		found = true;
		batadv_tt_orig_list_entry_free_ref(orig_entry);
	}

1351 1352 1353
	return found;
}

1354
static void
1355
batadv_tt_global_orig_entry_add(struct batadv_tt_global_entry *tt_global,
1356
				struct batadv_orig_node *orig_node, int ttvn)
1357
{
1358
	struct batadv_tt_orig_list_entry *orig_entry;
1359

1360
	orig_entry = batadv_tt_global_orig_entry_find(tt_global, orig_node);
1361 1362 1363 1364 1365
	if (orig_entry) {
		/* refresh the ttvn: the current value could be a bogus one that
		 * was added during a "temporary client detection"
		 */
		orig_entry->ttvn = ttvn;
1366
		goto out;
1367
	}
1368

1369 1370
	orig_entry = kzalloc(sizeof(*orig_entry), GFP_ATOMIC);
	if (!orig_entry)
1371
		goto out;
1372 1373

	INIT_HLIST_NODE(&orig_entry->list);
1374
	kref_get(&orig_node->refcount);
1375
	batadv_tt_global_size_inc(orig_node, tt_global->common.vid);
1376 1377
	orig_entry->orig_node = orig_node;
	orig_entry->ttvn = ttvn;
1378 1379
	kref_init(&orig_entry->refcount);
	kref_get(&orig_entry->refcount);
1380

1381
	spin_lock_bh(&tt_global->list_lock);
1382
	hlist_add_head_rcu(&orig_entry->list,
1383 1384
			   &tt_global->orig_list);
	spin_unlock_bh(&tt_global->list_lock);
1385 1386
	atomic_inc(&tt_global->orig_list_count);

1387 1388 1389
out:
	if (orig_entry)
		batadv_tt_orig_list_entry_free_ref(orig_entry);
1390 1391
}

1392 1393 1394 1395 1396
/**
 * batadv_tt_global_add - add a new TT global entry or update an existing one
 * @bat_priv: the bat priv with all the soft interface information
 * @orig_node: the originator announcing the client
 * @tt_addr: the mac address of the non-mesh client
1397
 * @vid: VLAN identifier
1398 1399 1400 1401 1402 1403 1404 1405 1406 1407
 * @flags: TT flags that have to be set for this non-mesh client
 * @ttvn: the tt version number ever announcing this non-mesh client
 *
 * Add a new TT global entry for the given originator. If the entry already
 * exists add a new reference to the given originator (a global entry can have
 * references to multiple originators) and adjust the flags attribute to reflect
 * the function argument.
 * If a TT local entry exists for this non-mesh client remove it.
 *
 * The caller must hold orig_node refcount.
1408
 *
1409
 * Return: true if the new entry has been added, false otherwise
1410
 */
1411 1412
static bool batadv_tt_global_add(struct batadv_priv *bat_priv,
				 struct batadv_orig_node *orig_node,
1413
				 const unsigned char *tt_addr,
1414
				 unsigned short vid, u16 flags, u8 ttvn)
1415
{
1416 1417
	struct batadv_tt_global_entry *tt_global_entry;
	struct batadv_tt_local_entry *tt_local_entry;
1418
	bool ret = false;
1419
	int hash_added;
1420
	struct batadv_tt_common_entry *common;
1421
	u16 local_flags;
1422

1423 1424 1425 1426
	/* ignore global entries from backbone nodes */
	if (batadv_bla_is_backbone_gw_orig(bat_priv, orig_node->orig, vid))
		return true;

1427 1428
	tt_global_entry = batadv_tt_global_hash_find(bat_priv, tt_addr, vid);
	tt_local_entry = batadv_tt_local_hash_find(bat_priv, tt_addr, vid);
1429 1430 1431 1432 1433 1434 1435 1436

	/* if the node already has a local client for this entry, it has to wait
	 * for a roaming advertisement instead of manually messing up the global
	 * table
	 */
	if ((flags & BATADV_TT_CLIENT_TEMP) && tt_local_entry &&
	    !(tt_local_entry->common.flags & BATADV_TT_CLIENT_NEW))
		goto out;
1437 1438

	if (!tt_global_entry) {
1439
		tt_global_entry = kzalloc(sizeof(*tt_global_entry), GFP_ATOMIC);
1440
		if (!tt_global_entry)
1441 1442
			goto out;

1443
		common = &tt_global_entry->common;
1444
		ether_addr_copy(common->addr, tt_addr);
1445
		common->vid = vid;
1446

1447
		common->flags = flags;
1448
		tt_global_entry->roam_at = 0;
1449 1450 1451 1452 1453 1454
		/* node must store current time in case of roaming. This is
		 * needed to purge this entry out on timeout (if nobody claims
		 * it)
		 */
		if (flags & BATADV_TT_CLIENT_ROAM)
			tt_global_entry->roam_at = jiffies;
1455 1456
		kref_init(&common->refcount);
		kref_get(&common->refcount);
1457
		common->added_at = jiffies;
1458 1459

		INIT_HLIST_HEAD(&tt_global_entry->orig_list);
1460
		atomic_set(&tt_global_entry->orig_list_count, 0);
1461
		spin_lock_init(&tt_global_entry->list_lock);
1462

1463
		hash_added = batadv_hash_add(bat_priv->tt.global_hash,
1464
					     batadv_compare_tt,
1465
					     batadv_choose_tt, common,
1466
					     &common->hash_entry);
1467 1468 1469

		if (unlikely(hash_added != 0)) {
			/* remove the reference for the hash */
1470
			batadv_tt_global_entry_free_ref(tt_global_entry);
1471 1472
			goto out_remove;
		}
1473
	} else {
1474
		common = &tt_global_entry->common;
1475 1476
		/* If there is already a global entry, we can use this one for
		 * our processing.
1477 1478 1479 1480 1481 1482 1483
		 * But if we are trying to add a temporary client then here are
		 * two options at this point:
		 * 1) the global client is not a temporary client: the global
		 *    client has to be left as it is, temporary information
		 *    should never override any already known client state
		 * 2) the global client is a temporary client: purge the
		 *    originator list and add the new one orig_entry
1484
		 */
1485 1486 1487 1488 1489 1490 1491 1492 1493
		if (flags & BATADV_TT_CLIENT_TEMP) {
			if (!(common->flags & BATADV_TT_CLIENT_TEMP))
				goto out;
			if (batadv_tt_global_entry_has_orig(tt_global_entry,
							    orig_node))
				goto out_remove;
			batadv_tt_global_del_orig_list(tt_global_entry);
			goto add_orig_entry;
		}
1494 1495

		/* if the client was temporary added before receiving the first
1496 1497 1498 1499
		 * OGM announcing it, we have to clear the TEMP flag. Also,
		 * remove the previous temporary orig node and re-add it
		 * if required. If the orig entry changed, the new one which
		 * is a non-temporary entry is preferred.
1500
		 */
1501 1502 1503 1504
		if (common->flags & BATADV_TT_CLIENT_TEMP) {
			batadv_tt_global_del_orig_list(tt_global_entry);
			common->flags &= ~BATADV_TT_CLIENT_TEMP;
		}
1505

1506 1507 1508 1509
		/* the change can carry possible "attribute" flags like the
		 * TT_CLIENT_WIFI, therefore they have to be copied in the
		 * client entry
		 */
1510
		common->flags |= flags;
1511

1512 1513
		/* If there is the BATADV_TT_CLIENT_ROAM flag set, there is only
		 * one originator left in the list and we previously received a
1514 1515 1516 1517 1518
		 * delete + roaming change for this originator.
		 *
		 * We should first delete the old originator before adding the
		 * new one.
		 */
1519
		if (common->flags & BATADV_TT_CLIENT_ROAM) {
1520
			batadv_tt_global_del_orig_list(tt_global_entry);
1521
			common->flags &= ~BATADV_TT_CLIENT_ROAM;
1522
			tt_global_entry->roam_at = 0;
1523 1524
		}
	}
1525
add_orig_entry:
1526
	/* add the new orig_entry (if needed) or update it */
1527
	batadv_tt_global_orig_entry_add(tt_global_entry, orig_node, ttvn);
1528

1529
	batadv_dbg(BATADV_DBG_TT, bat_priv,
1530 1531 1532
		   "Creating new global tt entry: %pM (vid: %d, via %pM)\n",
		   common->addr, BATADV_PRINT_VID(common->vid),
		   orig_node->orig);
1533
	ret = true;
1534

1535
out_remove:
1536 1537 1538 1539 1540
	/* Do not remove multicast addresses from the local hash on
	 * global additions
	 */
	if (is_multicast_ether_addr(tt_addr))
		goto out;
1541

1542
	/* remove address from local hash if present */
1543
	local_flags = batadv_tt_local_remove(bat_priv, tt_addr, vid,
1544
					     "global tt received",
1545
					     flags & BATADV_TT_CLIENT_ROAM);
1546 1547
	tt_global_entry->common.flags |= local_flags & BATADV_TT_CLIENT_WIFI;

1548 1549 1550 1551 1552 1553
	if (!(flags & BATADV_TT_CLIENT_ROAM))
		/* this is a normal global add. Therefore the client is not in a
		 * roaming state anymore.
		 */
		tt_global_entry->common.flags &= ~BATADV_TT_CLIENT_ROAM;

1554 1555
out:
	if (tt_global_entry)
1556
		batadv_tt_global_entry_free_ref(tt_global_entry);
1557 1558
	if (tt_local_entry)
		batadv_tt_local_entry_free_ref(tt_local_entry);
1559
	return ret;
1560 1561
}

1562 1563
/**
 * batadv_transtable_best_orig - Get best originator list entry from tt entry
1564
 * @bat_priv: the bat priv with all the soft interface information
1565 1566 1567
 * @tt_global_entry: global translation table entry to be analyzed
 *
 * This functon assumes the caller holds rcu_read_lock().
1568
 * Return: best originator list entry or NULL on errors.
1569 1570
 */
static struct batadv_tt_orig_list_entry *
1571 1572
batadv_transtable_best_orig(struct batadv_priv *bat_priv,
			    struct batadv_tt_global_entry *tt_global_entry)
1573
{
1574 1575
	struct batadv_neigh_node *router, *best_router = NULL;
	struct batadv_algo_ops *bao = bat_priv->bat_algo_ops;
1576 1577 1578 1579
	struct hlist_head *head;
	struct batadv_tt_orig_list_entry *orig_entry, *best_entry = NULL;

	head = &tt_global_entry->orig_list;
1580
	hlist_for_each_entry_rcu(orig_entry, head, list) {
1581 1582
		router = batadv_orig_router_get(orig_entry->orig_node,
						BATADV_IF_DEFAULT);
1583 1584 1585
		if (!router)
			continue;

1586
		if (best_router &&
1587 1588
		    bao->bat_neigh_cmp(router, BATADV_IF_DEFAULT,
				       best_router, BATADV_IF_DEFAULT) <= 0) {
1589
			batadv_neigh_node_put(router);
1590
			continue;
1591 1592
		}

1593 1594
		/* release the refcount for the "old" best */
		if (best_router)
1595
			batadv_neigh_node_put(best_router);
1596 1597 1598

		best_entry = orig_entry;
		best_router = router;
1599 1600
	}

1601
	if (best_router)
1602
		batadv_neigh_node_put(best_router);
1603

1604 1605 1606
	return best_entry;
}

1607 1608 1609
/**
 * batadv_tt_global_print_entry - print all orig nodes who announce the address
 *  for this global entry
1610
 * @bat_priv: the bat priv with all the soft interface information
1611 1612 1613 1614
 * @tt_global_entry: global translation table entry to be printed
 * @seq: debugfs table seq_file struct
 *
 * This functon assumes the caller holds rcu_read_lock().
1615
 */
1616
static void
1617 1618
batadv_tt_global_print_entry(struct batadv_priv *bat_priv,
			     struct batadv_tt_global_entry *tt_global_entry,
1619
			     struct seq_file *seq)
1620
{
1621
	struct batadv_tt_orig_list_entry *orig_entry, *best_entry;
1622
	struct batadv_tt_common_entry *tt_common_entry;
1623 1624
	struct batadv_orig_node_vlan *vlan;
	struct hlist_head *head;
1625 1626
	u8 last_ttvn;
	u16 flags;
1627 1628

	tt_common_entry = &tt_global_entry->common;
1629 1630
	flags = tt_common_entry->flags;

1631
	best_entry = batadv_transtable_best_orig(bat_priv, tt_global_entry);
1632
	if (best_entry) {
1633 1634 1635 1636 1637 1638 1639 1640 1641 1642
		vlan = batadv_orig_node_vlan_get(best_entry->orig_node,
						 tt_common_entry->vid);
		if (!vlan) {
			seq_printf(seq,
				   " * Cannot retrieve VLAN %d for originator %pM\n",
				   BATADV_PRINT_VID(tt_common_entry->vid),
				   best_entry->orig_node->orig);
			goto print_list;
		}

1643
		last_ttvn = atomic_read(&best_entry->orig_node->last_ttvn);
1644
		seq_printf(seq,
1645
			   " %c %pM %4i   (%3u) via %pM     (%3u)   (%#.8x) [%c%c%c%c]\n",
1646
			   '*', tt_global_entry->common.addr,
1647
			   BATADV_PRINT_VID(tt_global_entry->common.vid),
1648
			   best_entry->ttvn, best_entry->orig_node->orig,
1649
			   last_ttvn, vlan->tt.crc,
1650 1651 1652 1653
			   ((flags & BATADV_TT_CLIENT_ROAM) ? 'R' : '.'),
			   ((flags & BATADV_TT_CLIENT_WIFI) ? 'W' : '.'),
			   ((flags & BATADV_TT_CLIENT_ISOLA) ? 'I' : '.'),
			   ((flags & BATADV_TT_CLIENT_TEMP) ? 'T' : '.'));
1654 1655

		batadv_orig_node_vlan_free_ref(vlan);
1656
	}
1657

1658
print_list:
1659 1660
	head = &tt_global_entry->orig_list;

1661
	hlist_for_each_entry_rcu(orig_entry, head, list) {
1662 1663 1664
		if (best_entry == orig_entry)
			continue;

1665 1666 1667 1668 1669 1670 1671 1672 1673 1674
		vlan = batadv_orig_node_vlan_get(orig_entry->orig_node,
						 tt_common_entry->vid);
		if (!vlan) {
			seq_printf(seq,
				   " + Cannot retrieve VLAN %d for originator %pM\n",
				   BATADV_PRINT_VID(tt_common_entry->vid),
				   orig_entry->orig_node->orig);
			continue;
		}

1675
		last_ttvn = atomic_read(&orig_entry->orig_node->last_ttvn);
1676
		seq_printf(seq,
1677
			   " %c %pM %4d   (%3u) via %pM     (%3u)   (%#.8x) [%c%c%c%c]\n",
1678
			   '+', tt_global_entry->common.addr,
1679
			   BATADV_PRINT_VID(tt_global_entry->common.vid),
1680
			   orig_entry->ttvn, orig_entry->orig_node->orig,
1681
			   last_ttvn, vlan->tt.crc,
1682 1683 1684 1685
			   ((flags & BATADV_TT_CLIENT_ROAM) ? 'R' : '.'),
			   ((flags & BATADV_TT_CLIENT_WIFI) ? 'W' : '.'),
			   ((flags & BATADV_TT_CLIENT_ISOLA) ? 'I' : '.'),
			   ((flags & BATADV_TT_CLIENT_TEMP) ? 'T' : '.'));
1686 1687

		batadv_orig_node_vlan_free_ref(vlan);
1688 1689 1690
	}
}

1691
int batadv_tt_global_seq_print_text(struct seq_file *seq, void *offset)
1692 1693
{
	struct net_device *net_dev = (struct net_device *)seq->private;
1694
	struct batadv_priv *bat_priv = netdev_priv(net_dev);
1695
	struct batadv_hashtable *hash = bat_priv->tt.global_hash;
1696 1697 1698
	struct batadv_tt_common_entry *tt_common_entry;
	struct batadv_tt_global_entry *tt_global;
	struct batadv_hard_iface *primary_if;
1699
	struct hlist_head *head;
1700
	u32 i;
1701

1702 1703
	primary_if = batadv_seq_print_text_primary_if_get(seq);
	if (!primary_if)
1704
		goto out;
1705

1706 1707
	seq_printf(seq,
		   "Globally announced TT entries received via the mesh %s\n",
1708
		   net_dev->name);
1709 1710 1711
	seq_printf(seq, "       %-13s  %s  %s       %-15s %s (%-10s) %s\n",
		   "Client", "VID", "(TTVN)", "Originator", "(Curr TTVN)",
		   "CRC", "Flags");
1712 1713 1714 1715

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];

1716
		rcu_read_lock();
1717
		hlist_for_each_entry_rcu(tt_common_entry,
1718
					 head, hash_entry) {
1719 1720 1721
			tt_global = container_of(tt_common_entry,
						 struct batadv_tt_global_entry,
						 common);
1722
			batadv_tt_global_print_entry(bat_priv, tt_global, seq);
1723
		}
1724
		rcu_read_unlock();
1725
	}
1726 1727
out:
	if (primary_if)
1728
		batadv_hardif_put(primary_if);
1729
	return 0;
1730 1731
}

1732
/**
1733
 * _batadv_tt_global_del_orig_entry - remove and free an orig_entry
1734 1735 1736 1737 1738
 * @tt_global_entry: the global entry to remove the orig_entry from
 * @orig_entry: the orig entry to remove and free
 *
 * Remove an orig_entry from its list in the given tt_global_entry and
 * free this orig_entry afterwards.
1739 1740 1741
 *
 * Caller must hold tt_global_entry->list_lock and ensure orig_entry->list is
 * part of a list.
1742 1743
 */
static void
1744 1745
_batadv_tt_global_del_orig_entry(struct batadv_tt_global_entry *tt_global_entry,
				 struct batadv_tt_orig_list_entry *orig_entry)
1746
{
1747 1748
	lockdep_assert_held(&tt_global_entry->list_lock);

1749 1750 1751
	batadv_tt_global_size_dec(orig_entry->orig_node,
				  tt_global_entry->common.vid);
	atomic_dec(&tt_global_entry->orig_list_count);
1752 1753 1754
	/* requires holding tt_global_entry->list_lock and orig_entry->list
	 * being part of a list
	 */
1755 1756 1757 1758
	hlist_del_rcu(&orig_entry->list);
	batadv_tt_orig_list_entry_free_ref(orig_entry);
}

1759
/* deletes the orig list of a tt_global_entry */
1760
static void
1761
batadv_tt_global_del_orig_list(struct batadv_tt_global_entry *tt_global_entry)
1762
{
1763
	struct hlist_head *head;
1764
	struct hlist_node *safe;
1765
	struct batadv_tt_orig_list_entry *orig_entry;
1766

1767 1768
	spin_lock_bh(&tt_global_entry->list_lock);
	head = &tt_global_entry->orig_list;
1769
	hlist_for_each_entry_safe(orig_entry, safe, head, list)
1770
		_batadv_tt_global_del_orig_entry(tt_global_entry, orig_entry);
1771 1772 1773
	spin_unlock_bh(&tt_global_entry->list_lock);
}

1774 1775 1776 1777 1778 1779 1780 1781 1782 1783
/**
 * batadv_tt_global_del_orig_node - remove orig_node from a global tt entry
 * @bat_priv: the bat priv with all the soft interface information
 * @tt_global_entry: the global entry to remove the orig_node from
 * @orig_node: the originator announcing the client
 * @message: message to append to the log on deletion
 *
 * Remove the given orig_node and its according orig_entry from the given
 * global tt entry.
 */
1784
static void
1785 1786 1787 1788
batadv_tt_global_del_orig_node(struct batadv_priv *bat_priv,
			       struct batadv_tt_global_entry *tt_global_entry,
			       struct batadv_orig_node *orig_node,
			       const char *message)
1789 1790
{
	struct hlist_head *head;
1791
	struct hlist_node *safe;
1792
	struct batadv_tt_orig_list_entry *orig_entry;
1793
	unsigned short vid;
1794 1795 1796

	spin_lock_bh(&tt_global_entry->list_lock);
	head = &tt_global_entry->orig_list;
1797
	hlist_for_each_entry_safe(orig_entry, safe, head, list) {
1798
		if (orig_entry->orig_node == orig_node) {
1799
			vid = tt_global_entry->common.vid;
1800
			batadv_dbg(BATADV_DBG_TT, bat_priv,
1801
				   "Deleting %pM from global tt entry %pM (vid: %d): %s\n",
1802
				   orig_node->orig,
1803 1804
				   tt_global_entry->common.addr,
				   BATADV_PRINT_VID(vid), message);
1805 1806
			_batadv_tt_global_del_orig_entry(tt_global_entry,
							 orig_entry);
1807 1808 1809 1810 1811 1812
		}
	}
	spin_unlock_bh(&tt_global_entry->list_lock);
}

/* If the client is to be deleted, we check if it is the last origantor entry
1813 1814
 * within tt_global entry. If yes, we set the BATADV_TT_CLIENT_ROAM flag and the
 * timer, otherwise we simply remove the originator scheduled for deletion.
1815
 */
1816
static void
1817 1818 1819 1820
batadv_tt_global_del_roaming(struct batadv_priv *bat_priv,
			     struct batadv_tt_global_entry *tt_global_entry,
			     struct batadv_orig_node *orig_node,
			     const char *message)
1821 1822 1823
{
	bool last_entry = true;
	struct hlist_head *head;
1824
	struct batadv_tt_orig_list_entry *orig_entry;
1825 1826 1827 1828 1829 1830 1831

	/* no local entry exists, case 1:
	 * Check if this is the last one or if other entries exist.
	 */

	rcu_read_lock();
	head = &tt_global_entry->orig_list;
1832
	hlist_for_each_entry_rcu(orig_entry, head, list) {
1833 1834 1835 1836 1837 1838 1839 1840 1841
		if (orig_entry->orig_node != orig_node) {
			last_entry = false;
			break;
		}
	}
	rcu_read_unlock();

	if (last_entry) {
		/* its the last one, mark for roaming. */
1842
		tt_global_entry->common.flags |= BATADV_TT_CLIENT_ROAM;
1843 1844 1845 1846 1847
		tt_global_entry->roam_at = jiffies;
	} else
		/* there is another entry, we can simply delete this
		 * one and can still use the other one.
		 */
1848 1849
		batadv_tt_global_del_orig_node(bat_priv, tt_global_entry,
					       orig_node, message);
1850 1851
}

1852 1853 1854 1855 1856 1857 1858 1859 1860 1861
/**
 * batadv_tt_global_del - remove a client from the global table
 * @bat_priv: the bat priv with all the soft interface information
 * @orig_node: an originator serving this client
 * @addr: the mac address of the client
 * @vid: VLAN identifier
 * @message: a message explaining the reason for deleting the client to print
 *  for debugging purpose
 * @roaming: true if the deletion has been triggered by a roaming event
 */
1862 1863
static void batadv_tt_global_del(struct batadv_priv *bat_priv,
				 struct batadv_orig_node *orig_node,
1864
				 const unsigned char *addr, unsigned short vid,
1865
				 const char *message, bool roaming)
1866
{
1867
	struct batadv_tt_global_entry *tt_global_entry;
1868
	struct batadv_tt_local_entry *local_entry = NULL;
1869

1870
	tt_global_entry = batadv_tt_global_hash_find(bat_priv, addr, vid);
1871
	if (!tt_global_entry)
1872
		goto out;
1873

1874
	if (!roaming) {
1875 1876
		batadv_tt_global_del_orig_node(bat_priv, tt_global_entry,
					       orig_node, message);
1877 1878

		if (hlist_empty(&tt_global_entry->orig_list))
1879 1880
			batadv_tt_global_free(bat_priv, tt_global_entry,
					      message);
1881 1882 1883

		goto out;
	}
1884 1885 1886

	/* if we are deleting a global entry due to a roam
	 * event, there are two possibilities:
1887 1888
	 * 1) the client roamed from node A to node B => if there
	 *    is only one originator left for this client, we mark
1889
	 *    it with BATADV_TT_CLIENT_ROAM, we start a timer and we
1890 1891
	 *    wait for node B to claim it. In case of timeout
	 *    the entry is purged.
1892 1893 1894
	 *
	 *    If there are other originators left, we directly delete
	 *    the originator.
1895
	 * 2) the client roamed to us => we can directly delete
1896 1897
	 *    the global entry, since it is useless now.
	 */
1898
	local_entry = batadv_tt_local_hash_find(bat_priv,
1899 1900
						tt_global_entry->common.addr,
						vid);
1901
	if (local_entry) {
1902
		/* local entry exists, case 2: client roamed to us. */
1903
		batadv_tt_global_del_orig_list(tt_global_entry);
1904
		batadv_tt_global_free(bat_priv, tt_global_entry, message);
1905 1906
	} else
		/* no local entry exists, case 1: check for roaming */
1907 1908
		batadv_tt_global_del_roaming(bat_priv, tt_global_entry,
					     orig_node, message);
1909

1910
out:
1911
	if (tt_global_entry)
1912 1913 1914
		batadv_tt_global_entry_free_ref(tt_global_entry);
	if (local_entry)
		batadv_tt_local_entry_free_ref(local_entry);
1915 1916
}

1917 1918 1919 1920 1921 1922 1923 1924 1925
/**
 * batadv_tt_global_del_orig - remove all the TT global entries belonging to the
 *  given originator matching the provided vid
 * @bat_priv: the bat priv with all the soft interface information
 * @orig_node: the originator owning the entries to remove
 * @match_vid: the VLAN identifier to match. If negative all the entries will be
 *  removed
 * @message: debug message to print as "reason"
 */
1926 1927
void batadv_tt_global_del_orig(struct batadv_priv *bat_priv,
			       struct batadv_orig_node *orig_node,
1928
			       s32 match_vid,
1929
			       const char *message)
1930
{
1931 1932
	struct batadv_tt_global_entry *tt_global;
	struct batadv_tt_common_entry *tt_common_entry;
1933
	u32 i;
1934
	struct batadv_hashtable *hash = bat_priv->tt.global_hash;
1935
	struct hlist_node *safe;
1936
	struct hlist_head *head;
1937
	spinlock_t *list_lock; /* protects write access to the hash lists */
1938
	unsigned short vid;
1939

1940 1941 1942
	if (!hash)
		return;

1943 1944
	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];
1945
		list_lock = &hash->list_locks[i];
1946

1947
		spin_lock_bh(list_lock);
1948
		hlist_for_each_entry_safe(tt_common_entry, safe,
1949
					  head, hash_entry) {
1950 1951 1952 1953
			/* remove only matching entries */
			if (match_vid >= 0 && tt_common_entry->vid != match_vid)
				continue;

1954 1955 1956
			tt_global = container_of(tt_common_entry,
						 struct batadv_tt_global_entry,
						 common);
1957

1958 1959
			batadv_tt_global_del_orig_node(bat_priv, tt_global,
						       orig_node, message);
1960

1961
			if (hlist_empty(&tt_global->orig_list)) {
1962
				vid = tt_global->common.vid;
1963
				batadv_dbg(BATADV_DBG_TT, bat_priv,
1964 1965 1966
					   "Deleting global tt entry %pM (vid: %d): %s\n",
					   tt_global->common.addr,
					   BATADV_PRINT_VID(vid), message);
1967
				hlist_del_rcu(&tt_common_entry->hash_entry);
1968
				batadv_tt_global_entry_free_ref(tt_global);
1969
			}
1970
		}
1971
		spin_unlock_bh(list_lock);
1972
	}
1973
	clear_bit(BATADV_ORIG_CAPA_HAS_TT, &orig_node->capa_initialized);
1974 1975
}

1976 1977
static bool batadv_tt_global_to_purge(struct batadv_tt_global_entry *tt_global,
				      char **msg)
1978
{
1979 1980 1981
	bool purge = false;
	unsigned long roam_timeout = BATADV_TT_CLIENT_ROAM_TIMEOUT;
	unsigned long temp_timeout = BATADV_TT_CLIENT_TEMP_TIMEOUT;
1982

1983 1984 1985 1986 1987
	if ((tt_global->common.flags & BATADV_TT_CLIENT_ROAM) &&
	    batadv_has_timed_out(tt_global->roam_at, roam_timeout)) {
		purge = true;
		*msg = "Roaming timeout\n";
	}
1988

1989 1990 1991 1992
	if ((tt_global->common.flags & BATADV_TT_CLIENT_TEMP) &&
	    batadv_has_timed_out(tt_global->common.added_at, temp_timeout)) {
		purge = true;
		*msg = "Temporary client timeout\n";
1993
	}
1994 1995

	return purge;
1996 1997
}

1998
static void batadv_tt_global_purge(struct batadv_priv *bat_priv)
1999
{
2000
	struct batadv_hashtable *hash = bat_priv->tt.global_hash;
2001
	struct hlist_head *head;
2002
	struct hlist_node *node_tmp;
2003
	spinlock_t *list_lock; /* protects write access to the hash lists */
2004
	u32 i;
2005 2006 2007
	char *msg = NULL;
	struct batadv_tt_common_entry *tt_common;
	struct batadv_tt_global_entry *tt_global;
2008 2009 2010

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];
2011
		list_lock = &hash->list_locks[i];
2012

2013
		spin_lock_bh(list_lock);
2014
		hlist_for_each_entry_safe(tt_common, node_tmp, head,
2015 2016 2017 2018 2019 2020 2021 2022 2023
					  hash_entry) {
			tt_global = container_of(tt_common,
						 struct batadv_tt_global_entry,
						 common);

			if (!batadv_tt_global_to_purge(tt_global, &msg))
				continue;

			batadv_dbg(BATADV_DBG_TT, bat_priv,
2024 2025 2026 2027
				   "Deleting global tt entry %pM (vid: %d): %s\n",
				   tt_global->common.addr,
				   BATADV_PRINT_VID(tt_global->common.vid),
				   msg);
2028

2029
			hlist_del_rcu(&tt_common->hash_entry);
2030 2031 2032

			batadv_tt_global_entry_free_ref(tt_global);
		}
2033
		spin_unlock_bh(list_lock);
2034 2035 2036
	}
}

2037
static void batadv_tt_global_table_free(struct batadv_priv *bat_priv)
2038
{
2039
	struct batadv_hashtable *hash;
2040
	spinlock_t *list_lock; /* protects write access to the hash lists */
2041 2042
	struct batadv_tt_common_entry *tt_common_entry;
	struct batadv_tt_global_entry *tt_global;
2043
	struct hlist_node *node_tmp;
2044
	struct hlist_head *head;
2045
	u32 i;
2046

2047
	if (!bat_priv->tt.global_hash)
2048 2049
		return;

2050
	hash = bat_priv->tt.global_hash;
2051 2052 2053 2054 2055 2056

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];
		list_lock = &hash->list_locks[i];

		spin_lock_bh(list_lock);
2057
		hlist_for_each_entry_safe(tt_common_entry, node_tmp,
2058
					  head, hash_entry) {
2059
			hlist_del_rcu(&tt_common_entry->hash_entry);
2060 2061 2062 2063
			tt_global = container_of(tt_common_entry,
						 struct batadv_tt_global_entry,
						 common);
			batadv_tt_global_entry_free_ref(tt_global);
2064 2065 2066 2067
		}
		spin_unlock_bh(list_lock);
	}

2068
	batadv_hash_destroy(hash);
2069

2070
	bat_priv->tt.global_hash = NULL;
2071 2072
}

2073 2074 2075
static bool
_batadv_is_ap_isolated(struct batadv_tt_local_entry *tt_local_entry,
		       struct batadv_tt_global_entry *tt_global_entry)
2076 2077 2078
{
	bool ret = false;

2079 2080
	if (tt_local_entry->common.flags & BATADV_TT_CLIENT_WIFI &&
	    tt_global_entry->common.flags & BATADV_TT_CLIENT_WIFI)
2081 2082
		ret = true;

2083 2084 2085 2086 2087
	/* check if the two clients are marked as isolated */
	if (tt_local_entry->common.flags & BATADV_TT_CLIENT_ISOLA &&
	    tt_global_entry->common.flags & BATADV_TT_CLIENT_ISOLA)
		ret = true;

2088 2089 2090
	return ret;
}

2091 2092 2093 2094 2095 2096 2097
/**
 * batadv_transtable_search - get the mesh destination for a given client
 * @bat_priv: the bat priv with all the soft interface information
 * @src: mac address of the source client
 * @addr: mac address of the destination client
 * @vid: VLAN identifier
 *
2098
 * Return: a pointer to the originator that was selected as destination in the
2099 2100 2101 2102 2103 2104
 * mesh for contacting the client 'addr', NULL otherwise.
 * In case of multiple originators serving the same client, the function returns
 * the best one (best in terms of metric towards the destination node).
 *
 * If the two clients are AP isolated the function returns NULL.
 */
2105
struct batadv_orig_node *batadv_transtable_search(struct batadv_priv *bat_priv,
2106 2107
						  const u8 *src,
						  const u8 *addr,
2108
						  unsigned short vid)
2109
{
2110 2111 2112
	struct batadv_tt_local_entry *tt_local_entry = NULL;
	struct batadv_tt_global_entry *tt_global_entry = NULL;
	struct batadv_orig_node *orig_node = NULL;
2113
	struct batadv_tt_orig_list_entry *best_entry;
2114

2115
	if (src && batadv_vlan_ap_isola_get(bat_priv, vid)) {
2116
		tt_local_entry = batadv_tt_local_hash_find(bat_priv, src, vid);
2117 2118
		if (!tt_local_entry ||
		    (tt_local_entry->common.flags & BATADV_TT_CLIENT_PENDING))
2119 2120
			goto out;
	}
2121

2122
	tt_global_entry = batadv_tt_global_hash_find(bat_priv, addr, vid);
2123
	if (!tt_global_entry)
2124
		goto out;
2125

2126
	/* check whether the clients should not communicate due to AP
2127 2128
	 * isolation
	 */
2129 2130
	if (tt_local_entry &&
	    _batadv_is_ap_isolated(tt_local_entry, tt_global_entry))
2131 2132
		goto out;

2133
	rcu_read_lock();
2134
	best_entry = batadv_transtable_best_orig(bat_priv, tt_global_entry);
2135
	/* found anything? */
2136 2137
	if (best_entry)
		orig_node = best_entry->orig_node;
2138
	if (orig_node && !kref_get_unless_zero(&orig_node->refcount))
2139 2140
		orig_node = NULL;
	rcu_read_unlock();
2141

2142
out:
2143
	if (tt_global_entry)
2144
		batadv_tt_global_entry_free_ref(tt_global_entry);
2145
	if (tt_local_entry)
2146
		batadv_tt_local_entry_free_ref(tt_local_entry);
2147

2148
	return orig_node;
2149
}
2150

2151 2152 2153 2154
/**
 * batadv_tt_global_crc - calculates the checksum of the local table belonging
 *  to the given orig_node
 * @bat_priv: the bat priv with all the soft interface information
2155
 * @orig_node: originator for which the CRC should be computed
2156
 * @vid: VLAN identifier for which the CRC32 has to be computed
2157 2158 2159 2160 2161 2162 2163 2164 2165 2166 2167 2168 2169 2170 2171 2172
 *
 * This function computes the checksum for the global table corresponding to a
 * specific originator. In particular, the checksum is computed as follows: For
 * each client connected to the originator the CRC32C of the MAC address and the
 * VID is computed and then all the CRC32Cs of the various clients are xor'ed
 * together.
 *
 * The idea behind is that CRC32C should be used as much as possible in order to
 * produce a unique hash of the table, but since the order which is used to feed
 * the CRC32C function affects the result and since every node in the network
 * probably sorts the clients differently, the hash function cannot be directly
 * computed over the entire table. Hence the CRC32C is used only on
 * the single client entry, while all the results are then xor'ed together
 * because the XOR operation can combine them all while trying to reduce the
 * noise as much as possible.
 *
2173
 * Return: the checksum of the global table of a given originator.
2174
 */
2175 2176 2177
static u32 batadv_tt_global_crc(struct batadv_priv *bat_priv,
				struct batadv_orig_node *orig_node,
				unsigned short vid)
2178
{
2179
	struct batadv_hashtable *hash = bat_priv->tt.global_hash;
2180 2181
	struct batadv_tt_common_entry *tt_common;
	struct batadv_tt_global_entry *tt_global;
2182
	struct hlist_head *head;
2183 2184
	u32 i, crc_tmp, crc = 0;
	u8 flags;
2185
	__be16 tmp_vid;
2186 2187 2188 2189 2190

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];

		rcu_read_lock();
2191
		hlist_for_each_entry_rcu(tt_common, head, hash_entry) {
2192 2193 2194
			tt_global = container_of(tt_common,
						 struct batadv_tt_global_entry,
						 common);
2195 2196 2197 2198 2199 2200
			/* compute the CRC only for entries belonging to the
			 * VLAN identified by the vid passed as parameter
			 */
			if (tt_common->vid != vid)
				continue;

2201 2202 2203 2204 2205
			/* Roaming clients are in the global table for
			 * consistency only. They don't have to be
			 * taken into account while computing the
			 * global crc
			 */
2206
			if (tt_common->flags & BATADV_TT_CLIENT_ROAM)
2207
				continue;
2208 2209 2210 2211 2212 2213
			/* Temporary clients have not been announced yet, so
			 * they have to be skipped while computing the global
			 * crc
			 */
			if (tt_common->flags & BATADV_TT_CLIENT_TEMP)
				continue;
2214 2215 2216 2217

			/* find out if this global entry is announced by this
			 * originator
			 */
2218
			if (!batadv_tt_global_entry_has_orig(tt_global,
2219
							     orig_node))
2220 2221
				continue;

2222 2223 2224 2225 2226
			/* use network order to read the VID: this ensures that
			 * every node reads the bytes in the same order.
			 */
			tmp_vid = htons(tt_common->vid);
			crc_tmp = crc32c(0, &tmp_vid, sizeof(tmp_vid));
2227 2228 2229 2230 2231 2232 2233

			/* compute the CRC on flags that have to be kept in sync
			 * among nodes
			 */
			flags = tt_common->flags & BATADV_TT_SYNC_MASK;
			crc_tmp = crc32c(crc_tmp, &flags, sizeof(flags));

2234
			crc ^= crc32c(crc_tmp, tt_common->addr, ETH_ALEN);
2235 2236 2237 2238
		}
		rcu_read_unlock();
	}

2239
	return crc;
2240 2241
}

2242 2243 2244
/**
 * batadv_tt_local_crc - calculates the checksum of the local table
 * @bat_priv: the bat priv with all the soft interface information
2245
 * @vid: VLAN identifier for which the CRC32 has to be computed
2246 2247 2248 2249
 *
 * For details about the computation, please refer to the documentation for
 * batadv_tt_global_crc().
 *
2250
 * Return: the checksum of the local table
2251
 */
2252 2253
static u32 batadv_tt_local_crc(struct batadv_priv *bat_priv,
			       unsigned short vid)
2254
{
2255
	struct batadv_hashtable *hash = bat_priv->tt.local_hash;
2256
	struct batadv_tt_common_entry *tt_common;
2257
	struct hlist_head *head;
2258 2259
	u32 i, crc_tmp, crc = 0;
	u8 flags;
2260
	__be16 tmp_vid;
2261 2262 2263 2264 2265

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];

		rcu_read_lock();
2266
		hlist_for_each_entry_rcu(tt_common, head, hash_entry) {
2267 2268 2269 2270 2271 2272
			/* compute the CRC only for entries belonging to the
			 * VLAN identified by vid
			 */
			if (tt_common->vid != vid)
				continue;

2273
			/* not yet committed clients have not to be taken into
2274 2275
			 * account while computing the CRC
			 */
2276
			if (tt_common->flags & BATADV_TT_CLIENT_NEW)
2277
				continue;
2278

2279 2280 2281 2282 2283
			/* use network order to read the VID: this ensures that
			 * every node reads the bytes in the same order.
			 */
			tmp_vid = htons(tt_common->vid);
			crc_tmp = crc32c(0, &tmp_vid, sizeof(tmp_vid));
2284 2285 2286 2287 2288 2289 2290

			/* compute the CRC on flags that have to be kept in sync
			 * among nodes
			 */
			flags = tt_common->flags & BATADV_TT_SYNC_MASK;
			crc_tmp = crc32c(crc_tmp, &flags, sizeof(flags));

2291
			crc ^= crc32c(crc_tmp, tt_common->addr, ETH_ALEN);
2292 2293 2294 2295
		}
		rcu_read_unlock();
	}

2296
	return crc;
2297 2298
}

2299
static void batadv_tt_req_list_free(struct batadv_priv *bat_priv)
2300
{
2301 2302
	struct batadv_tt_req_node *node;
	struct hlist_node *safe;
2303

2304
	spin_lock_bh(&bat_priv->tt.req_list_lock);
2305

2306 2307
	hlist_for_each_entry_safe(node, safe, &bat_priv->tt.req_list, list) {
		hlist_del_init(&node->list);
2308 2309 2310
		kfree(node);
	}

2311
	spin_unlock_bh(&bat_priv->tt.req_list_lock);
2312 2313
}

2314 2315
static void batadv_tt_save_orig_buffer(struct batadv_priv *bat_priv,
				       struct batadv_orig_node *orig_node,
2316
				       const void *tt_buff,
2317
				       u16 tt_buff_len)
2318 2319
{
	/* Replace the old buffer only if I received something in the
2320 2321
	 * last OGM (the OGM could carry no changes)
	 */
2322 2323 2324 2325 2326 2327 2328 2329 2330 2331 2332 2333 2334
	spin_lock_bh(&orig_node->tt_buff_lock);
	if (tt_buff_len > 0) {
		kfree(orig_node->tt_buff);
		orig_node->tt_buff_len = 0;
		orig_node->tt_buff = kmalloc(tt_buff_len, GFP_ATOMIC);
		if (orig_node->tt_buff) {
			memcpy(orig_node->tt_buff, tt_buff, tt_buff_len);
			orig_node->tt_buff_len = tt_buff_len;
		}
	}
	spin_unlock_bh(&orig_node->tt_buff_lock);
}

2335
static void batadv_tt_req_purge(struct batadv_priv *bat_priv)
2336
{
2337 2338
	struct batadv_tt_req_node *node;
	struct hlist_node *safe;
2339

2340
	spin_lock_bh(&bat_priv->tt.req_list_lock);
2341
	hlist_for_each_entry_safe(node, safe, &bat_priv->tt.req_list, list) {
2342 2343
		if (batadv_has_timed_out(node->issued_at,
					 BATADV_TT_REQUEST_TIMEOUT)) {
2344
			hlist_del_init(&node->list);
2345 2346 2347
			kfree(node);
		}
	}
2348
	spin_unlock_bh(&bat_priv->tt.req_list_lock);
2349 2350
}

2351 2352 2353 2354 2355
/**
 * batadv_tt_req_node_new - search and possibly create a tt_req_node object
 * @bat_priv: the bat priv with all the soft interface information
 * @orig_node: orig node this request is being issued for
 *
2356
 * Return: the pointer to the new tt_req_node struct if no request
2357
 * has already been issued for this orig_node, NULL otherwise.
2358
 */
2359
static struct batadv_tt_req_node *
2360
batadv_tt_req_node_new(struct batadv_priv *bat_priv,
2361
		       struct batadv_orig_node *orig_node)
2362
{
2363
	struct batadv_tt_req_node *tt_req_node_tmp, *tt_req_node = NULL;
2364

2365
	spin_lock_bh(&bat_priv->tt.req_list_lock);
2366
	hlist_for_each_entry(tt_req_node_tmp, &bat_priv->tt.req_list, list) {
2367 2368
		if (batadv_compare_eth(tt_req_node_tmp, orig_node) &&
		    !batadv_has_timed_out(tt_req_node_tmp->issued_at,
2369
					  BATADV_TT_REQUEST_TIMEOUT))
2370 2371 2372 2373 2374 2375 2376
			goto unlock;
	}

	tt_req_node = kmalloc(sizeof(*tt_req_node), GFP_ATOMIC);
	if (!tt_req_node)
		goto unlock;

2377
	ether_addr_copy(tt_req_node->addr, orig_node->orig);
2378 2379
	tt_req_node->issued_at = jiffies;

2380
	hlist_add_head(&tt_req_node->list, &bat_priv->tt.req_list);
2381
unlock:
2382
	spin_unlock_bh(&bat_priv->tt.req_list_lock);
2383 2384 2385
	return tt_req_node;
}

2386 2387 2388 2389 2390
/**
 * batadv_tt_local_valid - verify that given tt entry is a valid one
 * @entry_ptr: to be checked local tt entry
 * @data_ptr: not used but definition required to satisfy the callback prototype
 *
2391
 * Return: 1 if the entry is a valid, 0 otherwise.
2392 2393
 */
static int batadv_tt_local_valid(const void *entry_ptr, const void *data_ptr)
2394
{
2395
	const struct batadv_tt_common_entry *tt_common_entry = entry_ptr;
2396

2397
	if (tt_common_entry->flags & BATADV_TT_CLIENT_NEW)
2398 2399 2400 2401
		return 0;
	return 1;
}

2402 2403
static int batadv_tt_global_valid(const void *entry_ptr,
				  const void *data_ptr)
2404
{
2405 2406 2407
	const struct batadv_tt_common_entry *tt_common_entry = entry_ptr;
	const struct batadv_tt_global_entry *tt_global_entry;
	const struct batadv_orig_node *orig_node = data_ptr;
2408

2409 2410
	if (tt_common_entry->flags & BATADV_TT_CLIENT_ROAM ||
	    tt_common_entry->flags & BATADV_TT_CLIENT_TEMP)
2411 2412
		return 0;

2413 2414
	tt_global_entry = container_of(tt_common_entry,
				       struct batadv_tt_global_entry,
2415 2416
				       common);

2417
	return batadv_tt_global_entry_has_orig(tt_global_entry, orig_node);
2418 2419
}

2420
/**
2421 2422
 * batadv_tt_tvlv_generate - fill the tvlv buff with the tt entries from the
 *  specified tt hash
2423 2424 2425
 * @bat_priv: the bat priv with all the soft interface information
 * @hash: hash table containing the tt entries
 * @tt_len: expected tvlv tt data buffer length in number of bytes
2426
 * @tvlv_buff: pointer to the buffer to fill with the TT data
2427 2428 2429
 * @valid_cb: function to filter tt change entries
 * @cb_data: data passed to the filter function as argument
 */
2430 2431
static void batadv_tt_tvlv_generate(struct batadv_priv *bat_priv,
				    struct batadv_hashtable *hash,
2432
				    void *tvlv_buff, u16 tt_len,
2433 2434
				    int (*valid_cb)(const void *, const void *),
				    void *cb_data)
2435
{
2436
	struct batadv_tt_common_entry *tt_common_entry;
2437
	struct batadv_tvlv_tt_change *tt_change;
2438
	struct hlist_head *head;
2439 2440
	u16 tt_tot, tt_num_entries = 0;
	u32 i;
2441

2442
	tt_tot = batadv_tt_entries(tt_len);
2443
	tt_change = (struct batadv_tvlv_tt_change *)tvlv_buff;
2444 2445 2446 2447 2448

	rcu_read_lock();
	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];

2449
		hlist_for_each_entry_rcu(tt_common_entry,
2450
					 head, hash_entry) {
2451
			if (tt_tot == tt_num_entries)
2452 2453
				break;

2454
			if ((valid_cb) && (!valid_cb(tt_common_entry, cb_data)))
2455 2456
				continue;

2457
			ether_addr_copy(tt_change->addr, tt_common_entry->addr);
2458
			tt_change->flags = tt_common_entry->flags;
2459
			tt_change->vid = htons(tt_common_entry->vid);
2460 2461
			memset(tt_change->reserved, 0,
			       sizeof(tt_change->reserved));
2462

2463
			tt_num_entries++;
2464 2465 2466 2467
			tt_change++;
		}
	}
	rcu_read_unlock();
2468
}
2469

2470 2471 2472 2473 2474 2475
/**
 * batadv_tt_global_check_crc - check if all the CRCs are correct
 * @orig_node: originator for which the CRCs have to be checked
 * @tt_vlan: pointer to the first tvlv VLAN entry
 * @num_vlan: number of tvlv VLAN entries
 *
2476
 * Return: true if all the received CRCs match the locally stored ones, false
2477 2478 2479 2480
 * otherwise
 */
static bool batadv_tt_global_check_crc(struct batadv_orig_node *orig_node,
				       struct batadv_tvlv_tt_vlan_data *tt_vlan,
2481
				       u16 num_vlan)
2482 2483 2484
{
	struct batadv_tvlv_tt_vlan_data *tt_vlan_tmp;
	struct batadv_orig_node_vlan *vlan;
2485
	int i, orig_num_vlan;
2486
	u32 crc;
2487 2488 2489 2490 2491 2492 2493 2494 2495

	/* check if each received CRC matches the locally stored one */
	for (i = 0; i < num_vlan; i++) {
		tt_vlan_tmp = tt_vlan + i;

		/* if orig_node is a backbone node for this VLAN, don't check
		 * the CRC as we ignore all the global entries over it
		 */
		if (batadv_bla_is_backbone_gw_orig(orig_node->bat_priv,
2496 2497
						   orig_node->orig,
						   ntohs(tt_vlan_tmp->vid)))
2498 2499 2500 2501 2502 2503 2504
			continue;

		vlan = batadv_orig_node_vlan_get(orig_node,
						 ntohs(tt_vlan_tmp->vid));
		if (!vlan)
			return false;

2505 2506 2507 2508
		crc = vlan->tt.crc;
		batadv_orig_node_vlan_free_ref(vlan);

		if (crc != ntohl(tt_vlan_tmp->crc))
2509 2510 2511
			return false;
	}

2512 2513 2514 2515 2516 2517 2518 2519 2520 2521 2522 2523
	/* check if any excess VLANs exist locally for the originator
	 * which are not mentioned in the TVLV from the originator.
	 */
	rcu_read_lock();
	orig_num_vlan = 0;
	hlist_for_each_entry_rcu(vlan, &orig_node->vlan_list, list)
		orig_num_vlan++;
	rcu_read_unlock();

	if (orig_num_vlan > num_vlan)
		return false;

2524 2525 2526 2527 2528 2529 2530 2531 2532 2533 2534 2535 2536 2537 2538 2539 2540 2541 2542 2543 2544 2545 2546 2547 2548 2549 2550 2551
	return true;
}

/**
 * batadv_tt_local_update_crc - update all the local CRCs
 * @bat_priv: the bat priv with all the soft interface information
 */
static void batadv_tt_local_update_crc(struct batadv_priv *bat_priv)
{
	struct batadv_softif_vlan *vlan;

	/* recompute the global CRC for each VLAN */
	rcu_read_lock();
	hlist_for_each_entry_rcu(vlan, &bat_priv->softif_vlan_list, list) {
		vlan->tt.crc = batadv_tt_local_crc(bat_priv, vlan->vid);
	}
	rcu_read_unlock();
}

/**
 * batadv_tt_global_update_crc - update all the global CRCs for this orig_node
 * @bat_priv: the bat priv with all the soft interface information
 * @orig_node: the orig_node for which the CRCs have to be updated
 */
static void batadv_tt_global_update_crc(struct batadv_priv *bat_priv,
					struct batadv_orig_node *orig_node)
{
	struct batadv_orig_node_vlan *vlan;
2552
	u32 crc;
2553 2554 2555

	/* recompute the global CRC for each VLAN */
	rcu_read_lock();
2556
	hlist_for_each_entry_rcu(vlan, &orig_node->vlan_list, list) {
2557 2558 2559
		/* if orig_node is a backbone node for this VLAN, don't compute
		 * the CRC as we ignore all the global entries over it
		 */
2560 2561
		if (batadv_bla_is_backbone_gw_orig(bat_priv, orig_node->orig,
						   vlan->vid))
2562 2563 2564 2565 2566 2567
			continue;

		crc = batadv_tt_global_crc(bat_priv, orig_node, vlan->vid);
		vlan->tt.crc = crc;
	}
	rcu_read_unlock();
2568 2569
}

2570 2571 2572 2573 2574
/**
 * batadv_send_tt_request - send a TT Request message to a given node
 * @bat_priv: the bat priv with all the soft interface information
 * @dst_orig_node: the destination of the message
 * @ttvn: the version number that the source of the message is looking for
2575 2576
 * @tt_vlan: pointer to the first tvlv VLAN object to request
 * @num_vlan: number of tvlv VLAN entries
2577 2578
 * @full_table: ask for the entire translation table if true, while only for the
 *  last TT diff otherwise
2579 2580
 *
 * Return: true if the TT Request was sent, false otherwise
2581
 */
2582 2583
static int batadv_send_tt_request(struct batadv_priv *bat_priv,
				  struct batadv_orig_node *dst_orig_node,
2584
				  u8 ttvn,
2585
				  struct batadv_tvlv_tt_vlan_data *tt_vlan,
2586
				  u16 num_vlan, bool full_table)
2587
{
2588
	struct batadv_tvlv_tt_data *tvlv_tt_data = NULL;
2589
	struct batadv_tt_req_node *tt_req_node = NULL;
2590 2591
	struct batadv_tvlv_tt_vlan_data *tt_vlan_req;
	struct batadv_hard_iface *primary_if;
2592
	bool ret = false;
2593
	int i, size;
2594

2595
	primary_if = batadv_primary_if_get_selected(bat_priv);
2596 2597 2598 2599
	if (!primary_if)
		goto out;

	/* The new tt_req will be issued only if I'm not waiting for a
2600 2601
	 * reply from the same orig_node yet
	 */
2602
	tt_req_node = batadv_tt_req_node_new(bat_priv, dst_orig_node);
2603 2604 2605
	if (!tt_req_node)
		goto out;

2606 2607
	size = sizeof(*tvlv_tt_data) + sizeof(*tt_vlan_req) * num_vlan;
	tvlv_tt_data = kzalloc(size, GFP_ATOMIC);
2608
	if (!tvlv_tt_data)
2609 2610
		goto out;

2611 2612
	tvlv_tt_data->flags = BATADV_TT_REQUEST;
	tvlv_tt_data->ttvn = ttvn;
2613 2614 2615 2616 2617 2618 2619 2620 2621 2622 2623 2624 2625
	tvlv_tt_data->num_vlan = htons(num_vlan);

	/* send all the CRCs within the request. This is needed by intermediate
	 * nodes to ensure they have the correct table before replying
	 */
	tt_vlan_req = (struct batadv_tvlv_tt_vlan_data *)(tvlv_tt_data + 1);
	for (i = 0; i < num_vlan; i++) {
		tt_vlan_req->vid = tt_vlan->vid;
		tt_vlan_req->crc = tt_vlan->crc;

		tt_vlan_req++;
		tt_vlan++;
	}
2626 2627

	if (full_table)
2628
		tvlv_tt_data->flags |= BATADV_TT_FULL_TABLE;
2629

2630
	batadv_dbg(BATADV_DBG_TT, bat_priv, "Sending TT_REQUEST to %pM [%c]\n",
2631
		   dst_orig_node->orig, full_table ? 'F' : '.');
2632

2633
	batadv_inc_counter(bat_priv, BATADV_CNT_TT_REQUEST_TX);
2634 2635
	batadv_tvlv_unicast_send(bat_priv, primary_if->net_dev->dev_addr,
				 dst_orig_node->orig, BATADV_TVLV_TT, 1,
2636
				 tvlv_tt_data, size);
2637
	ret = true;
2638 2639 2640

out:
	if (primary_if)
2641
		batadv_hardif_put(primary_if);
2642
	if (ret && tt_req_node) {
2643
		spin_lock_bh(&bat_priv->tt.req_list_lock);
2644 2645
		/* hlist_del_init() verifies tt_req_node still is in the list */
		hlist_del_init(&tt_req_node->list);
2646
		spin_unlock_bh(&bat_priv->tt.req_list_lock);
2647 2648
		kfree(tt_req_node);
	}
2649
	kfree(tvlv_tt_data);
2650 2651 2652
	return ret;
}

2653 2654 2655 2656 2657 2658 2659 2660
/**
 * batadv_send_other_tt_response - send reply to tt request concerning another
 *  node's translation table
 * @bat_priv: the bat priv with all the soft interface information
 * @tt_data: tt data containing the tt request information
 * @req_src: mac address of tt request sender
 * @req_dst: mac address of tt request recipient
 *
2661
 * Return: true if tt request reply was sent, false otherwise.
2662 2663 2664
 */
static bool batadv_send_other_tt_response(struct batadv_priv *bat_priv,
					  struct batadv_tvlv_tt_data *tt_data,
2665
					  u8 *req_src, u8 *req_dst)
2666
{
2667
	struct batadv_orig_node *req_dst_orig_node;
2668
	struct batadv_orig_node *res_dst_orig_node = NULL;
2669
	struct batadv_tvlv_tt_change *tt_change;
2670
	struct batadv_tvlv_tt_data *tvlv_tt_data = NULL;
2671
	struct batadv_tvlv_tt_vlan_data *tt_vlan;
2672
	bool ret = false, full_table;
2673 2674 2675
	u8 orig_ttvn, req_ttvn;
	u16 tvlv_len;
	s32 tt_len;
2676

2677
	batadv_dbg(BATADV_DBG_TT, bat_priv,
2678
		   "Received TT_REQUEST from %pM for ttvn: %u (%pM) [%c]\n",
2679
		   req_src, tt_data->ttvn, req_dst,
2680
		   ((tt_data->flags & BATADV_TT_FULL_TABLE) ? 'F' : '.'));
2681 2682

	/* Let's get the orig node of the REAL destination */
2683
	req_dst_orig_node = batadv_orig_hash_find(bat_priv, req_dst);
2684 2685 2686
	if (!req_dst_orig_node)
		goto out;

2687
	res_dst_orig_node = batadv_orig_hash_find(bat_priv, req_src);
2688 2689 2690
	if (!res_dst_orig_node)
		goto out;

2691
	orig_ttvn = (u8)atomic_read(&req_dst_orig_node->last_ttvn);
2692
	req_ttvn = tt_data->ttvn;
2693

2694
	tt_vlan = (struct batadv_tvlv_tt_vlan_data *)(tt_data + 1);
2695
	/* this node doesn't have the requested data */
2696
	if (orig_ttvn != req_ttvn ||
2697 2698
	    !batadv_tt_global_check_crc(req_dst_orig_node, tt_vlan,
					ntohs(tt_data->num_vlan)))
2699 2700
		goto out;

2701
	/* If the full table has been explicitly requested */
2702
	if (tt_data->flags & BATADV_TT_FULL_TABLE ||
2703 2704 2705 2706 2707
	    !req_dst_orig_node->tt_buff)
		full_table = true;
	else
		full_table = false;

2708 2709
	/* TT fragmentation hasn't been implemented yet, so send as many
	 * TT entries fit a single packet as possible only
2710
	 */
2711 2712 2713 2714
	if (!full_table) {
		spin_lock_bh(&req_dst_orig_node->tt_buff_lock);
		tt_len = req_dst_orig_node->tt_buff_len;

2715 2716 2717 2718 2719
		tvlv_len = batadv_tt_prepare_tvlv_global_data(req_dst_orig_node,
							      &tvlv_tt_data,
							      &tt_change,
							      &tt_len);
		if (!tt_len)
2720 2721 2722
			goto unlock;

		/* Copy the last orig_node's OGM buffer */
2723
		memcpy(tt_change, req_dst_orig_node->tt_buff,
2724 2725 2726
		       req_dst_orig_node->tt_buff_len);
		spin_unlock_bh(&req_dst_orig_node->tt_buff_lock);
	} else {
2727 2728 2729 2730 2731 2732 2733 2734 2735
		/* allocate the tvlv, put the tt_data and all the tt_vlan_data
		 * in the initial part
		 */
		tt_len = -1;
		tvlv_len = batadv_tt_prepare_tvlv_global_data(req_dst_orig_node,
							      &tvlv_tt_data,
							      &tt_change,
							      &tt_len);
		if (!tt_len)
2736
			goto out;
2737 2738 2739 2740 2741 2742

		/* fill the rest of the tvlv with the real TT entries */
		batadv_tt_tvlv_generate(bat_priv, bat_priv->tt.global_hash,
					tt_change, tt_len,
					batadv_tt_global_valid,
					req_dst_orig_node);
2743 2744
	}

2745 2746 2747 2748 2749 2750 2751 2752 2753
	/* Don't send the response, if larger than fragmented packet. */
	tt_len = sizeof(struct batadv_unicast_tvlv_packet) + tvlv_len;
	if (tt_len > atomic_read(&bat_priv->packet_size_max)) {
		net_ratelimited_function(batadv_info, bat_priv->soft_iface,
					 "Ignoring TT_REQUEST from %pM; Response size exceeds max packet size.\n",
					 res_dst_orig_node->orig);
		goto out;
	}

2754 2755
	tvlv_tt_data->flags = BATADV_TT_RESPONSE;
	tvlv_tt_data->ttvn = req_ttvn;
2756 2757

	if (full_table)
2758
		tvlv_tt_data->flags |= BATADV_TT_FULL_TABLE;
2759

2760
	batadv_dbg(BATADV_DBG_TT, bat_priv,
2761 2762 2763
		   "Sending TT_RESPONSE %pM for %pM [%c] (ttvn: %u)\n",
		   res_dst_orig_node->orig, req_dst_orig_node->orig,
		   full_table ? 'F' : '.', req_ttvn);
2764

2765
	batadv_inc_counter(bat_priv, BATADV_CNT_TT_RESPONSE_TX);
2766

2767
	batadv_tvlv_unicast_send(bat_priv, req_dst_orig_node->orig,
2768 2769
				 req_src, BATADV_TVLV_TT, 1, tvlv_tt_data,
				 tvlv_len);
2770

2771
	ret = true;
2772 2773 2774 2775 2776 2777 2778
	goto out;

unlock:
	spin_unlock_bh(&req_dst_orig_node->tt_buff_lock);

out:
	if (res_dst_orig_node)
2779
		batadv_orig_node_put(res_dst_orig_node);
2780
	if (req_dst_orig_node)
2781
		batadv_orig_node_put(req_dst_orig_node);
2782
	kfree(tvlv_tt_data);
2783 2784
	return ret;
}
2785

2786 2787 2788 2789 2790 2791 2792
/**
 * batadv_send_my_tt_response - send reply to tt request concerning this node's
 *  translation table
 * @bat_priv: the bat priv with all the soft interface information
 * @tt_data: tt data containing the tt request information
 * @req_src: mac address of tt request sender
 *
2793
 * Return: true if tt request reply was sent, false otherwise.
2794 2795 2796
 */
static bool batadv_send_my_tt_response(struct batadv_priv *bat_priv,
				       struct batadv_tvlv_tt_data *tt_data,
2797
				       u8 *req_src)
2798
{
2799
	struct batadv_tvlv_tt_data *tvlv_tt_data = NULL;
2800
	struct batadv_hard_iface *primary_if = NULL;
2801 2802
	struct batadv_tvlv_tt_change *tt_change;
	struct batadv_orig_node *orig_node;
2803 2804
	u8 my_ttvn, req_ttvn;
	u16 tvlv_len;
2805
	bool full_table;
2806
	s32 tt_len;
2807

2808
	batadv_dbg(BATADV_DBG_TT, bat_priv,
2809
		   "Received TT_REQUEST from %pM for ttvn: %u (me) [%c]\n",
2810
		   req_src, tt_data->ttvn,
2811
		   ((tt_data->flags & BATADV_TT_FULL_TABLE) ? 'F' : '.'));
2812

2813
	spin_lock_bh(&bat_priv->tt.commit_lock);
2814

2815
	my_ttvn = (u8)atomic_read(&bat_priv->tt.vn);
2816
	req_ttvn = tt_data->ttvn;
2817

2818
	orig_node = batadv_orig_hash_find(bat_priv, req_src);
2819 2820 2821
	if (!orig_node)
		goto out;

2822
	primary_if = batadv_primary_if_get_selected(bat_priv);
2823 2824 2825 2826
	if (!primary_if)
		goto out;

	/* If the full table has been explicitly requested or the gap
2827 2828
	 * is too big send the whole local translation table
	 */
2829
	if (tt_data->flags & BATADV_TT_FULL_TABLE || my_ttvn != req_ttvn ||
2830
	    !bat_priv->tt.last_changeset)
2831 2832 2833 2834
		full_table = true;
	else
		full_table = false;

2835 2836
	/* TT fragmentation hasn't been implemented yet, so send as many
	 * TT entries fit a single packet as possible only
2837
	 */
2838
	if (!full_table) {
2839
		spin_lock_bh(&bat_priv->tt.last_changeset_lock);
2840

2841 2842 2843 2844 2845 2846
		tt_len = bat_priv->tt.last_changeset_len;
		tvlv_len = batadv_tt_prepare_tvlv_local_data(bat_priv,
							     &tvlv_tt_data,
							     &tt_change,
							     &tt_len);
		if (!tt_len)
2847 2848
			goto unlock;

2849
		/* Copy the last orig_node's OGM buffer */
2850
		memcpy(tt_change, bat_priv->tt.last_changeset,
2851 2852
		       bat_priv->tt.last_changeset_len);
		spin_unlock_bh(&bat_priv->tt.last_changeset_lock);
2853
	} else {
2854
		req_ttvn = (u8)atomic_read(&bat_priv->tt.vn);
2855

2856 2857 2858 2859 2860 2861 2862 2863 2864
		/* allocate the tvlv, put the tt_data and all the tt_vlan_data
		 * in the initial part
		 */
		tt_len = -1;
		tvlv_len = batadv_tt_prepare_tvlv_local_data(bat_priv,
							     &tvlv_tt_data,
							     &tt_change,
							     &tt_len);
		if (!tt_len)
2865
			goto out;
2866 2867 2868 2869 2870

		/* fill the rest of the tvlv with the real TT entries */
		batadv_tt_tvlv_generate(bat_priv, bat_priv->tt.local_hash,
					tt_change, tt_len,
					batadv_tt_local_valid, NULL);
2871 2872
	}

2873 2874
	tvlv_tt_data->flags = BATADV_TT_RESPONSE;
	tvlv_tt_data->ttvn = req_ttvn;
2875 2876

	if (full_table)
2877
		tvlv_tt_data->flags |= BATADV_TT_FULL_TABLE;
2878

2879
	batadv_dbg(BATADV_DBG_TT, bat_priv,
2880 2881
		   "Sending TT_RESPONSE to %pM [%c] (ttvn: %u)\n",
		   orig_node->orig, full_table ? 'F' : '.', req_ttvn);
2882

2883
	batadv_inc_counter(bat_priv, BATADV_CNT_TT_RESPONSE_TX);
2884

2885
	batadv_tvlv_unicast_send(bat_priv, primary_if->net_dev->dev_addr,
2886 2887
				 req_src, BATADV_TVLV_TT, 1, tvlv_tt_data,
				 tvlv_len);
2888

2889 2890 2891
	goto out;

unlock:
2892
	spin_unlock_bh(&bat_priv->tt.last_changeset_lock);
2893
out:
2894
	spin_unlock_bh(&bat_priv->tt.commit_lock);
2895
	if (orig_node)
2896
		batadv_orig_node_put(orig_node);
2897
	if (primary_if)
2898
		batadv_hardif_put(primary_if);
2899 2900
	kfree(tvlv_tt_data);
	/* The packet was for this host, so it doesn't need to be re-routed */
2901 2902 2903
	return true;
}

2904 2905 2906 2907 2908 2909 2910
/**
 * batadv_send_tt_response - send reply to tt request
 * @bat_priv: the bat priv with all the soft interface information
 * @tt_data: tt data containing the tt request information
 * @req_src: mac address of tt request sender
 * @req_dst: mac address of tt request recipient
 *
2911
 * Return: true if tt request reply was sent, false otherwise.
2912 2913 2914
 */
static bool batadv_send_tt_response(struct batadv_priv *bat_priv,
				    struct batadv_tvlv_tt_data *tt_data,
2915
				    u8 *req_src, u8 *req_dst)
2916
{
2917
	if (batadv_is_my_mac(bat_priv, req_dst))
2918
		return batadv_send_my_tt_response(bat_priv, tt_data, req_src);
2919 2920
	return batadv_send_other_tt_response(bat_priv, tt_data, req_src,
					     req_dst);
2921 2922
}

2923 2924
static void _batadv_tt_update_changes(struct batadv_priv *bat_priv,
				      struct batadv_orig_node *orig_node,
2925
				      struct batadv_tvlv_tt_change *tt_change,
2926
				      u16 tt_num_changes, u8 ttvn)
2927 2928
{
	int i;
2929
	int roams;
2930 2931

	for (i = 0; i < tt_num_changes; i++) {
2932 2933
		if ((tt_change + i)->flags & BATADV_TT_CLIENT_DEL) {
			roams = (tt_change + i)->flags & BATADV_TT_CLIENT_ROAM;
2934 2935
			batadv_tt_global_del(bat_priv, orig_node,
					     (tt_change + i)->addr,
2936
					     ntohs((tt_change + i)->vid),
2937 2938
					     "tt removed by changes",
					     roams);
2939 2940
		} else {
			if (!batadv_tt_global_add(bat_priv, orig_node,
2941
						  (tt_change + i)->addr,
2942
						  ntohs((tt_change + i)->vid),
2943
						  (tt_change + i)->flags, ttvn))
2944 2945 2946 2947 2948 2949 2950
				/* In case of problem while storing a
				 * global_entry, we stop the updating
				 * procedure without committing the
				 * ttvn change. This will avoid to send
				 * corrupted data on tt_request
				 */
				return;
2951
		}
2952
	}
2953
	set_bit(BATADV_ORIG_CAPA_HAS_TT, &orig_node->capa_initialized);
2954 2955
}

2956
static void batadv_tt_fill_gtable(struct batadv_priv *bat_priv,
2957
				  struct batadv_tvlv_tt_change *tt_change,
2958 2959
				  u8 ttvn, u8 *resp_src,
				  u16 num_entries)
2960
{
2961
	struct batadv_orig_node *orig_node;
2962

2963
	orig_node = batadv_orig_hash_find(bat_priv, resp_src);
2964 2965 2966 2967
	if (!orig_node)
		goto out;

	/* Purge the old table first.. */
2968 2969
	batadv_tt_global_del_orig(bat_priv, orig_node, -1,
				  "Received full table");
2970

2971 2972
	_batadv_tt_update_changes(bat_priv, orig_node, tt_change, num_entries,
				  ttvn);
2973 2974 2975 2976 2977 2978 2979

	spin_lock_bh(&orig_node->tt_buff_lock);
	kfree(orig_node->tt_buff);
	orig_node->tt_buff_len = 0;
	orig_node->tt_buff = NULL;
	spin_unlock_bh(&orig_node->tt_buff_lock);

2980
	atomic_set(&orig_node->last_ttvn, ttvn);
2981 2982 2983

out:
	if (orig_node)
2984
		batadv_orig_node_put(orig_node);
2985 2986
}

2987 2988
static void batadv_tt_update_changes(struct batadv_priv *bat_priv,
				     struct batadv_orig_node *orig_node,
2989
				     u16 tt_num_changes, u8 ttvn,
2990
				     struct batadv_tvlv_tt_change *tt_change)
2991
{
2992 2993
	_batadv_tt_update_changes(bat_priv, orig_node, tt_change,
				  tt_num_changes, ttvn);
2994

2995 2996
	batadv_tt_save_orig_buffer(bat_priv, orig_node, tt_change,
				   batadv_tt_len(tt_num_changes));
2997 2998 2999
	atomic_set(&orig_node->last_ttvn, ttvn);
}

3000 3001 3002
/**
 * batadv_is_my_client - check if a client is served by the local node
 * @bat_priv: the bat priv with all the soft interface information
3003
 * @addr: the mac address of the client to check
3004 3005
 * @vid: VLAN identifier
 *
3006
 * Return: true if the client is served by this node, false otherwise.
3007
 */
3008
bool batadv_is_my_client(struct batadv_priv *bat_priv, const u8 *addr,
3009
			 unsigned short vid)
3010
{
3011
	struct batadv_tt_local_entry *tt_local_entry;
3012
	bool ret = false;
3013

3014
	tt_local_entry = batadv_tt_local_hash_find(bat_priv, addr, vid);
3015 3016
	if (!tt_local_entry)
		goto out;
3017
	/* Check if the client has been logically deleted (but is kept for
3018 3019
	 * consistency purpose)
	 */
3020 3021
	if ((tt_local_entry->common.flags & BATADV_TT_CLIENT_PENDING) ||
	    (tt_local_entry->common.flags & BATADV_TT_CLIENT_ROAM))
3022
		goto out;
3023 3024
	ret = true;
out:
3025
	if (tt_local_entry)
3026
		batadv_tt_local_entry_free_ref(tt_local_entry);
3027
	return ret;
3028 3029
}

3030 3031 3032 3033 3034 3035 3036 3037 3038
/**
 * batadv_handle_tt_response - process incoming tt reply
 * @bat_priv: the bat priv with all the soft interface information
 * @tt_data: tt data containing the tt request information
 * @resp_src: mac address of tt reply sender
 * @num_entries: number of tt change entries appended to the tt data
 */
static void batadv_handle_tt_response(struct batadv_priv *bat_priv,
				      struct batadv_tvlv_tt_data *tt_data,
3039
				      u8 *resp_src, u16 num_entries)
3040
{
3041 3042
	struct batadv_tt_req_node *node;
	struct hlist_node *safe;
3043
	struct batadv_orig_node *orig_node = NULL;
3044
	struct batadv_tvlv_tt_change *tt_change;
3045 3046
	u8 *tvlv_ptr = (u8 *)tt_data;
	u16 change_offset;
3047

3048
	batadv_dbg(BATADV_DBG_TT, bat_priv,
3049
		   "Received TT_RESPONSE from %pM for ttvn %d t_size: %d [%c]\n",
3050
		   resp_src, tt_data->ttvn, num_entries,
3051
		   ((tt_data->flags & BATADV_TT_FULL_TABLE) ? 'F' : '.'));
3052

3053
	orig_node = batadv_orig_hash_find(bat_priv, resp_src);
3054 3055 3056
	if (!orig_node)
		goto out;

3057 3058
	spin_lock_bh(&orig_node->tt_lock);

3059 3060 3061 3062 3063 3064
	change_offset = sizeof(struct batadv_tvlv_tt_vlan_data);
	change_offset *= ntohs(tt_data->num_vlan);
	change_offset += sizeof(*tt_data);
	tvlv_ptr += change_offset;

	tt_change = (struct batadv_tvlv_tt_change *)tvlv_ptr;
3065
	if (tt_data->flags & BATADV_TT_FULL_TABLE) {
3066 3067
		batadv_tt_fill_gtable(bat_priv, tt_change, tt_data->ttvn,
				      resp_src, num_entries);
3068
	} else {
3069 3070
		batadv_tt_update_changes(bat_priv, orig_node, num_entries,
					 tt_data->ttvn, tt_change);
3071
	}
3072

3073
	/* Recalculate the CRC for this orig_node and store it */
3074
	batadv_tt_global_update_crc(bat_priv, orig_node);
3075 3076 3077

	spin_unlock_bh(&orig_node->tt_lock);

3078
	/* Delete the tt_req_node from pending tt_requests list */
3079
	spin_lock_bh(&bat_priv->tt.req_list_lock);
3080
	hlist_for_each_entry_safe(node, safe, &bat_priv->tt.req_list, list) {
3081
		if (!batadv_compare_eth(node->addr, resp_src))
3082
			continue;
3083
		hlist_del_init(&node->list);
3084 3085
		kfree(node);
	}
3086

3087
	spin_unlock_bh(&bat_priv->tt.req_list_lock);
3088 3089
out:
	if (orig_node)
3090
		batadv_orig_node_put(orig_node);
3091 3092
}

3093
static void batadv_tt_roam_list_free(struct batadv_priv *bat_priv)
3094
{
3095
	struct batadv_tt_roam_node *node, *safe;
3096

3097
	spin_lock_bh(&bat_priv->tt.roam_list_lock);
3098

3099
	list_for_each_entry_safe(node, safe, &bat_priv->tt.roam_list, list) {
3100 3101 3102 3103
		list_del(&node->list);
		kfree(node);
	}

3104
	spin_unlock_bh(&bat_priv->tt.roam_list_lock);
3105 3106
}

3107
static void batadv_tt_roam_purge(struct batadv_priv *bat_priv)
3108
{
3109
	struct batadv_tt_roam_node *node, *safe;
3110

3111 3112
	spin_lock_bh(&bat_priv->tt.roam_list_lock);
	list_for_each_entry_safe(node, safe, &bat_priv->tt.roam_list, list) {
3113 3114
		if (!batadv_has_timed_out(node->first_time,
					  BATADV_ROAMING_MAX_TIME))
3115 3116 3117 3118 3119
			continue;

		list_del(&node->list);
		kfree(node);
	}
3120
	spin_unlock_bh(&bat_priv->tt.roam_list_lock);
3121 3122
}

3123
/**
3124 3125 3126
 * batadv_tt_check_roam_count - check if a client has roamed too frequently
 * @bat_priv: the bat priv with all the soft interface information
 * @client: mac address of the roaming client
3127 3128
 *
 * This function checks whether the client already reached the
3129 3130 3131
 * maximum number of possible roaming phases. In this case the ROAMING_ADV
 * will not be sent.
 *
3132
 * Return: true if the ROAMING_ADV can be sent, false otherwise
3133
 */
3134
static bool batadv_tt_check_roam_count(struct batadv_priv *bat_priv, u8 *client)
3135
{
3136
	struct batadv_tt_roam_node *tt_roam_node;
3137 3138
	bool ret = false;

3139
	spin_lock_bh(&bat_priv->tt.roam_list_lock);
3140
	/* The new tt_req will be issued only if I'm not waiting for a
3141 3142
	 * reply from the same orig_node yet
	 */
3143
	list_for_each_entry(tt_roam_node, &bat_priv->tt.roam_list, list) {
3144
		if (!batadv_compare_eth(tt_roam_node->addr, client))
3145 3146
			continue;

3147
		if (batadv_has_timed_out(tt_roam_node->first_time,
3148
					 BATADV_ROAMING_MAX_TIME))
3149 3150
			continue;

3151
		if (!batadv_atomic_dec_not_zero(&tt_roam_node->counter))
3152 3153 3154 3155 3156 3157 3158 3159 3160 3161 3162 3163
			/* Sorry, you roamed too many times! */
			goto unlock;
		ret = true;
		break;
	}

	if (!ret) {
		tt_roam_node = kmalloc(sizeof(*tt_roam_node), GFP_ATOMIC);
		if (!tt_roam_node)
			goto unlock;

		tt_roam_node->first_time = jiffies;
3164 3165
		atomic_set(&tt_roam_node->counter,
			   BATADV_ROAMING_MAX_COUNT - 1);
3166
		ether_addr_copy(tt_roam_node->addr, client);
3167

3168
		list_add(&tt_roam_node->list, &bat_priv->tt.roam_list);
3169 3170 3171 3172
		ret = true;
	}

unlock:
3173
	spin_unlock_bh(&bat_priv->tt.roam_list_lock);
3174 3175 3176
	return ret;
}

3177 3178 3179 3180 3181 3182 3183 3184 3185 3186 3187 3188
/**
 * batadv_send_roam_adv - send a roaming advertisement message
 * @bat_priv: the bat priv with all the soft interface information
 * @client: mac address of the roaming client
 * @vid: VLAN identifier
 * @orig_node: message destination
 *
 * Send a ROAMING_ADV message to the node which was previously serving this
 * client. This is done to inform the node that from now on all traffic destined
 * for this particular roamed client has to be forwarded to the sender of the
 * roaming message.
 */
3189
static void batadv_send_roam_adv(struct batadv_priv *bat_priv, u8 *client,
3190
				 unsigned short vid,
3191
				 struct batadv_orig_node *orig_node)
3192
{
3193
	struct batadv_hard_iface *primary_if;
3194 3195 3196 3197 3198
	struct batadv_tvlv_roam_adv tvlv_roam;

	primary_if = batadv_primary_if_get_selected(bat_priv);
	if (!primary_if)
		goto out;
3199 3200

	/* before going on we have to check whether the client has
3201 3202
	 * already roamed to us too many times
	 */
3203
	if (!batadv_tt_check_roam_count(bat_priv, client))
3204 3205
		goto out;

3206
	batadv_dbg(BATADV_DBG_TT, bat_priv,
3207 3208
		   "Sending ROAMING_ADV to %pM (client %pM, vid: %d)\n",
		   orig_node->orig, client, BATADV_PRINT_VID(vid));
3209

3210
	batadv_inc_counter(bat_priv, BATADV_CNT_TT_ROAM_ADV_TX);
3211

3212
	memcpy(tvlv_roam.client, client, sizeof(tvlv_roam.client));
3213
	tvlv_roam.vid = htons(vid);
3214 3215 3216 3217

	batadv_tvlv_unicast_send(bat_priv, primary_if->net_dev->dev_addr,
				 orig_node->orig, BATADV_TVLV_ROAM, 1,
				 &tvlv_roam, sizeof(tvlv_roam));
3218 3219

out:
3220
	if (primary_if)
3221
		batadv_hardif_put(primary_if);
3222 3223
}

3224
static void batadv_tt_purge(struct work_struct *work)
3225
{
3226
	struct delayed_work *delayed_work;
3227
	struct batadv_priv_tt *priv_tt;
3228 3229 3230
	struct batadv_priv *bat_priv;

	delayed_work = container_of(work, struct delayed_work, work);
3231 3232
	priv_tt = container_of(delayed_work, struct batadv_priv_tt, work);
	bat_priv = container_of(priv_tt, struct batadv_priv, tt);
3233

3234
	batadv_tt_local_purge(bat_priv, BATADV_TT_LOCAL_TIMEOUT);
3235
	batadv_tt_global_purge(bat_priv);
3236 3237
	batadv_tt_req_purge(bat_priv);
	batadv_tt_roam_purge(bat_priv);
3238

3239 3240
	queue_delayed_work(batadv_event_workqueue, &bat_priv->tt.work,
			   msecs_to_jiffies(BATADV_TT_WORK_PERIOD));
3241
}
3242

3243
void batadv_tt_free(struct batadv_priv *bat_priv)
3244
{
3245 3246 3247
	batadv_tvlv_container_unregister(bat_priv, BATADV_TVLV_TT, 1);
	batadv_tvlv_handler_unregister(bat_priv, BATADV_TVLV_TT, 1);

3248
	cancel_delayed_work_sync(&bat_priv->tt.work);
3249

3250 3251 3252 3253 3254
	batadv_tt_local_table_free(bat_priv);
	batadv_tt_global_table_free(bat_priv);
	batadv_tt_req_list_free(bat_priv);
	batadv_tt_changes_list_free(bat_priv);
	batadv_tt_roam_list_free(bat_priv);
3255

3256
	kfree(bat_priv->tt.last_changeset);
3257
}
3258

3259 3260 3261 3262 3263 3264 3265
/**
 * batadv_tt_local_set_flags - set or unset the specified flags on the local
 *  table and possibly count them in the TT size
 * @bat_priv: the bat priv with all the soft interface information
 * @flags: the flag to switch
 * @enable: whether to set or unset the flag
 * @count: whether to increase the TT size by the number of changed entries
3266
 */
3267 3268
static void batadv_tt_local_set_flags(struct batadv_priv *bat_priv, u16 flags,
				      bool enable, bool count)
3269
{
3270 3271
	struct batadv_hashtable *hash = bat_priv->tt.local_hash;
	struct batadv_tt_common_entry *tt_common_entry;
3272
	u16 changed_num = 0;
3273
	struct hlist_head *head;
3274
	u32 i;
3275 3276

	if (!hash)
3277
		return;
3278 3279 3280 3281 3282

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];

		rcu_read_lock();
3283
		hlist_for_each_entry_rcu(tt_common_entry,
3284
					 head, hash_entry) {
3285 3286 3287 3288 3289 3290 3291 3292 3293 3294
			if (enable) {
				if ((tt_common_entry->flags & flags) == flags)
					continue;
				tt_common_entry->flags |= flags;
			} else {
				if (!(tt_common_entry->flags & flags))
					continue;
				tt_common_entry->flags &= ~flags;
			}
			changed_num++;
3295 3296 3297 3298 3299 3300

			if (!count)
				continue;

			batadv_tt_local_size_inc(bat_priv,
						 tt_common_entry->vid);
3301 3302 3303 3304 3305
		}
		rcu_read_unlock();
	}
}

3306
/* Purge out all the tt local entries marked with BATADV_TT_CLIENT_PENDING */
3307
static void batadv_tt_local_purge_pending_clients(struct batadv_priv *bat_priv)
3308
{
3309
	struct batadv_hashtable *hash = bat_priv->tt.local_hash;
3310 3311
	struct batadv_tt_common_entry *tt_common;
	struct batadv_tt_local_entry *tt_local;
3312
	struct batadv_softif_vlan *vlan;
3313
	struct hlist_node *node_tmp;
3314 3315
	struct hlist_head *head;
	spinlock_t *list_lock; /* protects write access to the hash lists */
3316
	u32 i;
3317 3318 3319 3320 3321 3322 3323 3324 3325

	if (!hash)
		return;

	for (i = 0; i < hash->size; i++) {
		head = &hash->table[i];
		list_lock = &hash->list_locks[i];

		spin_lock_bh(list_lock);
3326
		hlist_for_each_entry_safe(tt_common, node_tmp, head,
3327 3328
					  hash_entry) {
			if (!(tt_common->flags & BATADV_TT_CLIENT_PENDING))
3329 3330
				continue;

3331
			batadv_dbg(BATADV_DBG_TT, bat_priv,
3332 3333 3334
				   "Deleting local tt entry (%pM, vid: %d): pending\n",
				   tt_common->addr,
				   BATADV_PRINT_VID(tt_common->vid));
3335

3336
			batadv_tt_local_size_dec(bat_priv, tt_common->vid);
3337
			hlist_del_rcu(&tt_common->hash_entry);
3338 3339 3340
			tt_local = container_of(tt_common,
						struct batadv_tt_local_entry,
						common);
3341 3342 3343

			/* decrease the reference held for this vlan */
			vlan = batadv_softif_vlan_get(bat_priv, tt_common->vid);
3344
			if (vlan) {
3345 3346
				batadv_softif_vlan_put(vlan);
				batadv_softif_vlan_put(vlan);
3347
			}
3348

3349
			batadv_tt_local_entry_free_ref(tt_local);
3350 3351 3352 3353 3354
		}
		spin_unlock_bh(list_lock);
	}
}

3355
/**
3356 3357
 * batadv_tt_local_commit_changes_nolock - commit all pending local tt changes
 *  which have been queued in the time since the last commit
3358
 * @bat_priv: the bat priv with all the soft interface information
3359 3360
 *
 * Caller must hold tt->commit_lock.
3361
 */
3362
static void batadv_tt_local_commit_changes_nolock(struct batadv_priv *bat_priv)
3363
{
3364 3365
	lockdep_assert_held(&bat_priv->tt.commit_lock);

3366 3367 3368
	/* Update multicast addresses in local translation table */
	batadv_mcast_mla_update(bat_priv);

3369 3370 3371
	if (atomic_read(&bat_priv->tt.local_changes) < 1) {
		if (!batadv_atomic_dec_not_zero(&bat_priv->tt.ogm_append_cnt))
			batadv_tt_tvlv_container_update(bat_priv);
3372
		return;
3373
	}
3374

3375
	batadv_tt_local_set_flags(bat_priv, BATADV_TT_CLIENT_NEW, false, true);
3376

3377
	batadv_tt_local_purge_pending_clients(bat_priv);
3378
	batadv_tt_local_update_crc(bat_priv);
3379 3380

	/* Increment the TTVN only once per OGM interval */
3381
	atomic_inc(&bat_priv->tt.vn);
3382
	batadv_dbg(BATADV_DBG_TT, bat_priv,
3383
		   "Local changes committed, updating to ttvn %u\n",
3384
		   (u8)atomic_read(&bat_priv->tt.vn));
3385 3386

	/* reset the sending counter */
3387
	atomic_set(&bat_priv->tt.ogm_append_cnt, BATADV_TT_OGM_APPEND_MAX);
3388
	batadv_tt_tvlv_container_update(bat_priv);
3389
}
3390

3391 3392 3393 3394 3395 3396 3397 3398 3399
/**
 * batadv_tt_local_commit_changes - commit all pending local tt changes which
 *  have been queued in the time since the last commit
 * @bat_priv: the bat priv with all the soft interface information
 */
void batadv_tt_local_commit_changes(struct batadv_priv *bat_priv)
{
	spin_lock_bh(&bat_priv->tt.commit_lock);
	batadv_tt_local_commit_changes_nolock(bat_priv);
3400
	spin_unlock_bh(&bat_priv->tt.commit_lock);
3401
}
3402

3403 3404
bool batadv_is_ap_isolated(struct batadv_priv *bat_priv, u8 *src, u8 *dst,
			   unsigned short vid)
3405
{
3406 3407
	struct batadv_tt_local_entry *tt_local_entry = NULL;
	struct batadv_tt_global_entry *tt_global_entry = NULL;
3408
	struct batadv_softif_vlan *vlan;
3409
	bool ret = false;
3410

3411
	vlan = batadv_softif_vlan_get(bat_priv, vid);
3412 3413 3414 3415
	if (!vlan)
		return false;

	if (!atomic_read(&vlan->ap_isolation))
3416
		goto out;
3417

3418
	tt_local_entry = batadv_tt_local_hash_find(bat_priv, dst, vid);
3419 3420 3421
	if (!tt_local_entry)
		goto out;

3422
	tt_global_entry = batadv_tt_global_hash_find(bat_priv, src, vid);
3423 3424 3425
	if (!tt_global_entry)
		goto out;

3426
	if (!_batadv_is_ap_isolated(tt_local_entry, tt_global_entry))
3427 3428
		goto out;

3429
	ret = true;
3430 3431

out:
3432
	batadv_softif_vlan_put(vlan);
3433
	if (tt_global_entry)
3434
		batadv_tt_global_entry_free_ref(tt_global_entry);
3435
	if (tt_local_entry)
3436
		batadv_tt_local_entry_free_ref(tt_local_entry);
3437 3438
	return ret;
}
3439

3440 3441 3442 3443
/**
 * batadv_tt_update_orig - update global translation table with new tt
 *  information received via ogms
 * @bat_priv: the bat priv with all the soft interface information
3444 3445
 * @orig_node: the orig_node of the ogm
 * @tt_buff: pointer to the first tvlv VLAN entry
3446 3447
 * @tt_num_vlan: number of tvlv VLAN entries
 * @tt_change: pointer to the first entry in the TT buffer
3448 3449 3450 3451 3452
 * @tt_num_changes: number of tt changes inside the tt buffer
 * @ttvn: translation table version number of this changeset
 */
static void batadv_tt_update_orig(struct batadv_priv *bat_priv,
				  struct batadv_orig_node *orig_node,
3453
				  const void *tt_buff, u16 tt_num_vlan,
3454
				  struct batadv_tvlv_tt_change *tt_change,
3455
				  u16 tt_num_changes, u8 ttvn)
3456
{
3457
	u8 orig_ttvn = (u8)atomic_read(&orig_node->last_ttvn);
3458
	struct batadv_tvlv_tt_vlan_data *tt_vlan;
3459
	bool full_table = true;
3460
	bool has_tt_init;
3461

3462
	tt_vlan = (struct batadv_tvlv_tt_vlan_data *)tt_buff;
3463 3464
	has_tt_init = test_bit(BATADV_ORIG_CAPA_HAS_TT,
			       &orig_node->capa_initialized);
3465

3466
	/* orig table not initialised AND first diff is in the OGM OR the ttvn
3467 3468
	 * increased by one -> we can apply the attached changes
	 */
3469
	if ((!has_tt_init && ttvn == 1) || ttvn - orig_ttvn == 1) {
3470
		/* the OGM could not contain the changes due to their size or
3471 3472
		 * because they have already been sent BATADV_TT_OGM_APPEND_MAX
		 * times.
3473 3474
		 * In this case send a tt request
		 */
3475 3476 3477 3478 3479
		if (!tt_num_changes) {
			full_table = false;
			goto request_table;
		}

3480 3481
		spin_lock_bh(&orig_node->tt_lock);

3482
		batadv_tt_update_changes(bat_priv, orig_node, tt_num_changes,
3483
					 ttvn, tt_change);
3484 3485 3486

		/* Even if we received the precomputed crc with the OGM, we
		 * prefer to recompute it to spot any possible inconsistency
3487 3488
		 * in the global table
		 */
3489
		batadv_tt_global_update_crc(bat_priv, orig_node);
3490

3491 3492
		spin_unlock_bh(&orig_node->tt_lock);

3493 3494 3495 3496 3497 3498 3499
		/* The ttvn alone is not enough to guarantee consistency
		 * because a single value could represent different states
		 * (due to the wrap around). Thus a node has to check whether
		 * the resulting table (after applying the changes) is still
		 * consistent or not. E.g. a node could disconnect while its
		 * ttvn is X and reconnect on ttvn = X + TTVN_MAX: in this case
		 * checking the CRC value is mandatory to detect the
3500 3501
		 * inconsistency
		 */
3502 3503
		if (!batadv_tt_global_check_crc(orig_node, tt_vlan,
						tt_num_vlan))
3504 3505 3506
			goto request_table;
	} else {
		/* if we missed more than one change or our tables are not
3507 3508
		 * in sync anymore -> request fresh tt data
		 */
3509
		if (!has_tt_init || ttvn != orig_ttvn ||
3510 3511
		    !batadv_tt_global_check_crc(orig_node, tt_vlan,
						tt_num_vlan)) {
3512
request_table:
3513
			batadv_dbg(BATADV_DBG_TT, bat_priv,
3514 3515 3516
				   "TT inconsistency for %pM. Need to retrieve the correct information (ttvn: %u last_ttvn: %u num_changes: %u)\n",
				   orig_node->orig, ttvn, orig_ttvn,
				   tt_num_changes);
3517
			batadv_send_tt_request(bat_priv, orig_node, ttvn,
3518 3519
					       tt_vlan, tt_num_vlan,
					       full_table);
3520 3521 3522 3523
			return;
		}
	}
}
3524

3525 3526 3527 3528 3529 3530
/**
 * batadv_tt_global_client_is_roaming - check if a client is marked as roaming
 * @bat_priv: the bat priv with all the soft interface information
 * @addr: the mac address of the client to check
 * @vid: VLAN identifier
 *
3531
 * Return: true if we know that the client has moved from its old originator
3532 3533
 * to another one. This entry is still kept for consistency purposes and will be
 * deleted later by a DEL or because of timeout
3534
 */
3535
bool batadv_tt_global_client_is_roaming(struct batadv_priv *bat_priv,
3536
					u8 *addr, unsigned short vid)
3537
{
3538
	struct batadv_tt_global_entry *tt_global_entry;
3539 3540
	bool ret = false;

3541
	tt_global_entry = batadv_tt_global_hash_find(bat_priv, addr, vid);
3542 3543 3544
	if (!tt_global_entry)
		goto out;

3545
	ret = tt_global_entry->common.flags & BATADV_TT_CLIENT_ROAM;
3546
	batadv_tt_global_entry_free_ref(tt_global_entry);
3547 3548 3549
out:
	return ret;
}
3550

3551 3552 3553
/**
 * batadv_tt_local_client_is_roaming - tells whether the client is roaming
 * @bat_priv: the bat priv with all the soft interface information
3554 3555
 * @addr: the mac address of the local client to query
 * @vid: VLAN identifier
3556
 *
3557
 * Return: true if the local client is known to be roaming (it is not served by
3558 3559 3560 3561
 * this node anymore) or not. If yes, the client is still present in the table
 * to keep the latter consistent with the node TTVN
 */
bool batadv_tt_local_client_is_roaming(struct batadv_priv *bat_priv,
3562
				       u8 *addr, unsigned short vid)
3563 3564 3565 3566
{
	struct batadv_tt_local_entry *tt_local_entry;
	bool ret = false;

3567
	tt_local_entry = batadv_tt_local_hash_find(bat_priv, addr, vid);
3568 3569 3570 3571 3572 3573 3574 3575 3576
	if (!tt_local_entry)
		goto out;

	ret = tt_local_entry->common.flags & BATADV_TT_CLIENT_ROAM;
	batadv_tt_local_entry_free_ref(tt_local_entry);
out:
	return ret;
}

3577 3578
bool batadv_tt_add_temporary_global_entry(struct batadv_priv *bat_priv,
					  struct batadv_orig_node *orig_node,
3579
					  const unsigned char *addr,
3580
					  unsigned short vid)
3581 3582 3583
{
	bool ret = false;

3584
	if (!batadv_tt_global_add(bat_priv, orig_node, addr, vid,
3585 3586 3587 3588 3589
				  BATADV_TT_CLIENT_TEMP,
				  atomic_read(&orig_node->last_ttvn)))
		goto out;

	batadv_dbg(BATADV_DBG_TT, bat_priv,
3590 3591
		   "Added temporary global client (addr: %pM, vid: %d, orig: %pM)\n",
		   addr, BATADV_PRINT_VID(vid), orig_node->orig);
3592 3593 3594 3595
	ret = true;
out:
	return ret;
}
3596

3597 3598 3599 3600 3601 3602 3603 3604 3605 3606 3607 3608 3609 3610 3611 3612 3613 3614 3615 3616 3617 3618 3619 3620 3621 3622 3623 3624 3625 3626 3627 3628 3629 3630 3631 3632 3633 3634 3635 3636 3637
/**
 * batadv_tt_local_resize_to_mtu - resize the local translation table fit the
 *  maximum packet size that can be transported through the mesh
 * @soft_iface: netdev struct of the mesh interface
 *
 * Remove entries older than 'timeout' and half timeout if more entries need
 * to be removed.
 */
void batadv_tt_local_resize_to_mtu(struct net_device *soft_iface)
{
	struct batadv_priv *bat_priv = netdev_priv(soft_iface);
	int packet_size_max = atomic_read(&bat_priv->packet_size_max);
	int table_size, timeout = BATADV_TT_LOCAL_TIMEOUT / 2;
	bool reduced = false;

	spin_lock_bh(&bat_priv->tt.commit_lock);

	while (true) {
		table_size = batadv_tt_local_table_transmit_size(bat_priv);
		if (packet_size_max >= table_size)
			break;

		batadv_tt_local_purge(bat_priv, timeout);
		batadv_tt_local_purge_pending_clients(bat_priv);

		timeout /= 2;
		reduced = true;
		net_ratelimited_function(batadv_info, soft_iface,
					 "Forced to purge local tt entries to fit new maximum fragment MTU (%i)\n",
					 packet_size_max);
	}

	/* commit these changes immediately, to avoid synchronization problem
	 * with the TTVN
	 */
	if (reduced)
		batadv_tt_local_commit_changes_nolock(bat_priv);

	spin_unlock_bh(&bat_priv->tt.commit_lock);
}

3638 3639 3640 3641 3642 3643 3644 3645 3646 3647
/**
 * batadv_tt_tvlv_ogm_handler_v1 - process incoming tt tvlv container
 * @bat_priv: the bat priv with all the soft interface information
 * @orig: the orig_node of the ogm
 * @flags: flags indicating the tvlv state (see batadv_tvlv_handler_flags)
 * @tvlv_value: tvlv buffer containing the gateway data
 * @tvlv_value_len: tvlv buffer length
 */
static void batadv_tt_tvlv_ogm_handler_v1(struct batadv_priv *bat_priv,
					  struct batadv_orig_node *orig,
3648 3649
					  u8 flags, void *tvlv_value,
					  u16 tvlv_value_len)
3650
{
3651 3652
	struct batadv_tvlv_tt_vlan_data *tt_vlan;
	struct batadv_tvlv_tt_change *tt_change;
3653
	struct batadv_tvlv_tt_data *tt_data;
3654
	u16 num_entries, num_vlan;
3655 3656 3657 3658 3659 3660 3661

	if (tvlv_value_len < sizeof(*tt_data))
		return;

	tt_data = (struct batadv_tvlv_tt_data *)tvlv_value;
	tvlv_value_len -= sizeof(*tt_data);

3662 3663 3664 3665 3666 3667 3668 3669 3670
	num_vlan = ntohs(tt_data->num_vlan);

	if (tvlv_value_len < sizeof(*tt_vlan) * num_vlan)
		return;

	tt_vlan = (struct batadv_tvlv_tt_vlan_data *)(tt_data + 1);
	tt_change = (struct batadv_tvlv_tt_change *)(tt_vlan + num_vlan);
	tvlv_value_len -= sizeof(*tt_vlan) * num_vlan;

3671
	num_entries = batadv_tt_entries(tvlv_value_len);
3672

3673 3674
	batadv_tt_update_orig(bat_priv, orig, tt_vlan, num_vlan, tt_change,
			      num_entries, tt_data->ttvn);
3675 3676
}

3677 3678 3679 3680 3681 3682 3683 3684 3685
/**
 * batadv_tt_tvlv_unicast_handler_v1 - process incoming (unicast) tt tvlv
 *  container
 * @bat_priv: the bat priv with all the soft interface information
 * @src: mac address of tt tvlv sender
 * @dst: mac address of tt tvlv recipient
 * @tvlv_value: tvlv buffer containing the tt data
 * @tvlv_value_len: tvlv buffer length
 *
3686
 * Return: NET_RX_DROP if the tt tvlv is to be re-routed, NET_RX_SUCCESS
3687 3688 3689
 * otherwise.
 */
static int batadv_tt_tvlv_unicast_handler_v1(struct batadv_priv *bat_priv,
3690
					     u8 *src, u8 *dst,
3691
					     void *tvlv_value,
3692
					     u16 tvlv_value_len)
3693 3694
{
	struct batadv_tvlv_tt_data *tt_data;
3695
	u16 tt_vlan_len, tt_num_entries;
3696 3697 3698 3699 3700 3701 3702 3703 3704
	char tt_flag;
	bool ret;

	if (tvlv_value_len < sizeof(*tt_data))
		return NET_RX_SUCCESS;

	tt_data = (struct batadv_tvlv_tt_data *)tvlv_value;
	tvlv_value_len -= sizeof(*tt_data);

3705 3706 3707 3708 3709 3710 3711 3712
	tt_vlan_len = sizeof(struct batadv_tvlv_tt_vlan_data);
	tt_vlan_len *= ntohs(tt_data->num_vlan);

	if (tvlv_value_len < tt_vlan_len)
		return NET_RX_SUCCESS;

	tvlv_value_len -= tt_vlan_len;
	tt_num_entries = batadv_tt_entries(tvlv_value_len);
3713 3714 3715 3716 3717 3718 3719 3720 3721 3722 3723 3724 3725 3726 3727 3728 3729 3730 3731 3732 3733 3734 3735 3736 3737 3738 3739

	switch (tt_data->flags & BATADV_TT_DATA_TYPE_MASK) {
	case BATADV_TT_REQUEST:
		batadv_inc_counter(bat_priv, BATADV_CNT_TT_REQUEST_RX);

		/* If this node cannot provide a TT response the tt_request is
		 * forwarded
		 */
		ret = batadv_send_tt_response(bat_priv, tt_data, src, dst);
		if (!ret) {
			if (tt_data->flags & BATADV_TT_FULL_TABLE)
				tt_flag = 'F';
			else
				tt_flag = '.';

			batadv_dbg(BATADV_DBG_TT, bat_priv,
				   "Routing TT_REQUEST to %pM [%c]\n",
				   dst, tt_flag);
			/* tvlv API will re-route the packet */
			return NET_RX_DROP;
		}
		break;
	case BATADV_TT_RESPONSE:
		batadv_inc_counter(bat_priv, BATADV_CNT_TT_RESPONSE_RX);

		if (batadv_is_my_mac(bat_priv, dst)) {
			batadv_handle_tt_response(bat_priv, tt_data,
3740
						  src, tt_num_entries);
3741 3742 3743 3744 3745 3746 3747 3748 3749 3750 3751 3752 3753 3754 3755 3756 3757 3758
			return NET_RX_SUCCESS;
		}

		if (tt_data->flags & BATADV_TT_FULL_TABLE)
			tt_flag =  'F';
		else
			tt_flag = '.';

		batadv_dbg(BATADV_DBG_TT, bat_priv,
			   "Routing TT_RESPONSE to %pM [%c]\n", dst, tt_flag);

		/* tvlv API will re-route the packet */
		return NET_RX_DROP;
	}

	return NET_RX_SUCCESS;
}

3759 3760 3761 3762 3763 3764 3765 3766
/**
 * batadv_roam_tvlv_unicast_handler_v1 - process incoming tt roam tvlv container
 * @bat_priv: the bat priv with all the soft interface information
 * @src: mac address of tt tvlv sender
 * @dst: mac address of tt tvlv recipient
 * @tvlv_value: tvlv buffer containing the tt data
 * @tvlv_value_len: tvlv buffer length
 *
3767
 * Return: NET_RX_DROP if the tt roam tvlv is to be re-routed, NET_RX_SUCCESS
3768 3769 3770
 * otherwise.
 */
static int batadv_roam_tvlv_unicast_handler_v1(struct batadv_priv *bat_priv,
3771
					       u8 *src, u8 *dst,
3772
					       void *tvlv_value,
3773
					       u16 tvlv_value_len)
3774 3775 3776 3777 3778 3779 3780 3781 3782 3783 3784 3785 3786 3787 3788 3789 3790 3791 3792 3793 3794 3795 3796 3797 3798 3799
{
	struct batadv_tvlv_roam_adv *roaming_adv;
	struct batadv_orig_node *orig_node = NULL;

	/* If this node is not the intended recipient of the
	 * roaming advertisement the packet is forwarded
	 * (the tvlv API will re-route the packet).
	 */
	if (!batadv_is_my_mac(bat_priv, dst))
		return NET_RX_DROP;

	if (tvlv_value_len < sizeof(*roaming_adv))
		goto out;

	orig_node = batadv_orig_hash_find(bat_priv, src);
	if (!orig_node)
		goto out;

	batadv_inc_counter(bat_priv, BATADV_CNT_TT_ROAM_ADV_RX);
	roaming_adv = (struct batadv_tvlv_roam_adv *)tvlv_value;

	batadv_dbg(BATADV_DBG_TT, bat_priv,
		   "Received ROAMING_ADV from %pM (client %pM)\n",
		   src, roaming_adv->client);

	batadv_tt_global_add(bat_priv, orig_node, roaming_adv->client,
3800
			     ntohs(roaming_adv->vid), BATADV_TT_CLIENT_ROAM,
3801 3802 3803 3804
			     atomic_read(&orig_node->last_ttvn) + 1);

out:
	if (orig_node)
3805
		batadv_orig_node_put(orig_node);
3806 3807 3808
	return NET_RX_SUCCESS;
}

3809 3810 3811 3812
/**
 * batadv_tt_init - initialise the translation table internals
 * @bat_priv: the bat priv with all the soft interface information
 *
3813
 * Return: 0 on success or negative error number in case of failure.
3814 3815 3816 3817 3818
 */
int batadv_tt_init(struct batadv_priv *bat_priv)
{
	int ret;

3819 3820 3821
	/* synchronized flags must be remote */
	BUILD_BUG_ON(!(BATADV_TT_SYNC_MASK & BATADV_TT_REMOTE_MASK));

3822 3823 3824 3825 3826 3827 3828 3829 3830
	ret = batadv_tt_local_init(bat_priv);
	if (ret < 0)
		return ret;

	ret = batadv_tt_global_init(bat_priv);
	if (ret < 0)
		return ret;

	batadv_tvlv_handler_register(bat_priv, batadv_tt_tvlv_ogm_handler_v1,
3831 3832
				     batadv_tt_tvlv_unicast_handler_v1,
				     BATADV_TVLV_TT, 1, BATADV_NO_FLAGS);
3833

3834 3835 3836 3837
	batadv_tvlv_handler_register(bat_priv, NULL,
				     batadv_roam_tvlv_unicast_handler_v1,
				     BATADV_TVLV_ROAM, 1, BATADV_NO_FLAGS);

3838 3839 3840 3841 3842 3843
	INIT_DELAYED_WORK(&bat_priv->tt.work, batadv_tt_purge);
	queue_delayed_work(batadv_event_workqueue, &bat_priv->tt.work,
			   msecs_to_jiffies(BATADV_TT_WORK_PERIOD));

	return 1;
}
3844 3845 3846 3847 3848 3849 3850

/**
 * batadv_tt_global_is_isolated - check if a client is marked as isolated
 * @bat_priv: the bat priv with all the soft interface information
 * @addr: the mac address of the client
 * @vid: the identifier of the VLAN where this client is connected
 *
3851
 * Return: true if the client is marked with the TT_CLIENT_ISOLA flag, false
3852 3853 3854
 * otherwise
 */
bool batadv_tt_global_is_isolated(struct batadv_priv *bat_priv,
3855
				  const u8 *addr, unsigned short vid)
3856 3857 3858 3859 3860 3861 3862 3863 3864 3865 3866 3867 3868 3869
{
	struct batadv_tt_global_entry *tt;
	bool ret;

	tt = batadv_tt_global_hash_find(bat_priv, addr, vid);
	if (!tt)
		return false;

	ret = tt->common.flags & BATADV_TT_CLIENT_ISOLA;

	batadv_tt_global_entry_free_ref(tt);

	return ret;
}