diff options
author | Holger Schurig <holgerschurig@gmail.com> | 2009-11-11 06:25:40 -0500 |
---|---|---|
committer | John W. Linville <linville@tuxdriver.com> | 2009-11-13 17:43:58 -0500 |
commit | 61fa713c751683da915fa0c1aa502be85822c357 (patch) | |
tree | a4f286a2bb59b9afd1f3b836cb62a8e81bafee98 /net | |
parent | a043897a314e8bcfc821d54fe4e591efed5936a3 (diff) |
cfg80211: return channel noise via survey API
This patch implements the NL80211_CMD_GET_SURVEY command and an get_survey()
ops that a driver can implement. The goal of this command is to allow a
drivers to report channel survey data (e.g. channel noise, channel
occupation).
For now, only the mechanism to report back channel noise has been
implemented.
In future, there will either be a survey-trigger command --- or the existing
scan-trigger command will be enhanced. This will allow user-space to
request survey for arbitrary channels.
Note: any driver that cannot report channel noise should not report
any value at all, e.g. made-up -92 dBm.
Signed-off-by: Holger Schurig <holgerschurig@gmail.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
Diffstat (limited to 'net')
-rw-r--r-- | net/wireless/nl80211.c | 105 |
1 files changed, 105 insertions, 0 deletions
diff --git a/net/wireless/nl80211.c b/net/wireless/nl80211.c index 4dc139cdba5c..5c8b3bfada4b 100644 --- a/net/wireless/nl80211.c +++ b/net/wireless/nl80211.c | |||
@@ -3245,6 +3245,106 @@ static int nl80211_dump_scan(struct sk_buff *skb, | |||
3245 | return err; | 3245 | return err; |
3246 | } | 3246 | } |
3247 | 3247 | ||
3248 | static int nl80211_send_survey(struct sk_buff *msg, u32 pid, u32 seq, | ||
3249 | int flags, struct net_device *dev, | ||
3250 | struct survey_info *survey) | ||
3251 | { | ||
3252 | void *hdr; | ||
3253 | struct nlattr *infoattr; | ||
3254 | |||
3255 | /* Survey without a channel doesn't make sense */ | ||
3256 | if (!survey->channel) | ||
3257 | return -EINVAL; | ||
3258 | |||
3259 | hdr = nl80211hdr_put(msg, pid, seq, flags, | ||
3260 | NL80211_CMD_NEW_SURVEY_RESULTS); | ||
3261 | if (!hdr) | ||
3262 | return -ENOMEM; | ||
3263 | |||
3264 | NLA_PUT_U32(msg, NL80211_ATTR_IFINDEX, dev->ifindex); | ||
3265 | |||
3266 | infoattr = nla_nest_start(msg, NL80211_ATTR_SURVEY_INFO); | ||
3267 | if (!infoattr) | ||
3268 | goto nla_put_failure; | ||
3269 | |||
3270 | NLA_PUT_U32(msg, NL80211_SURVEY_INFO_FREQUENCY, | ||
3271 | survey->channel->center_freq); | ||
3272 | if (survey->filled & SURVEY_INFO_NOISE_DBM) | ||
3273 | NLA_PUT_U8(msg, NL80211_SURVEY_INFO_NOISE, | ||
3274 | survey->noise); | ||
3275 | |||
3276 | nla_nest_end(msg, infoattr); | ||
3277 | |||
3278 | return genlmsg_end(msg, hdr); | ||
3279 | |||
3280 | nla_put_failure: | ||
3281 | genlmsg_cancel(msg, hdr); | ||
3282 | return -EMSGSIZE; | ||
3283 | } | ||
3284 | |||
3285 | static int nl80211_dump_survey(struct sk_buff *skb, | ||
3286 | struct netlink_callback *cb) | ||
3287 | { | ||
3288 | struct survey_info survey; | ||
3289 | struct cfg80211_registered_device *dev; | ||
3290 | struct net_device *netdev; | ||
3291 | int ifidx = cb->args[0]; | ||
3292 | int survey_idx = cb->args[1]; | ||
3293 | int res; | ||
3294 | |||
3295 | if (!ifidx) | ||
3296 | ifidx = nl80211_get_ifidx(cb); | ||
3297 | if (ifidx < 0) | ||
3298 | return ifidx; | ||
3299 | cb->args[0] = ifidx; | ||
3300 | |||
3301 | rtnl_lock(); | ||
3302 | |||
3303 | netdev = __dev_get_by_index(sock_net(skb->sk), ifidx); | ||
3304 | if (!netdev) { | ||
3305 | res = -ENODEV; | ||
3306 | goto out_rtnl; | ||
3307 | } | ||
3308 | |||
3309 | dev = cfg80211_get_dev_from_ifindex(sock_net(skb->sk), ifidx); | ||
3310 | if (IS_ERR(dev)) { | ||
3311 | res = PTR_ERR(dev); | ||
3312 | goto out_rtnl; | ||
3313 | } | ||
3314 | |||
3315 | if (!dev->ops->dump_survey) { | ||
3316 | res = -EOPNOTSUPP; | ||
3317 | goto out_err; | ||
3318 | } | ||
3319 | |||
3320 | while (1) { | ||
3321 | res = dev->ops->dump_survey(&dev->wiphy, netdev, survey_idx, | ||
3322 | &survey); | ||
3323 | if (res == -ENOENT) | ||
3324 | break; | ||
3325 | if (res) | ||
3326 | goto out_err; | ||
3327 | |||
3328 | if (nl80211_send_survey(skb, | ||
3329 | NETLINK_CB(cb->skb).pid, | ||
3330 | cb->nlh->nlmsg_seq, NLM_F_MULTI, | ||
3331 | netdev, | ||
3332 | &survey) < 0) | ||
3333 | goto out; | ||
3334 | survey_idx++; | ||
3335 | } | ||
3336 | |||
3337 | out: | ||
3338 | cb->args[1] = survey_idx; | ||
3339 | res = skb->len; | ||
3340 | out_err: | ||
3341 | cfg80211_unlock_rdev(dev); | ||
3342 | out_rtnl: | ||
3343 | rtnl_unlock(); | ||
3344 | |||
3345 | return res; | ||
3346 | } | ||
3347 | |||
3248 | static bool nl80211_valid_auth_type(enum nl80211_auth_type auth_type) | 3348 | static bool nl80211_valid_auth_type(enum nl80211_auth_type auth_type) |
3249 | { | 3349 | { |
3250 | return auth_type <= NL80211_AUTHTYPE_MAX; | 3350 | return auth_type <= NL80211_AUTHTYPE_MAX; |
@@ -4322,6 +4422,11 @@ static struct genl_ops nl80211_ops[] = { | |||
4322 | .policy = nl80211_policy, | 4422 | .policy = nl80211_policy, |
4323 | .flags = GENL_ADMIN_PERM, | 4423 | .flags = GENL_ADMIN_PERM, |
4324 | }, | 4424 | }, |
4425 | { | ||
4426 | .cmd = NL80211_CMD_GET_SURVEY, | ||
4427 | .policy = nl80211_policy, | ||
4428 | .dumpit = nl80211_dump_survey, | ||
4429 | }, | ||
4325 | }; | 4430 | }; |
4326 | static struct genl_multicast_group nl80211_mlme_mcgrp = { | 4431 | static struct genl_multicast_group nl80211_mlme_mcgrp = { |
4327 | .name = "mlme", | 4432 | .name = "mlme", |