From: Antonio Quartulli antonio@open-mesh.com
Since batman-adv is now fully VLAN-aware, a proper framework able to handle per-vlan-interface attributes is now needed. Those attributes will affect the associated VLAN interface only, rather than the real soft_iface (which would result in every vlan interface having the same attribute configuration)
Signed-off-by: Antonio Quartulli antonio@open-mesh.com --- compat.c | 9 ++++++++ main.c | 44 ++++++++++++++++++++++++++++++++++++ main.h | 3 +++ soft-interface.c | 69 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++ types.h | 21 +++++++++++++++++ 5 files changed, 146 insertions(+)
diff --git a/compat.c b/compat.c index da556a4..b99d505 100644 --- a/compat.c +++ b/compat.c @@ -27,6 +27,15 @@
#if LINUX_VERSION_CODE < KERNEL_VERSION(3, 0, 0)
+void batadv_free_rcu_priv_vlan(struct rcu_head *rcu) +{ + struct batadv_priv_vlan *vlan; + + vlan = container_of(rcu, struct batadv_priv_vlan, rcu); + + kfree(vlan); +} + void batadv_free_rcu_tt_global_entry(struct rcu_head *rcu) { struct batadv_tt_global_entry *global; diff --git a/main.c b/main.c index 08ee7f3..41a9996 100644 --- a/main.c +++ b/main.c @@ -109,6 +109,7 @@ int batadv_mesh_init(struct net_device *soft_iface) spin_lock_init(&bat_priv->gw.list_lock); spin_lock_init(&bat_priv->tvlv.container_list_lock); spin_lock_init(&bat_priv->tvlv.handler_list_lock); + spin_lock_init(&bat_priv->priv_vlan_list_lock);
INIT_HLIST_HEAD(&bat_priv->forw_bat_list); INIT_HLIST_HEAD(&bat_priv->forw_bcast_list); @@ -118,6 +119,7 @@ int batadv_mesh_init(struct net_device *soft_iface) INIT_LIST_HEAD(&bat_priv->tt.roam_list); INIT_HLIST_HEAD(&bat_priv->tvlv.container_list); INIT_HLIST_HEAD(&bat_priv->tvlv.handler_list); + INIT_LIST_HEAD(&bat_priv->priv_vlan_list);
ret = batadv_originator_init(bat_priv); if (ret < 0) @@ -1136,6 +1138,48 @@ unsigned short batadv_get_vid(struct sk_buff *skb, size_t header_len) return vid; }
+/** + * batadv_priv_vlan_free_ref - decrease vlan refcounter and possibly free it + * @priv_vlan: the vlan object to release + */ +void batadv_priv_vlan_free_ref(struct batadv_priv_vlan *priv_vlan) +{ + if (atomic_dec_and_test(&priv_vlan->refcount)) + kfree_rcu(priv_vlan, rcu); +} + +/** + * batadv_get_priv_vlan - get VLAN priv data + * @bat_priv: the bat priv with all the soft interface information + * @vid: the identifier of the VLAN data to retrieve + * + * Return the private data of the VLAN matching vid and increments its + * refcounter or NULL otherwise. + */ +struct batadv_priv_vlan *batadv_get_priv_vlan(struct batadv_priv *bat_priv, + unsigned short vid) +{ + struct batadv_priv_vlan *vlan, *vlan_tmp = NULL; + + if (!(vid & BATADV_VLAN_HAS_TAG)) + return NULL; + + rcu_read_lock(); + list_for_each_entry_rcu(vlan, &bat_priv->priv_vlan_list, list) { + if (vlan->vid != (vid & VLAN_VID_MASK)) + continue; + + if (!atomic_inc_not_zero(&vlan->refcount)) + continue; + + vlan_tmp = vlan; + break; + } + rcu_read_unlock(); + + return vlan_tmp; +} + static int batadv_param_set_ra(const char *val, const struct kernel_param *kp) { struct batadv_algo_ops *bat_algo_ops; diff --git a/main.h b/main.h index fb46cbb..c4fbe9f 100644 --- a/main.h +++ b/main.h @@ -197,6 +197,9 @@ int batadv_algo_register(struct batadv_algo_ops *bat_algo_ops); int batadv_algo_select(struct batadv_priv *bat_priv, char *name); int batadv_algo_seq_print_text(struct seq_file *seq, void *offset); __be32 batadv_skb_crc32(struct sk_buff *skb, u8 *payload_ptr); +void batadv_priv_vlan_free_ref(struct batadv_priv_vlan *priv_vlan); +struct batadv_priv_vlan *batadv_get_priv_vlan(struct batadv_priv *bat_priv, + unsigned short vid);
/** * enum batadv_dbg_level - available log levels diff --git a/soft-interface.c b/soft-interface.c index 3162ccb..c90f8fa 100644 --- a/soft-interface.c +++ b/soft-interface.c @@ -372,6 +372,72 @@ out: return; }
+/** + * batadv_add_vid - ndo_add_vid API implementation + * @dev: the netdev of the mesh interface + * @vid: identifier of the new VLAN + * + * Set up all the internal structures for handling the new VLAN on top of the + * mesh interface + */ +static int batadv_add_vid(struct net_device *dev, unsigned short vid) +{ + struct batadv_priv *bat_priv = netdev_priv(dev); + struct batadv_priv_vlan *vlan; + + vlan = kmalloc(sizeof(*vlan), GFP_ATOMIC); + if (!vlan) + return -ENOMEM; + + vlan->vid = vid; + atomic_set(&vlan->refcount, 1); + + /* add a new TT local entry. This one will be marked with the NOPURGE + * flag + */ + batadv_tt_local_add(dev, dev->dev_addr, vid | BATADV_VLAN_HAS_TAG, + BATADV_NULL_IFINDEX); + + spin_lock_bh(&bat_priv->priv_vlan_list_lock); + list_add_rcu(&vlan->list, &bat_priv->priv_vlan_list); + spin_unlock_bh(&bat_priv->priv_vlan_list_lock); + + return 0; +} + +/** + * batadv_kill_vid - ndo_kill_vid API implementation + * @dev: the netdev of the mesh interface + * @vid: identifier of the deleted VLAN + * + * Destroy all the internal structures used to handle the VLAN identified by vid + * on top of the mesh interface + */ +static int batadv_kill_vid(struct net_device *dev, unsigned short vid) +{ + struct batadv_priv *bat_priv = netdev_priv(dev); + struct batadv_priv_vlan *vlan, *vlan_tmp = NULL; + + list_for_each_entry_rcu(vlan, &bat_priv->priv_vlan_list, list) { + if (vlan->vid != vid) + continue; + vlan_tmp = vlan; + break; + } + + spin_lock_bh(&bat_priv->priv_vlan_list_lock); + list_del_rcu(&vlan->list); + spin_unlock_bh(&bat_priv->priv_vlan_list_lock); + + /* explicitly remove the associated TT local entry because it is marked + * with the NOPURGE flag + */ + batadv_tt_local_remove(bat_priv, dev->dev_addr, vid, + "vlan interface destroyed", false); + + return 0; +} + /* batman-adv network devices have devices nesting below it and are a special * "super class" of normal network devices; split their locks off into a * separate class since they always nest. @@ -573,6 +639,8 @@ static const struct net_device_ops batadv_netdev_ops = { .ndo_open = batadv_interface_open, .ndo_stop = batadv_interface_release, .ndo_get_stats = batadv_interface_stats, + .ndo_vlan_rx_add_vid = batadv_add_vid, + .ndo_vlan_rx_kill_vid = batadv_kill_vid, .ndo_set_mac_address = batadv_interface_set_mac_addr, .ndo_change_mtu = batadv_interface_change_mtu, .ndo_start_xmit = batadv_interface_tx, @@ -611,6 +679,7 @@ static void batadv_softif_init_early(struct net_device *dev)
dev->netdev_ops = &batadv_netdev_ops; dev->destructor = batadv_softif_free; + dev->features |= NETIF_F_HW_VLAN_FILTER; dev->tx_queue_len = 0;
/* can't call min_mtu, because the needed variables diff --git a/types.h b/types.h index 736f822..e737423 100644 --- a/types.h +++ b/types.h @@ -537,6 +537,22 @@ struct batadv_priv_nc { struct batadv_hashtable *decoding_hash; };
+/* + * struct batadv_priv_vlan - per VLAN attributes set + * @vid: VLAN identifier + * @kobj: kobject for sysfs vlan subdirectory + * @list: list node for bat_priv::priv_vlan_list + * @refcount: number of context where this object is currently in use + * @rcu: struct used for freeing in a RCU-safe manner + */ +struct batadv_priv_vlan { + unsigned short vid; + struct kobject *kobj; + struct list_head list; + atomic_t refcount; + struct rcu_head rcu; +}; + /** * struct batadv_priv - per mesh interface data * @mesh_state: current status of the mesh (inactive/active/deactivating) @@ -573,6 +589,9 @@ struct batadv_priv_nc { * @primary_if: one of the hard interfaces assigned to this mesh interface * becomes the primary interface * @bat_algo_ops: routing algorithm used by this mesh interface + * @priv_vlan_list: a list of priv_vlan structs, one per VLAN created on top of + * the mesh interface represented by this object + * @priv_vlan_list_lock: lock protecting priv_vlan_list * @bla: bridge loope avoidance data * @debug_log: holding debug logging relevant data * @gw: gateway data @@ -620,6 +639,8 @@ struct batadv_priv { struct work_struct cleanup_work; struct batadv_hard_iface __rcu *primary_if; /* rcu protected pointer */ struct batadv_algo_ops *bat_algo_ops; + struct list_head priv_vlan_list; + spinlock_t priv_vlan_list_lock; /* protects priv_vlan_list */ #ifdef CONFIG_BATMAN_ADV_BLA struct batadv_priv_bla bla; #endif