batman-adv: send GW_DEL event when the gw client mode is deselected
authorAntonio Quartulli <antonio@open-mesh.com>
Fri, 12 Jul 2013 22:06:00 +0000 (00:06 +0200)
committerAntonio Quartulli <ordex@autistici.org>
Wed, 28 Aug 2013 09:33:00 +0000 (11:33 +0200)
Whenever the GW client mode is deselected, a DEL event has
to be sent in order to tell userspace that the current
gateway has been lost. Send the uevent on state change only
if a gateway was currently selected.

Reported-by: Marek Lindner <lindner_marek@yahoo.de>
Signed-off-by: Antonio Quartulli <antonio@open-mesh.com>
Signed-off-by: Marek Lindner <lindner_marek@yahoo.de>
net/batman-adv/gateway_client.c
net/batman-adv/gateway_client.h
net/batman-adv/sysfs.c

index 7614af31daff3678eb2bb75447810c0e27850f88..1ce4b8763ef289f3679177c5a1fdc96df6e6126f 100644 (file)
@@ -190,6 +190,33 @@ next:
        return curr_gw;
 }
 
+/**
+ * batadv_gw_check_client_stop - check if client mode has been switched off
+ * @bat_priv: the bat priv with all the soft interface information
+ *
+ * This function assumes the caller has checked that the gw state *is actually
+ * changing*. This function is not supposed to be called when there is no state
+ * change.
+ */
+void batadv_gw_check_client_stop(struct batadv_priv *bat_priv)
+{
+       struct batadv_gw_node *curr_gw;
+
+       if (atomic_read(&bat_priv->gw_mode) != BATADV_GW_MODE_CLIENT)
+               return;
+
+       curr_gw = batadv_gw_get_selected_gw_node(bat_priv);
+       if (!curr_gw)
+               return;
+
+       /* if batman-adv is switching the gw client mode off and a gateway was
+        * already selected, send a DEL uevent
+        */
+       batadv_throw_uevent(bat_priv, BATADV_UEV_GW, BATADV_UEV_DEL, NULL);
+
+       batadv_gw_node_free_ref(curr_gw);
+}
+
 void batadv_gw_election(struct batadv_priv *bat_priv)
 {
        struct batadv_gw_node *curr_gw = NULL, *next_gw = NULL;
index 1037d75da51f35c1867cee72cc64bafd2859e3a8..ceef4ebe8bcd6a89711000c0d52b20e536c5375a 100644 (file)
@@ -20,6 +20,7 @@
 #ifndef _NET_BATMAN_ADV_GATEWAY_CLIENT_H_
 #define _NET_BATMAN_ADV_GATEWAY_CLIENT_H_
 
+void batadv_gw_check_client_stop(struct batadv_priv *bat_priv);
 void batadv_gw_deselect(struct batadv_priv *bat_priv);
 void batadv_gw_election(struct batadv_priv *bat_priv);
 struct batadv_orig_node *
index 929e304dacb254c333dd3dfbb94ba8de832153c6..4114b961bc2c8400acc6bd776e2732313674b0e4 100644 (file)
@@ -385,6 +385,10 @@ static ssize_t batadv_store_gw_mode(struct kobject *kobj,
                    curr_gw_mode_str, buff);
 
        batadv_gw_deselect(bat_priv);
+       /* always call batadv_gw_check_client_stop() before changing the gateway
+        * state
+        */
+       batadv_gw_check_client_stop(bat_priv);
        atomic_set(&bat_priv->gw_mode, (unsigned int)gw_mode_tmp);
        return count;
 }