aboutsummaryrefslogtreecommitdiffstats
path: root/net/wireless
diff options
context:
space:
mode:
Diffstat (limited to 'net/wireless')
-rw-r--r--net/wireless/nl80211.c61
1 files changed, 38 insertions, 23 deletions
diff --git a/net/wireless/nl80211.c b/net/wireless/nl80211.c
index cc2e5d6163de..c8d4d53fc450 100644
--- a/net/wireless/nl80211.c
+++ b/net/wireless/nl80211.c
@@ -2613,14 +2613,6 @@ static int nl80211_get_mesh_params(struct sk_buff *skb,
2613 return -ENOBUFS; 2613 return -ENOBUFS;
2614} 2614}
2615 2615
2616#define FILL_IN_MESH_PARAM_IF_SET(table, cfg, param, mask, attr_num, nla_fn) \
2617do {\
2618 if (table[attr_num]) {\
2619 cfg.param = nla_fn(table[attr_num]); \
2620 mask |= (1 << (attr_num - 1)); \
2621 } \
2622} while (0);\
2623
2624static const struct nla_policy nl80211_meshconf_params_policy[NL80211_MESHCONF_ATTR_MAX+1] = { 2616static const struct nla_policy nl80211_meshconf_params_policy[NL80211_MESHCONF_ATTR_MAX+1] = {
2625 [NL80211_MESHCONF_RETRY_TIMEOUT] = { .type = NLA_U16 }, 2617 [NL80211_MESHCONF_RETRY_TIMEOUT] = { .type = NLA_U16 },
2626 [NL80211_MESHCONF_CONFIRM_TIMEOUT] = { .type = NLA_U16 }, 2618 [NL80211_MESHCONF_CONFIRM_TIMEOUT] = { .type = NLA_U16 },
@@ -2639,31 +2631,34 @@ static const struct nla_policy nl80211_meshconf_params_policy[NL80211_MESHCONF_A
2639 [NL80211_MESHCONF_HWMP_NET_DIAM_TRVS_TIME] = { .type = NLA_U16 }, 2631 [NL80211_MESHCONF_HWMP_NET_DIAM_TRVS_TIME] = { .type = NLA_U16 },
2640}; 2632};
2641 2633
2642static int nl80211_set_mesh_params(struct sk_buff *skb, struct genl_info *info) 2634static int nl80211_parse_mesh_params(struct genl_info *info,
2635 struct mesh_config *cfg,
2636 u32 *mask_out)
2643{ 2637{
2644 u32 mask;
2645 struct cfg80211_registered_device *rdev = info->user_ptr[0];
2646 struct net_device *dev = info->user_ptr[1];
2647 struct mesh_config cfg;
2648 struct nlattr *tb[NL80211_MESHCONF_ATTR_MAX + 1]; 2638 struct nlattr *tb[NL80211_MESHCONF_ATTR_MAX + 1];
2649 struct nlattr *parent_attr; 2639 u32 mask = 0;
2650 2640
2651 parent_attr = info->attrs[NL80211_ATTR_MESH_PARAMS]; 2641#define FILL_IN_MESH_PARAM_IF_SET(table, cfg, param, mask, attr_num, nla_fn) \
2652 if (!parent_attr) 2642do {\
2643 if (table[attr_num]) {\
2644 cfg->param = nla_fn(table[attr_num]); \
2645 mask |= (1 << (attr_num - 1)); \
2646 } \
2647} while (0);\
2648
2649
2650 if (!info->attrs[NL80211_ATTR_MESH_PARAMS])
2653 return -EINVAL; 2651 return -EINVAL;
2654 if (nla_parse_nested(tb, NL80211_MESHCONF_ATTR_MAX, 2652 if (nla_parse_nested(tb, NL80211_MESHCONF_ATTR_MAX,
2655 parent_attr, nl80211_meshconf_params_policy)) 2653 info->attrs[NL80211_ATTR_MESH_PARAMS],
2654 nl80211_meshconf_params_policy))
2656 return -EINVAL; 2655 return -EINVAL;
2657 2656
2658 if (!rdev->ops->set_mesh_params)
2659 return -EOPNOTSUPP;
2660
2661 /* This makes sure that there aren't more than 32 mesh config 2657 /* This makes sure that there aren't more than 32 mesh config
2662 * parameters (otherwise our bitfield scheme would not work.) */ 2658 * parameters (otherwise our bitfield scheme would not work.) */
2663 BUILD_BUG_ON(NL80211_MESHCONF_ATTR_MAX > 32); 2659 BUILD_BUG_ON(NL80211_MESHCONF_ATTR_MAX > 32);
2664 2660
2665 /* Fill in the params struct */ 2661 /* Fill in the params struct */
2666 mask = 0;
2667 FILL_IN_MESH_PARAM_IF_SET(tb, cfg, dot11MeshRetryTimeout, 2662 FILL_IN_MESH_PARAM_IF_SET(tb, cfg, dot11MeshRetryTimeout,
2668 mask, NL80211_MESHCONF_RETRY_TIMEOUT, nla_get_u16); 2663 mask, NL80211_MESHCONF_RETRY_TIMEOUT, nla_get_u16);
2669 FILL_IN_MESH_PARAM_IF_SET(tb, cfg, dot11MeshConfirmTimeout, 2664 FILL_IN_MESH_PARAM_IF_SET(tb, cfg, dot11MeshConfirmTimeout,
@@ -2703,12 +2698,32 @@ static int nl80211_set_mesh_params(struct sk_buff *skb, struct genl_info *info)
2703 NL80211_MESHCONF_HWMP_ROOTMODE, 2698 NL80211_MESHCONF_HWMP_ROOTMODE,
2704 nla_get_u8); 2699 nla_get_u8);
2705 2700
2701 if (mask_out)
2702 *mask_out = mask;
2703 return 0;
2704
2705#undef FILL_IN_MESH_PARAM_IF_SET
2706}
2707
2708static int nl80211_set_mesh_params(struct sk_buff *skb, struct genl_info *info)
2709{
2710 struct cfg80211_registered_device *rdev = info->user_ptr[0];
2711 struct net_device *dev = info->user_ptr[1];
2712 struct mesh_config cfg;
2713 u32 mask;
2714 int err;
2715
2716 if (!rdev->ops->set_mesh_params)
2717 return -EOPNOTSUPP;
2718
2719 err = nl80211_parse_mesh_params(info, &cfg, &mask);
2720 if (err)
2721 return err;
2722
2706 /* Apply changes */ 2723 /* Apply changes */
2707 return rdev->ops->set_mesh_params(&rdev->wiphy, dev, &cfg, mask); 2724 return rdev->ops->set_mesh_params(&rdev->wiphy, dev, &cfg, mask);
2708} 2725}
2709 2726
2710#undef FILL_IN_MESH_PARAM_IF_SET
2711
2712static int nl80211_get_reg(struct sk_buff *skb, struct genl_info *info) 2727static int nl80211_get_reg(struct sk_buff *skb, struct genl_info *info)
2713{ 2728{
2714 struct sk_buff *msg; 2729 struct sk_buff *msg;