diff options
| author | Linus Lüssing <linus.luessing@web.de> | 2014-02-15 11:47:49 -0500 |
|---|---|---|
| committer | Antonio Quartulli <antonio@meshcoding.com> | 2014-03-22 04:18:56 -0400 |
| commit | c5caf4ef34e2779c9a90bf4cbb57fbdf57dc8cbc (patch) | |
| tree | 85fef7bf4b8eb836560710fb97214d97c16528dd | |
| parent | c5d3a652a3cf180e7a4b670d73517a0dfbbefebc (diff) | |
batman-adv: Multicast Listener Announcements via Translation Table
With this patch a node which has no bridge interface on top of its soft
interface announces its local multicast listeners via the translation
table.
Signed-off-by: Linus Lüssing <linus.luessing@web.de>
Signed-off-by: Marek Lindner <mareklindner@neomailbox.ch>
Signed-off-by: Antonio Quartulli <antonio@meshcoding.com>
| -rw-r--r-- | net/batman-adv/Kconfig | 9 | ||||
| -rw-r--r-- | net/batman-adv/Makefile | 1 | ||||
| -rw-r--r-- | net/batman-adv/main.c | 6 | ||||
| -rw-r--r-- | net/batman-adv/main.h | 1 | ||||
| -rw-r--r-- | net/batman-adv/multicast.c | 218 | ||||
| -rw-r--r-- | net/batman-adv/multicast.h | 41 | ||||
| -rw-r--r-- | net/batman-adv/translation-table.c | 22 | ||||
| -rw-r--r-- | net/batman-adv/types.h | 24 |
8 files changed, 318 insertions, 4 deletions
diff --git a/net/batman-adv/Kconfig b/net/batman-adv/Kconfig index fa780b76630e..11660a3aab5a 100644 --- a/net/batman-adv/Kconfig +++ b/net/batman-adv/Kconfig | |||
| @@ -50,6 +50,15 @@ config BATMAN_ADV_NC | |||
| 50 | If you think that your network does not need this feature you | 50 | If you think that your network does not need this feature you |
| 51 | can safely disable it and save some space. | 51 | can safely disable it and save some space. |
| 52 | 52 | ||
| 53 | config BATMAN_ADV_MCAST | ||
| 54 | bool "Multicast optimisation" | ||
| 55 | depends on BATMAN_ADV | ||
| 56 | default n | ||
| 57 | help | ||
| 58 | This option enables the multicast optimisation which aims to | ||
| 59 | reduce the air overhead while improving the reliability of | ||
| 60 | multicast messages. | ||
| 61 | |||
| 53 | config BATMAN_ADV_DEBUG | 62 | config BATMAN_ADV_DEBUG |
| 54 | bool "B.A.T.M.A.N. debugging" | 63 | bool "B.A.T.M.A.N. debugging" |
| 55 | depends on BATMAN_ADV | 64 | depends on BATMAN_ADV |
diff --git a/net/batman-adv/Makefile b/net/batman-adv/Makefile index 42df18f877e9..eb7d8c0388e4 100644 --- a/net/batman-adv/Makefile +++ b/net/batman-adv/Makefile | |||
| @@ -36,3 +36,4 @@ batman-adv-y += send.o | |||
| 36 | batman-adv-y += soft-interface.o | 36 | batman-adv-y += soft-interface.o |
| 37 | batman-adv-y += sysfs.o | 37 | batman-adv-y += sysfs.o |
| 38 | batman-adv-y += translation-table.o | 38 | batman-adv-y += translation-table.o |
| 39 | batman-adv-$(CONFIG_BATMAN_ADV_MCAST) += multicast.o | ||
diff --git a/net/batman-adv/main.c b/net/batman-adv/main.c index fbeaebde080d..58e98c89a68f 100644 --- a/net/batman-adv/main.c +++ b/net/batman-adv/main.c | |||
| @@ -34,6 +34,7 @@ | |||
| 34 | #include "gateway_client.h" | 34 | #include "gateway_client.h" |
| 35 | #include "bridge_loop_avoidance.h" | 35 | #include "bridge_loop_avoidance.h" |
| 36 | #include "distributed-arp-table.h" | 36 | #include "distributed-arp-table.h" |
| 37 | #include "multicast.h" | ||
| 37 | #include "gateway_common.h" | 38 | #include "gateway_common.h" |
| 38 | #include "hash.h" | 39 | #include "hash.h" |
| 39 | #include "bat_algo.h" | 40 | #include "bat_algo.h" |
| @@ -120,6 +121,9 @@ int batadv_mesh_init(struct net_device *soft_iface) | |||
| 120 | INIT_LIST_HEAD(&bat_priv->tt.changes_list); | 121 | INIT_LIST_HEAD(&bat_priv->tt.changes_list); |
| 121 | INIT_LIST_HEAD(&bat_priv->tt.req_list); | 122 | INIT_LIST_HEAD(&bat_priv->tt.req_list); |
| 122 | INIT_LIST_HEAD(&bat_priv->tt.roam_list); | 123 | INIT_LIST_HEAD(&bat_priv->tt.roam_list); |
| 124 | #ifdef CONFIG_BATMAN_ADV_MCAST | ||
| 125 | INIT_HLIST_HEAD(&bat_priv->mcast.mla_list); | ||
| 126 | #endif | ||
| 123 | INIT_HLIST_HEAD(&bat_priv->tvlv.container_list); | 127 | INIT_HLIST_HEAD(&bat_priv->tvlv.container_list); |
| 124 | INIT_HLIST_HEAD(&bat_priv->tvlv.handler_list); | 128 | INIT_HLIST_HEAD(&bat_priv->tvlv.handler_list); |
| 125 | INIT_HLIST_HEAD(&bat_priv->softif_vlan_list); | 129 | INIT_HLIST_HEAD(&bat_priv->softif_vlan_list); |
| @@ -169,6 +173,8 @@ void batadv_mesh_free(struct net_device *soft_iface) | |||
| 169 | batadv_dat_free(bat_priv); | 173 | batadv_dat_free(bat_priv); |
| 170 | batadv_bla_free(bat_priv); | 174 | batadv_bla_free(bat_priv); |
| 171 | 175 | ||
| 176 | batadv_mcast_free(bat_priv); | ||
| 177 | |||
| 172 | /* Free the TT and the originator tables only after having terminated | 178 | /* Free the TT and the originator tables only after having terminated |
| 173 | * all the other depending components which may use these structures for | 179 | * all the other depending components which may use these structures for |
| 174 | * their purposes. | 180 | * their purposes. |
diff --git a/net/batman-adv/main.h b/net/batman-adv/main.h index 9374f1a51348..aa4f73a044c5 100644 --- a/net/batman-adv/main.h +++ b/net/batman-adv/main.h | |||
| @@ -176,6 +176,7 @@ enum batadv_uev_type { | |||
| 176 | #include <linux/percpu.h> | 176 | #include <linux/percpu.h> |
| 177 | #include <linux/slab.h> | 177 | #include <linux/slab.h> |
| 178 | #include <net/sock.h> /* struct sock */ | 178 | #include <net/sock.h> /* struct sock */ |
| 179 | #include <net/addrconf.h> /* ipv6 address stuff */ | ||
| 179 | #include <net/rtnetlink.h> | 180 | #include <net/rtnetlink.h> |
| 180 | #include <linux/jiffies.h> | 181 | #include <linux/jiffies.h> |
| 181 | #include <linux/seq_file.h> | 182 | #include <linux/seq_file.h> |
diff --git a/net/batman-adv/multicast.c b/net/batman-adv/multicast.c new file mode 100644 index 000000000000..e099fd67403c --- /dev/null +++ b/net/batman-adv/multicast.c | |||
| @@ -0,0 +1,218 @@ | |||
| 1 | /* Copyright (C) 2014 B.A.T.M.A.N. contributors: | ||
| 2 | * | ||
| 3 | * Linus Lüssing | ||
| 4 | * | ||
| 5 | * This program is free software; you can redistribute it and/or | ||
| 6 | * modify it under the terms of version 2 of the GNU General Public | ||
| 7 | * License as published by the Free Software Foundation. | ||
| 8 | * | ||
| 9 | * This program is distributed in the hope that it will be useful, but | ||
| 10 | * WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 12 | * General Public License for more details. | ||
| 13 | * | ||
| 14 | * You should have received a copy of the GNU General Public License | ||
| 15 | * along with this program; if not, see <http://www.gnu.org/licenses/>. | ||
| 16 | */ | ||
| 17 | |||
| 18 | #include "main.h" | ||
| 19 | #include "multicast.h" | ||
| 20 | #include "originator.h" | ||
| 21 | #include "hard-interface.h" | ||
| 22 | #include "translation-table.h" | ||
| 23 | |||
| 24 | /** | ||
| 25 | * batadv_mcast_mla_softif_get - get softif multicast listeners | ||
| 26 | * @dev: the device to collect multicast addresses from | ||
| 27 | * @mcast_list: a list to put found addresses into | ||
| 28 | * | ||
| 29 | * Collect multicast addresses of the local multicast listeners | ||
| 30 | * on the given soft interface, dev, in the given mcast_list. | ||
| 31 | * | ||
| 32 | * Returns -ENOMEM on memory allocation error or the number of | ||
| 33 | * items added to the mcast_list otherwise. | ||
| 34 | */ | ||
| 35 | static int batadv_mcast_mla_softif_get(struct net_device *dev, | ||
| 36 | struct hlist_head *mcast_list) | ||
| 37 | { | ||
| 38 | struct netdev_hw_addr *mc_list_entry; | ||
| 39 | struct batadv_hw_addr *new; | ||
| 40 | int ret = 0; | ||
| 41 | |||
| 42 | netif_addr_lock_bh(dev); | ||
| 43 | netdev_for_each_mc_addr(mc_list_entry, dev) { | ||
| 44 | new = kmalloc(sizeof(*new), GFP_ATOMIC); | ||
| 45 | if (!new) { | ||
| 46 | ret = -ENOMEM; | ||
| 47 | break; | ||
| 48 | } | ||
| 49 | |||
| 50 | ether_addr_copy(new->addr, mc_list_entry->addr); | ||
| 51 | hlist_add_head(&new->list, mcast_list); | ||
| 52 | ret++; | ||
| 53 | } | ||
| 54 | netif_addr_unlock_bh(dev); | ||
| 55 | |||
| 56 | return ret; | ||
| 57 | } | ||
| 58 | |||
| 59 | /** | ||
| 60 | * batadv_mcast_mla_is_duplicate - check whether an address is in a list | ||
| 61 | * @mcast_addr: the multicast address to check | ||
| 62 | * @mcast_list: the list with multicast addresses to search in | ||
| 63 | * | ||
| 64 | * Returns true if the given address is already in the given list. | ||
| 65 | * Otherwise returns false. | ||
| 66 | */ | ||
| 67 | static bool batadv_mcast_mla_is_duplicate(uint8_t *mcast_addr, | ||
| 68 | struct hlist_head *mcast_list) | ||
| 69 | { | ||
| 70 | struct batadv_hw_addr *mcast_entry; | ||
| 71 | |||
| 72 | hlist_for_each_entry(mcast_entry, mcast_list, list) | ||
| 73 | if (batadv_compare_eth(mcast_entry->addr, mcast_addr)) | ||
| 74 | return true; | ||
| 75 | |||
| 76 | return false; | ||
| 77 | } | ||
| 78 | |||
| 79 | /** | ||
| 80 | * batadv_mcast_mla_list_free - free a list of multicast addresses | ||
| 81 | * @mcast_list: the list to free | ||
| 82 | * | ||
| 83 | * Removes and frees all items in the given mcast_list. | ||
| 84 | */ | ||
| 85 | static void batadv_mcast_mla_list_free(struct hlist_head *mcast_list) | ||
| 86 | { | ||
| 87 | struct batadv_hw_addr *mcast_entry; | ||
| 88 | struct hlist_node *tmp; | ||
| 89 | |||
| 90 | hlist_for_each_entry_safe(mcast_entry, tmp, mcast_list, list) { | ||
| 91 | hlist_del(&mcast_entry->list); | ||
| 92 | kfree(mcast_entry); | ||
| 93 | } | ||
| 94 | } | ||
| 95 | |||
| 96 | /** | ||
| 97 | * batadv_mcast_mla_tt_retract - clean up multicast listener announcements | ||
| 98 | * @bat_priv: the bat priv with all the soft interface information | ||
| 99 | * @mcast_list: a list of addresses which should _not_ be removed | ||
| 100 | * | ||
| 101 | * Retracts the announcement of any multicast listener from the | ||
| 102 | * translation table except the ones listed in the given mcast_list. | ||
| 103 | * | ||
| 104 | * If mcast_list is NULL then all are retracted. | ||
| 105 | */ | ||
| 106 | static void batadv_mcast_mla_tt_retract(struct batadv_priv *bat_priv, | ||
| 107 | struct hlist_head *mcast_list) | ||
| 108 | { | ||
| 109 | struct batadv_hw_addr *mcast_entry; | ||
| 110 | struct hlist_node *tmp; | ||
| 111 | |||
| 112 | hlist_for_each_entry_safe(mcast_entry, tmp, &bat_priv->mcast.mla_list, | ||
| 113 | list) { | ||
| 114 | if (mcast_list && | ||
| 115 | batadv_mcast_mla_is_duplicate(mcast_entry->addr, | ||
| 116 | mcast_list)) | ||
| 117 | continue; | ||
| 118 | |||
| 119 | batadv_tt_local_remove(bat_priv, mcast_entry->addr, | ||
| 120 | BATADV_NO_FLAGS, | ||
| 121 | "mcast TT outdated", false); | ||
| 122 | |||
| 123 | hlist_del(&mcast_entry->list); | ||
| 124 | kfree(mcast_entry); | ||
| 125 | } | ||
| 126 | } | ||
| 127 | |||
| 128 | /** | ||
| 129 | * batadv_mcast_mla_tt_add - add multicast listener announcements | ||
| 130 | * @bat_priv: the bat priv with all the soft interface information | ||
| 131 | * @mcast_list: a list of addresses which are going to get added | ||
| 132 | * | ||
| 133 | * Adds multicast listener announcements from the given mcast_list to the | ||
| 134 | * translation table if they have not been added yet. | ||
| 135 | */ | ||
| 136 | static void batadv_mcast_mla_tt_add(struct batadv_priv *bat_priv, | ||
| 137 | struct hlist_head *mcast_list) | ||
| 138 | { | ||
| 139 | struct batadv_hw_addr *mcast_entry; | ||
| 140 | struct hlist_node *tmp; | ||
| 141 | |||
| 142 | if (!mcast_list) | ||
| 143 | return; | ||
| 144 | |||
| 145 | hlist_for_each_entry_safe(mcast_entry, tmp, mcast_list, list) { | ||
| 146 | if (batadv_mcast_mla_is_duplicate(mcast_entry->addr, | ||
| 147 | &bat_priv->mcast.mla_list)) | ||
| 148 | continue; | ||
| 149 | |||
| 150 | if (!batadv_tt_local_add(bat_priv->soft_iface, | ||
| 151 | mcast_entry->addr, BATADV_NO_FLAGS, | ||
| 152 | BATADV_NULL_IFINDEX, BATADV_NO_MARK)) | ||
| 153 | continue; | ||
| 154 | |||
| 155 | hlist_del(&mcast_entry->list); | ||
| 156 | hlist_add_head(&mcast_entry->list, &bat_priv->mcast.mla_list); | ||
| 157 | } | ||
| 158 | } | ||
| 159 | |||
| 160 | /** | ||
| 161 | * batadv_mcast_has_bridge - check whether the soft-iface is bridged | ||
| 162 | * @bat_priv: the bat priv with all the soft interface information | ||
| 163 | * | ||
| 164 | * Checks whether there is a bridge on top of our soft interface. Returns | ||
| 165 | * true if so, false otherwise. | ||
| 166 | */ | ||
| 167 | static bool batadv_mcast_has_bridge(struct batadv_priv *bat_priv) | ||
| 168 | { | ||
| 169 | struct net_device *upper = bat_priv->soft_iface; | ||
| 170 | |||
| 171 | rcu_read_lock(); | ||
| 172 | do { | ||
| 173 | upper = netdev_master_upper_dev_get_rcu(upper); | ||
| 174 | } while (upper && !(upper->priv_flags & IFF_EBRIDGE)); | ||
| 175 | rcu_read_unlock(); | ||
| 176 | |||
| 177 | return upper; | ||
| 178 | } | ||
| 179 | |||
| 180 | /** | ||
| 181 | * batadv_mcast_mla_update - update the own MLAs | ||
| 182 | * @bat_priv: the bat priv with all the soft interface information | ||
| 183 | * | ||
| 184 | * Update the own multicast listener announcements in the translation | ||
| 185 | * table. | ||
| 186 | */ | ||
| 187 | void batadv_mcast_mla_update(struct batadv_priv *bat_priv) | ||
| 188 | { | ||
| 189 | struct net_device *soft_iface = bat_priv->soft_iface; | ||
| 190 | struct hlist_head mcast_list = HLIST_HEAD_INIT; | ||
| 191 | int ret; | ||
| 192 | |||
| 193 | /* Avoid attaching MLAs, if there is a bridge on top of our soft | ||
| 194 | * interface, we don't support that yet (TODO) | ||
| 195 | */ | ||
| 196 | if (batadv_mcast_has_bridge(bat_priv)) | ||
| 197 | goto update; | ||
| 198 | |||
| 199 | ret = batadv_mcast_mla_softif_get(soft_iface, &mcast_list); | ||
| 200 | if (ret < 0) | ||
| 201 | goto out; | ||
| 202 | |||
| 203 | update: | ||
| 204 | batadv_mcast_mla_tt_retract(bat_priv, &mcast_list); | ||
| 205 | batadv_mcast_mla_tt_add(bat_priv, &mcast_list); | ||
| 206 | |||
| 207 | out: | ||
| 208 | batadv_mcast_mla_list_free(&mcast_list); | ||
| 209 | } | ||
| 210 | |||
| 211 | /** | ||
| 212 | * batadv_mcast_free - free the multicast optimizations structures | ||
| 213 | * @bat_priv: the bat priv with all the soft interface information | ||
| 214 | */ | ||
| 215 | void batadv_mcast_free(struct batadv_priv *bat_priv) | ||
| 216 | { | ||
| 217 | batadv_mcast_mla_tt_retract(bat_priv, NULL); | ||
| 218 | } | ||
diff --git a/net/batman-adv/multicast.h b/net/batman-adv/multicast.h new file mode 100644 index 000000000000..7513e024b807 --- /dev/null +++ b/net/batman-adv/multicast.h | |||
| @@ -0,0 +1,41 @@ | |||
| 1 | /* Copyright (C) 2014 B.A.T.M.A.N. contributors: | ||
| 2 | * | ||
| 3 | * Linus Lüssing | ||
| 4 | * | ||
| 5 | * This program is free software; you can redistribute it and/or | ||
| 6 | * modify it under the terms of version 2 of the GNU General Public | ||
| 7 | * License as published by the Free Software Foundation. | ||
| 8 | * | ||
| 9 | * This program is distributed in the hope that it will be useful, but | ||
| 10 | * WITHOUT ANY WARRANTY; without even the implied warranty of | ||
| 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU | ||
| 12 | * General Public License for more details. | ||
| 13 | * | ||
| 14 | * You should have received a copy of the GNU General Public License | ||
| 15 | * along with this program; if not, see <http://www.gnu.org/licenses/>. | ||
| 16 | */ | ||
| 17 | |||
| 18 | #ifndef _NET_BATMAN_ADV_MULTICAST_H_ | ||
| 19 | #define _NET_BATMAN_ADV_MULTICAST_H_ | ||
| 20 | |||
| 21 | #ifdef CONFIG_BATMAN_ADV_MCAST | ||
| 22 | |||
| 23 | void batadv_mcast_mla_update(struct batadv_priv *bat_priv); | ||
| 24 | |||
| 25 | void batadv_mcast_free(struct batadv_priv *bat_priv); | ||
| 26 | |||
| 27 | #else | ||
| 28 | |||
| 29 | static inline void batadv_mcast_mla_update(struct batadv_priv *bat_priv) | ||
| 30 | { | ||
| 31 | return; | ||
| 32 | } | ||
| 33 | |||
| 34 | static inline void batadv_mcast_free(struct batadv_priv *bat_priv) | ||
| 35 | { | ||
| 36 | return; | ||
| 37 | } | ||
| 38 | |||
| 39 | #endif /* CONFIG_BATMAN_ADV_MCAST */ | ||
| 40 | |||
| 41 | #endif /* _NET_BATMAN_ADV_MULTICAST_H_ */ | ||
diff --git a/net/batman-adv/translation-table.c b/net/batman-adv/translation-table.c index 6d0da58d755e..4082d05a93a9 100644 --- a/net/batman-adv/translation-table.c +++ b/net/batman-adv/translation-table.c | |||
| @@ -24,6 +24,7 @@ | |||
| 24 | #include "originator.h" | 24 | #include "originator.h" |
| 25 | #include "routing.h" | 25 | #include "routing.h" |
| 26 | #include "bridge_loop_avoidance.h" | 26 | #include "bridge_loop_avoidance.h" |
| 27 | #include "multicast.h" | ||
| 27 | 28 | ||
| 28 | #include <linux/crc32c.h> | 29 | #include <linux/crc32c.h> |
| 29 | 30 | ||
| @@ -484,7 +485,7 @@ bool batadv_tt_local_add(struct net_device *soft_iface, const uint8_t *addr, | |||
| 484 | { | 485 | { |
| 485 | struct batadv_priv *bat_priv = netdev_priv(soft_iface); | 486 | struct batadv_priv *bat_priv = netdev_priv(soft_iface); |
| 486 | struct batadv_tt_local_entry *tt_local; | 487 | struct batadv_tt_local_entry *tt_local; |
| 487 | struct batadv_tt_global_entry *tt_global; | 488 | struct batadv_tt_global_entry *tt_global = NULL; |
| 488 | struct net_device *in_dev = NULL; | 489 | struct net_device *in_dev = NULL; |
| 489 | struct hlist_head *head; | 490 | struct hlist_head *head; |
| 490 | struct batadv_tt_orig_list_entry *orig_entry; | 491 | struct batadv_tt_orig_list_entry *orig_entry; |
| @@ -497,7 +498,9 @@ bool batadv_tt_local_add(struct net_device *soft_iface, const uint8_t *addr, | |||
| 497 | in_dev = dev_get_by_index(&init_net, ifindex); | 498 | in_dev = dev_get_by_index(&init_net, ifindex); |
| 498 | 499 | ||
| 499 | tt_local = batadv_tt_local_hash_find(bat_priv, addr, vid); | 500 | tt_local = batadv_tt_local_hash_find(bat_priv, addr, vid); |
| 500 | tt_global = batadv_tt_global_hash_find(bat_priv, addr, vid); | 501 | |
| 502 | if (!is_multicast_ether_addr(addr)) | ||
| 503 | tt_global = batadv_tt_global_hash_find(bat_priv, addr, vid); | ||
| 501 | 504 | ||
| 502 | if (tt_local) { | 505 | if (tt_local) { |
| 503 | tt_local->last_seen = jiffies; | 506 | tt_local->last_seen = jiffies; |
| @@ -562,8 +565,11 @@ bool batadv_tt_local_add(struct net_device *soft_iface, const uint8_t *addr, | |||
| 562 | tt_local->last_seen = jiffies; | 565 | tt_local->last_seen = jiffies; |
| 563 | tt_local->common.added_at = tt_local->last_seen; | 566 | tt_local->common.added_at = tt_local->last_seen; |
| 564 | 567 | ||
| 565 | /* the batman interface mac address should never be purged */ | 568 | /* the batman interface mac and multicast addresses should never be |
| 566 | if (batadv_compare_eth(addr, soft_iface->dev_addr)) | 569 | * purged |
| 570 | */ | ||
| 571 | if (batadv_compare_eth(addr, soft_iface->dev_addr) || | ||
| 572 | is_multicast_ether_addr(addr)) | ||
| 567 | tt_local->common.flags |= BATADV_TT_CLIENT_NOPURGE; | 573 | tt_local->common.flags |= BATADV_TT_CLIENT_NOPURGE; |
| 568 | 574 | ||
| 569 | hash_added = batadv_hash_add(bat_priv->tt.local_hash, batadv_compare_tt, | 575 | hash_added = batadv_hash_add(bat_priv->tt.local_hash, batadv_compare_tt, |
| @@ -1361,6 +1367,11 @@ add_orig_entry: | |||
| 1361 | ret = true; | 1367 | ret = true; |
| 1362 | 1368 | ||
| 1363 | out_remove: | 1369 | out_remove: |
| 1370 | /* Do not remove multicast addresses from the local hash on | ||
| 1371 | * global additions | ||
| 1372 | */ | ||
| 1373 | if (is_multicast_ether_addr(tt_addr)) | ||
| 1374 | goto out; | ||
| 1364 | 1375 | ||
| 1365 | /* remove address from local hash if present */ | 1376 | /* remove address from local hash if present */ |
| 1366 | local_flags = batadv_tt_local_remove(bat_priv, tt_addr, vid, | 1377 | local_flags = batadv_tt_local_remove(bat_priv, tt_addr, vid, |
| @@ -3120,6 +3131,9 @@ static void batadv_tt_local_purge_pending_clients(struct batadv_priv *bat_priv) | |||
| 3120 | */ | 3131 | */ |
| 3121 | static void batadv_tt_local_commit_changes_nolock(struct batadv_priv *bat_priv) | 3132 | static void batadv_tt_local_commit_changes_nolock(struct batadv_priv *bat_priv) |
| 3122 | { | 3133 | { |
| 3134 | /* Update multicast addresses in local translation table */ | ||
| 3135 | batadv_mcast_mla_update(bat_priv); | ||
| 3136 | |||
| 3123 | if (atomic_read(&bat_priv->tt.local_changes) < 1) { | 3137 | if (atomic_read(&bat_priv->tt.local_changes) < 1) { |
| 3124 | if (!batadv_atomic_dec_not_zero(&bat_priv->tt.ogm_append_cnt)) | 3138 | if (!batadv_atomic_dec_not_zero(&bat_priv->tt.ogm_append_cnt)) |
| 3125 | batadv_tt_tvlv_container_update(bat_priv); | 3139 | batadv_tt_tvlv_container_update(bat_priv); |
diff --git a/net/batman-adv/types.h b/net/batman-adv/types.h index 9f52517d0e74..d553264135d3 100644 --- a/net/batman-adv/types.h +++ b/net/batman-adv/types.h | |||
| @@ -607,6 +607,16 @@ struct batadv_priv_dat { | |||
| 607 | }; | 607 | }; |
| 608 | #endif | 608 | #endif |
| 609 | 609 | ||
| 610 | #ifdef CONFIG_BATMAN_ADV_MCAST | ||
| 611 | /** | ||
| 612 | * struct batadv_priv_mcast - per mesh interface mcast data | ||
| 613 | * @mla_list: list of multicast addresses we are currently announcing via TT | ||
| 614 | */ | ||
| 615 | struct batadv_priv_mcast { | ||
| 616 | struct hlist_head mla_list; | ||
| 617 | }; | ||
| 618 | #endif | ||
| 619 | |||
| 610 | /** | 620 | /** |
| 611 | * struct batadv_priv_nc - per mesh interface network coding private data | 621 | * struct batadv_priv_nc - per mesh interface network coding private data |
| 612 | * @work: work queue callback item for cleanup | 622 | * @work: work queue callback item for cleanup |
| @@ -702,6 +712,7 @@ struct batadv_softif_vlan { | |||
| 702 | * @tt: translation table data | 712 | * @tt: translation table data |
| 703 | * @tvlv: type-version-length-value data | 713 | * @tvlv: type-version-length-value data |
| 704 | * @dat: distributed arp table data | 714 | * @dat: distributed arp table data |
| 715 | * @mcast: multicast data | ||
| 705 | * @network_coding: bool indicating whether network coding is enabled | 716 | * @network_coding: bool indicating whether network coding is enabled |
| 706 | * @batadv_priv_nc: network coding data | 717 | * @batadv_priv_nc: network coding data |
| 707 | */ | 718 | */ |
| @@ -759,6 +770,9 @@ struct batadv_priv { | |||
| 759 | #ifdef CONFIG_BATMAN_ADV_DAT | 770 | #ifdef CONFIG_BATMAN_ADV_DAT |
| 760 | struct batadv_priv_dat dat; | 771 | struct batadv_priv_dat dat; |
| 761 | #endif | 772 | #endif |
| 773 | #ifdef CONFIG_BATMAN_ADV_MCAST | ||
| 774 | struct batadv_priv_mcast mcast; | ||
| 775 | #endif | ||
| 762 | #ifdef CONFIG_BATMAN_ADV_NC | 776 | #ifdef CONFIG_BATMAN_ADV_NC |
| 763 | atomic_t network_coding; | 777 | atomic_t network_coding; |
| 764 | struct batadv_priv_nc nc; | 778 | struct batadv_priv_nc nc; |
| @@ -1116,6 +1130,16 @@ struct batadv_dat_entry { | |||
| 1116 | }; | 1130 | }; |
| 1117 | 1131 | ||
| 1118 | /** | 1132 | /** |
| 1133 | * struct batadv_hw_addr - a list entry for a MAC address | ||
| 1134 | * @list: list node for the linking of entries | ||
| 1135 | * @addr: the MAC address of this list entry | ||
| 1136 | */ | ||
| 1137 | struct batadv_hw_addr { | ||
| 1138 | struct hlist_node list; | ||
| 1139 | unsigned char addr[ETH_ALEN]; | ||
| 1140 | }; | ||
| 1141 | |||
| 1142 | /** | ||
| 1119 | * struct batadv_dat_candidate - candidate destination for DAT operations | 1143 | * struct batadv_dat_candidate - candidate destination for DAT operations |
| 1120 | * @type: the type of the selected candidate. It can one of the following: | 1144 | * @type: the type of the selected candidate. It can one of the following: |
| 1121 | * - BATADV_DAT_CANDIDATE_NOT_FOUND | 1145 | * - BATADV_DAT_CANDIDATE_NOT_FOUND |
