nl80211.c 29 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282
  1. /*
  2. * This is the new netlink-based wireless configuration interface.
  3. *
  4. * Copyright 2006, 2007 Johannes Berg <johannes@sipsolutions.net>
  5. */
  6. #include <linux/if.h>
  7. #include <linux/module.h>
  8. #include <linux/err.h>
  9. #include <linux/mutex.h>
  10. #include <linux/list.h>
  11. #include <linux/if_ether.h>
  12. #include <linux/ieee80211.h>
  13. #include <linux/nl80211.h>
  14. #include <linux/rtnetlink.h>
  15. #include <linux/netlink.h>
  16. #include <net/genetlink.h>
  17. #include <net/cfg80211.h>
  18. #include "core.h"
  19. #include "nl80211.h"
  20. /* the netlink family */
  21. static struct genl_family nl80211_fam = {
  22. .id = GENL_ID_GENERATE, /* don't bother with a hardcoded ID */
  23. .name = "nl80211", /* have users key off the name instead */
  24. .hdrsize = 0, /* no private header */
  25. .version = 1, /* no particular meaning now */
  26. .maxattr = NL80211_ATTR_MAX,
  27. };
  28. /* internal helper: get drv and dev */
  29. static int get_drv_dev_by_info_ifindex(struct genl_info *info,
  30. struct cfg80211_registered_device **drv,
  31. struct net_device **dev)
  32. {
  33. int ifindex;
  34. if (!info->attrs[NL80211_ATTR_IFINDEX])
  35. return -EINVAL;
  36. ifindex = nla_get_u32(info->attrs[NL80211_ATTR_IFINDEX]);
  37. *dev = dev_get_by_index(&init_net, ifindex);
  38. if (!*dev)
  39. return -ENODEV;
  40. *drv = cfg80211_get_dev_from_ifindex(ifindex);
  41. if (IS_ERR(*drv)) {
  42. dev_put(*dev);
  43. return PTR_ERR(*drv);
  44. }
  45. return 0;
  46. }
  47. /* policy for the attributes */
  48. static struct nla_policy nl80211_policy[NL80211_ATTR_MAX+1] __read_mostly = {
  49. [NL80211_ATTR_WIPHY] = { .type = NLA_U32 },
  50. [NL80211_ATTR_WIPHY_NAME] = { .type = NLA_NUL_STRING,
  51. .len = BUS_ID_SIZE-1 },
  52. [NL80211_ATTR_IFTYPE] = { .type = NLA_U32 },
  53. [NL80211_ATTR_IFINDEX] = { .type = NLA_U32 },
  54. [NL80211_ATTR_IFNAME] = { .type = NLA_NUL_STRING, .len = IFNAMSIZ-1 },
  55. [NL80211_ATTR_MAC] = { .type = NLA_BINARY, .len = ETH_ALEN },
  56. [NL80211_ATTR_KEY_DATA] = { .type = NLA_BINARY,
  57. .len = WLAN_MAX_KEY_LEN },
  58. [NL80211_ATTR_KEY_IDX] = { .type = NLA_U8 },
  59. [NL80211_ATTR_KEY_CIPHER] = { .type = NLA_U32 },
  60. [NL80211_ATTR_KEY_DEFAULT] = { .type = NLA_FLAG },
  61. [NL80211_ATTR_BEACON_INTERVAL] = { .type = NLA_U32 },
  62. [NL80211_ATTR_DTIM_PERIOD] = { .type = NLA_U32 },
  63. [NL80211_ATTR_BEACON_HEAD] = { .type = NLA_BINARY,
  64. .len = IEEE80211_MAX_DATA_LEN },
  65. [NL80211_ATTR_BEACON_TAIL] = { .type = NLA_BINARY,
  66. .len = IEEE80211_MAX_DATA_LEN },
  67. [NL80211_ATTR_STA_AID] = { .type = NLA_U16 },
  68. [NL80211_ATTR_STA_FLAGS] = { .type = NLA_NESTED },
  69. [NL80211_ATTR_STA_LISTEN_INTERVAL] = { .type = NLA_U16 },
  70. [NL80211_ATTR_STA_SUPPORTED_RATES] = { .type = NLA_BINARY,
  71. .len = NL80211_MAX_SUPP_RATES },
  72. [NL80211_ATTR_STA_VLAN] = { .type = NLA_U32 },
  73. [NL80211_ATTR_MNTR_FLAGS] = { .type = NLA_NESTED },
  74. };
  75. /* message building helper */
  76. static inline void *nl80211hdr_put(struct sk_buff *skb, u32 pid, u32 seq,
  77. int flags, u8 cmd)
  78. {
  79. /* since there is no private header just add the generic one */
  80. return genlmsg_put(skb, pid, seq, &nl80211_fam, flags, cmd);
  81. }
  82. /* netlink command implementations */
  83. static int nl80211_send_wiphy(struct sk_buff *msg, u32 pid, u32 seq, int flags,
  84. struct cfg80211_registered_device *dev)
  85. {
  86. void *hdr;
  87. struct nlattr *nl_bands, *nl_band;
  88. struct nlattr *nl_freqs, *nl_freq;
  89. struct nlattr *nl_rates, *nl_rate;
  90. enum ieee80211_band band;
  91. struct ieee80211_channel *chan;
  92. struct ieee80211_rate *rate;
  93. int i;
  94. hdr = nl80211hdr_put(msg, pid, seq, flags, NL80211_CMD_NEW_WIPHY);
  95. if (!hdr)
  96. return -1;
  97. NLA_PUT_U32(msg, NL80211_ATTR_WIPHY, dev->idx);
  98. NLA_PUT_STRING(msg, NL80211_ATTR_WIPHY_NAME, wiphy_name(&dev->wiphy));
  99. nl_bands = nla_nest_start(msg, NL80211_ATTR_WIPHY_BANDS);
  100. if (!nl_bands)
  101. goto nla_put_failure;
  102. for (band = 0; band < IEEE80211_NUM_BANDS; band++) {
  103. if (!dev->wiphy.bands[band])
  104. continue;
  105. nl_band = nla_nest_start(msg, band);
  106. if (!nl_band)
  107. goto nla_put_failure;
  108. /* add frequencies */
  109. nl_freqs = nla_nest_start(msg, NL80211_BAND_ATTR_FREQS);
  110. if (!nl_freqs)
  111. goto nla_put_failure;
  112. for (i = 0; i < dev->wiphy.bands[band]->n_channels; i++) {
  113. nl_freq = nla_nest_start(msg, i);
  114. if (!nl_freq)
  115. goto nla_put_failure;
  116. chan = &dev->wiphy.bands[band]->channels[i];
  117. NLA_PUT_U32(msg, NL80211_FREQUENCY_ATTR_FREQ,
  118. chan->center_freq);
  119. if (chan->flags & IEEE80211_CHAN_DISABLED)
  120. NLA_PUT_FLAG(msg, NL80211_FREQUENCY_ATTR_DISABLED);
  121. if (chan->flags & IEEE80211_CHAN_PASSIVE_SCAN)
  122. NLA_PUT_FLAG(msg, NL80211_FREQUENCY_ATTR_PASSIVE_SCAN);
  123. if (chan->flags & IEEE80211_CHAN_NO_IBSS)
  124. NLA_PUT_FLAG(msg, NL80211_FREQUENCY_ATTR_NO_IBSS);
  125. if (chan->flags & IEEE80211_CHAN_RADAR)
  126. NLA_PUT_FLAG(msg, NL80211_FREQUENCY_ATTR_RADAR);
  127. nla_nest_end(msg, nl_freq);
  128. }
  129. nla_nest_end(msg, nl_freqs);
  130. /* add bitrates */
  131. nl_rates = nla_nest_start(msg, NL80211_BAND_ATTR_RATES);
  132. if (!nl_rates)
  133. goto nla_put_failure;
  134. for (i = 0; i < dev->wiphy.bands[band]->n_bitrates; i++) {
  135. nl_rate = nla_nest_start(msg, i);
  136. if (!nl_rate)
  137. goto nla_put_failure;
  138. rate = &dev->wiphy.bands[band]->bitrates[i];
  139. NLA_PUT_U32(msg, NL80211_BITRATE_ATTR_RATE,
  140. rate->bitrate);
  141. if (rate->flags & IEEE80211_RATE_SHORT_PREAMBLE)
  142. NLA_PUT_FLAG(msg,
  143. NL80211_BITRATE_ATTR_2GHZ_SHORTPREAMBLE);
  144. nla_nest_end(msg, nl_rate);
  145. }
  146. nla_nest_end(msg, nl_rates);
  147. nla_nest_end(msg, nl_band);
  148. }
  149. nla_nest_end(msg, nl_bands);
  150. return genlmsg_end(msg, hdr);
  151. nla_put_failure:
  152. return genlmsg_cancel(msg, hdr);
  153. }
  154. static int nl80211_dump_wiphy(struct sk_buff *skb, struct netlink_callback *cb)
  155. {
  156. int idx = 0;
  157. int start = cb->args[0];
  158. struct cfg80211_registered_device *dev;
  159. mutex_lock(&cfg80211_drv_mutex);
  160. list_for_each_entry(dev, &cfg80211_drv_list, list) {
  161. if (++idx < start)
  162. continue;
  163. if (nl80211_send_wiphy(skb, NETLINK_CB(cb->skb).pid,
  164. cb->nlh->nlmsg_seq, NLM_F_MULTI,
  165. dev) < 0)
  166. break;
  167. }
  168. mutex_unlock(&cfg80211_drv_mutex);
  169. cb->args[0] = idx;
  170. return skb->len;
  171. }
  172. static int nl80211_get_wiphy(struct sk_buff *skb, struct genl_info *info)
  173. {
  174. struct sk_buff *msg;
  175. struct cfg80211_registered_device *dev;
  176. dev = cfg80211_get_dev_from_info(info);
  177. if (IS_ERR(dev))
  178. return PTR_ERR(dev);
  179. msg = nlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
  180. if (!msg)
  181. goto out_err;
  182. if (nl80211_send_wiphy(msg, info->snd_pid, info->snd_seq, 0, dev) < 0)
  183. goto out_free;
  184. cfg80211_put_dev(dev);
  185. return genlmsg_unicast(msg, info->snd_pid);
  186. out_free:
  187. nlmsg_free(msg);
  188. out_err:
  189. cfg80211_put_dev(dev);
  190. return -ENOBUFS;
  191. }
  192. static int nl80211_set_wiphy(struct sk_buff *skb, struct genl_info *info)
  193. {
  194. struct cfg80211_registered_device *rdev;
  195. int result;
  196. if (!info->attrs[NL80211_ATTR_WIPHY_NAME])
  197. return -EINVAL;
  198. rdev = cfg80211_get_dev_from_info(info);
  199. if (IS_ERR(rdev))
  200. return PTR_ERR(rdev);
  201. result = cfg80211_dev_rename(rdev, nla_data(info->attrs[NL80211_ATTR_WIPHY_NAME]));
  202. cfg80211_put_dev(rdev);
  203. return result;
  204. }
  205. static int nl80211_send_iface(struct sk_buff *msg, u32 pid, u32 seq, int flags,
  206. struct net_device *dev)
  207. {
  208. void *hdr;
  209. hdr = nl80211hdr_put(msg, pid, seq, flags, NL80211_CMD_NEW_INTERFACE);
  210. if (!hdr)
  211. return -1;
  212. NLA_PUT_U32(msg, NL80211_ATTR_IFINDEX, dev->ifindex);
  213. NLA_PUT_STRING(msg, NL80211_ATTR_IFNAME, dev->name);
  214. /* TODO: interface type */
  215. return genlmsg_end(msg, hdr);
  216. nla_put_failure:
  217. return genlmsg_cancel(msg, hdr);
  218. }
  219. static int nl80211_dump_interface(struct sk_buff *skb, struct netlink_callback *cb)
  220. {
  221. int wp_idx = 0;
  222. int if_idx = 0;
  223. int wp_start = cb->args[0];
  224. int if_start = cb->args[1];
  225. struct cfg80211_registered_device *dev;
  226. struct wireless_dev *wdev;
  227. mutex_lock(&cfg80211_drv_mutex);
  228. list_for_each_entry(dev, &cfg80211_drv_list, list) {
  229. if (++wp_idx < wp_start)
  230. continue;
  231. if_idx = 0;
  232. mutex_lock(&dev->devlist_mtx);
  233. list_for_each_entry(wdev, &dev->netdev_list, list) {
  234. if (++if_idx < if_start)
  235. continue;
  236. if (nl80211_send_iface(skb, NETLINK_CB(cb->skb).pid,
  237. cb->nlh->nlmsg_seq, NLM_F_MULTI,
  238. wdev->netdev) < 0)
  239. break;
  240. }
  241. mutex_unlock(&dev->devlist_mtx);
  242. }
  243. mutex_unlock(&cfg80211_drv_mutex);
  244. cb->args[0] = wp_idx;
  245. cb->args[1] = if_idx;
  246. return skb->len;
  247. }
  248. static int nl80211_get_interface(struct sk_buff *skb, struct genl_info *info)
  249. {
  250. struct sk_buff *msg;
  251. struct cfg80211_registered_device *dev;
  252. struct net_device *netdev;
  253. int err;
  254. err = get_drv_dev_by_info_ifindex(info, &dev, &netdev);
  255. if (err)
  256. return err;
  257. msg = nlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
  258. if (!msg)
  259. goto out_err;
  260. if (nl80211_send_iface(msg, info->snd_pid, info->snd_seq, 0, netdev) < 0)
  261. goto out_free;
  262. dev_put(netdev);
  263. cfg80211_put_dev(dev);
  264. return genlmsg_unicast(msg, info->snd_pid);
  265. out_free:
  266. nlmsg_free(msg);
  267. out_err:
  268. dev_put(netdev);
  269. cfg80211_put_dev(dev);
  270. return -ENOBUFS;
  271. }
  272. static const struct nla_policy mntr_flags_policy[NL80211_MNTR_FLAG_MAX + 1] = {
  273. [NL80211_MNTR_FLAG_FCSFAIL] = { .type = NLA_FLAG },
  274. [NL80211_MNTR_FLAG_PLCPFAIL] = { .type = NLA_FLAG },
  275. [NL80211_MNTR_FLAG_CONTROL] = { .type = NLA_FLAG },
  276. [NL80211_MNTR_FLAG_OTHER_BSS] = { .type = NLA_FLAG },
  277. [NL80211_MNTR_FLAG_COOK_FRAMES] = { .type = NLA_FLAG },
  278. };
  279. static int parse_monitor_flags(struct nlattr *nla, u32 *mntrflags)
  280. {
  281. struct nlattr *flags[NL80211_MNTR_FLAG_MAX + 1];
  282. int flag;
  283. *mntrflags = 0;
  284. if (!nla)
  285. return -EINVAL;
  286. if (nla_parse_nested(flags, NL80211_MNTR_FLAG_MAX,
  287. nla, mntr_flags_policy))
  288. return -EINVAL;
  289. for (flag = 1; flag <= NL80211_MNTR_FLAG_MAX; flag++)
  290. if (flags[flag])
  291. *mntrflags |= (1<<flag);
  292. return 0;
  293. }
  294. static int nl80211_set_interface(struct sk_buff *skb, struct genl_info *info)
  295. {
  296. struct cfg80211_registered_device *drv;
  297. int err, ifindex;
  298. enum nl80211_iftype type;
  299. struct net_device *dev;
  300. u32 flags;
  301. if (info->attrs[NL80211_ATTR_IFTYPE]) {
  302. type = nla_get_u32(info->attrs[NL80211_ATTR_IFTYPE]);
  303. if (type > NL80211_IFTYPE_MAX)
  304. return -EINVAL;
  305. } else
  306. return -EINVAL;
  307. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  308. if (err)
  309. return err;
  310. ifindex = dev->ifindex;
  311. dev_put(dev);
  312. if (!drv->ops->change_virtual_intf) {
  313. err = -EOPNOTSUPP;
  314. goto unlock;
  315. }
  316. rtnl_lock();
  317. err = parse_monitor_flags(type == NL80211_IFTYPE_MONITOR ?
  318. info->attrs[NL80211_ATTR_MNTR_FLAGS] : NULL,
  319. &flags);
  320. err = drv->ops->change_virtual_intf(&drv->wiphy, ifindex,
  321. type, err ? NULL : &flags);
  322. rtnl_unlock();
  323. unlock:
  324. cfg80211_put_dev(drv);
  325. return err;
  326. }
  327. static int nl80211_new_interface(struct sk_buff *skb, struct genl_info *info)
  328. {
  329. struct cfg80211_registered_device *drv;
  330. int err;
  331. enum nl80211_iftype type = NL80211_IFTYPE_UNSPECIFIED;
  332. u32 flags;
  333. if (!info->attrs[NL80211_ATTR_IFNAME])
  334. return -EINVAL;
  335. if (info->attrs[NL80211_ATTR_IFTYPE]) {
  336. type = nla_get_u32(info->attrs[NL80211_ATTR_IFTYPE]);
  337. if (type > NL80211_IFTYPE_MAX)
  338. return -EINVAL;
  339. }
  340. drv = cfg80211_get_dev_from_info(info);
  341. if (IS_ERR(drv))
  342. return PTR_ERR(drv);
  343. if (!drv->ops->add_virtual_intf) {
  344. err = -EOPNOTSUPP;
  345. goto unlock;
  346. }
  347. rtnl_lock();
  348. err = parse_monitor_flags(type == NL80211_IFTYPE_MONITOR ?
  349. info->attrs[NL80211_ATTR_MNTR_FLAGS] : NULL,
  350. &flags);
  351. err = drv->ops->add_virtual_intf(&drv->wiphy,
  352. nla_data(info->attrs[NL80211_ATTR_IFNAME]),
  353. type, err ? NULL : &flags);
  354. rtnl_unlock();
  355. unlock:
  356. cfg80211_put_dev(drv);
  357. return err;
  358. }
  359. static int nl80211_del_interface(struct sk_buff *skb, struct genl_info *info)
  360. {
  361. struct cfg80211_registered_device *drv;
  362. int ifindex, err;
  363. struct net_device *dev;
  364. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  365. if (err)
  366. return err;
  367. ifindex = dev->ifindex;
  368. dev_put(dev);
  369. if (!drv->ops->del_virtual_intf) {
  370. err = -EOPNOTSUPP;
  371. goto out;
  372. }
  373. rtnl_lock();
  374. err = drv->ops->del_virtual_intf(&drv->wiphy, ifindex);
  375. rtnl_unlock();
  376. out:
  377. cfg80211_put_dev(drv);
  378. return err;
  379. }
  380. struct get_key_cookie {
  381. struct sk_buff *msg;
  382. int error;
  383. };
  384. static void get_key_callback(void *c, struct key_params *params)
  385. {
  386. struct get_key_cookie *cookie = c;
  387. if (params->key)
  388. NLA_PUT(cookie->msg, NL80211_ATTR_KEY_DATA,
  389. params->key_len, params->key);
  390. if (params->seq)
  391. NLA_PUT(cookie->msg, NL80211_ATTR_KEY_SEQ,
  392. params->seq_len, params->seq);
  393. if (params->cipher)
  394. NLA_PUT_U32(cookie->msg, NL80211_ATTR_KEY_CIPHER,
  395. params->cipher);
  396. return;
  397. nla_put_failure:
  398. cookie->error = 1;
  399. }
  400. static int nl80211_get_key(struct sk_buff *skb, struct genl_info *info)
  401. {
  402. struct cfg80211_registered_device *drv;
  403. int err;
  404. struct net_device *dev;
  405. u8 key_idx = 0;
  406. u8 *mac_addr = NULL;
  407. struct get_key_cookie cookie = {
  408. .error = 0,
  409. };
  410. void *hdr;
  411. struct sk_buff *msg;
  412. if (info->attrs[NL80211_ATTR_KEY_IDX])
  413. key_idx = nla_get_u8(info->attrs[NL80211_ATTR_KEY_IDX]);
  414. if (key_idx > 3)
  415. return -EINVAL;
  416. if (info->attrs[NL80211_ATTR_MAC])
  417. mac_addr = nla_data(info->attrs[NL80211_ATTR_MAC]);
  418. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  419. if (err)
  420. return err;
  421. if (!drv->ops->get_key) {
  422. err = -EOPNOTSUPP;
  423. goto out;
  424. }
  425. msg = nlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
  426. if (!msg) {
  427. err = -ENOMEM;
  428. goto out;
  429. }
  430. hdr = nl80211hdr_put(msg, info->snd_pid, info->snd_seq, 0,
  431. NL80211_CMD_NEW_KEY);
  432. if (IS_ERR(hdr)) {
  433. err = PTR_ERR(hdr);
  434. goto out;
  435. }
  436. cookie.msg = msg;
  437. NLA_PUT_U32(msg, NL80211_ATTR_IFINDEX, dev->ifindex);
  438. NLA_PUT_U8(msg, NL80211_ATTR_KEY_IDX, key_idx);
  439. if (mac_addr)
  440. NLA_PUT(msg, NL80211_ATTR_MAC, ETH_ALEN, mac_addr);
  441. rtnl_lock();
  442. err = drv->ops->get_key(&drv->wiphy, dev, key_idx, mac_addr,
  443. &cookie, get_key_callback);
  444. rtnl_unlock();
  445. if (err)
  446. goto out;
  447. if (cookie.error)
  448. goto nla_put_failure;
  449. genlmsg_end(msg, hdr);
  450. err = genlmsg_unicast(msg, info->snd_pid);
  451. goto out;
  452. nla_put_failure:
  453. err = -ENOBUFS;
  454. nlmsg_free(msg);
  455. out:
  456. cfg80211_put_dev(drv);
  457. dev_put(dev);
  458. return err;
  459. }
  460. static int nl80211_set_key(struct sk_buff *skb, struct genl_info *info)
  461. {
  462. struct cfg80211_registered_device *drv;
  463. int err;
  464. struct net_device *dev;
  465. u8 key_idx;
  466. if (!info->attrs[NL80211_ATTR_KEY_IDX])
  467. return -EINVAL;
  468. key_idx = nla_get_u8(info->attrs[NL80211_ATTR_KEY_IDX]);
  469. if (key_idx > 3)
  470. return -EINVAL;
  471. /* currently only support setting default key */
  472. if (!info->attrs[NL80211_ATTR_KEY_DEFAULT])
  473. return -EINVAL;
  474. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  475. if (err)
  476. return err;
  477. if (!drv->ops->set_default_key) {
  478. err = -EOPNOTSUPP;
  479. goto out;
  480. }
  481. rtnl_lock();
  482. err = drv->ops->set_default_key(&drv->wiphy, dev, key_idx);
  483. rtnl_unlock();
  484. out:
  485. cfg80211_put_dev(drv);
  486. dev_put(dev);
  487. return err;
  488. }
  489. static int nl80211_new_key(struct sk_buff *skb, struct genl_info *info)
  490. {
  491. struct cfg80211_registered_device *drv;
  492. int err;
  493. struct net_device *dev;
  494. struct key_params params;
  495. u8 key_idx = 0;
  496. u8 *mac_addr = NULL;
  497. memset(&params, 0, sizeof(params));
  498. if (!info->attrs[NL80211_ATTR_KEY_CIPHER])
  499. return -EINVAL;
  500. if (info->attrs[NL80211_ATTR_KEY_DATA]) {
  501. params.key = nla_data(info->attrs[NL80211_ATTR_KEY_DATA]);
  502. params.key_len = nla_len(info->attrs[NL80211_ATTR_KEY_DATA]);
  503. }
  504. if (info->attrs[NL80211_ATTR_KEY_IDX])
  505. key_idx = nla_get_u8(info->attrs[NL80211_ATTR_KEY_IDX]);
  506. params.cipher = nla_get_u32(info->attrs[NL80211_ATTR_KEY_CIPHER]);
  507. if (info->attrs[NL80211_ATTR_MAC])
  508. mac_addr = nla_data(info->attrs[NL80211_ATTR_MAC]);
  509. if (key_idx > 3)
  510. return -EINVAL;
  511. /*
  512. * Disallow pairwise keys with non-zero index unless it's WEP
  513. * (because current deployments use pairwise WEP keys with
  514. * non-zero indizes but 802.11i clearly specifies to use zero)
  515. */
  516. if (mac_addr && key_idx &&
  517. params.cipher != WLAN_CIPHER_SUITE_WEP40 &&
  518. params.cipher != WLAN_CIPHER_SUITE_WEP104)
  519. return -EINVAL;
  520. /* TODO: add definitions for the lengths to linux/ieee80211.h */
  521. switch (params.cipher) {
  522. case WLAN_CIPHER_SUITE_WEP40:
  523. if (params.key_len != 5)
  524. return -EINVAL;
  525. break;
  526. case WLAN_CIPHER_SUITE_TKIP:
  527. if (params.key_len != 32)
  528. return -EINVAL;
  529. break;
  530. case WLAN_CIPHER_SUITE_CCMP:
  531. if (params.key_len != 16)
  532. return -EINVAL;
  533. break;
  534. case WLAN_CIPHER_SUITE_WEP104:
  535. if (params.key_len != 13)
  536. return -EINVAL;
  537. break;
  538. default:
  539. return -EINVAL;
  540. }
  541. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  542. if (err)
  543. return err;
  544. if (!drv->ops->add_key) {
  545. err = -EOPNOTSUPP;
  546. goto out;
  547. }
  548. rtnl_lock();
  549. err = drv->ops->add_key(&drv->wiphy, dev, key_idx, mac_addr, &params);
  550. rtnl_unlock();
  551. out:
  552. cfg80211_put_dev(drv);
  553. dev_put(dev);
  554. return err;
  555. }
  556. static int nl80211_del_key(struct sk_buff *skb, struct genl_info *info)
  557. {
  558. struct cfg80211_registered_device *drv;
  559. int err;
  560. struct net_device *dev;
  561. u8 key_idx = 0;
  562. u8 *mac_addr = NULL;
  563. if (info->attrs[NL80211_ATTR_KEY_IDX])
  564. key_idx = nla_get_u8(info->attrs[NL80211_ATTR_KEY_IDX]);
  565. if (key_idx > 3)
  566. return -EINVAL;
  567. if (info->attrs[NL80211_ATTR_MAC])
  568. mac_addr = nla_data(info->attrs[NL80211_ATTR_MAC]);
  569. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  570. if (err)
  571. return err;
  572. if (!drv->ops->del_key) {
  573. err = -EOPNOTSUPP;
  574. goto out;
  575. }
  576. rtnl_lock();
  577. err = drv->ops->del_key(&drv->wiphy, dev, key_idx, mac_addr);
  578. rtnl_unlock();
  579. out:
  580. cfg80211_put_dev(drv);
  581. dev_put(dev);
  582. return err;
  583. }
  584. static int nl80211_addset_beacon(struct sk_buff *skb, struct genl_info *info)
  585. {
  586. int (*call)(struct wiphy *wiphy, struct net_device *dev,
  587. struct beacon_parameters *info);
  588. struct cfg80211_registered_device *drv;
  589. int err;
  590. struct net_device *dev;
  591. struct beacon_parameters params;
  592. int haveinfo = 0;
  593. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  594. if (err)
  595. return err;
  596. switch (info->genlhdr->cmd) {
  597. case NL80211_CMD_NEW_BEACON:
  598. /* these are required for NEW_BEACON */
  599. if (!info->attrs[NL80211_ATTR_BEACON_INTERVAL] ||
  600. !info->attrs[NL80211_ATTR_DTIM_PERIOD] ||
  601. !info->attrs[NL80211_ATTR_BEACON_HEAD]) {
  602. err = -EINVAL;
  603. goto out;
  604. }
  605. call = drv->ops->add_beacon;
  606. break;
  607. case NL80211_CMD_SET_BEACON:
  608. call = drv->ops->set_beacon;
  609. break;
  610. default:
  611. WARN_ON(1);
  612. err = -EOPNOTSUPP;
  613. goto out;
  614. }
  615. if (!call) {
  616. err = -EOPNOTSUPP;
  617. goto out;
  618. }
  619. memset(&params, 0, sizeof(params));
  620. if (info->attrs[NL80211_ATTR_BEACON_INTERVAL]) {
  621. params.interval =
  622. nla_get_u32(info->attrs[NL80211_ATTR_BEACON_INTERVAL]);
  623. haveinfo = 1;
  624. }
  625. if (info->attrs[NL80211_ATTR_DTIM_PERIOD]) {
  626. params.dtim_period =
  627. nla_get_u32(info->attrs[NL80211_ATTR_DTIM_PERIOD]);
  628. haveinfo = 1;
  629. }
  630. if (info->attrs[NL80211_ATTR_BEACON_HEAD]) {
  631. params.head = nla_data(info->attrs[NL80211_ATTR_BEACON_HEAD]);
  632. params.head_len =
  633. nla_len(info->attrs[NL80211_ATTR_BEACON_HEAD]);
  634. haveinfo = 1;
  635. }
  636. if (info->attrs[NL80211_ATTR_BEACON_TAIL]) {
  637. params.tail = nla_data(info->attrs[NL80211_ATTR_BEACON_TAIL]);
  638. params.tail_len =
  639. nla_len(info->attrs[NL80211_ATTR_BEACON_TAIL]);
  640. haveinfo = 1;
  641. }
  642. if (!haveinfo) {
  643. err = -EINVAL;
  644. goto out;
  645. }
  646. rtnl_lock();
  647. err = call(&drv->wiphy, dev, &params);
  648. rtnl_unlock();
  649. out:
  650. cfg80211_put_dev(drv);
  651. dev_put(dev);
  652. return err;
  653. }
  654. static int nl80211_del_beacon(struct sk_buff *skb, struct genl_info *info)
  655. {
  656. struct cfg80211_registered_device *drv;
  657. int err;
  658. struct net_device *dev;
  659. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  660. if (err)
  661. return err;
  662. if (!drv->ops->del_beacon) {
  663. err = -EOPNOTSUPP;
  664. goto out;
  665. }
  666. rtnl_lock();
  667. err = drv->ops->del_beacon(&drv->wiphy, dev);
  668. rtnl_unlock();
  669. out:
  670. cfg80211_put_dev(drv);
  671. dev_put(dev);
  672. return err;
  673. }
  674. static const struct nla_policy sta_flags_policy[NL80211_STA_FLAG_MAX + 1] = {
  675. [NL80211_STA_FLAG_AUTHORIZED] = { .type = NLA_FLAG },
  676. [NL80211_STA_FLAG_SHORT_PREAMBLE] = { .type = NLA_FLAG },
  677. [NL80211_STA_FLAG_WME] = { .type = NLA_FLAG },
  678. };
  679. static int parse_station_flags(struct nlattr *nla, u32 *staflags)
  680. {
  681. struct nlattr *flags[NL80211_STA_FLAG_MAX + 1];
  682. int flag;
  683. *staflags = 0;
  684. if (!nla)
  685. return 0;
  686. if (nla_parse_nested(flags, NL80211_STA_FLAG_MAX,
  687. nla, sta_flags_policy))
  688. return -EINVAL;
  689. *staflags = STATION_FLAG_CHANGED;
  690. for (flag = 1; flag <= NL80211_STA_FLAG_MAX; flag++)
  691. if (flags[flag])
  692. *staflags |= (1<<flag);
  693. return 0;
  694. }
  695. static int nl80211_send_station(struct sk_buff *msg, u32 pid, u32 seq,
  696. int flags, struct net_device *dev,
  697. u8 *mac_addr, struct station_stats *stats)
  698. {
  699. void *hdr;
  700. struct nlattr *statsattr;
  701. hdr = nl80211hdr_put(msg, pid, seq, flags, NL80211_CMD_NEW_STATION);
  702. if (!hdr)
  703. return -1;
  704. NLA_PUT_U32(msg, NL80211_ATTR_IFINDEX, dev->ifindex);
  705. NLA_PUT(msg, NL80211_ATTR_MAC, ETH_ALEN, mac_addr);
  706. statsattr = nla_nest_start(msg, NL80211_ATTR_STA_STATS);
  707. if (!statsattr)
  708. goto nla_put_failure;
  709. if (stats->filled & STATION_STAT_INACTIVE_TIME)
  710. NLA_PUT_U32(msg, NL80211_STA_STAT_INACTIVE_TIME,
  711. stats->inactive_time);
  712. if (stats->filled & STATION_STAT_RX_BYTES)
  713. NLA_PUT_U32(msg, NL80211_STA_STAT_RX_BYTES,
  714. stats->rx_bytes);
  715. if (stats->filled & STATION_STAT_TX_BYTES)
  716. NLA_PUT_U32(msg, NL80211_STA_STAT_TX_BYTES,
  717. stats->tx_bytes);
  718. nla_nest_end(msg, statsattr);
  719. return genlmsg_end(msg, hdr);
  720. nla_put_failure:
  721. return genlmsg_cancel(msg, hdr);
  722. }
  723. static int nl80211_get_station(struct sk_buff *skb, struct genl_info *info)
  724. {
  725. struct cfg80211_registered_device *drv;
  726. int err;
  727. struct net_device *dev;
  728. struct station_stats stats;
  729. struct sk_buff *msg;
  730. u8 *mac_addr = NULL;
  731. memset(&stats, 0, sizeof(stats));
  732. if (!info->attrs[NL80211_ATTR_MAC])
  733. return -EINVAL;
  734. mac_addr = nla_data(info->attrs[NL80211_ATTR_MAC]);
  735. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  736. if (err)
  737. return err;
  738. if (!drv->ops->get_station) {
  739. err = -EOPNOTSUPP;
  740. goto out;
  741. }
  742. rtnl_lock();
  743. err = drv->ops->get_station(&drv->wiphy, dev, mac_addr, &stats);
  744. rtnl_unlock();
  745. msg = nlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
  746. if (!msg)
  747. goto out;
  748. if (nl80211_send_station(msg, info->snd_pid, info->snd_seq, 0,
  749. dev, mac_addr, &stats) < 0)
  750. goto out_free;
  751. err = genlmsg_unicast(msg, info->snd_pid);
  752. goto out;
  753. out_free:
  754. nlmsg_free(msg);
  755. out:
  756. cfg80211_put_dev(drv);
  757. dev_put(dev);
  758. return err;
  759. }
  760. /*
  761. * Get vlan interface making sure it is on the right wiphy.
  762. */
  763. static int get_vlan(struct nlattr *vlanattr,
  764. struct cfg80211_registered_device *rdev,
  765. struct net_device **vlan)
  766. {
  767. *vlan = NULL;
  768. if (vlanattr) {
  769. *vlan = dev_get_by_index(&init_net, nla_get_u32(vlanattr));
  770. if (!*vlan)
  771. return -ENODEV;
  772. if (!(*vlan)->ieee80211_ptr)
  773. return -EINVAL;
  774. if ((*vlan)->ieee80211_ptr->wiphy != &rdev->wiphy)
  775. return -EINVAL;
  776. }
  777. return 0;
  778. }
  779. static int nl80211_set_station(struct sk_buff *skb, struct genl_info *info)
  780. {
  781. struct cfg80211_registered_device *drv;
  782. int err;
  783. struct net_device *dev;
  784. struct station_parameters params;
  785. u8 *mac_addr = NULL;
  786. memset(&params, 0, sizeof(params));
  787. params.listen_interval = -1;
  788. if (info->attrs[NL80211_ATTR_STA_AID])
  789. return -EINVAL;
  790. if (!info->attrs[NL80211_ATTR_MAC])
  791. return -EINVAL;
  792. mac_addr = nla_data(info->attrs[NL80211_ATTR_MAC]);
  793. if (info->attrs[NL80211_ATTR_STA_SUPPORTED_RATES]) {
  794. params.supported_rates =
  795. nla_data(info->attrs[NL80211_ATTR_STA_SUPPORTED_RATES]);
  796. params.supported_rates_len =
  797. nla_len(info->attrs[NL80211_ATTR_STA_SUPPORTED_RATES]);
  798. }
  799. if (info->attrs[NL80211_ATTR_STA_LISTEN_INTERVAL])
  800. params.listen_interval =
  801. nla_get_u16(info->attrs[NL80211_ATTR_STA_LISTEN_INTERVAL]);
  802. if (parse_station_flags(info->attrs[NL80211_ATTR_STA_FLAGS],
  803. &params.station_flags))
  804. return -EINVAL;
  805. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  806. if (err)
  807. return err;
  808. err = get_vlan(info->attrs[NL80211_ATTR_STA_VLAN], drv, &params.vlan);
  809. if (err)
  810. goto out;
  811. if (!drv->ops->change_station) {
  812. err = -EOPNOTSUPP;
  813. goto out;
  814. }
  815. rtnl_lock();
  816. err = drv->ops->change_station(&drv->wiphy, dev, mac_addr, &params);
  817. rtnl_unlock();
  818. out:
  819. if (params.vlan)
  820. dev_put(params.vlan);
  821. cfg80211_put_dev(drv);
  822. dev_put(dev);
  823. return err;
  824. }
  825. static int nl80211_new_station(struct sk_buff *skb, struct genl_info *info)
  826. {
  827. struct cfg80211_registered_device *drv;
  828. int err;
  829. struct net_device *dev;
  830. struct station_parameters params;
  831. u8 *mac_addr = NULL;
  832. memset(&params, 0, sizeof(params));
  833. if (!info->attrs[NL80211_ATTR_MAC])
  834. return -EINVAL;
  835. if (!info->attrs[NL80211_ATTR_STA_AID])
  836. return -EINVAL;
  837. if (!info->attrs[NL80211_ATTR_STA_LISTEN_INTERVAL])
  838. return -EINVAL;
  839. if (!info->attrs[NL80211_ATTR_STA_SUPPORTED_RATES])
  840. return -EINVAL;
  841. mac_addr = nla_data(info->attrs[NL80211_ATTR_MAC]);
  842. params.supported_rates =
  843. nla_data(info->attrs[NL80211_ATTR_STA_SUPPORTED_RATES]);
  844. params.supported_rates_len =
  845. nla_len(info->attrs[NL80211_ATTR_STA_SUPPORTED_RATES]);
  846. params.listen_interval =
  847. nla_get_u16(info->attrs[NL80211_ATTR_STA_LISTEN_INTERVAL]);
  848. params.listen_interval = nla_get_u16(info->attrs[NL80211_ATTR_STA_AID]);
  849. if (parse_station_flags(info->attrs[NL80211_ATTR_STA_FLAGS],
  850. &params.station_flags))
  851. return -EINVAL;
  852. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  853. if (err)
  854. return err;
  855. err = get_vlan(info->attrs[NL80211_ATTR_STA_VLAN], drv, &params.vlan);
  856. if (err)
  857. goto out;
  858. if (!drv->ops->add_station) {
  859. err = -EOPNOTSUPP;
  860. goto out;
  861. }
  862. rtnl_lock();
  863. err = drv->ops->add_station(&drv->wiphy, dev, mac_addr, &params);
  864. rtnl_unlock();
  865. out:
  866. if (params.vlan)
  867. dev_put(params.vlan);
  868. cfg80211_put_dev(drv);
  869. dev_put(dev);
  870. return err;
  871. }
  872. static int nl80211_del_station(struct sk_buff *skb, struct genl_info *info)
  873. {
  874. struct cfg80211_registered_device *drv;
  875. int err;
  876. struct net_device *dev;
  877. u8 *mac_addr = NULL;
  878. if (info->attrs[NL80211_ATTR_MAC])
  879. mac_addr = nla_data(info->attrs[NL80211_ATTR_MAC]);
  880. err = get_drv_dev_by_info_ifindex(info, &drv, &dev);
  881. if (err)
  882. return err;
  883. if (!drv->ops->del_station) {
  884. err = -EOPNOTSUPP;
  885. goto out;
  886. }
  887. rtnl_lock();
  888. err = drv->ops->del_station(&drv->wiphy, dev, mac_addr);
  889. rtnl_unlock();
  890. out:
  891. cfg80211_put_dev(drv);
  892. dev_put(dev);
  893. return err;
  894. }
  895. static struct genl_ops nl80211_ops[] = {
  896. {
  897. .cmd = NL80211_CMD_GET_WIPHY,
  898. .doit = nl80211_get_wiphy,
  899. .dumpit = nl80211_dump_wiphy,
  900. .policy = nl80211_policy,
  901. /* can be retrieved by unprivileged users */
  902. },
  903. {
  904. .cmd = NL80211_CMD_SET_WIPHY,
  905. .doit = nl80211_set_wiphy,
  906. .policy = nl80211_policy,
  907. .flags = GENL_ADMIN_PERM,
  908. },
  909. {
  910. .cmd = NL80211_CMD_GET_INTERFACE,
  911. .doit = nl80211_get_interface,
  912. .dumpit = nl80211_dump_interface,
  913. .policy = nl80211_policy,
  914. /* can be retrieved by unprivileged users */
  915. },
  916. {
  917. .cmd = NL80211_CMD_SET_INTERFACE,
  918. .doit = nl80211_set_interface,
  919. .policy = nl80211_policy,
  920. .flags = GENL_ADMIN_PERM,
  921. },
  922. {
  923. .cmd = NL80211_CMD_NEW_INTERFACE,
  924. .doit = nl80211_new_interface,
  925. .policy = nl80211_policy,
  926. .flags = GENL_ADMIN_PERM,
  927. },
  928. {
  929. .cmd = NL80211_CMD_DEL_INTERFACE,
  930. .doit = nl80211_del_interface,
  931. .policy = nl80211_policy,
  932. .flags = GENL_ADMIN_PERM,
  933. },
  934. {
  935. .cmd = NL80211_CMD_GET_KEY,
  936. .doit = nl80211_get_key,
  937. .policy = nl80211_policy,
  938. .flags = GENL_ADMIN_PERM,
  939. },
  940. {
  941. .cmd = NL80211_CMD_SET_KEY,
  942. .doit = nl80211_set_key,
  943. .policy = nl80211_policy,
  944. .flags = GENL_ADMIN_PERM,
  945. },
  946. {
  947. .cmd = NL80211_CMD_NEW_KEY,
  948. .doit = nl80211_new_key,
  949. .policy = nl80211_policy,
  950. .flags = GENL_ADMIN_PERM,
  951. },
  952. {
  953. .cmd = NL80211_CMD_DEL_KEY,
  954. .doit = nl80211_del_key,
  955. .policy = nl80211_policy,
  956. .flags = GENL_ADMIN_PERM,
  957. },
  958. {
  959. .cmd = NL80211_CMD_SET_BEACON,
  960. .policy = nl80211_policy,
  961. .flags = GENL_ADMIN_PERM,
  962. .doit = nl80211_addset_beacon,
  963. },
  964. {
  965. .cmd = NL80211_CMD_NEW_BEACON,
  966. .policy = nl80211_policy,
  967. .flags = GENL_ADMIN_PERM,
  968. .doit = nl80211_addset_beacon,
  969. },
  970. {
  971. .cmd = NL80211_CMD_DEL_BEACON,
  972. .policy = nl80211_policy,
  973. .flags = GENL_ADMIN_PERM,
  974. .doit = nl80211_del_beacon,
  975. },
  976. {
  977. .cmd = NL80211_CMD_GET_STATION,
  978. .doit = nl80211_get_station,
  979. /* TODO: implement dumpit */
  980. .policy = nl80211_policy,
  981. .flags = GENL_ADMIN_PERM,
  982. },
  983. {
  984. .cmd = NL80211_CMD_SET_STATION,
  985. .doit = nl80211_set_station,
  986. .policy = nl80211_policy,
  987. .flags = GENL_ADMIN_PERM,
  988. },
  989. {
  990. .cmd = NL80211_CMD_NEW_STATION,
  991. .doit = nl80211_new_station,
  992. .policy = nl80211_policy,
  993. .flags = GENL_ADMIN_PERM,
  994. },
  995. {
  996. .cmd = NL80211_CMD_DEL_STATION,
  997. .doit = nl80211_del_station,
  998. .policy = nl80211_policy,
  999. .flags = GENL_ADMIN_PERM,
  1000. },
  1001. };
  1002. /* multicast groups */
  1003. static struct genl_multicast_group nl80211_config_mcgrp = {
  1004. .name = "config",
  1005. };
  1006. /* notification functions */
  1007. void nl80211_notify_dev_rename(struct cfg80211_registered_device *rdev)
  1008. {
  1009. struct sk_buff *msg;
  1010. msg = nlmsg_new(NLMSG_GOODSIZE, GFP_KERNEL);
  1011. if (!msg)
  1012. return;
  1013. if (nl80211_send_wiphy(msg, 0, 0, 0, rdev) < 0) {
  1014. nlmsg_free(msg);
  1015. return;
  1016. }
  1017. genlmsg_multicast(msg, 0, nl80211_config_mcgrp.id, GFP_KERNEL);
  1018. }
  1019. /* initialisation/exit functions */
  1020. int nl80211_init(void)
  1021. {
  1022. int err, i;
  1023. err = genl_register_family(&nl80211_fam);
  1024. if (err)
  1025. return err;
  1026. for (i = 0; i < ARRAY_SIZE(nl80211_ops); i++) {
  1027. err = genl_register_ops(&nl80211_fam, &nl80211_ops[i]);
  1028. if (err)
  1029. goto err_out;
  1030. }
  1031. err = genl_register_mc_group(&nl80211_fam, &nl80211_config_mcgrp);
  1032. if (err)
  1033. goto err_out;
  1034. return 0;
  1035. err_out:
  1036. genl_unregister_family(&nl80211_fam);
  1037. return err;
  1038. }
  1039. void nl80211_exit(void)
  1040. {
  1041. genl_unregister_family(&nl80211_fam);
  1042. }