Commit 62759361 authored by Jukka Rissanen's avatar Jukka Rissanen Committed by Johannes Berg

mac80211-hwsim: Provide multicast event for HWSIM_CMD_NEW_RADIO

When adding new radio via HWSIM_CMD_NEW_RADIO then listeners on the
multicast group "config" are informed.

While at it, refactor the configuration parameters to be able to
pass them directly and have fewer function arguments.
Signed-off-by: default avatarJukka Rissanen <jukka.rissanen@linux.intel.com>
Signed-off-by: default avatarJohannes Berg <johannes.berg@intel.com>
parent de29eda8
...@@ -487,6 +487,14 @@ static struct genl_family hwsim_genl_family = { ...@@ -487,6 +487,14 @@ static struct genl_family hwsim_genl_family = {
.maxattr = HWSIM_ATTR_MAX, .maxattr = HWSIM_ATTR_MAX,
}; };
enum hwsim_multicast_groups {
HWSIM_MCGRP_CONFIG,
};
static const struct genl_multicast_group hwsim_mcgrps[] = {
[HWSIM_MCGRP_CONFIG] = { .name = "config", },
};
/* MAC80211_HWSIM netlink policy */ /* MAC80211_HWSIM netlink policy */
static const struct nla_policy hwsim_genl_policy[HWSIM_ATTR_MAX + 1] = { static const struct nla_policy hwsim_genl_policy[HWSIM_ATTR_MAX + 1] = {
...@@ -2025,12 +2033,124 @@ static const struct ieee80211_ops mac80211_hwsim_ops = { ...@@ -2025,12 +2033,124 @@ static const struct ieee80211_ops mac80211_hwsim_ops = {
static struct ieee80211_ops mac80211_hwsim_mchan_ops; static struct ieee80211_ops mac80211_hwsim_mchan_ops;
static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2, struct hwsim_new_radio_params {
const struct ieee80211_regdomain *regd, unsigned int channels;
bool reg_strict, bool p2p_device, const char *reg_alpha2;
bool use_chanctx, bool destroy_on_close, const struct ieee80211_regdomain *regd;
u32 portid, const char *hwname, bool reg_strict;
bool no_vif) bool p2p_device;
bool use_chanctx;
bool destroy_on_close;
const char *hwname;
bool no_vif;
};
static void hwsim_mcast_config_msg(struct sk_buff *mcast_skb,
struct genl_info *info)
{
if (info)
genl_notify(&hwsim_genl_family, mcast_skb,
genl_info_net(info), info->snd_portid,
HWSIM_MCGRP_CONFIG, info->nlhdr, GFP_KERNEL);
else
genlmsg_multicast(&hwsim_genl_family, mcast_skb, 0,
HWSIM_MCGRP_CONFIG, GFP_KERNEL);
}
static struct sk_buff *build_radio_msg(int cmd, int id,
struct hwsim_new_radio_params *param)
{
struct sk_buff *skb;
void *data;
int ret;
skb = genlmsg_new(GENLMSG_DEFAULT_SIZE, GFP_KERNEL);
if (!skb)
return NULL;
data = genlmsg_put(skb, 0, 0, &hwsim_genl_family, 0, cmd);
if (!data)
goto error;
ret = nla_put_u32(skb, HWSIM_ATTR_RADIO_ID, id);
if (ret < 0)
goto error;
if (param->channels) {
ret = nla_put_u32(skb, HWSIM_ATTR_CHANNELS, param->channels);
if (ret < 0)
goto error;
}
if (param->reg_alpha2) {
ret = nla_put(skb, HWSIM_ATTR_REG_HINT_ALPHA2, 2,
param->reg_alpha2);
if (ret < 0)
goto error;
}
if (param->regd) {
int i;
for (i = 0; hwsim_world_regdom_custom[i] != param->regd &&
i < ARRAY_SIZE(hwsim_world_regdom_custom); i++)
;
if (i < ARRAY_SIZE(hwsim_world_regdom_custom)) {
ret = nla_put_u32(skb, HWSIM_ATTR_REG_CUSTOM_REG, i);
if (ret < 0)
goto error;
}
}
if (param->reg_strict) {
ret = nla_put_flag(skb, HWSIM_ATTR_REG_STRICT_REG);
if (ret < 0)
goto error;
}
if (param->p2p_device) {
ret = nla_put_flag(skb, HWSIM_ATTR_SUPPORT_P2P_DEVICE);
if (ret < 0)
goto error;
}
if (param->use_chanctx) {
ret = nla_put_flag(skb, HWSIM_ATTR_USE_CHANCTX);
if (ret < 0)
goto error;
}
if (param->hwname) {
ret = nla_put(skb, HWSIM_ATTR_RADIO_NAME,
strlen(param->hwname), param->hwname);
if (ret < 0)
goto error;
}
genlmsg_end(skb, data);
return skb;
error:
nlmsg_free(skb);
return NULL;
}
static void hswim_mcast_new_radio(int id, struct genl_info *info,
struct hwsim_new_radio_params *param)
{
struct sk_buff *mcast_skb;
mcast_skb = build_radio_msg(HWSIM_CMD_NEW_RADIO, id, param);
if (!mcast_skb)
return;
hwsim_mcast_config_msg(mcast_skb, info);
}
static int mac80211_hwsim_new_radio(struct genl_info *info,
struct hwsim_new_radio_params *param)
{ {
int err; int err;
u8 addr[ETH_ALEN]; u8 addr[ETH_ALEN];
...@@ -2040,16 +2160,16 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2, ...@@ -2040,16 +2160,16 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2,
const struct ieee80211_ops *ops = &mac80211_hwsim_ops; const struct ieee80211_ops *ops = &mac80211_hwsim_ops;
int idx; int idx;
if (WARN_ON(channels > 1 && !use_chanctx)) if (WARN_ON(param->channels > 1 && !param->use_chanctx))
return -EINVAL; return -EINVAL;
spin_lock_bh(&hwsim_radio_lock); spin_lock_bh(&hwsim_radio_lock);
idx = hwsim_radio_idx++; idx = hwsim_radio_idx++;
spin_unlock_bh(&hwsim_radio_lock); spin_unlock_bh(&hwsim_radio_lock);
if (use_chanctx) if (param->use_chanctx)
ops = &mac80211_hwsim_mchan_ops; ops = &mac80211_hwsim_mchan_ops;
hw = ieee80211_alloc_hw_nm(sizeof(*data), ops, hwname); hw = ieee80211_alloc_hw_nm(sizeof(*data), ops, param->hwname);
if (!hw) { if (!hw) {
printk(KERN_DEBUG "mac80211_hwsim: ieee80211_alloc_hw failed\n"); printk(KERN_DEBUG "mac80211_hwsim: ieee80211_alloc_hw failed\n");
err = -ENOMEM; err = -ENOMEM;
...@@ -2087,11 +2207,12 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2, ...@@ -2087,11 +2207,12 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2,
hw->wiphy->n_addresses = 2; hw->wiphy->n_addresses = 2;
hw->wiphy->addresses = data->addresses; hw->wiphy->addresses = data->addresses;
data->channels = channels; data->channels = param->channels;
data->use_chanctx = use_chanctx; data->use_chanctx = param->use_chanctx;
data->idx = idx; data->idx = idx;
data->destroy_on_close = destroy_on_close; data->destroy_on_close = param->destroy_on_close;
data->portid = portid; if (info)
data->portid = info->snd_portid;
if (data->use_chanctx) { if (data->use_chanctx) {
hw->wiphy->max_scan_ssids = 255; hw->wiphy->max_scan_ssids = 255;
...@@ -2100,12 +2221,12 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2, ...@@ -2100,12 +2221,12 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2,
/* For channels > 1 DFS is not allowed */ /* For channels > 1 DFS is not allowed */
hw->wiphy->n_iface_combinations = 1; hw->wiphy->n_iface_combinations = 1;
hw->wiphy->iface_combinations = &data->if_combination; hw->wiphy->iface_combinations = &data->if_combination;
if (p2p_device) if (param->p2p_device)
data->if_combination = hwsim_if_comb_p2p_dev[0]; data->if_combination = hwsim_if_comb_p2p_dev[0];
else else
data->if_combination = hwsim_if_comb[0]; data->if_combination = hwsim_if_comb[0];
data->if_combination.num_different_channels = data->channels; data->if_combination.num_different_channels = data->channels;
} else if (p2p_device) { } else if (param->p2p_device) {
hw->wiphy->iface_combinations = hwsim_if_comb_p2p_dev; hw->wiphy->iface_combinations = hwsim_if_comb_p2p_dev;
hw->wiphy->n_iface_combinations = hw->wiphy->n_iface_combinations =
ARRAY_SIZE(hwsim_if_comb_p2p_dev); ARRAY_SIZE(hwsim_if_comb_p2p_dev);
...@@ -2126,7 +2247,7 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2, ...@@ -2126,7 +2247,7 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2,
BIT(NL80211_IFTYPE_ADHOC) | BIT(NL80211_IFTYPE_ADHOC) |
BIT(NL80211_IFTYPE_MESH_POINT); BIT(NL80211_IFTYPE_MESH_POINT);
if (p2p_device) if (param->p2p_device)
hw->wiphy->interface_modes |= BIT(NL80211_IFTYPE_P2P_DEVICE); hw->wiphy->interface_modes |= BIT(NL80211_IFTYPE_P2P_DEVICE);
hw->flags = IEEE80211_HW_MFP_CAPABLE | hw->flags = IEEE80211_HW_MFP_CAPABLE |
...@@ -2229,16 +2350,16 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2, ...@@ -2229,16 +2350,16 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2,
hw->max_rates = 4; hw->max_rates = 4;
hw->max_rate_tries = 11; hw->max_rate_tries = 11;
if (reg_strict) if (param->reg_strict)
hw->wiphy->regulatory_flags |= REGULATORY_STRICT_REG; hw->wiphy->regulatory_flags |= REGULATORY_STRICT_REG;
if (regd) { if (param->regd) {
hw->wiphy->regulatory_flags |= REGULATORY_CUSTOM_REG; hw->wiphy->regulatory_flags |= REGULATORY_CUSTOM_REG;
wiphy_apply_custom_regulatory(hw->wiphy, regd); wiphy_apply_custom_regulatory(hw->wiphy, param->regd);
/* give the regulatory workqueue a chance to run */ /* give the regulatory workqueue a chance to run */
schedule_timeout_interruptible(1); schedule_timeout_interruptible(1);
} }
if (no_vif) if (param->no_vif)
hw->flags |= IEEE80211_HW_NO_AUTO_VIF; hw->flags |= IEEE80211_HW_NO_AUTO_VIF;
err = ieee80211_register_hw(hw); err = ieee80211_register_hw(hw);
...@@ -2250,8 +2371,8 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2, ...@@ -2250,8 +2371,8 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2,
wiphy_debug(hw->wiphy, "hwaddr %pM registered\n", hw->wiphy->perm_addr); wiphy_debug(hw->wiphy, "hwaddr %pM registered\n", hw->wiphy->perm_addr);
if (reg_alpha2) if (param->reg_alpha2)
regulatory_hint(hw->wiphy, reg_alpha2); regulatory_hint(hw->wiphy, param->reg_alpha2);
data->debugfs = debugfs_create_dir("hwsim", hw->wiphy->debugfsdir); data->debugfs = debugfs_create_dir("hwsim", hw->wiphy->debugfsdir);
debugfs_create_file("ps", 0666, data->debugfs, data, &hwsim_fops_ps); debugfs_create_file("ps", 0666, data->debugfs, data, &hwsim_fops_ps);
...@@ -2270,6 +2391,9 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2, ...@@ -2270,6 +2391,9 @@ static int mac80211_hwsim_create_radio(int channels, const char *reg_alpha2,
list_add_tail(&data->list, &hwsim_radios); list_add_tail(&data->list, &hwsim_radios);
spin_unlock_bh(&hwsim_radio_lock); spin_unlock_bh(&hwsim_radio_lock);
if (idx > 0)
hswim_mcast_new_radio(idx, info, param);
return idx; return idx;
failed_hw: failed_hw:
...@@ -2520,47 +2644,43 @@ static int hwsim_register_received_nl(struct sk_buff *skb_2, ...@@ -2520,47 +2644,43 @@ static int hwsim_register_received_nl(struct sk_buff *skb_2,
return 0; return 0;
} }
static int hwsim_create_radio_nl(struct sk_buff *msg, struct genl_info *info) static int hwsim_new_radio_nl(struct sk_buff *msg, struct genl_info *info)
{ {
unsigned int chans = channels; struct hwsim_new_radio_params param = { 0 };
const char *alpha2 = NULL;
const struct ieee80211_regdomain *regd = NULL; param.reg_strict = info->attrs[HWSIM_ATTR_REG_STRICT_REG];
bool reg_strict = info->attrs[HWSIM_ATTR_REG_STRICT_REG]; param.p2p_device = info->attrs[HWSIM_ATTR_SUPPORT_P2P_DEVICE];
bool p2p_device = info->attrs[HWSIM_ATTR_SUPPORT_P2P_DEVICE]; param.channels = channels;
bool destroy_on_close = info->attrs[HWSIM_ATTR_DESTROY_RADIO_ON_CLOSE]; param.destroy_on_close =
bool use_chanctx; info->attrs[HWSIM_ATTR_DESTROY_RADIO_ON_CLOSE];
bool no_vif = false;
const char *hwname = NULL;
if (info->attrs[HWSIM_ATTR_CHANNELS]) if (info->attrs[HWSIM_ATTR_CHANNELS])
chans = nla_get_u32(info->attrs[HWSIM_ATTR_CHANNELS]); param.channels = nla_get_u32(info->attrs[HWSIM_ATTR_CHANNELS]);
if (info->attrs[HWSIM_ATTR_NO_VIF]) if (info->attrs[HWSIM_ATTR_NO_VIF])
no_vif = true; param.no_vif = true;
if (info->attrs[HWSIM_ATTR_RADIO_NAME]) if (info->attrs[HWSIM_ATTR_RADIO_NAME])
hwname = nla_data(info->attrs[HWSIM_ATTR_RADIO_NAME]); param.hwname = nla_data(info->attrs[HWSIM_ATTR_RADIO_NAME]);
if (info->attrs[HWSIM_ATTR_USE_CHANCTX]) if (info->attrs[HWSIM_ATTR_USE_CHANCTX])
use_chanctx = true; param.use_chanctx = true;
else else
use_chanctx = (chans > 1); param.use_chanctx = (param.channels > 1);
if (info->attrs[HWSIM_ATTR_REG_HINT_ALPHA2]) if (info->attrs[HWSIM_ATTR_REG_HINT_ALPHA2])
alpha2 = nla_data(info->attrs[HWSIM_ATTR_REG_HINT_ALPHA2]); param.reg_alpha2 =
nla_data(info->attrs[HWSIM_ATTR_REG_HINT_ALPHA2]);
if (info->attrs[HWSIM_ATTR_REG_CUSTOM_REG]) { if (info->attrs[HWSIM_ATTR_REG_CUSTOM_REG]) {
u32 idx = nla_get_u32(info->attrs[HWSIM_ATTR_REG_CUSTOM_REG]); u32 idx = nla_get_u32(info->attrs[HWSIM_ATTR_REG_CUSTOM_REG]);
if (idx >= ARRAY_SIZE(hwsim_world_regdom_custom)) if (idx >= ARRAY_SIZE(hwsim_world_regdom_custom))
return -EINVAL; return -EINVAL;
regd = hwsim_world_regdom_custom[idx]; param.regd = hwsim_world_regdom_custom[idx];
} }
return mac80211_hwsim_create_radio(chans, alpha2, regd, reg_strict, return mac80211_hwsim_new_radio(info, &param);
p2p_device, use_chanctx,
destroy_on_close, info->snd_portid,
hwname, no_vif);
} }
static int hwsim_destroy_radio_nl(struct sk_buff *msg, struct genl_info *info) static int hwsim_destroy_radio_nl(struct sk_buff *msg, struct genl_info *info)
...@@ -2616,9 +2736,9 @@ static const struct genl_ops hwsim_ops[] = { ...@@ -2616,9 +2736,9 @@ static const struct genl_ops hwsim_ops[] = {
.doit = hwsim_tx_info_frame_received_nl, .doit = hwsim_tx_info_frame_received_nl,
}, },
{ {
.cmd = HWSIM_CMD_CREATE_RADIO, .cmd = HWSIM_CMD_NEW_RADIO,
.policy = hwsim_genl_policy, .policy = hwsim_genl_policy,
.doit = hwsim_create_radio_nl, .doit = hwsim_new_radio_nl,
.flags = GENL_ADMIN_PERM, .flags = GENL_ADMIN_PERM,
}, },
{ {
...@@ -2682,7 +2802,9 @@ static int hwsim_init_netlink(void) ...@@ -2682,7 +2802,9 @@ static int hwsim_init_netlink(void)
printk(KERN_INFO "mac80211_hwsim: initializing netlink\n"); printk(KERN_INFO "mac80211_hwsim: initializing netlink\n");
rc = genl_register_family_with_ops(&hwsim_genl_family, hwsim_ops); rc = genl_register_family_with_ops_groups(&hwsim_genl_family,
hwsim_ops,
hwsim_mcgrps);
if (rc) if (rc)
goto failure; goto failure;
...@@ -2743,69 +2865,73 @@ static int __init init_mac80211_hwsim(void) ...@@ -2743,69 +2865,73 @@ static int __init init_mac80211_hwsim(void)
goto out_unregister_driver; goto out_unregister_driver;
} }
err = hwsim_init_netlink();
if (err < 0)
goto out_unregister_driver;
for (i = 0; i < radios; i++) { for (i = 0; i < radios; i++) {
const char *reg_alpha2 = NULL; struct hwsim_new_radio_params param = { 0 };
const struct ieee80211_regdomain *regd = NULL;
bool reg_strict = false; param.channels = channels;
switch (regtest) { switch (regtest) {
case HWSIM_REGTEST_DIFF_COUNTRY: case HWSIM_REGTEST_DIFF_COUNTRY:
if (i < ARRAY_SIZE(hwsim_alpha2s)) if (i < ARRAY_SIZE(hwsim_alpha2s))
reg_alpha2 = hwsim_alpha2s[i]; param.reg_alpha2 = hwsim_alpha2s[i];
break; break;
case HWSIM_REGTEST_DRIVER_REG_FOLLOW: case HWSIM_REGTEST_DRIVER_REG_FOLLOW:
if (!i) if (!i)
reg_alpha2 = hwsim_alpha2s[0]; param.reg_alpha2 = hwsim_alpha2s[0];
break; break;
case HWSIM_REGTEST_STRICT_ALL: case HWSIM_REGTEST_STRICT_ALL:
reg_strict = true; param.reg_strict = true;
case HWSIM_REGTEST_DRIVER_REG_ALL: case HWSIM_REGTEST_DRIVER_REG_ALL:
reg_alpha2 = hwsim_alpha2s[0]; param.reg_alpha2 = hwsim_alpha2s[0];
break; break;
case HWSIM_REGTEST_WORLD_ROAM: case HWSIM_REGTEST_WORLD_ROAM:
if (i == 0) if (i == 0)
regd = &hwsim_world_regdom_custom_01; param.regd = &hwsim_world_regdom_custom_01;
break; break;
case HWSIM_REGTEST_CUSTOM_WORLD: case HWSIM_REGTEST_CUSTOM_WORLD:
regd = &hwsim_world_regdom_custom_01; param.regd = &hwsim_world_regdom_custom_01;
break; break;
case HWSIM_REGTEST_CUSTOM_WORLD_2: case HWSIM_REGTEST_CUSTOM_WORLD_2:
if (i == 0) if (i == 0)
regd = &hwsim_world_regdom_custom_01; param.regd = &hwsim_world_regdom_custom_01;
else if (i == 1) else if (i == 1)
regd = &hwsim_world_regdom_custom_02; param.regd = &hwsim_world_regdom_custom_02;
break; break;
case HWSIM_REGTEST_STRICT_FOLLOW: case HWSIM_REGTEST_STRICT_FOLLOW:
if (i == 0) { if (i == 0) {
reg_strict = true; param.reg_strict = true;
reg_alpha2 = hwsim_alpha2s[0]; param.reg_alpha2 = hwsim_alpha2s[0];
} }
break; break;
case HWSIM_REGTEST_STRICT_AND_DRIVER_REG: case HWSIM_REGTEST_STRICT_AND_DRIVER_REG:
if (i == 0) { if (i == 0) {
reg_strict = true; param.reg_strict = true;
reg_alpha2 = hwsim_alpha2s[0]; param.reg_alpha2 = hwsim_alpha2s[0];
} else if (i == 1) { } else if (i == 1) {
reg_alpha2 = hwsim_alpha2s[1]; param.reg_alpha2 = hwsim_alpha2s[1];
} }
break; break;
case HWSIM_REGTEST_ALL: case HWSIM_REGTEST_ALL:
switch (i) { switch (i) {
case 0: case 0:
regd = &hwsim_world_regdom_custom_01; param.regd = &hwsim_world_regdom_custom_01;
break; break;
case 1: case 1:
regd = &hwsim_world_regdom_custom_02; param.regd = &hwsim_world_regdom_custom_02;
break; break;
case 2: case 2:
reg_alpha2 = hwsim_alpha2s[0]; param.reg_alpha2 = hwsim_alpha2s[0];
break; break;
case 3: case 3:
reg_alpha2 = hwsim_alpha2s[1]; param.reg_alpha2 = hwsim_alpha2s[1];
break; break;
case 4: case 4:
reg_strict = true; param.reg_strict = true;
reg_alpha2 = hwsim_alpha2s[2]; param.reg_alpha2 = hwsim_alpha2s[2];
break; break;
} }
break; break;
...@@ -2813,11 +2939,10 @@ static int __init init_mac80211_hwsim(void) ...@@ -2813,11 +2939,10 @@ static int __init init_mac80211_hwsim(void)
break; break;
} }
err = mac80211_hwsim_create_radio(channels, reg_alpha2, param.p2p_device = support_p2p_device;
regd, reg_strict, param.use_chanctx = channels > 1;
support_p2p_device,
channels > 1, false, 0, NULL, err = mac80211_hwsim_new_radio(NULL, &param);
false);
if (err < 0) if (err < 0)
goto out_free_radios; goto out_free_radios;
} }
...@@ -2843,10 +2968,6 @@ static int __init init_mac80211_hwsim(void) ...@@ -2843,10 +2968,6 @@ static int __init init_mac80211_hwsim(void)
} }
rtnl_unlock(); rtnl_unlock();
err = hwsim_init_netlink();
if (err < 0)
goto out_free_mon;
return 0; return 0;
out_free_mon: out_free_mon:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment