From: Antonio Quartulli <antonio@meshcoding.com>
To: b.a.t.m.a.n@lists.open-mesh.org
Cc: Antonio Quartulli <antonio@open-mesh.com>
Subject: [B.A.T.M.A.N.] [RFC 09/23] batman-adv: OGMv2 - purge obsolete potential routers
Date: Tue, 11 Feb 2014 13:48:09 +0100 [thread overview]
Message-ID: <1392122903-805-10-git-send-email-antonio@meshcoding.com> (raw)
In-Reply-To: <1392122903-805-1-git-send-email-antonio@meshcoding.com>
From: Antonio Quartulli <antonio@open-mesh.com>
If a neighbor is not forwarding OGMs since OGM_SEQ_RANGE
sequence numbers then it can be considered obsolete and
removed from the potential routers list. This may lead to a
route switch.
Signed-off-by: Antonio Quartulli <antonio@open-mesh.com>
---
bat_v_ogm.c | 92 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++--
main.h | 3 ++
originator.c | 63 +++++++++++++++++++++++------------------
originator.h | 2 ++
types.h | 2 ++
5 files changed, 131 insertions(+), 31 deletions(-)
diff --git a/bat_v_ogm.c b/bat_v_ogm.c
index 4981d7c..5876164 100644
--- a/bat_v_ogm.c
+++ b/bat_v_ogm.c
@@ -79,8 +79,6 @@ batadv_v_ogm_neigh_new(struct batadv_priv *bat_priv,
return NULL;
}
- neigh_node->bat_v.elp_neigh = elp_neigh;
-
spin_lock_bh(&orig_node->neigh_list_lock);
tmp_neigh_node = batadv_neigh_node_get(orig_node, hard_iface, addr);
if (!tmp_neigh_node) {
@@ -98,6 +96,8 @@ batadv_v_ogm_neigh_new(struct batadv_priv *bat_priv,
"Creating new neighbor %pM for orig_node %pM on interface %s\n",
addr, orig_node->orig, hard_iface->net_dev->name);
+ neigh_node->bat_v.elp_neigh = elp_neigh;
+
return neigh_node;
}
@@ -235,6 +235,86 @@ void batadv_v_ogm_primary_iface_set(struct batadv_hard_iface *hard_iface)
rcu_read_unlock();
}
+static bool batadv_ogm_deselect_router(struct batadv_priv *bat_priv,
+ struct batadv_orig_node *orig_node,
+ struct batadv_neigh_node *neigh)
+{
+ struct batadv_hard_iface *if_outgoing;
+ struct batadv_neigh_node *router;
+ bool removed = false;
+
+ router = batadv_orig_router_get(orig_node, BATADV_IF_DEFAULT);
+ if (router == neigh)
+ batadv_update_route(bat_priv, orig_node, BATADV_IF_DEFAULT,
+ NULL);
+ if (router)
+ batadv_neigh_node_free_ref(router);
+
+ rcu_read_lock();
+ list_for_each_entry_rcu(if_outgoing, &batadv_hardif_list, list) {
+ if (if_outgoing->if_status != BATADV_IF_ACTIVE)
+ continue;
+
+ if (if_outgoing->soft_iface != bat_priv->soft_iface)
+ continue;
+
+ router = batadv_orig_router_get(orig_node, if_outgoing);
+ /* if this is the current router, unset it */
+ if (router == neigh) {
+ batadv_update_route(bat_priv, orig_node, if_outgoing,
+ NULL);
+ removed = true;
+ }
+ if (router)
+ batadv_neigh_node_free_ref(router);
+ }
+ rcu_read_unlock();
+
+ return removed;
+}
+
+/**
+ * batadv_v_ogm_purge_neighs - purge obsolete potential routers
+ * @bat_priv:
+ * @orig_node:
+ * @seqno:
+ *
+ * Purge routers from which this node did not receive at least
+ * BATADV_OGM_SEQ_RANGE OGMs
+ */
+static void batadv_v_ogm_purge_neighs(struct batadv_priv *bat_priv,
+ struct batadv_orig_node *orig_node,
+ uint32_t seqno)
+{
+ struct batadv_neigh_node *neigh;
+ struct hlist_node *node_tmp;
+ int32_t diff;
+
+ rcu_read_lock();
+ spin_lock_bh(&orig_node->neigh_list_lock);
+ hlist_for_each_entry_safe(neigh, node_tmp, &orig_node->neigh_list,
+ list) {
+ diff = seqno - neigh->bat_v.last_recv_seqno;
+ if (diff < BATADV_OGM_SEQ_RANGE)
+ continue;
+
+ hlist_del_rcu(&neigh->list);
+ batadv_neigh_node_free_ref(neigh);
+ }
+ spin_unlock_bh(&orig_node->neigh_list_lock);
+
+ /* check if this neighbour is a router for any outgoing
+ * interface and possibly deselect it
+ */
+ if (batadv_ogm_deselect_router(bat_priv, orig_node, neigh))
+ goto out;
+
+ batadv_router_selection(bat_priv, orig_node);
+out:
+ rcu_read_unlock();
+ return;
+}
+
static void
batadv_v_ogm_orig_update(struct batadv_priv *bat_priv,
struct batadv_orig_node *orig_node,
@@ -262,7 +342,7 @@ batadv_v_ogm_orig_update(struct batadv_priv *bat_priv,
batadv_orig_ifinfo_free_ref(orig_ifinfo);
/* if this neighbor already is our next hop there is nothing
- * to change
+ * to change. The router may have been changed by the purge routine
*/
router = batadv_orig_router_get(orig_node, if_outgoing);
if (router == neigh_node)
@@ -547,6 +627,7 @@ int batadv_v_ogm_packet_recv(struct sk_buff *skb,
/* update the neigh_node information */
neigh_node->last_seen = jiffies;
+ neigh_node->bat_v.last_recv_seqno = ntohl(ogm2->seqno);
/* update the received metric to match the node topology: if this node
* is the first hop traversed by the OGM, then the metric is substituted
@@ -579,6 +660,11 @@ int batadv_v_ogm_packet_recv(struct sk_buff *skb,
}
rcu_read_unlock();
+ /* purge potential neighs not forwarding OGMs anymore.
+ * This may lead to a router change
+ */
+ batadv_v_ogm_purge_neighs(bat_priv, orig_node, ntohl(ogm2->seqno));
+
ret = NET_RX_SUCCESS;
consume_skb(skb);
diff --git a/main.h b/main.h
index 4badd01..47fb9ed 100644
--- a/main.h
+++ b/main.h
@@ -62,6 +62,9 @@
/* number of OGMs sent with the last tt diff */
#define BATADV_TT_OGM_APPEND_MAX 3
+/* number of missing OGMs to wait before considering a router unusable */
+#define BATADV_OGM_SEQ_RANGE 5
+
/* Time in which a client can roam at most ROAMING_MAX_COUNT times in
* milliseconds
*/
diff --git a/originator.c b/originator.c
index 9f2da37..447f417 100644
--- a/originator.c
+++ b/originator.c
@@ -837,6 +837,40 @@ batadv_find_best_neighbor(struct batadv_priv *bat_priv,
return best;
}
+void batadv_router_selection(struct batadv_priv *bat_priv,
+ struct batadv_orig_node *orig_node)
+{
+ struct batadv_neigh_node *best_neigh_node;
+ struct batadv_hard_iface *hard_iface;
+
+ /* First for the default routing table ... */
+ best_neigh_node = batadv_find_best_neighbor(bat_priv, orig_node,
+ BATADV_IF_DEFAULT);
+ batadv_update_route(bat_priv, orig_node, BATADV_IF_DEFAULT,
+ best_neigh_node);
+ if (best_neigh_node)
+ batadv_neigh_node_free_ref(best_neigh_node);
+
+ /* ... then for all other outgoing interfaces. */
+ rcu_read_lock();
+ list_for_each_entry_rcu(hard_iface, &batadv_hardif_list, list) {
+ if (hard_iface->if_status != BATADV_IF_ACTIVE)
+ continue;
+
+ if (hard_iface->soft_iface != bat_priv->soft_iface)
+ continue;
+
+ best_neigh_node = batadv_find_best_neighbor(bat_priv,
+ orig_node,
+ hard_iface);
+ batadv_update_route(bat_priv, orig_node, hard_iface,
+ best_neigh_node);
+ if (best_neigh_node)
+ batadv_neigh_node_free_ref(best_neigh_node);
+ }
+ rcu_read_unlock();
+}
+
/**
* batadv_purge_orig_node - purges obsolete information from an orig_node
* @bat_priv: the bat priv with all the soft interface information
@@ -850,8 +884,6 @@ batadv_find_best_neighbor(struct batadv_priv *bat_priv,
static bool batadv_purge_orig_node(struct batadv_priv *bat_priv,
struct batadv_orig_node *orig_node)
{
- struct batadv_neigh_node *best_neigh_node;
- struct batadv_hard_iface *hard_iface;
bool changed;
if (batadv_has_timed_out(orig_node->last_seen,
@@ -868,32 +900,7 @@ static bool batadv_purge_orig_node(struct batadv_priv *bat_priv,
if (!changed)
return false;
- /* first for NULL ... */
- best_neigh_node = batadv_find_best_neighbor(bat_priv, orig_node,
- BATADV_IF_DEFAULT);
- batadv_update_route(bat_priv, orig_node, BATADV_IF_DEFAULT,
- best_neigh_node);
- if (best_neigh_node)
- batadv_neigh_node_free_ref(best_neigh_node);
-
- /* ... then for all other interfaces. */
- rcu_read_lock();
- list_for_each_entry_rcu(hard_iface, &batadv_hardif_list, list) {
- if (hard_iface->if_status != BATADV_IF_ACTIVE)
- continue;
-
- if (hard_iface->soft_iface != bat_priv->soft_iface)
- continue;
-
- best_neigh_node = batadv_find_best_neighbor(bat_priv,
- orig_node,
- hard_iface);
- batadv_update_route(bat_priv, orig_node, hard_iface,
- best_neigh_node);
- if (best_neigh_node)
- batadv_neigh_node_free_ref(best_neigh_node);
- }
- rcu_read_unlock();
+ batadv_router_selection(bat_priv, orig_node);
return false;
}
diff --git a/originator.h b/originator.h
index db3a9ed..62e8e68 100644
--- a/originator.h
+++ b/originator.h
@@ -55,6 +55,8 @@ struct batadv_orig_ifinfo *
batadv_orig_ifinfo_new(struct batadv_orig_node *orig_node,
struct batadv_hard_iface *if_outgoing);
void batadv_orig_ifinfo_free_ref(struct batadv_orig_ifinfo *orig_ifinfo);
+void batadv_router_selection(struct batadv_priv *bat_priv,
+ struct batadv_orig_node *orig_node);
int batadv_orig_seq_print_text(struct seq_file *seq, void *offset);
int batadv_orig_hardif_seq_print_text(struct seq_file *seq, void *offset);
diff --git a/types.h b/types.h
index e9149d1..28f2b85 100644
--- a/types.h
+++ b/types.h
@@ -358,9 +358,11 @@ struct batadv_elp_neigh_node {
/**
* batadv_neigh_node_bat_v - B.A.T.M.A.N. V private neighbor information
* @elp_neigh: ELP private neighbour data
+ * @last_recv_seqno: last OGM seqno received through this neighbour
*/
struct batadv_neigh_node_bat_v {
struct batadv_elp_neigh_node *elp_neigh;
+ uint32_t last_recv_seqno;
};
/**
--
1.8.5.3
next prev parent reply other threads:[~2014-02-11 12:48 UTC|newest]
Thread overview: 68+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-02-11 12:48 [B.A.T.M.A.N.] [RFC 00/23] Introducing a new routing protocol: B.A.T.M.A.N. V Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 01/23] batman-adv: invoke ogm_schedule() only when the interface is activated Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 02/23] batman-adv: Add hard_iface specific sysfs wrapper macros for UINT Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 03/23] batman-adv: ELP - adding basic infrastructure Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 04/23] batman-adv: ELP - creating neighbor structures Antonio Quartulli
2014-02-11 15:32 ` Andrew Lunn
2014-02-11 16:02 ` Antonio Quartulli
2014-02-11 16:11 ` Lew Pitcher
2014-02-11 16:26 ` Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 05/23] batman-adv: ELP - exporting neighbor list via debugfs Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 06/23] batman-adv: ELP - adding sysfs parameter for elp interval Antonio Quartulli
2014-02-11 16:59 ` Andrew Lunn
2014-02-11 17:08 ` Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 07/23] batman-adv: OGMv2 - add basic infrastructure Antonio Quartulli
2014-02-11 17:12 ` Andrew Lunn
2014-02-11 17:52 ` Antonio Quartulli
2014-02-12 7:44 ` Andrew Lunn
2014-02-12 7:58 ` Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 08/23] batman-adv: OGMv2 - implement originators logic Antonio Quartulli
2014-02-11 17:22 ` Andrew Lunn
2014-02-11 17:30 ` Antonio Quartulli
2014-02-11 12:48 ` Antonio Quartulli [this message]
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 10/23] batman-adv: split name from variable for uint mesh attributes Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 11/23] batman-adv: add throughput attribute to hard_ifaces Antonio Quartulli
2014-02-12 8:42 ` Andrew Lunn
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 12/23] batman-adv: add base throughput attribute Antonio Quartulli
2014-02-12 8:40 ` Andrew Lunn
2014-02-12 12:20 ` Antonio Quartulli
2014-02-13 9:36 ` Andrew Lunn
2014-02-13 9:53 ` Antonio Quartulli
2014-02-13 9:57 ` Andrew Lunn
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 13/23] batman-adv: add last_unicast_tx to struct neigh_node_elp Antonio Quartulli
2014-02-12 8:49 ` Andrew Lunn
2014-02-12 12:25 ` Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 14/23] batman-adv: ELP - compute the metric based on the estimated throughput Antonio Quartulli
2014-02-12 8:58 ` Andrew Lunn
2014-02-12 12:27 ` Antonio Quartulli
2014-02-12 15:44 ` Antonio Quartulli
2014-02-13 9:45 ` Andrew Lunn
2014-02-13 9:46 ` Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 15/23] batman-adv: ELP - send unicast ELP packets for throughput sampling Antonio Quartulli
2014-02-12 9:12 ` Andrew Lunn
2014-02-12 12:12 ` Antonio Quartulli
2014-02-12 12:54 ` Felix Fietkau
2014-02-12 12:56 ` Antonio Quartulli
2014-02-12 13:02 ` Antonio Quartulli
2014-02-13 9:55 ` Andrew Lunn
2014-02-13 10:02 ` Antonio Quartulli
2014-02-13 10:09 ` Andrew Lunn
2014-02-13 10:13 ` Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 16/23] batman-adv: ELP - read estimated throughput from cfg80211 Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 17/23] batman-adv: ELP - implement dead neigh node detection Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 18/23] batman-adv: ELP - use phydev to determine link characteristics Antonio Quartulli
2014-02-13 8:17 ` Antonio Quartulli
2014-02-13 8:19 ` Antonio Quartulli
2014-02-13 10:52 ` Andrew Lunn
2014-02-13 11:02 ` Antonio Quartulli
2014-02-13 11:44 ` Andrew Lunn
2014-02-14 8:24 ` Antonio Quartulli
2014-02-14 17:38 ` Andrew Lunn
2014-02-14 17:46 ` Antonio Quartulli
2014-02-14 18:18 ` Andrew Lunn
2014-02-14 19:18 ` Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 19/23] batman-adv: add bat_neigh_free() API Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 20/23] batman-adv: B.A.T.M.A.N. V - implement " Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 21/23] batman-adv: B.A.T.M.A.N. V - implement neigh_is_equiv_or_better API Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 22/23] batman-adv: B.A.T.M.A.N. V - implement bat_neigh_cmp API Antonio Quartulli
2014-02-11 12:48 ` [B.A.T.M.A.N.] [RFC 23/23] batman-adv: B.A.T.M.A.N. V - implement bat_orig_print API Antonio Quartulli
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=1392122903-805-10-git-send-email-antonio@meshcoding.com \
--to=antonio@meshcoding.com \
--cc=antonio@open-mesh.com \
--cc=b.a.t.m.a.n@lists.open-mesh.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.