Search Linux Wireless

Re: [Patch]mac80211: Add support for mesh proxy path dump

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Tested with ath9k and working fine for me.

----
ChunYeow

On Mon, Sep 1, 2014 at 8:03 PM, Henning Rogge <hrogge@xxxxxxxxx> wrote:
> Hi,
>
> I forgot to add that the patch applies to the head of
> wireless-testing. I have also tested it on OpenWRT Barrier Breaker on
> a Ubiquiti Bullet M5 with the ath9k driver.
>
> Henning Rogge
>
> On Mon, Sep 1, 2014 at 1:26 PM, Henning Rogge <hrogge@xxxxxxxxx> wrote:
>> The following patch adds NL80211_CMD_GET_MPP as a new nl80211 command that
>> allows to query the content of the 'mesh proxy path' table of mac80211s via
>> 'get' or 'dump'  operation.
>>
>> Signed-off-by: Henning Rogge <henning.rogge@xxxxxxxxxxxxxxxxxx>
>> ---
>>  include/net/cfg80211.h       |  7 ++++
>>  include/uapi/linux/nl80211.h |  6 +++
>>  net/mac80211/cfg.c           | 53 ++++++++++++++++++++++++
>>  net/mac80211/mesh.h          |  3 ++
>>  net/mac80211/mesh_pathtbl.c  | 31 ++++++++++++++
>>  net/wireless/nl80211.c       | 99 ++++++++++++++++++++++++++++++++++++++++++++
>>  net/wireless/rdev-ops.h      | 27 +++++++++++-
>>  net/wireless/trace.h         | 45 ++++++++++++++++++++
>>  8 files changed, 270 insertions(+), 1 deletion(-)
>>
>> diff --git a/include/net/cfg80211.h b/include/net/cfg80211.h
>> index 0a080c4..68fbcf5 100644
>> --- a/include/net/cfg80211.h
>> +++ b/include/net/cfg80211.h
>> @@ -2139,6 +2139,8 @@ struct cfg80211_qos_map {
>>   * @change_mpath: change a given mesh path
>>   * @get_mpath: get a mesh path for the given parameters
>>   * @dump_mpath: dump mesh path callback -- resume dump at index @idx
>> + * @get_mpp: get a mesh proxy path for the given parameters
>> + * @dump_mpp: dump mesh proxy path callback -- resume dump at index @idx
>>   * @join_mesh: join the mesh network with the specified parameters
>>   *     (invoked with the wireless_dev mutex held)
>>   * @leave_mesh: leave the current mesh network
>> @@ -2378,6 +2380,11 @@ struct cfg80211_ops {
>>         int     (*dump_mpath)(struct wiphy *wiphy, struct net_device *dev,
>>                               int idx, u8 *dst, u8 *next_hop,
>>                               struct mpath_info *pinfo);
>> +       int     (*get_mpp)(struct wiphy *wiphy, struct net_device *dev,
>> +                            u8 *dst, u8 *mpp, struct mpath_info *pinfo);
>> +       int     (*dump_mpp)(struct wiphy *wiphy, struct net_device *dev,
>> +                             int idx, u8 *dst, u8 *mpp,
>> +                             struct mpath_info *pinfo);
>>         int     (*get_mesh_config)(struct wiphy *wiphy,
>>                                 struct net_device *dev,
>>                                 struct mesh_config *conf);
>> diff --git a/include/uapi/linux/nl80211.h b/include/uapi/linux/nl80211.h
>> index f1db15b..80cff48 100644
>> --- a/include/uapi/linux/nl80211.h
>> +++ b/include/uapi/linux/nl80211.h
>> @@ -722,6 +722,10 @@
>>   *     QoS mapping is relevant for IP packets, it is only valid during an
>>   *     association. This is cleared on disassociation and AP restart.
>>   *
>> + * @NL80211_CMD_GET_MPP: Get mesh path attributes for mesh proxy path to
>> + *     destination %NL80211_ATTR_MAC on the interface identified by
>> + *     %NL80211_ATTR_IFINDEX.
>> + *
>>   * @NL80211_CMD_MAX: highest used command number
>>   * @__NL80211_CMD_AFTER_LAST: internal use
>>   */
>> @@ -893,6 +897,8 @@ enum nl80211_commands {
>>
>>         NL80211_CMD_SET_QOS_MAP,
>>
>> +       NL80211_CMD_GET_MPP,
>> +
>>         /* add new commands above here */
>>
>>         /* used to define NL80211_CMD_MAX below */
>> diff --git a/net/mac80211/cfg.c b/net/mac80211/cfg.c
>> index 927b4ea..2b85eed 100644
>> --- a/net/mac80211/cfg.c
>> +++ b/net/mac80211/cfg.c
>> @@ -1511,6 +1511,57 @@ static int ieee80211_dump_mpath(struct wiphy *wiphy, struct net_device *dev,
>>         return 0;
>>  }
>>
>> +static void mpp_set_pinfo(struct mesh_path *mpath, u8 *mpp,
>> +                         struct mpath_info *pinfo)
>> +{
>> +       memset(pinfo, 0, sizeof(*pinfo));
>> +       memcpy(mpp, mpath->mpp, ETH_ALEN);
>> +
>> +       pinfo->generation = mpp_paths_generation;
>> +}
>> +
>> +static int ieee80211_get_mpp(struct wiphy *wiphy, struct net_device *dev,
>> +                            u8 *dst, u8 *mpp, struct mpath_info *pinfo)
>> +
>> +{
>> +       struct ieee80211_sub_if_data *sdata;
>> +       struct mesh_path *mpath;
>> +
>> +       sdata = IEEE80211_DEV_TO_SUB_IF(dev);
>> +
>> +       rcu_read_lock();
>> +       mpath = mpp_path_lookup(sdata, dst);
>> +       if (!mpath) {
>> +               rcu_read_unlock();
>> +               return -ENOENT;
>> +       }
>> +       memcpy(dst, mpath->dst, ETH_ALEN);
>> +       mpp_set_pinfo(mpath, mpp, pinfo);
>> +       rcu_read_unlock();
>> +       return 0;
>> +}
>> +
>> +static int ieee80211_dump_mpp(struct wiphy *wiphy, struct net_device *dev,
>> +                             int idx, u8 *dst, u8 *mpp,
>> +                             struct mpath_info *pinfo)
>> +{
>> +       struct ieee80211_sub_if_data *sdata;
>> +       struct mesh_path *mpath;
>> +
>> +       sdata = IEEE80211_DEV_TO_SUB_IF(dev);
>> +
>> +       rcu_read_lock();
>> +       mpath = mpp_path_lookup_by_idx(sdata, idx);
>> +       if (!mpath) {
>> +               rcu_read_unlock();
>> +               return -ENOENT;
>> +       }
>> +       memcpy(dst, mpath->dst, ETH_ALEN);
>> +       mpp_set_pinfo(mpath, mpp, pinfo);
>> +       rcu_read_unlock();
>> +       return 0;
>> +}
>> +
>>  static int ieee80211_get_mesh_config(struct wiphy *wiphy,
>>                                 struct net_device *dev,
>>                                 struct mesh_config *conf)
>> @@ -3507,6 +3558,8 @@ const struct cfg80211_ops mac80211_config_ops = {
>>         .change_mpath = ieee80211_change_mpath,
>>         .get_mpath = ieee80211_get_mpath,
>>         .dump_mpath = ieee80211_dump_mpath,
>> +       .get_mpp = ieee80211_get_mpp,
>> +       .dump_mpp = ieee80211_dump_mpp,
>>         .update_mesh_config = ieee80211_update_mesh_config,
>>         .get_mesh_config = ieee80211_get_mesh_config,
>>         .join_mesh = ieee80211_join_mesh,
>> diff --git a/net/mac80211/mesh.h b/net/mac80211/mesh.h
>> index f39a19f..50c8473 100644
>> --- a/net/mac80211/mesh.h
>> +++ b/net/mac80211/mesh.h
>> @@ -270,6 +270,8 @@ int mpp_path_add(struct ieee80211_sub_if_data *sdata,
>>                  const u8 *dst, const u8 *mpp);
>>  struct mesh_path *
>>  mesh_path_lookup_by_idx(struct ieee80211_sub_if_data *sdata, int idx);
>> +struct mesh_path *
>> +mpp_path_lookup_by_idx(struct ieee80211_sub_if_data *sdata, int idx);
>>  void mesh_path_fix_nexthop(struct mesh_path *mpath, struct sta_info *next_hop);
>>  void mesh_path_expire(struct ieee80211_sub_if_data *sdata);
>>  void mesh_rx_path_sel_frame(struct ieee80211_sub_if_data *sdata,
>> @@ -317,6 +319,7 @@ void mesh_path_tx_root_frame(struct ieee80211_sub_if_data *sdata);
>>
>>  bool mesh_action_is_path_sel(struct ieee80211_mgmt *mgmt);
>>  extern int mesh_paths_generation;
>> +extern int mpp_paths_generation;
>>
>>  #ifdef CONFIG_MAC80211_MESH
>>  static inline
>> diff --git a/net/mac80211/mesh_pathtbl.c b/net/mac80211/mesh_pathtbl.c
>> index cf032a8..8630963 100644
>> --- a/net/mac80211/mesh_pathtbl.c
>> +++ b/net/mac80211/mesh_pathtbl.c
>> @@ -44,6 +44,7 @@ static struct mesh_table __rcu *mesh_paths;
>>  static struct mesh_table __rcu *mpp_paths; /* Store paths for MPP&MAP */
>>
>>  int mesh_paths_generation;
>> +int mpp_paths_generation;
>>
>>  /* This lock will have the grow table function as writer and add / delete nodes
>>   * as readers. RCU provides sufficient protection only when reading the table
>> @@ -410,6 +411,33 @@ mesh_path_lookup_by_idx(struct ieee80211_sub_if_data *sdata, int idx)
>>  }
>>
>>  /**
>> + * mpp_path_lookup_by_idx - look up a path in the proxy path table by its index
>> + * @idx: index
>> + * @sdata: local subif, or NULL for all entries
>> + *
>> + * Returns: pointer to the proxy path structure, or NULL if not found.
>> + *
>> + * Locking: must be called within a read rcu section.
>> + */
>> +struct mesh_path *
>> +mpp_path_lookup_by_idx(struct ieee80211_sub_if_data *sdata, int idx)
>> +{
>> +       struct mesh_table *tbl = rcu_dereference(mpp_paths);
>> +       struct mpath_node *node;
>> +       int i;
>> +       int j = 0;
>> +
>> +       for_each_mesh_entry(tbl, node, i) {
>> +               if (sdata && node->mpath->sdata != sdata)
>> +                       continue;
>> +               if (j++ == idx)
>> +                       return node->mpath;
>> +       }
>> +
>> +       return NULL;
>> +}
>> +
>> +/**
>>   * mesh_path_add_gate - add the given mpath to a mesh gate to our path table
>>   * @mpath: gate path to add to table
>>   */
>> @@ -691,6 +719,9 @@ int mpp_path_add(struct ieee80211_sub_if_data *sdata,
>>
>>         spin_unlock(&tbl->hashwlock[hash_idx]);
>>         read_unlock_bh(&pathtbl_resize_lock);
>> +
>> +       mpp_paths_generation++;
>> +
>>         if (grow) {
>>                 set_bit(MESH_WORK_GROW_MPP_TABLE,  &ifmsh->wrkq_flags);
>>                 ieee80211_queue_work(&local->hw, &sdata->work);
>> diff --git a/net/wireless/nl80211.c b/net/wireless/nl80211.c
>> index df7b133..1e33f2f 100644
>> --- a/net/wireless/nl80211.c
>> +++ b/net/wireless/nl80211.c
>> @@ -4581,6 +4581,96 @@ static int nl80211_del_mpath(struct sk_buff *skb, struct genl_info *info)
>>         return rdev_del_mpath(rdev, dev, dst);
>>  }
>>
>> +static int nl80211_get_mpp(struct sk_buff *skb, struct genl_info *info)
>> +{
>> +       struct cfg80211_registered_device *rdev = info->user_ptr[0];
>> +       int err;
>> +       struct net_device *dev = info->user_ptr[1];
>> +       struct mpath_info pinfo;
>> +       struct sk_buff *msg;
>> +       u8 *dst = NULL;
>> +       u8 mpp[ETH_ALEN];
>> +
>> +       memset(&pinfo, 0, sizeof(pinfo));
>> +
>> +       if (!info->attrs[NL80211_ATTR_MAC])
>> +               return -EINVAL;
>> +
>> +       dst = nla_data(info->attrs[NL80211_ATTR_MAC]);
>> +
>> +       if (!rdev->ops->get_mpp)
>> +               return -EOPNOTSUPP;
>> +
>> +       if (dev->ieee80211_ptr->iftype != NL80211_IFTYPE_MESH_POINT)
>> +               return -EOPNOTSUPP;
>> +
>> +       err = rdev_get_mpp(rdev, dev, dst, mpp, &pinfo);
>> +       if (err)
>> +               return err;
>> +
>> +       msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
>> +       if (!msg)
>> +               return -ENOMEM;
>> +
>> +       if (nl80211_send_mpath(msg, info->snd_portid, info->snd_seq, 0,
>> +                              dev, dst, mpp, &pinfo) < 0) {
>> +               nlmsg_free(msg);
>> +               return -ENOBUFS;
>> +       }
>> +
>> +       return genlmsg_reply(msg, info);
>> +}
>> +
>> +static int nl80211_dump_mpp(struct sk_buff *skb,
>> +                           struct netlink_callback *cb)
>> +{
>> +       struct mpath_info pinfo;
>> +       struct cfg80211_registered_device *rdev;
>> +       struct wireless_dev *wdev;
>> +       u8 dst[ETH_ALEN];
>> +       u8 mpp[ETH_ALEN];
>> +       int path_idx = cb->args[2];
>> +       int err;
>> +
>> +       err = nl80211_prepare_wdev_dump(skb, cb, &rdev, &wdev);
>> +       if (err)
>> +               return err;
>> +
>> +       if (!rdev->ops->dump_mpp) {
>> +               err = -EOPNOTSUPP;
>> +               goto out_err;
>> +       }
>> +
>> +       if (wdev->iftype != NL80211_IFTYPE_MESH_POINT) {
>> +               err = -EOPNOTSUPP;
>> +               goto out_err;
>> +       }
>> +
>> +       while (1) {
>> +               err = rdev_dump_mpp(rdev, wdev->netdev, path_idx, dst,
>> +                                   mpp, &pinfo);
>> +               if (err == -ENOENT)
>> +                       break;
>> +               if (err)
>> +                       goto out_err;
>> +
>> +               if (nl80211_send_mpath(skb, NETLINK_CB(cb->skb).portid,
>> +                                      cb->nlh->nlmsg_seq, NLM_F_MULTI,
>> +                                      wdev->netdev, dst, mpp,
>> +                                      &pinfo) < 0)
>> +                       goto out;
>> +
>> +               path_idx++;
>> +       }
>> +
>> + out:
>> +       cb->args[2] = path_idx;
>> +       err = skb->len;
>> + out_err:
>> +       nl80211_finish_wdev_dump(rdev);
>> +       return err;
>> +}
>> +
>>  static int nl80211_set_bss(struct sk_buff *skb, struct genl_info *info)
>>  {
>>         struct cfg80211_registered_device *rdev = info->user_ptr[0];
>> @@ -9603,6 +9693,15 @@ static const struct genl_ops nl80211_ops[] = {
>>                                   NL80211_FLAG_NEED_RTNL,
>>         },
>>         {
>> +               .cmd = NL80211_CMD_GET_MPP,
>> +               .doit = nl80211_get_mpp,
>> +               .dumpit = nl80211_dump_mpp,
>> +               .policy = nl80211_policy,
>> +               .flags = GENL_ADMIN_PERM,
>> +               .internal_flags = NL80211_FLAG_NEED_NETDEV_UP |
>> +                                 NL80211_FLAG_NEED_RTNL,
>> +       },
>> +       {
>>                 .cmd = NL80211_CMD_SET_MPATH,
>>                 .doit = nl80211_set_mpath,
>>                 .policy = nl80211_policy,
>> diff --git a/net/wireless/rdev-ops.h b/net/wireless/rdev-ops.h
>> index 56c2240..e5560d5 100644
>> --- a/net/wireless/rdev-ops.h
>> +++ b/net/wireless/rdev-ops.h
>> @@ -263,6 +263,18 @@ static inline int rdev_get_mpath(struct cfg80211_registered_device *rdev,
>>
>>  }
>>
>> +static inline int rdev_get_mpp(struct cfg80211_registered_device *rdev,
>> +                              struct net_device *dev, u8 *dst, u8 *mpp,
>> +                              struct mpath_info *pinfo)
>> +{
>> +       int ret;
>> +
>> +       trace_rdev_get_mpp(&rdev->wiphy, dev, dst, mpp);
>> +       ret = rdev->ops->get_mpp(&rdev->wiphy, dev, dst, mpp, pinfo);
>> +       trace_rdev_return_int_mpath_info(&rdev->wiphy, ret, pinfo);
>> +       return ret;
>> +}
>> +
>>  static inline int rdev_dump_mpath(struct cfg80211_registered_device *rdev,
>>                                   struct net_device *dev, int idx, u8 *dst,
>>                                   u8 *next_hop, struct mpath_info *pinfo)
>> @@ -271,7 +283,20 @@ static inline int rdev_dump_mpath(struct cfg80211_registered_device *rdev,
>>         int ret;
>>         trace_rdev_dump_mpath(&rdev->wiphy, dev, idx, dst, next_hop);
>>         ret = rdev->ops->dump_mpath(&rdev->wiphy, dev, idx, dst, next_hop,
>> -                                    pinfo);
>> +                                   pinfo);
>> +       trace_rdev_return_int_mpath_info(&rdev->wiphy, ret, pinfo);
>> +       return ret;
>> +}
>> +
>> +static inline int rdev_dump_mpp(struct cfg80211_registered_device *rdev,
>> +                               struct net_device *dev, int idx, u8 *dst,
>> +                               u8 *mpp, struct mpath_info *pinfo)
>> +
>> +{
>> +       int ret;
>> +
>> +       trace_rdev_dump_mpp(&rdev->wiphy, dev, idx, dst, mpp);
>> +       ret = rdev->ops->dump_mpp(&rdev->wiphy, dev, idx, dst, mpp, pinfo);
>>         trace_rdev_return_int_mpath_info(&rdev->wiphy, ret, pinfo);
>>         return ret;
>>  }
>> diff --git a/net/wireless/trace.h b/net/wireless/trace.h
>> index 0c524cd..57ab727 100644
>> --- a/net/wireless/trace.h
>> +++ b/net/wireless/trace.h
>> @@ -801,6 +801,51 @@ TRACE_EVENT(rdev_dump_mpath,
>>                   MAC_PR_ARG(next_hop))
>>  );
>>
>> +TRACE_EVENT(rdev_get_mpp,
>> +       TP_PROTO(struct wiphy *wiphy, struct net_device *netdev,
>> +                u8 *dst, u8 *mpp),
>> +       TP_ARGS(wiphy, netdev, dst, mpp),
>> +       TP_STRUCT__entry(
>> +               WIPHY_ENTRY
>> +               NETDEV_ENTRY
>> +               MAC_ENTRY(dst)
>> +               MAC_ENTRY(mpp)
>> +       ),
>> +       TP_fast_assign(
>> +               WIPHY_ASSIGN;
>> +               NETDEV_ASSIGN;
>> +               MAC_ASSIGN(dst, dst);
>> +               MAC_ASSIGN(mpp, mpp);
>> +       ),
>> +       TP_printk(WIPHY_PR_FMT ", " NETDEV_PR_FMT ", destination: " MAC_PR_FMT
>> +                 ", mpp: " MAC_PR_FMT, WIPHY_PR_ARG, NETDEV_PR_ARG,
>> +                 MAC_PR_ARG(dst), MAC_PR_ARG(mpp))
>> +);
>> +
>> +TRACE_EVENT(rdev_dump_mpp,
>> +       TP_PROTO(struct wiphy *wiphy, struct net_device *netdev, int idx,
>> +                u8 *dst, u8 *mpp),
>> +       TP_ARGS(wiphy, netdev, idx, mpp, dst),
>> +       TP_STRUCT__entry(
>> +               WIPHY_ENTRY
>> +               NETDEV_ENTRY
>> +               MAC_ENTRY(dst)
>> +               MAC_ENTRY(mpp)
>> +               __field(int, idx)
>> +       ),
>> +       TP_fast_assign(
>> +               WIPHY_ASSIGN;
>> +               NETDEV_ASSIGN;
>> +               MAC_ASSIGN(dst, dst);
>> +               MAC_ASSIGN(mpp, mpp);
>> +               __entry->idx = idx;
>> +       ),
>> +       TP_printk(WIPHY_PR_FMT ", " NETDEV_PR_FMT ", index: %d, destination: "
>> +                 MAC_PR_FMT ", mpp: " MAC_PR_FMT,
>> +                 WIPHY_PR_ARG, NETDEV_PR_ARG, __entry->idx, MAC_PR_ARG(dst),
>> +                 MAC_PR_ARG(mpp))
>> +);
>> +
>>  TRACE_EVENT(rdev_return_int_mpath_info,
>>         TP_PROTO(struct wiphy *wiphy, int ret, struct mpath_info *pinfo),
>>         TP_ARGS(wiphy, ret, pinfo),
>> --
>> 1.9.1
>>
>>
--
To unsubscribe from this list: send the line "unsubscribe linux-wireless" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html




[Index of Archives]     [Linux Host AP]     [ATH6KL]     [Linux Wireless Personal Area Network]     [Linux Bluetooth]     [Linux Netdev]     [Kernel Newbies]     [Linux Kernel]     [IDE]     [Git]     [Netfilter]     [Bugtraq]     [Yosemite Hiking]     [MIPS Linux]     [ARM Linux]     [Linux RAID]

  Powered by Linux