|
@@ -4624,6 +4624,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];
|
|
@@ -9773,6 +9863,15 @@ static const struct genl_ops nl80211_ops[] = {
|
|
|
.internal_flags = NL80211_FLAG_NEED_NETDEV_UP |
|
|
|
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,
|