nl802154.c 84 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. *
  4. * Authors:
  5. * Alexander Aring <aar@pengutronix.de>
  6. *
  7. * Based on: net/wireless/nl80211.c
  8. */
  9. #include <linux/rtnetlink.h>
  10. #include <net/cfg802154.h>
  11. #include <net/genetlink.h>
  12. #include <net/mac802154.h>
  13. #include <net/netlink.h>
  14. #include <net/nl802154.h>
  15. #include <net/sock.h>
  16. #include "nl802154.h"
  17. #include "rdev-ops.h"
  18. #include "core.h"
  19. /* the netlink family */
  20. static struct genl_family nl802154_fam;
  21. /* multicast groups */
  22. enum nl802154_multicast_groups {
  23. NL802154_MCGRP_CONFIG,
  24. NL802154_MCGRP_SCAN,
  25. };
  26. static const struct genl_multicast_group nl802154_mcgrps[] = {
  27. [NL802154_MCGRP_CONFIG] = { .name = "config", },
  28. [NL802154_MCGRP_SCAN] = { .name = "scan", },
  29. };
  30. /* returns ERR_PTR values */
  31. static struct wpan_dev *
  32. __cfg802154_wpan_dev_from_attrs(struct net *netns, struct nlattr **attrs)
  33. {
  34. struct cfg802154_registered_device *rdev;
  35. struct wpan_dev *result = NULL;
  36. bool have_ifidx = attrs[NL802154_ATTR_IFINDEX];
  37. bool have_wpan_dev_id = attrs[NL802154_ATTR_WPAN_DEV];
  38. u64 wpan_dev_id;
  39. int wpan_phy_idx = -1;
  40. int ifidx = -1;
  41. ASSERT_RTNL();
  42. if (!have_ifidx && !have_wpan_dev_id)
  43. return ERR_PTR(-EINVAL);
  44. if (have_ifidx)
  45. ifidx = nla_get_u32(attrs[NL802154_ATTR_IFINDEX]);
  46. if (have_wpan_dev_id) {
  47. wpan_dev_id = nla_get_u64(attrs[NL802154_ATTR_WPAN_DEV]);
  48. wpan_phy_idx = wpan_dev_id >> 32;
  49. }
  50. list_for_each_entry(rdev, &cfg802154_rdev_list, list) {
  51. struct wpan_dev *wpan_dev;
  52. if (wpan_phy_net(&rdev->wpan_phy) != netns)
  53. continue;
  54. if (have_wpan_dev_id && rdev->wpan_phy_idx != wpan_phy_idx)
  55. continue;
  56. list_for_each_entry(wpan_dev, &rdev->wpan_dev_list, list) {
  57. if (have_ifidx && wpan_dev->netdev &&
  58. wpan_dev->netdev->ifindex == ifidx) {
  59. result = wpan_dev;
  60. break;
  61. }
  62. if (have_wpan_dev_id &&
  63. wpan_dev->identifier == (u32)wpan_dev_id) {
  64. result = wpan_dev;
  65. break;
  66. }
  67. }
  68. if (result)
  69. break;
  70. }
  71. if (result)
  72. return result;
  73. return ERR_PTR(-ENODEV);
  74. }
  75. static struct cfg802154_registered_device *
  76. __cfg802154_rdev_from_attrs(struct net *netns, struct nlattr **attrs)
  77. {
  78. struct cfg802154_registered_device *rdev = NULL, *tmp;
  79. struct net_device *netdev;
  80. ASSERT_RTNL();
  81. if (!attrs[NL802154_ATTR_WPAN_PHY] &&
  82. !attrs[NL802154_ATTR_IFINDEX] &&
  83. !attrs[NL802154_ATTR_WPAN_DEV])
  84. return ERR_PTR(-EINVAL);
  85. if (attrs[NL802154_ATTR_WPAN_PHY])
  86. rdev = cfg802154_rdev_by_wpan_phy_idx(
  87. nla_get_u32(attrs[NL802154_ATTR_WPAN_PHY]));
  88. if (attrs[NL802154_ATTR_WPAN_DEV]) {
  89. u64 wpan_dev_id = nla_get_u64(attrs[NL802154_ATTR_WPAN_DEV]);
  90. struct wpan_dev *wpan_dev;
  91. bool found = false;
  92. tmp = cfg802154_rdev_by_wpan_phy_idx(wpan_dev_id >> 32);
  93. if (tmp) {
  94. /* make sure wpan_dev exists */
  95. list_for_each_entry(wpan_dev, &tmp->wpan_dev_list, list) {
  96. if (wpan_dev->identifier != (u32)wpan_dev_id)
  97. continue;
  98. found = true;
  99. break;
  100. }
  101. if (!found)
  102. tmp = NULL;
  103. if (rdev && tmp != rdev)
  104. return ERR_PTR(-EINVAL);
  105. rdev = tmp;
  106. }
  107. }
  108. if (attrs[NL802154_ATTR_IFINDEX]) {
  109. int ifindex = nla_get_u32(attrs[NL802154_ATTR_IFINDEX]);
  110. netdev = __dev_get_by_index(netns, ifindex);
  111. if (netdev) {
  112. if (netdev->ieee802154_ptr)
  113. tmp = wpan_phy_to_rdev(
  114. netdev->ieee802154_ptr->wpan_phy);
  115. else
  116. tmp = NULL;
  117. /* not wireless device -- return error */
  118. if (!tmp)
  119. return ERR_PTR(-EINVAL);
  120. /* mismatch -- return error */
  121. if (rdev && tmp != rdev)
  122. return ERR_PTR(-EINVAL);
  123. rdev = tmp;
  124. }
  125. }
  126. if (!rdev)
  127. return ERR_PTR(-ENODEV);
  128. if (netns != wpan_phy_net(&rdev->wpan_phy))
  129. return ERR_PTR(-ENODEV);
  130. return rdev;
  131. }
  132. /* This function returns a pointer to the driver
  133. * that the genl_info item that is passed refers to.
  134. *
  135. * The result of this can be a PTR_ERR and hence must
  136. * be checked with IS_ERR() for errors.
  137. */
  138. static struct cfg802154_registered_device *
  139. cfg802154_get_dev_from_info(struct net *netns, struct genl_info *info)
  140. {
  141. return __cfg802154_rdev_from_attrs(netns, info->attrs);
  142. }
  143. /* policy for the attributes */
  144. static const struct nla_policy nl802154_policy[NL802154_ATTR_MAX+1] = {
  145. [NL802154_ATTR_WPAN_PHY] = { .type = NLA_U32 },
  146. [NL802154_ATTR_WPAN_PHY_NAME] = { .type = NLA_NUL_STRING,
  147. .len = 20-1 },
  148. [NL802154_ATTR_IFINDEX] = { .type = NLA_U32 },
  149. [NL802154_ATTR_IFTYPE] = { .type = NLA_U32 },
  150. [NL802154_ATTR_IFNAME] = { .type = NLA_NUL_STRING, .len = IFNAMSIZ-1 },
  151. [NL802154_ATTR_WPAN_DEV] = { .type = NLA_U64 },
  152. [NL802154_ATTR_PAGE] = NLA_POLICY_MAX(NLA_U8, IEEE802154_MAX_PAGE),
  153. [NL802154_ATTR_CHANNEL] = NLA_POLICY_MAX(NLA_U8, IEEE802154_MAX_CHANNEL),
  154. [NL802154_ATTR_TX_POWER] = { .type = NLA_S32, },
  155. [NL802154_ATTR_CCA_MODE] = { .type = NLA_U32, },
  156. [NL802154_ATTR_CCA_OPT] = { .type = NLA_U32, },
  157. [NL802154_ATTR_CCA_ED_LEVEL] = { .type = NLA_S32, },
  158. [NL802154_ATTR_SUPPORTED_CHANNEL] = { .type = NLA_U32, },
  159. [NL802154_ATTR_PAN_ID] = { .type = NLA_U16, },
  160. [NL802154_ATTR_EXTENDED_ADDR] = { .type = NLA_U64 },
  161. [NL802154_ATTR_SHORT_ADDR] = { .type = NLA_U16, },
  162. [NL802154_ATTR_MIN_BE] = { .type = NLA_U8, },
  163. [NL802154_ATTR_MAX_BE] = { .type = NLA_U8, },
  164. [NL802154_ATTR_MAX_CSMA_BACKOFFS] = { .type = NLA_U8, },
  165. [NL802154_ATTR_MAX_FRAME_RETRIES] = { .type = NLA_S8, },
  166. [NL802154_ATTR_LBT_MODE] = { .type = NLA_U8, },
  167. [NL802154_ATTR_WPAN_PHY_CAPS] = { .type = NLA_NESTED },
  168. [NL802154_ATTR_SUPPORTED_COMMANDS] = { .type = NLA_NESTED },
  169. [NL802154_ATTR_ACKREQ_DEFAULT] = { .type = NLA_U8 },
  170. [NL802154_ATTR_PID] = { .type = NLA_U32 },
  171. [NL802154_ATTR_NETNS_FD] = { .type = NLA_U32 },
  172. [NL802154_ATTR_COORDINATOR] = { .type = NLA_NESTED },
  173. [NL802154_ATTR_SCAN_TYPE] =
  174. NLA_POLICY_RANGE(NLA_U8, NL802154_SCAN_ED, NL802154_SCAN_RIT_PASSIVE),
  175. [NL802154_ATTR_SCAN_CHANNELS] =
  176. NLA_POLICY_MASK(NLA_U32, GENMASK(IEEE802154_MAX_CHANNEL, 0)),
  177. [NL802154_ATTR_SCAN_PREAMBLE_CODES] = { .type = NLA_REJECT },
  178. [NL802154_ATTR_SCAN_MEAN_PRF] = { .type = NLA_REJECT },
  179. [NL802154_ATTR_SCAN_DURATION] =
  180. NLA_POLICY_MAX(NLA_U8, IEEE802154_MAX_SCAN_DURATION),
  181. [NL802154_ATTR_SCAN_DONE_REASON] =
  182. NLA_POLICY_RANGE(NLA_U8, NL802154_SCAN_DONE_REASON_FINISHED,
  183. NL802154_SCAN_DONE_REASON_ABORTED),
  184. [NL802154_ATTR_BEACON_INTERVAL] =
  185. NLA_POLICY_MAX(NLA_U8, IEEE802154_ACTIVE_SCAN_DURATION),
  186. [NL802154_ATTR_MAX_ASSOCIATIONS] = { .type = NLA_U32 },
  187. [NL802154_ATTR_PEER] = { .type = NLA_NESTED },
  188. #ifdef CONFIG_IEEE802154_NL802154_EXPERIMENTAL
  189. [NL802154_ATTR_SEC_ENABLED] = { .type = NLA_U8, },
  190. [NL802154_ATTR_SEC_OUT_LEVEL] = { .type = NLA_U32, },
  191. [NL802154_ATTR_SEC_OUT_KEY_ID] = { .type = NLA_NESTED, },
  192. [NL802154_ATTR_SEC_FRAME_COUNTER] = { .type = NLA_U32 },
  193. [NL802154_ATTR_SEC_LEVEL] = { .type = NLA_NESTED },
  194. [NL802154_ATTR_SEC_DEVICE] = { .type = NLA_NESTED },
  195. [NL802154_ATTR_SEC_DEVKEY] = { .type = NLA_NESTED },
  196. [NL802154_ATTR_SEC_KEY] = { .type = NLA_NESTED },
  197. #endif /* CONFIG_IEEE802154_NL802154_EXPERIMENTAL */
  198. };
  199. static int
  200. nl802154_prepare_wpan_dev_dump(struct sk_buff *skb,
  201. struct netlink_callback *cb,
  202. struct cfg802154_registered_device **rdev,
  203. struct wpan_dev **wpan_dev)
  204. {
  205. const struct genl_dumpit_info *info = genl_dumpit_info(cb);
  206. int err;
  207. rtnl_lock();
  208. if (!cb->args[0]) {
  209. *wpan_dev = __cfg802154_wpan_dev_from_attrs(sock_net(skb->sk),
  210. info->info.attrs);
  211. if (IS_ERR(*wpan_dev)) {
  212. err = PTR_ERR(*wpan_dev);
  213. goto out_unlock;
  214. }
  215. *rdev = wpan_phy_to_rdev((*wpan_dev)->wpan_phy);
  216. /* 0 is the first index - add 1 to parse only once */
  217. cb->args[0] = (*rdev)->wpan_phy_idx + 1;
  218. cb->args[1] = (*wpan_dev)->identifier;
  219. } else {
  220. /* subtract the 1 again here */
  221. struct wpan_phy *wpan_phy = wpan_phy_idx_to_wpan_phy(cb->args[0] - 1);
  222. struct wpan_dev *tmp;
  223. if (!wpan_phy) {
  224. err = -ENODEV;
  225. goto out_unlock;
  226. }
  227. *rdev = wpan_phy_to_rdev(wpan_phy);
  228. *wpan_dev = NULL;
  229. list_for_each_entry(tmp, &(*rdev)->wpan_dev_list, list) {
  230. if (tmp->identifier == cb->args[1]) {
  231. *wpan_dev = tmp;
  232. break;
  233. }
  234. }
  235. if (!*wpan_dev) {
  236. err = -ENODEV;
  237. goto out_unlock;
  238. }
  239. }
  240. return 0;
  241. out_unlock:
  242. rtnl_unlock();
  243. return err;
  244. }
  245. static void
  246. nl802154_finish_wpan_dev_dump(struct cfg802154_registered_device *rdev)
  247. {
  248. rtnl_unlock();
  249. }
  250. /* message building helper */
  251. static inline void *nl802154hdr_put(struct sk_buff *skb, u32 portid, u32 seq,
  252. int flags, u8 cmd)
  253. {
  254. /* since there is no private header just add the generic one */
  255. return genlmsg_put(skb, portid, seq, &nl802154_fam, flags, cmd);
  256. }
  257. static int
  258. nl802154_put_flags(struct sk_buff *msg, int attr, u32 mask)
  259. {
  260. struct nlattr *nl_flags = nla_nest_start_noflag(msg, attr);
  261. int i;
  262. if (!nl_flags)
  263. return -ENOBUFS;
  264. i = 0;
  265. while (mask) {
  266. if ((mask & 1) && nla_put_flag(msg, i))
  267. return -ENOBUFS;
  268. mask >>= 1;
  269. i++;
  270. }
  271. nla_nest_end(msg, nl_flags);
  272. return 0;
  273. }
  274. static int
  275. nl802154_send_wpan_phy_channels(struct cfg802154_registered_device *rdev,
  276. struct sk_buff *msg)
  277. {
  278. struct nlattr *nl_page;
  279. unsigned long page;
  280. nl_page = nla_nest_start_noflag(msg, NL802154_ATTR_CHANNELS_SUPPORTED);
  281. if (!nl_page)
  282. return -ENOBUFS;
  283. for (page = 0; page <= IEEE802154_MAX_PAGE; page++) {
  284. if (nla_put_u32(msg, NL802154_ATTR_SUPPORTED_CHANNEL,
  285. rdev->wpan_phy.supported.channels[page]))
  286. return -ENOBUFS;
  287. }
  288. nla_nest_end(msg, nl_page);
  289. return 0;
  290. }
  291. static int
  292. nl802154_put_capabilities(struct sk_buff *msg,
  293. struct cfg802154_registered_device *rdev)
  294. {
  295. const struct wpan_phy_supported *caps = &rdev->wpan_phy.supported;
  296. struct nlattr *nl_caps, *nl_channels;
  297. int i;
  298. nl_caps = nla_nest_start_noflag(msg, NL802154_ATTR_WPAN_PHY_CAPS);
  299. if (!nl_caps)
  300. return -ENOBUFS;
  301. nl_channels = nla_nest_start_noflag(msg, NL802154_CAP_ATTR_CHANNELS);
  302. if (!nl_channels)
  303. return -ENOBUFS;
  304. for (i = 0; i <= IEEE802154_MAX_PAGE; i++) {
  305. if (caps->channels[i]) {
  306. if (nl802154_put_flags(msg, i, caps->channels[i]))
  307. return -ENOBUFS;
  308. }
  309. }
  310. nla_nest_end(msg, nl_channels);
  311. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_CCA_ED_LEVEL) {
  312. struct nlattr *nl_ed_lvls;
  313. nl_ed_lvls = nla_nest_start_noflag(msg,
  314. NL802154_CAP_ATTR_CCA_ED_LEVELS);
  315. if (!nl_ed_lvls)
  316. return -ENOBUFS;
  317. for (i = 0; i < caps->cca_ed_levels_size; i++) {
  318. if (nla_put_s32(msg, i, caps->cca_ed_levels[i]))
  319. return -ENOBUFS;
  320. }
  321. nla_nest_end(msg, nl_ed_lvls);
  322. }
  323. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_TXPOWER) {
  324. struct nlattr *nl_tx_pwrs;
  325. nl_tx_pwrs = nla_nest_start_noflag(msg,
  326. NL802154_CAP_ATTR_TX_POWERS);
  327. if (!nl_tx_pwrs)
  328. return -ENOBUFS;
  329. for (i = 0; i < caps->tx_powers_size; i++) {
  330. if (nla_put_s32(msg, i, caps->tx_powers[i]))
  331. return -ENOBUFS;
  332. }
  333. nla_nest_end(msg, nl_tx_pwrs);
  334. }
  335. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_CCA_MODE) {
  336. if (nl802154_put_flags(msg, NL802154_CAP_ATTR_CCA_MODES,
  337. caps->cca_modes) ||
  338. nl802154_put_flags(msg, NL802154_CAP_ATTR_CCA_OPTS,
  339. caps->cca_opts))
  340. return -ENOBUFS;
  341. }
  342. if (nla_put_u8(msg, NL802154_CAP_ATTR_MIN_MINBE, caps->min_minbe) ||
  343. nla_put_u8(msg, NL802154_CAP_ATTR_MAX_MINBE, caps->max_minbe) ||
  344. nla_put_u8(msg, NL802154_CAP_ATTR_MIN_MAXBE, caps->min_maxbe) ||
  345. nla_put_u8(msg, NL802154_CAP_ATTR_MAX_MAXBE, caps->max_maxbe) ||
  346. nla_put_u8(msg, NL802154_CAP_ATTR_MIN_CSMA_BACKOFFS,
  347. caps->min_csma_backoffs) ||
  348. nla_put_u8(msg, NL802154_CAP_ATTR_MAX_CSMA_BACKOFFS,
  349. caps->max_csma_backoffs) ||
  350. nla_put_s8(msg, NL802154_CAP_ATTR_MIN_FRAME_RETRIES,
  351. caps->min_frame_retries) ||
  352. nla_put_s8(msg, NL802154_CAP_ATTR_MAX_FRAME_RETRIES,
  353. caps->max_frame_retries) ||
  354. nl802154_put_flags(msg, NL802154_CAP_ATTR_IFTYPES,
  355. caps->iftypes) ||
  356. nla_put_u32(msg, NL802154_CAP_ATTR_LBT, caps->lbt))
  357. return -ENOBUFS;
  358. nla_nest_end(msg, nl_caps);
  359. return 0;
  360. }
  361. static int nl802154_send_wpan_phy(struct cfg802154_registered_device *rdev,
  362. enum nl802154_commands cmd,
  363. struct sk_buff *msg, u32 portid, u32 seq,
  364. int flags)
  365. {
  366. struct nlattr *nl_cmds;
  367. void *hdr;
  368. int i;
  369. hdr = nl802154hdr_put(msg, portid, seq, flags, cmd);
  370. if (!hdr)
  371. return -ENOBUFS;
  372. if (nla_put_u32(msg, NL802154_ATTR_WPAN_PHY, rdev->wpan_phy_idx) ||
  373. nla_put_string(msg, NL802154_ATTR_WPAN_PHY_NAME,
  374. wpan_phy_name(&rdev->wpan_phy)) ||
  375. nla_put_u32(msg, NL802154_ATTR_GENERATION,
  376. cfg802154_rdev_list_generation))
  377. goto nla_put_failure;
  378. if (cmd != NL802154_CMD_NEW_WPAN_PHY)
  379. goto finish;
  380. /* DUMP PHY PIB */
  381. /* current channel settings */
  382. if (nla_put_u8(msg, NL802154_ATTR_PAGE,
  383. rdev->wpan_phy.current_page) ||
  384. nla_put_u8(msg, NL802154_ATTR_CHANNEL,
  385. rdev->wpan_phy.current_channel))
  386. goto nla_put_failure;
  387. /* TODO remove this behaviour, we still keep support it for a while
  388. * so users can change the behaviour to the new one.
  389. */
  390. if (nl802154_send_wpan_phy_channels(rdev, msg))
  391. goto nla_put_failure;
  392. /* cca mode */
  393. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_CCA_MODE) {
  394. if (nla_put_u32(msg, NL802154_ATTR_CCA_MODE,
  395. rdev->wpan_phy.cca.mode))
  396. goto nla_put_failure;
  397. if (rdev->wpan_phy.cca.mode == NL802154_CCA_ENERGY_CARRIER) {
  398. if (nla_put_u32(msg, NL802154_ATTR_CCA_OPT,
  399. rdev->wpan_phy.cca.opt))
  400. goto nla_put_failure;
  401. }
  402. }
  403. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_TXPOWER) {
  404. if (nla_put_s32(msg, NL802154_ATTR_TX_POWER,
  405. rdev->wpan_phy.transmit_power))
  406. goto nla_put_failure;
  407. }
  408. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_CCA_ED_LEVEL) {
  409. if (nla_put_s32(msg, NL802154_ATTR_CCA_ED_LEVEL,
  410. rdev->wpan_phy.cca_ed_level))
  411. goto nla_put_failure;
  412. }
  413. if (nl802154_put_capabilities(msg, rdev))
  414. goto nla_put_failure;
  415. nl_cmds = nla_nest_start_noflag(msg, NL802154_ATTR_SUPPORTED_COMMANDS);
  416. if (!nl_cmds)
  417. goto nla_put_failure;
  418. i = 0;
  419. #define CMD(op, n) \
  420. do { \
  421. if (rdev->ops->op) { \
  422. i++; \
  423. if (nla_put_u32(msg, i, NL802154_CMD_ ## n)) \
  424. goto nla_put_failure; \
  425. } \
  426. } while (0)
  427. CMD(add_virtual_intf, NEW_INTERFACE);
  428. CMD(del_virtual_intf, DEL_INTERFACE);
  429. CMD(set_channel, SET_CHANNEL);
  430. CMD(set_pan_id, SET_PAN_ID);
  431. CMD(set_short_addr, SET_SHORT_ADDR);
  432. CMD(set_backoff_exponent, SET_BACKOFF_EXPONENT);
  433. CMD(set_max_csma_backoffs, SET_MAX_CSMA_BACKOFFS);
  434. CMD(set_max_frame_retries, SET_MAX_FRAME_RETRIES);
  435. CMD(set_lbt_mode, SET_LBT_MODE);
  436. CMD(set_ackreq_default, SET_ACKREQ_DEFAULT);
  437. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_TXPOWER)
  438. CMD(set_tx_power, SET_TX_POWER);
  439. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_CCA_ED_LEVEL)
  440. CMD(set_cca_ed_level, SET_CCA_ED_LEVEL);
  441. if (rdev->wpan_phy.flags & WPAN_PHY_FLAG_CCA_MODE)
  442. CMD(set_cca_mode, SET_CCA_MODE);
  443. #undef CMD
  444. nla_nest_end(msg, nl_cmds);
  445. finish:
  446. genlmsg_end(msg, hdr);
  447. return 0;
  448. nla_put_failure:
  449. genlmsg_cancel(msg, hdr);
  450. return -EMSGSIZE;
  451. }
  452. struct nl802154_dump_wpan_phy_state {
  453. s64 filter_wpan_phy;
  454. long start;
  455. };
  456. static int nl802154_dump_wpan_phy_parse(struct sk_buff *skb,
  457. struct netlink_callback *cb,
  458. struct nl802154_dump_wpan_phy_state *state)
  459. {
  460. const struct genl_dumpit_info *info = genl_dumpit_info(cb);
  461. struct nlattr **tb = info->info.attrs;
  462. if (tb[NL802154_ATTR_WPAN_PHY])
  463. state->filter_wpan_phy = nla_get_u32(tb[NL802154_ATTR_WPAN_PHY]);
  464. if (tb[NL802154_ATTR_WPAN_DEV])
  465. state->filter_wpan_phy = nla_get_u64(tb[NL802154_ATTR_WPAN_DEV]) >> 32;
  466. if (tb[NL802154_ATTR_IFINDEX]) {
  467. struct net_device *netdev;
  468. struct cfg802154_registered_device *rdev;
  469. int ifidx = nla_get_u32(tb[NL802154_ATTR_IFINDEX]);
  470. netdev = __dev_get_by_index(&init_net, ifidx);
  471. if (!netdev)
  472. return -ENODEV;
  473. if (netdev->ieee802154_ptr) {
  474. rdev = wpan_phy_to_rdev(
  475. netdev->ieee802154_ptr->wpan_phy);
  476. state->filter_wpan_phy = rdev->wpan_phy_idx;
  477. }
  478. }
  479. return 0;
  480. }
  481. static int
  482. nl802154_dump_wpan_phy(struct sk_buff *skb, struct netlink_callback *cb)
  483. {
  484. int idx = 0, ret;
  485. struct nl802154_dump_wpan_phy_state *state = (void *)cb->args[0];
  486. struct cfg802154_registered_device *rdev;
  487. rtnl_lock();
  488. if (!state) {
  489. state = kzalloc_obj(*state);
  490. if (!state) {
  491. rtnl_unlock();
  492. return -ENOMEM;
  493. }
  494. state->filter_wpan_phy = -1;
  495. ret = nl802154_dump_wpan_phy_parse(skb, cb, state);
  496. if (ret) {
  497. kfree(state);
  498. rtnl_unlock();
  499. return ret;
  500. }
  501. cb->args[0] = (long)state;
  502. }
  503. list_for_each_entry(rdev, &cfg802154_rdev_list, list) {
  504. if (!net_eq(wpan_phy_net(&rdev->wpan_phy), sock_net(skb->sk)))
  505. continue;
  506. if (++idx <= state->start)
  507. continue;
  508. if (state->filter_wpan_phy != -1 &&
  509. state->filter_wpan_phy != rdev->wpan_phy_idx)
  510. continue;
  511. /* attempt to fit multiple wpan_phy data chunks into the skb */
  512. ret = nl802154_send_wpan_phy(rdev,
  513. NL802154_CMD_NEW_WPAN_PHY,
  514. skb,
  515. NETLINK_CB(cb->skb).portid,
  516. cb->nlh->nlmsg_seq, NLM_F_MULTI);
  517. if (ret < 0) {
  518. if ((ret == -ENOBUFS || ret == -EMSGSIZE) &&
  519. !skb->len && cb->min_dump_alloc < 4096) {
  520. cb->min_dump_alloc = 4096;
  521. rtnl_unlock();
  522. return 1;
  523. }
  524. idx--;
  525. break;
  526. }
  527. break;
  528. }
  529. rtnl_unlock();
  530. state->start = idx;
  531. return skb->len;
  532. }
  533. static int nl802154_dump_wpan_phy_done(struct netlink_callback *cb)
  534. {
  535. kfree((void *)cb->args[0]);
  536. return 0;
  537. }
  538. static int nl802154_get_wpan_phy(struct sk_buff *skb, struct genl_info *info)
  539. {
  540. struct sk_buff *msg;
  541. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  542. msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
  543. if (!msg)
  544. return -ENOMEM;
  545. if (nl802154_send_wpan_phy(rdev, NL802154_CMD_NEW_WPAN_PHY, msg,
  546. info->snd_portid, info->snd_seq, 0) < 0) {
  547. nlmsg_free(msg);
  548. return -ENOBUFS;
  549. }
  550. return genlmsg_reply(msg, info);
  551. }
  552. static inline u64 wpan_dev_id(struct wpan_dev *wpan_dev)
  553. {
  554. return (u64)wpan_dev->identifier |
  555. ((u64)wpan_phy_to_rdev(wpan_dev->wpan_phy)->wpan_phy_idx << 32);
  556. }
  557. #ifdef CONFIG_IEEE802154_NL802154_EXPERIMENTAL
  558. #include <net/ieee802154_netdev.h>
  559. static int
  560. ieee802154_llsec_send_key_id(struct sk_buff *msg,
  561. const struct ieee802154_llsec_key_id *desc)
  562. {
  563. struct nlattr *nl_dev_addr;
  564. if (nla_put_u32(msg, NL802154_KEY_ID_ATTR_MODE, desc->mode))
  565. return -ENOBUFS;
  566. switch (desc->mode) {
  567. case NL802154_KEY_ID_MODE_IMPLICIT:
  568. nl_dev_addr = nla_nest_start_noflag(msg,
  569. NL802154_KEY_ID_ATTR_IMPLICIT);
  570. if (!nl_dev_addr)
  571. return -ENOBUFS;
  572. if (nla_put_le16(msg, NL802154_DEV_ADDR_ATTR_PAN_ID,
  573. desc->device_addr.pan_id) ||
  574. nla_put_u32(msg, NL802154_DEV_ADDR_ATTR_MODE,
  575. desc->device_addr.mode))
  576. return -ENOBUFS;
  577. switch (desc->device_addr.mode) {
  578. case NL802154_DEV_ADDR_SHORT:
  579. if (nla_put_le16(msg, NL802154_DEV_ADDR_ATTR_SHORT,
  580. desc->device_addr.short_addr))
  581. return -ENOBUFS;
  582. break;
  583. case NL802154_DEV_ADDR_EXTENDED:
  584. if (nla_put_le64(msg, NL802154_DEV_ADDR_ATTR_EXTENDED,
  585. desc->device_addr.extended_addr,
  586. NL802154_DEV_ADDR_ATTR_PAD))
  587. return -ENOBUFS;
  588. break;
  589. default:
  590. /* userspace should handle unknown */
  591. break;
  592. }
  593. nla_nest_end(msg, nl_dev_addr);
  594. break;
  595. case NL802154_KEY_ID_MODE_INDEX:
  596. break;
  597. case NL802154_KEY_ID_MODE_INDEX_SHORT:
  598. /* TODO renmae short_source? */
  599. if (nla_put_le32(msg, NL802154_KEY_ID_ATTR_SOURCE_SHORT,
  600. desc->short_source))
  601. return -ENOBUFS;
  602. break;
  603. case NL802154_KEY_ID_MODE_INDEX_EXTENDED:
  604. if (nla_put_le64(msg, NL802154_KEY_ID_ATTR_SOURCE_EXTENDED,
  605. desc->extended_source,
  606. NL802154_KEY_ID_ATTR_PAD))
  607. return -ENOBUFS;
  608. break;
  609. default:
  610. /* userspace should handle unknown */
  611. break;
  612. }
  613. /* TODO key_id to key_idx ? Check naming */
  614. if (desc->mode != NL802154_KEY_ID_MODE_IMPLICIT) {
  615. if (nla_put_u8(msg, NL802154_KEY_ID_ATTR_INDEX, desc->id))
  616. return -ENOBUFS;
  617. }
  618. return 0;
  619. }
  620. static int nl802154_get_llsec_params(struct sk_buff *msg,
  621. struct cfg802154_registered_device *rdev,
  622. struct wpan_dev *wpan_dev)
  623. {
  624. struct nlattr *nl_key_id;
  625. struct ieee802154_llsec_params params;
  626. int ret;
  627. ret = rdev_get_llsec_params(rdev, wpan_dev, &params);
  628. if (ret < 0)
  629. return ret;
  630. if (nla_put_u8(msg, NL802154_ATTR_SEC_ENABLED, params.enabled) ||
  631. nla_put_u32(msg, NL802154_ATTR_SEC_OUT_LEVEL, params.out_level) ||
  632. nla_put_be32(msg, NL802154_ATTR_SEC_FRAME_COUNTER,
  633. params.frame_counter))
  634. return -ENOBUFS;
  635. nl_key_id = nla_nest_start_noflag(msg, NL802154_ATTR_SEC_OUT_KEY_ID);
  636. if (!nl_key_id)
  637. return -ENOBUFS;
  638. ret = ieee802154_llsec_send_key_id(msg, &params.out_key);
  639. if (ret < 0)
  640. return ret;
  641. nla_nest_end(msg, nl_key_id);
  642. return 0;
  643. }
  644. #endif /* CONFIG_IEEE802154_NL802154_EXPERIMENTAL */
  645. static int
  646. nl802154_send_iface(struct sk_buff *msg, u32 portid, u32 seq, int flags,
  647. struct cfg802154_registered_device *rdev,
  648. struct wpan_dev *wpan_dev)
  649. {
  650. struct net_device *dev = wpan_dev->netdev;
  651. void *hdr;
  652. hdr = nl802154hdr_put(msg, portid, seq, flags,
  653. NL802154_CMD_NEW_INTERFACE);
  654. if (!hdr)
  655. return -1;
  656. if (dev &&
  657. (nla_put_u32(msg, NL802154_ATTR_IFINDEX, dev->ifindex) ||
  658. nla_put_string(msg, NL802154_ATTR_IFNAME, dev->name)))
  659. goto nla_put_failure;
  660. if (nla_put_u32(msg, NL802154_ATTR_WPAN_PHY, rdev->wpan_phy_idx) ||
  661. nla_put_u32(msg, NL802154_ATTR_IFTYPE, wpan_dev->iftype) ||
  662. nla_put_u64_64bit(msg, NL802154_ATTR_WPAN_DEV,
  663. wpan_dev_id(wpan_dev), NL802154_ATTR_PAD) ||
  664. nla_put_u32(msg, NL802154_ATTR_GENERATION,
  665. rdev->devlist_generation ^
  666. (cfg802154_rdev_list_generation << 2)))
  667. goto nla_put_failure;
  668. /* address settings */
  669. if (nla_put_le64(msg, NL802154_ATTR_EXTENDED_ADDR,
  670. wpan_dev->extended_addr,
  671. NL802154_ATTR_PAD) ||
  672. nla_put_le16(msg, NL802154_ATTR_SHORT_ADDR,
  673. wpan_dev->short_addr) ||
  674. nla_put_le16(msg, NL802154_ATTR_PAN_ID, wpan_dev->pan_id))
  675. goto nla_put_failure;
  676. /* ARET handling */
  677. if (nla_put_s8(msg, NL802154_ATTR_MAX_FRAME_RETRIES,
  678. wpan_dev->frame_retries) ||
  679. nla_put_u8(msg, NL802154_ATTR_MAX_BE, wpan_dev->max_be) ||
  680. nla_put_u8(msg, NL802154_ATTR_MAX_CSMA_BACKOFFS,
  681. wpan_dev->csma_retries) ||
  682. nla_put_u8(msg, NL802154_ATTR_MIN_BE, wpan_dev->min_be))
  683. goto nla_put_failure;
  684. /* listen before transmit */
  685. if (nla_put_u8(msg, NL802154_ATTR_LBT_MODE, wpan_dev->lbt))
  686. goto nla_put_failure;
  687. /* ackreq default behaviour */
  688. if (nla_put_u8(msg, NL802154_ATTR_ACKREQ_DEFAULT, wpan_dev->ackreq))
  689. goto nla_put_failure;
  690. #ifdef CONFIG_IEEE802154_NL802154_EXPERIMENTAL
  691. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  692. goto out;
  693. if (nl802154_get_llsec_params(msg, rdev, wpan_dev) < 0)
  694. goto nla_put_failure;
  695. out:
  696. #endif /* CONFIG_IEEE802154_NL802154_EXPERIMENTAL */
  697. genlmsg_end(msg, hdr);
  698. return 0;
  699. nla_put_failure:
  700. genlmsg_cancel(msg, hdr);
  701. return -EMSGSIZE;
  702. }
  703. static int
  704. nl802154_dump_interface(struct sk_buff *skb, struct netlink_callback *cb)
  705. {
  706. int wp_idx = 0;
  707. int if_idx = 0;
  708. int wp_start = cb->args[0];
  709. int if_start = cb->args[1];
  710. struct cfg802154_registered_device *rdev;
  711. struct wpan_dev *wpan_dev;
  712. rtnl_lock();
  713. list_for_each_entry(rdev, &cfg802154_rdev_list, list) {
  714. if (!net_eq(wpan_phy_net(&rdev->wpan_phy), sock_net(skb->sk)))
  715. continue;
  716. if (wp_idx < wp_start) {
  717. wp_idx++;
  718. continue;
  719. }
  720. if_idx = 0;
  721. list_for_each_entry(wpan_dev, &rdev->wpan_dev_list, list) {
  722. if (if_idx < if_start) {
  723. if_idx++;
  724. continue;
  725. }
  726. if (nl802154_send_iface(skb, NETLINK_CB(cb->skb).portid,
  727. cb->nlh->nlmsg_seq, NLM_F_MULTI,
  728. rdev, wpan_dev) < 0) {
  729. goto out;
  730. }
  731. if_idx++;
  732. }
  733. wp_idx++;
  734. }
  735. out:
  736. rtnl_unlock();
  737. cb->args[0] = wp_idx;
  738. cb->args[1] = if_idx;
  739. return skb->len;
  740. }
  741. static int nl802154_get_interface(struct sk_buff *skb, struct genl_info *info)
  742. {
  743. struct sk_buff *msg;
  744. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  745. struct wpan_dev *wdev = info->user_ptr[1];
  746. msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
  747. if (!msg)
  748. return -ENOMEM;
  749. if (nl802154_send_iface(msg, info->snd_portid, info->snd_seq, 0,
  750. rdev, wdev) < 0) {
  751. nlmsg_free(msg);
  752. return -ENOBUFS;
  753. }
  754. return genlmsg_reply(msg, info);
  755. }
  756. static int nl802154_new_interface(struct sk_buff *skb, struct genl_info *info)
  757. {
  758. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  759. enum nl802154_iftype type = NL802154_IFTYPE_UNSPEC;
  760. __le64 extended_addr = cpu_to_le64(0x0000000000000000ULL);
  761. /* TODO avoid failing a new interface
  762. * creation due to pending removal?
  763. */
  764. if (!info->attrs[NL802154_ATTR_IFNAME])
  765. return -EINVAL;
  766. if (info->attrs[NL802154_ATTR_IFTYPE]) {
  767. type = nla_get_u32(info->attrs[NL802154_ATTR_IFTYPE]);
  768. if (type > NL802154_IFTYPE_MAX ||
  769. !(rdev->wpan_phy.supported.iftypes & BIT(type)))
  770. return -EINVAL;
  771. }
  772. if (info->attrs[NL802154_ATTR_EXTENDED_ADDR])
  773. extended_addr = nla_get_le64(info->attrs[NL802154_ATTR_EXTENDED_ADDR]);
  774. if (!rdev->ops->add_virtual_intf)
  775. return -EOPNOTSUPP;
  776. return rdev_add_virtual_intf(rdev,
  777. nla_data(info->attrs[NL802154_ATTR_IFNAME]),
  778. NET_NAME_USER, type, extended_addr);
  779. }
  780. static int nl802154_del_interface(struct sk_buff *skb, struct genl_info *info)
  781. {
  782. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  783. struct wpan_dev *wpan_dev = info->user_ptr[1];
  784. if (!rdev->ops->del_virtual_intf)
  785. return -EOPNOTSUPP;
  786. /* If we remove a wpan device without a netdev then clear
  787. * user_ptr[1] so that nl802154_post_doit won't dereference it
  788. * to check if it needs to do dev_put(). Otherwise it crashes
  789. * since the wpan_dev has been freed, unlike with a netdev where
  790. * we need the dev_put() for the netdev to really be freed.
  791. */
  792. if (!wpan_dev->netdev)
  793. info->user_ptr[1] = NULL;
  794. return rdev_del_virtual_intf(rdev, wpan_dev);
  795. }
  796. static int nl802154_set_channel(struct sk_buff *skb, struct genl_info *info)
  797. {
  798. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  799. u8 channel, page;
  800. if (!info->attrs[NL802154_ATTR_PAGE] ||
  801. !info->attrs[NL802154_ATTR_CHANNEL])
  802. return -EINVAL;
  803. page = nla_get_u8(info->attrs[NL802154_ATTR_PAGE]);
  804. channel = nla_get_u8(info->attrs[NL802154_ATTR_CHANNEL]);
  805. /* check 802.15.4 constraints */
  806. if (!ieee802154_chan_is_valid(&rdev->wpan_phy, page, channel))
  807. return -EINVAL;
  808. return rdev_set_channel(rdev, page, channel);
  809. }
  810. static int nl802154_set_cca_mode(struct sk_buff *skb, struct genl_info *info)
  811. {
  812. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  813. struct wpan_phy_cca cca;
  814. if (!(rdev->wpan_phy.flags & WPAN_PHY_FLAG_CCA_MODE))
  815. return -EOPNOTSUPP;
  816. if (!info->attrs[NL802154_ATTR_CCA_MODE])
  817. return -EINVAL;
  818. cca.mode = nla_get_u32(info->attrs[NL802154_ATTR_CCA_MODE]);
  819. /* checking 802.15.4 constraints */
  820. if (cca.mode < NL802154_CCA_ENERGY ||
  821. cca.mode > NL802154_CCA_ATTR_MAX ||
  822. !(rdev->wpan_phy.supported.cca_modes & BIT(cca.mode)))
  823. return -EINVAL;
  824. if (cca.mode == NL802154_CCA_ENERGY_CARRIER) {
  825. if (!info->attrs[NL802154_ATTR_CCA_OPT])
  826. return -EINVAL;
  827. cca.opt = nla_get_u32(info->attrs[NL802154_ATTR_CCA_OPT]);
  828. if (cca.opt > NL802154_CCA_OPT_ATTR_MAX ||
  829. !(rdev->wpan_phy.supported.cca_opts & BIT(cca.opt)))
  830. return -EINVAL;
  831. }
  832. return rdev_set_cca_mode(rdev, &cca);
  833. }
  834. static int nl802154_set_cca_ed_level(struct sk_buff *skb, struct genl_info *info)
  835. {
  836. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  837. s32 ed_level;
  838. int i;
  839. if (!(rdev->wpan_phy.flags & WPAN_PHY_FLAG_CCA_ED_LEVEL))
  840. return -EOPNOTSUPP;
  841. if (!info->attrs[NL802154_ATTR_CCA_ED_LEVEL])
  842. return -EINVAL;
  843. ed_level = nla_get_s32(info->attrs[NL802154_ATTR_CCA_ED_LEVEL]);
  844. for (i = 0; i < rdev->wpan_phy.supported.cca_ed_levels_size; i++) {
  845. if (ed_level == rdev->wpan_phy.supported.cca_ed_levels[i])
  846. return rdev_set_cca_ed_level(rdev, ed_level);
  847. }
  848. return -EINVAL;
  849. }
  850. static int nl802154_set_tx_power(struct sk_buff *skb, struct genl_info *info)
  851. {
  852. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  853. s32 power;
  854. int i;
  855. if (!(rdev->wpan_phy.flags & WPAN_PHY_FLAG_TXPOWER))
  856. return -EOPNOTSUPP;
  857. if (!info->attrs[NL802154_ATTR_TX_POWER])
  858. return -EINVAL;
  859. power = nla_get_s32(info->attrs[NL802154_ATTR_TX_POWER]);
  860. for (i = 0; i < rdev->wpan_phy.supported.tx_powers_size; i++) {
  861. if (power == rdev->wpan_phy.supported.tx_powers[i])
  862. return rdev_set_tx_power(rdev, power);
  863. }
  864. return -EINVAL;
  865. }
  866. static int nl802154_set_pan_id(struct sk_buff *skb, struct genl_info *info)
  867. {
  868. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  869. struct net_device *dev = info->user_ptr[1];
  870. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  871. __le16 pan_id;
  872. /* conflict here while tx/rx calls */
  873. if (netif_running(dev))
  874. return -EBUSY;
  875. if (wpan_dev->lowpan_dev) {
  876. if (netif_running(wpan_dev->lowpan_dev))
  877. return -EBUSY;
  878. }
  879. /* don't change address fields on monitor */
  880. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR ||
  881. !info->attrs[NL802154_ATTR_PAN_ID])
  882. return -EINVAL;
  883. pan_id = nla_get_le16(info->attrs[NL802154_ATTR_PAN_ID]);
  884. /* Only allow changing the PAN ID when the device has no more
  885. * associations ongoing to avoid confusing peers.
  886. */
  887. if (cfg802154_device_is_associated(wpan_dev)) {
  888. NL_SET_ERR_MSG(info->extack,
  889. "Existing associations, changing PAN ID forbidden");
  890. return -EINVAL;
  891. }
  892. return rdev_set_pan_id(rdev, wpan_dev, pan_id);
  893. }
  894. static int nl802154_set_short_addr(struct sk_buff *skb, struct genl_info *info)
  895. {
  896. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  897. struct net_device *dev = info->user_ptr[1];
  898. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  899. __le16 short_addr;
  900. /* conflict here while tx/rx calls */
  901. if (netif_running(dev))
  902. return -EBUSY;
  903. if (wpan_dev->lowpan_dev) {
  904. if (netif_running(wpan_dev->lowpan_dev))
  905. return -EBUSY;
  906. }
  907. /* don't change address fields on monitor */
  908. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR ||
  909. !info->attrs[NL802154_ATTR_SHORT_ADDR])
  910. return -EINVAL;
  911. short_addr = nla_get_le16(info->attrs[NL802154_ATTR_SHORT_ADDR]);
  912. /* The short address only has a meaning when part of a PAN, after a
  913. * proper association procedure. However, we want to still offer the
  914. * possibility to create static networks so changing the short address
  915. * is only allowed when not already associated to other devices with
  916. * the official handshake.
  917. */
  918. if (cfg802154_device_is_associated(wpan_dev)) {
  919. NL_SET_ERR_MSG(info->extack,
  920. "Existing associations, changing short address forbidden");
  921. return -EINVAL;
  922. }
  923. return rdev_set_short_addr(rdev, wpan_dev, short_addr);
  924. }
  925. static int
  926. nl802154_set_backoff_exponent(struct sk_buff *skb, struct genl_info *info)
  927. {
  928. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  929. struct net_device *dev = info->user_ptr[1];
  930. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  931. u8 min_be, max_be;
  932. /* should be set on netif open inside phy settings */
  933. if (netif_running(dev))
  934. return -EBUSY;
  935. if (!info->attrs[NL802154_ATTR_MIN_BE] ||
  936. !info->attrs[NL802154_ATTR_MAX_BE])
  937. return -EINVAL;
  938. min_be = nla_get_u8(info->attrs[NL802154_ATTR_MIN_BE]);
  939. max_be = nla_get_u8(info->attrs[NL802154_ATTR_MAX_BE]);
  940. /* check 802.15.4 constraints */
  941. if (min_be < rdev->wpan_phy.supported.min_minbe ||
  942. min_be > rdev->wpan_phy.supported.max_minbe ||
  943. max_be < rdev->wpan_phy.supported.min_maxbe ||
  944. max_be > rdev->wpan_phy.supported.max_maxbe ||
  945. min_be > max_be)
  946. return -EINVAL;
  947. return rdev_set_backoff_exponent(rdev, wpan_dev, min_be, max_be);
  948. }
  949. static int
  950. nl802154_set_max_csma_backoffs(struct sk_buff *skb, struct genl_info *info)
  951. {
  952. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  953. struct net_device *dev = info->user_ptr[1];
  954. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  955. u8 max_csma_backoffs;
  956. /* conflict here while other running iface settings */
  957. if (netif_running(dev))
  958. return -EBUSY;
  959. if (!info->attrs[NL802154_ATTR_MAX_CSMA_BACKOFFS])
  960. return -EINVAL;
  961. max_csma_backoffs = nla_get_u8(
  962. info->attrs[NL802154_ATTR_MAX_CSMA_BACKOFFS]);
  963. /* check 802.15.4 constraints */
  964. if (max_csma_backoffs < rdev->wpan_phy.supported.min_csma_backoffs ||
  965. max_csma_backoffs > rdev->wpan_phy.supported.max_csma_backoffs)
  966. return -EINVAL;
  967. return rdev_set_max_csma_backoffs(rdev, wpan_dev, max_csma_backoffs);
  968. }
  969. static int
  970. nl802154_set_max_frame_retries(struct sk_buff *skb, struct genl_info *info)
  971. {
  972. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  973. struct net_device *dev = info->user_ptr[1];
  974. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  975. s8 max_frame_retries;
  976. if (netif_running(dev))
  977. return -EBUSY;
  978. if (!info->attrs[NL802154_ATTR_MAX_FRAME_RETRIES])
  979. return -EINVAL;
  980. max_frame_retries = nla_get_s8(
  981. info->attrs[NL802154_ATTR_MAX_FRAME_RETRIES]);
  982. /* check 802.15.4 constraints */
  983. if (max_frame_retries < rdev->wpan_phy.supported.min_frame_retries ||
  984. max_frame_retries > rdev->wpan_phy.supported.max_frame_retries)
  985. return -EINVAL;
  986. return rdev_set_max_frame_retries(rdev, wpan_dev, max_frame_retries);
  987. }
  988. static int nl802154_set_lbt_mode(struct sk_buff *skb, struct genl_info *info)
  989. {
  990. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  991. struct net_device *dev = info->user_ptr[1];
  992. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  993. int mode;
  994. if (netif_running(dev))
  995. return -EBUSY;
  996. if (!info->attrs[NL802154_ATTR_LBT_MODE])
  997. return -EINVAL;
  998. mode = nla_get_u8(info->attrs[NL802154_ATTR_LBT_MODE]);
  999. if (mode != 0 && mode != 1)
  1000. return -EINVAL;
  1001. if (!wpan_phy_supported_bool(mode, rdev->wpan_phy.supported.lbt))
  1002. return -EINVAL;
  1003. return rdev_set_lbt_mode(rdev, wpan_dev, mode);
  1004. }
  1005. static int
  1006. nl802154_set_ackreq_default(struct sk_buff *skb, struct genl_info *info)
  1007. {
  1008. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1009. struct net_device *dev = info->user_ptr[1];
  1010. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1011. int ackreq;
  1012. if (netif_running(dev))
  1013. return -EBUSY;
  1014. if (!info->attrs[NL802154_ATTR_ACKREQ_DEFAULT])
  1015. return -EINVAL;
  1016. ackreq = nla_get_u8(info->attrs[NL802154_ATTR_ACKREQ_DEFAULT]);
  1017. if (ackreq != 0 && ackreq != 1)
  1018. return -EINVAL;
  1019. return rdev_set_ackreq_default(rdev, wpan_dev, ackreq);
  1020. }
  1021. static int nl802154_wpan_phy_netns(struct sk_buff *skb, struct genl_info *info)
  1022. {
  1023. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1024. struct net *net;
  1025. int err;
  1026. if (info->attrs[NL802154_ATTR_PID]) {
  1027. u32 pid = nla_get_u32(info->attrs[NL802154_ATTR_PID]);
  1028. net = get_net_ns_by_pid(pid);
  1029. } else if (info->attrs[NL802154_ATTR_NETNS_FD]) {
  1030. u32 fd = nla_get_u32(info->attrs[NL802154_ATTR_NETNS_FD]);
  1031. net = get_net_ns_by_fd(fd);
  1032. } else {
  1033. return -EINVAL;
  1034. }
  1035. if (IS_ERR(net))
  1036. return PTR_ERR(net);
  1037. err = 0;
  1038. /* check if anything to do */
  1039. if (!net_eq(wpan_phy_net(&rdev->wpan_phy), net))
  1040. err = cfg802154_switch_netns(rdev, net);
  1041. put_net(net);
  1042. return err;
  1043. }
  1044. static int nl802154_prep_scan_event_msg(struct sk_buff *msg,
  1045. struct cfg802154_registered_device *rdev,
  1046. struct wpan_dev *wpan_dev,
  1047. u32 portid, u32 seq, int flags, u8 cmd,
  1048. struct ieee802154_coord_desc *desc)
  1049. {
  1050. struct nlattr *nla;
  1051. void *hdr;
  1052. hdr = nl802154hdr_put(msg, portid, seq, flags, cmd);
  1053. if (!hdr)
  1054. return -ENOBUFS;
  1055. if (nla_put_u32(msg, NL802154_ATTR_WPAN_PHY, rdev->wpan_phy_idx))
  1056. goto nla_put_failure;
  1057. if (wpan_dev->netdev &&
  1058. nla_put_u32(msg, NL802154_ATTR_IFINDEX, wpan_dev->netdev->ifindex))
  1059. goto nla_put_failure;
  1060. if (nla_put_u64_64bit(msg, NL802154_ATTR_WPAN_DEV,
  1061. wpan_dev_id(wpan_dev), NL802154_ATTR_PAD))
  1062. goto nla_put_failure;
  1063. nla = nla_nest_start_noflag(msg, NL802154_ATTR_COORDINATOR);
  1064. if (!nla)
  1065. goto nla_put_failure;
  1066. if (nla_put(msg, NL802154_COORD_PANID, IEEE802154_PAN_ID_LEN,
  1067. &desc->addr.pan_id))
  1068. goto nla_put_failure;
  1069. if (desc->addr.mode == IEEE802154_ADDR_SHORT) {
  1070. if (nla_put(msg, NL802154_COORD_ADDR,
  1071. IEEE802154_SHORT_ADDR_LEN,
  1072. &desc->addr.short_addr))
  1073. goto nla_put_failure;
  1074. } else {
  1075. if (nla_put(msg, NL802154_COORD_ADDR,
  1076. IEEE802154_EXTENDED_ADDR_LEN,
  1077. &desc->addr.extended_addr))
  1078. goto nla_put_failure;
  1079. }
  1080. if (nla_put_u8(msg, NL802154_COORD_CHANNEL, desc->channel))
  1081. goto nla_put_failure;
  1082. if (nla_put_u8(msg, NL802154_COORD_PAGE, desc->page))
  1083. goto nla_put_failure;
  1084. if (nla_put_u16(msg, NL802154_COORD_SUPERFRAME_SPEC,
  1085. desc->superframe_spec))
  1086. goto nla_put_failure;
  1087. if (nla_put_u8(msg, NL802154_COORD_LINK_QUALITY, desc->link_quality))
  1088. goto nla_put_failure;
  1089. if (desc->gts_permit && nla_put_flag(msg, NL802154_COORD_GTS_PERMIT))
  1090. goto nla_put_failure;
  1091. /* TODO: NL802154_COORD_PAYLOAD_DATA if any */
  1092. nla_nest_end(msg, nla);
  1093. genlmsg_end(msg, hdr);
  1094. return 0;
  1095. nla_put_failure:
  1096. genlmsg_cancel(msg, hdr);
  1097. return -EMSGSIZE;
  1098. }
  1099. int nl802154_scan_event(struct wpan_phy *wpan_phy, struct wpan_dev *wpan_dev,
  1100. struct ieee802154_coord_desc *desc)
  1101. {
  1102. struct cfg802154_registered_device *rdev = wpan_phy_to_rdev(wpan_phy);
  1103. struct sk_buff *msg;
  1104. int ret;
  1105. msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_ATOMIC);
  1106. if (!msg)
  1107. return -ENOMEM;
  1108. ret = nl802154_prep_scan_event_msg(msg, rdev, wpan_dev, 0, 0, 0,
  1109. NL802154_CMD_SCAN_EVENT,
  1110. desc);
  1111. if (ret < 0) {
  1112. nlmsg_free(msg);
  1113. return ret;
  1114. }
  1115. return genlmsg_multicast_netns(&nl802154_fam, wpan_phy_net(wpan_phy),
  1116. msg, 0, NL802154_MCGRP_SCAN, GFP_ATOMIC);
  1117. }
  1118. EXPORT_SYMBOL_GPL(nl802154_scan_event);
  1119. static int nl802154_trigger_scan(struct sk_buff *skb, struct genl_info *info)
  1120. {
  1121. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1122. struct net_device *dev = info->user_ptr[1];
  1123. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1124. struct wpan_phy *wpan_phy = &rdev->wpan_phy;
  1125. struct cfg802154_scan_request *request;
  1126. u8 type;
  1127. int err;
  1128. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR) {
  1129. NL_SET_ERR_MSG(info->extack, "Monitors are not allowed to perform scans");
  1130. return -EOPNOTSUPP;
  1131. }
  1132. if (!info->attrs[NL802154_ATTR_SCAN_TYPE]) {
  1133. NL_SET_ERR_MSG(info->extack, "Malformed request, missing scan type");
  1134. return -EINVAL;
  1135. }
  1136. if (wpan_phy->flags & WPAN_PHY_FLAG_DATAGRAMS_ONLY) {
  1137. NL_SET_ERR_MSG(info->extack, "PHY only supports datagrams");
  1138. return -EOPNOTSUPP;
  1139. }
  1140. request = kzalloc_obj(*request);
  1141. if (!request)
  1142. return -ENOMEM;
  1143. request->wpan_dev = wpan_dev;
  1144. request->wpan_phy = wpan_phy;
  1145. type = nla_get_u8(info->attrs[NL802154_ATTR_SCAN_TYPE]);
  1146. switch (type) {
  1147. case NL802154_SCAN_ACTIVE:
  1148. case NL802154_SCAN_PASSIVE:
  1149. request->type = type;
  1150. break;
  1151. default:
  1152. NL_SET_ERR_MSG_FMT(info->extack, "Unsupported scan type: %d", type);
  1153. err = -EINVAL;
  1154. goto free_request;
  1155. }
  1156. /* Use current page by default */
  1157. request->page = nla_get_u8_default(info->attrs[NL802154_ATTR_PAGE],
  1158. wpan_phy->current_page);
  1159. /* Scan all supported channels by default */
  1160. request->channels =
  1161. nla_get_u32_default(info->attrs[NL802154_ATTR_SCAN_CHANNELS],
  1162. wpan_phy->supported.channels[request->page]);
  1163. /* Use maximum duration order by default */
  1164. request->duration =
  1165. nla_get_u8_default(info->attrs[NL802154_ATTR_SCAN_DURATION],
  1166. IEEE802154_MAX_SCAN_DURATION);
  1167. err = rdev_trigger_scan(rdev, request);
  1168. if (err) {
  1169. pr_err("Failure starting scanning (%d)\n", err);
  1170. goto free_request;
  1171. }
  1172. return 0;
  1173. free_request:
  1174. kfree(request);
  1175. return err;
  1176. }
  1177. static int nl802154_prep_scan_msg(struct sk_buff *msg,
  1178. struct cfg802154_registered_device *rdev,
  1179. struct wpan_dev *wpan_dev, u32 portid,
  1180. u32 seq, int flags, u8 cmd, u8 arg)
  1181. {
  1182. void *hdr;
  1183. hdr = nl802154hdr_put(msg, portid, seq, flags, cmd);
  1184. if (!hdr)
  1185. return -ENOBUFS;
  1186. if (nla_put_u32(msg, NL802154_ATTR_WPAN_PHY, rdev->wpan_phy_idx))
  1187. goto nla_put_failure;
  1188. if (wpan_dev->netdev &&
  1189. nla_put_u32(msg, NL802154_ATTR_IFINDEX, wpan_dev->netdev->ifindex))
  1190. goto nla_put_failure;
  1191. if (nla_put_u64_64bit(msg, NL802154_ATTR_WPAN_DEV,
  1192. wpan_dev_id(wpan_dev), NL802154_ATTR_PAD))
  1193. goto nla_put_failure;
  1194. if (cmd == NL802154_CMD_SCAN_DONE &&
  1195. nla_put_u8(msg, NL802154_ATTR_SCAN_DONE_REASON, arg))
  1196. goto nla_put_failure;
  1197. genlmsg_end(msg, hdr);
  1198. return 0;
  1199. nla_put_failure:
  1200. genlmsg_cancel(msg, hdr);
  1201. return -EMSGSIZE;
  1202. }
  1203. static int nl802154_send_scan_msg(struct cfg802154_registered_device *rdev,
  1204. struct wpan_dev *wpan_dev, u8 cmd, u8 arg)
  1205. {
  1206. struct sk_buff *msg;
  1207. int ret;
  1208. msg = nlmsg_new(NLMSG_DEFAULT_SIZE, GFP_KERNEL);
  1209. if (!msg)
  1210. return -ENOMEM;
  1211. ret = nl802154_prep_scan_msg(msg, rdev, wpan_dev, 0, 0, 0, cmd, arg);
  1212. if (ret < 0) {
  1213. nlmsg_free(msg);
  1214. return ret;
  1215. }
  1216. return genlmsg_multicast_netns(&nl802154_fam,
  1217. wpan_phy_net(&rdev->wpan_phy), msg, 0,
  1218. NL802154_MCGRP_SCAN, GFP_KERNEL);
  1219. }
  1220. int nl802154_scan_started(struct wpan_phy *wpan_phy, struct wpan_dev *wpan_dev)
  1221. {
  1222. struct cfg802154_registered_device *rdev = wpan_phy_to_rdev(wpan_phy);
  1223. int err;
  1224. /* Ignore errors when there are no listeners */
  1225. err = nl802154_send_scan_msg(rdev, wpan_dev, NL802154_CMD_TRIGGER_SCAN, 0);
  1226. if (err == -ESRCH)
  1227. err = 0;
  1228. return err;
  1229. }
  1230. EXPORT_SYMBOL_GPL(nl802154_scan_started);
  1231. int nl802154_scan_done(struct wpan_phy *wpan_phy, struct wpan_dev *wpan_dev,
  1232. enum nl802154_scan_done_reasons reason)
  1233. {
  1234. struct cfg802154_registered_device *rdev = wpan_phy_to_rdev(wpan_phy);
  1235. int err;
  1236. /* Ignore errors when there are no listeners */
  1237. err = nl802154_send_scan_msg(rdev, wpan_dev, NL802154_CMD_SCAN_DONE, reason);
  1238. if (err == -ESRCH)
  1239. err = 0;
  1240. return err;
  1241. }
  1242. EXPORT_SYMBOL_GPL(nl802154_scan_done);
  1243. static int nl802154_abort_scan(struct sk_buff *skb, struct genl_info *info)
  1244. {
  1245. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1246. struct net_device *dev = info->user_ptr[1];
  1247. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1248. /* Resources are released in the notification helper above */
  1249. return rdev_abort_scan(rdev, wpan_dev);
  1250. }
  1251. static int
  1252. nl802154_send_beacons(struct sk_buff *skb, struct genl_info *info)
  1253. {
  1254. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1255. struct net_device *dev = info->user_ptr[1];
  1256. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1257. struct wpan_phy *wpan_phy = &rdev->wpan_phy;
  1258. struct cfg802154_beacon_request *request;
  1259. int err;
  1260. if (wpan_dev->iftype != NL802154_IFTYPE_COORD) {
  1261. NL_SET_ERR_MSG(info->extack, "Only coordinators can send beacons");
  1262. return -EOPNOTSUPP;
  1263. }
  1264. if (wpan_dev->pan_id == cpu_to_le16(IEEE802154_PANID_BROADCAST)) {
  1265. NL_SET_ERR_MSG(info->extack, "Device is not part of any PAN");
  1266. return -EPERM;
  1267. }
  1268. if (wpan_phy->flags & WPAN_PHY_FLAG_DATAGRAMS_ONLY) {
  1269. NL_SET_ERR_MSG(info->extack, "PHY only supports datagrams");
  1270. return -EOPNOTSUPP;
  1271. }
  1272. request = kzalloc_obj(*request);
  1273. if (!request)
  1274. return -ENOMEM;
  1275. request->wpan_dev = wpan_dev;
  1276. request->wpan_phy = wpan_phy;
  1277. /* Use maximum duration order by default */
  1278. request->interval = nla_get_u8_default(info->attrs[NL802154_ATTR_BEACON_INTERVAL],
  1279. IEEE802154_MAX_SCAN_DURATION);
  1280. err = rdev_send_beacons(rdev, request);
  1281. if (err) {
  1282. pr_err("Failure starting sending beacons (%d)\n", err);
  1283. goto free_request;
  1284. }
  1285. return 0;
  1286. free_request:
  1287. kfree(request);
  1288. return err;
  1289. }
  1290. void nl802154_beaconing_done(struct wpan_dev *wpan_dev)
  1291. {
  1292. /* NOP */
  1293. }
  1294. EXPORT_SYMBOL_GPL(nl802154_beaconing_done);
  1295. static int
  1296. nl802154_stop_beacons(struct sk_buff *skb, struct genl_info *info)
  1297. {
  1298. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1299. struct net_device *dev = info->user_ptr[1];
  1300. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1301. /* Resources are released in the notification helper above */
  1302. return rdev_stop_beacons(rdev, wpan_dev);
  1303. }
  1304. static int nl802154_associate(struct sk_buff *skb, struct genl_info *info)
  1305. {
  1306. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1307. struct net_device *dev = info->user_ptr[1];
  1308. struct wpan_dev *wpan_dev;
  1309. struct wpan_phy *wpan_phy;
  1310. struct ieee802154_addr coord;
  1311. int err;
  1312. wpan_dev = dev->ieee802154_ptr;
  1313. wpan_phy = &rdev->wpan_phy;
  1314. if (wpan_phy->flags & WPAN_PHY_FLAG_DATAGRAMS_ONLY) {
  1315. NL_SET_ERR_MSG(info->extack, "PHY only supports datagrams");
  1316. return -EOPNOTSUPP;
  1317. }
  1318. if (!info->attrs[NL802154_ATTR_PAN_ID] ||
  1319. !info->attrs[NL802154_ATTR_EXTENDED_ADDR])
  1320. return -EINVAL;
  1321. coord.pan_id = nla_get_le16(info->attrs[NL802154_ATTR_PAN_ID]);
  1322. coord.mode = IEEE802154_ADDR_LONG;
  1323. coord.extended_addr = nla_get_le64(info->attrs[NL802154_ATTR_EXTENDED_ADDR]);
  1324. mutex_lock(&wpan_dev->association_lock);
  1325. err = rdev_associate(rdev, wpan_dev, &coord);
  1326. mutex_unlock(&wpan_dev->association_lock);
  1327. if (err)
  1328. pr_err("Association with PAN ID 0x%x failed (%d)\n",
  1329. le16_to_cpu(coord.pan_id), err);
  1330. return err;
  1331. }
  1332. static int nl802154_disassociate(struct sk_buff *skb, struct genl_info *info)
  1333. {
  1334. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1335. struct net_device *dev = info->user_ptr[1];
  1336. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1337. struct wpan_phy *wpan_phy = &rdev->wpan_phy;
  1338. struct ieee802154_addr target;
  1339. if (wpan_phy->flags & WPAN_PHY_FLAG_DATAGRAMS_ONLY) {
  1340. NL_SET_ERR_MSG(info->extack, "PHY only supports datagrams");
  1341. return -EOPNOTSUPP;
  1342. }
  1343. target.pan_id = wpan_dev->pan_id;
  1344. if (info->attrs[NL802154_ATTR_EXTENDED_ADDR]) {
  1345. target.mode = IEEE802154_ADDR_LONG;
  1346. target.extended_addr = nla_get_le64(info->attrs[NL802154_ATTR_EXTENDED_ADDR]);
  1347. } else if (info->attrs[NL802154_ATTR_SHORT_ADDR]) {
  1348. target.mode = IEEE802154_ADDR_SHORT;
  1349. target.short_addr = nla_get_le16(info->attrs[NL802154_ATTR_SHORT_ADDR]);
  1350. } else {
  1351. NL_SET_ERR_MSG(info->extack, "Device address is missing");
  1352. return -EINVAL;
  1353. }
  1354. mutex_lock(&wpan_dev->association_lock);
  1355. rdev_disassociate(rdev, wpan_dev, &target);
  1356. mutex_unlock(&wpan_dev->association_lock);
  1357. return 0;
  1358. }
  1359. static int nl802154_set_max_associations(struct sk_buff *skb, struct genl_info *info)
  1360. {
  1361. struct net_device *dev = info->user_ptr[1];
  1362. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1363. unsigned int max_assoc;
  1364. if (!info->attrs[NL802154_ATTR_MAX_ASSOCIATIONS]) {
  1365. NL_SET_ERR_MSG(info->extack, "No maximum number of association given");
  1366. return -EINVAL;
  1367. }
  1368. max_assoc = nla_get_u32(info->attrs[NL802154_ATTR_MAX_ASSOCIATIONS]);
  1369. mutex_lock(&wpan_dev->association_lock);
  1370. cfg802154_set_max_associations(wpan_dev, max_assoc);
  1371. mutex_unlock(&wpan_dev->association_lock);
  1372. return 0;
  1373. }
  1374. static int nl802154_send_peer_info(struct sk_buff *msg,
  1375. struct netlink_callback *cb,
  1376. u32 seq, int flags,
  1377. struct cfg802154_registered_device *rdev,
  1378. struct wpan_dev *wpan_dev,
  1379. struct ieee802154_pan_device *peer,
  1380. enum nl802154_peer_type type)
  1381. {
  1382. struct nlattr *nla;
  1383. void *hdr;
  1384. ASSERT_RTNL();
  1385. hdr = nl802154hdr_put(msg, NETLINK_CB(cb->skb).portid, seq, flags,
  1386. NL802154_CMD_LIST_ASSOCIATIONS);
  1387. if (!hdr)
  1388. return -ENOBUFS;
  1389. genl_dump_check_consistent(cb, hdr);
  1390. nla = nla_nest_start_noflag(msg, NL802154_ATTR_PEER);
  1391. if (!nla)
  1392. goto nla_put_failure;
  1393. if (nla_put_u8(msg, NL802154_DEV_ADDR_ATTR_PEER_TYPE, type))
  1394. goto nla_put_failure;
  1395. if (nla_put_u8(msg, NL802154_DEV_ADDR_ATTR_MODE, peer->mode))
  1396. goto nla_put_failure;
  1397. if (nla_put(msg, NL802154_DEV_ADDR_ATTR_SHORT,
  1398. IEEE802154_SHORT_ADDR_LEN, &peer->short_addr))
  1399. goto nla_put_failure;
  1400. if (nla_put(msg, NL802154_DEV_ADDR_ATTR_EXTENDED,
  1401. IEEE802154_EXTENDED_ADDR_LEN, &peer->extended_addr))
  1402. goto nla_put_failure;
  1403. nla_nest_end(msg, nla);
  1404. genlmsg_end(msg, hdr);
  1405. return 0;
  1406. nla_put_failure:
  1407. genlmsg_cancel(msg, hdr);
  1408. return -EMSGSIZE;
  1409. }
  1410. static int nl802154_list_associations(struct sk_buff *skb,
  1411. struct netlink_callback *cb)
  1412. {
  1413. struct cfg802154_registered_device *rdev;
  1414. struct ieee802154_pan_device *child;
  1415. struct wpan_dev *wpan_dev;
  1416. int err;
  1417. err = nl802154_prepare_wpan_dev_dump(skb, cb, &rdev, &wpan_dev);
  1418. if (err)
  1419. return err;
  1420. mutex_lock(&wpan_dev->association_lock);
  1421. if (cb->args[2])
  1422. goto out;
  1423. if (wpan_dev->parent) {
  1424. err = nl802154_send_peer_info(skb, cb, cb->nlh->nlmsg_seq,
  1425. NLM_F_MULTI, rdev, wpan_dev,
  1426. wpan_dev->parent,
  1427. NL802154_PEER_TYPE_PARENT);
  1428. if (err < 0)
  1429. goto out_err;
  1430. }
  1431. list_for_each_entry(child, &wpan_dev->children, node) {
  1432. err = nl802154_send_peer_info(skb, cb, cb->nlh->nlmsg_seq,
  1433. NLM_F_MULTI, rdev, wpan_dev,
  1434. child,
  1435. NL802154_PEER_TYPE_CHILD);
  1436. if (err < 0)
  1437. goto out_err;
  1438. }
  1439. cb->args[2] = 1;
  1440. out:
  1441. err = skb->len;
  1442. out_err:
  1443. mutex_unlock(&wpan_dev->association_lock);
  1444. nl802154_finish_wpan_dev_dump(rdev);
  1445. return err;
  1446. }
  1447. #ifdef CONFIG_IEEE802154_NL802154_EXPERIMENTAL
  1448. static const struct nla_policy nl802154_dev_addr_policy[NL802154_DEV_ADDR_ATTR_MAX + 1] = {
  1449. [NL802154_DEV_ADDR_ATTR_PAN_ID] = { .type = NLA_U16 },
  1450. [NL802154_DEV_ADDR_ATTR_MODE] = { .type = NLA_U32 },
  1451. [NL802154_DEV_ADDR_ATTR_SHORT] = { .type = NLA_U16 },
  1452. [NL802154_DEV_ADDR_ATTR_EXTENDED] = { .type = NLA_U64 },
  1453. };
  1454. static int
  1455. ieee802154_llsec_parse_dev_addr(struct nlattr *nla,
  1456. struct ieee802154_addr *addr)
  1457. {
  1458. struct nlattr *attrs[NL802154_DEV_ADDR_ATTR_MAX + 1];
  1459. if (!nla || nla_parse_nested_deprecated(attrs, NL802154_DEV_ADDR_ATTR_MAX, nla, nl802154_dev_addr_policy, NULL))
  1460. return -EINVAL;
  1461. if (!attrs[NL802154_DEV_ADDR_ATTR_PAN_ID] || !attrs[NL802154_DEV_ADDR_ATTR_MODE])
  1462. return -EINVAL;
  1463. addr->pan_id = nla_get_le16(attrs[NL802154_DEV_ADDR_ATTR_PAN_ID]);
  1464. addr->mode = nla_get_u32(attrs[NL802154_DEV_ADDR_ATTR_MODE]);
  1465. switch (addr->mode) {
  1466. case NL802154_DEV_ADDR_SHORT:
  1467. if (!attrs[NL802154_DEV_ADDR_ATTR_SHORT])
  1468. return -EINVAL;
  1469. addr->short_addr = nla_get_le16(attrs[NL802154_DEV_ADDR_ATTR_SHORT]);
  1470. break;
  1471. case NL802154_DEV_ADDR_EXTENDED:
  1472. if (!attrs[NL802154_DEV_ADDR_ATTR_EXTENDED])
  1473. return -EINVAL;
  1474. addr->extended_addr = nla_get_le64(attrs[NL802154_DEV_ADDR_ATTR_EXTENDED]);
  1475. break;
  1476. default:
  1477. return -EINVAL;
  1478. }
  1479. return 0;
  1480. }
  1481. static const struct nla_policy nl802154_key_id_policy[NL802154_KEY_ID_ATTR_MAX + 1] = {
  1482. [NL802154_KEY_ID_ATTR_MODE] = { .type = NLA_U32 },
  1483. [NL802154_KEY_ID_ATTR_INDEX] = { .type = NLA_U8 },
  1484. [NL802154_KEY_ID_ATTR_IMPLICIT] = { .type = NLA_NESTED },
  1485. [NL802154_KEY_ID_ATTR_SOURCE_SHORT] = { .type = NLA_U32 },
  1486. [NL802154_KEY_ID_ATTR_SOURCE_EXTENDED] = { .type = NLA_U64 },
  1487. };
  1488. static int
  1489. ieee802154_llsec_parse_key_id(struct nlattr *nla,
  1490. struct ieee802154_llsec_key_id *desc)
  1491. {
  1492. struct nlattr *attrs[NL802154_KEY_ID_ATTR_MAX + 1];
  1493. if (!nla || nla_parse_nested_deprecated(attrs, NL802154_KEY_ID_ATTR_MAX, nla, nl802154_key_id_policy, NULL))
  1494. return -EINVAL;
  1495. if (!attrs[NL802154_KEY_ID_ATTR_MODE])
  1496. return -EINVAL;
  1497. desc->mode = nla_get_u32(attrs[NL802154_KEY_ID_ATTR_MODE]);
  1498. switch (desc->mode) {
  1499. case NL802154_KEY_ID_MODE_IMPLICIT:
  1500. if (!attrs[NL802154_KEY_ID_ATTR_IMPLICIT])
  1501. return -EINVAL;
  1502. if (ieee802154_llsec_parse_dev_addr(attrs[NL802154_KEY_ID_ATTR_IMPLICIT],
  1503. &desc->device_addr) < 0)
  1504. return -EINVAL;
  1505. break;
  1506. case NL802154_KEY_ID_MODE_INDEX:
  1507. break;
  1508. case NL802154_KEY_ID_MODE_INDEX_SHORT:
  1509. if (!attrs[NL802154_KEY_ID_ATTR_SOURCE_SHORT])
  1510. return -EINVAL;
  1511. desc->short_source = nla_get_le32(attrs[NL802154_KEY_ID_ATTR_SOURCE_SHORT]);
  1512. break;
  1513. case NL802154_KEY_ID_MODE_INDEX_EXTENDED:
  1514. if (!attrs[NL802154_KEY_ID_ATTR_SOURCE_EXTENDED])
  1515. return -EINVAL;
  1516. desc->extended_source = nla_get_le64(attrs[NL802154_KEY_ID_ATTR_SOURCE_EXTENDED]);
  1517. break;
  1518. default:
  1519. return -EINVAL;
  1520. }
  1521. if (desc->mode != NL802154_KEY_ID_MODE_IMPLICIT) {
  1522. if (!attrs[NL802154_KEY_ID_ATTR_INDEX])
  1523. return -EINVAL;
  1524. /* TODO change id to idx */
  1525. desc->id = nla_get_u8(attrs[NL802154_KEY_ID_ATTR_INDEX]);
  1526. }
  1527. return 0;
  1528. }
  1529. static int nl802154_set_llsec_params(struct sk_buff *skb,
  1530. struct genl_info *info)
  1531. {
  1532. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1533. struct net_device *dev = info->user_ptr[1];
  1534. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1535. struct ieee802154_llsec_params params;
  1536. u32 changed = 0;
  1537. int ret;
  1538. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  1539. return -EOPNOTSUPP;
  1540. if (info->attrs[NL802154_ATTR_SEC_ENABLED]) {
  1541. u8 enabled;
  1542. enabled = nla_get_u8(info->attrs[NL802154_ATTR_SEC_ENABLED]);
  1543. if (enabled != 0 && enabled != 1)
  1544. return -EINVAL;
  1545. params.enabled = nla_get_u8(info->attrs[NL802154_ATTR_SEC_ENABLED]);
  1546. changed |= IEEE802154_LLSEC_PARAM_ENABLED;
  1547. }
  1548. if (info->attrs[NL802154_ATTR_SEC_OUT_KEY_ID]) {
  1549. ret = ieee802154_llsec_parse_key_id(info->attrs[NL802154_ATTR_SEC_OUT_KEY_ID],
  1550. &params.out_key);
  1551. if (ret < 0)
  1552. return ret;
  1553. changed |= IEEE802154_LLSEC_PARAM_OUT_KEY;
  1554. }
  1555. if (info->attrs[NL802154_ATTR_SEC_OUT_LEVEL]) {
  1556. params.out_level = nla_get_u32(info->attrs[NL802154_ATTR_SEC_OUT_LEVEL]);
  1557. if (params.out_level > NL802154_SECLEVEL_MAX)
  1558. return -EINVAL;
  1559. changed |= IEEE802154_LLSEC_PARAM_OUT_LEVEL;
  1560. }
  1561. if (info->attrs[NL802154_ATTR_SEC_FRAME_COUNTER]) {
  1562. params.frame_counter = nla_get_be32(info->attrs[NL802154_ATTR_SEC_FRAME_COUNTER]);
  1563. changed |= IEEE802154_LLSEC_PARAM_FRAME_COUNTER;
  1564. }
  1565. return rdev_set_llsec_params(rdev, wpan_dev, &params, changed);
  1566. }
  1567. static int nl802154_send_key(struct sk_buff *msg, u32 cmd, u32 portid,
  1568. u32 seq, int flags,
  1569. struct cfg802154_registered_device *rdev,
  1570. struct net_device *dev,
  1571. const struct ieee802154_llsec_key_entry *key)
  1572. {
  1573. void *hdr;
  1574. u32 commands[NL802154_CMD_FRAME_NR_IDS / 32];
  1575. struct nlattr *nl_key, *nl_key_id;
  1576. hdr = nl802154hdr_put(msg, portid, seq, flags, cmd);
  1577. if (!hdr)
  1578. return -ENOBUFS;
  1579. if (nla_put_u32(msg, NL802154_ATTR_IFINDEX, dev->ifindex))
  1580. goto nla_put_failure;
  1581. nl_key = nla_nest_start_noflag(msg, NL802154_ATTR_SEC_KEY);
  1582. if (!nl_key)
  1583. goto nla_put_failure;
  1584. nl_key_id = nla_nest_start_noflag(msg, NL802154_KEY_ATTR_ID);
  1585. if (!nl_key_id)
  1586. goto nla_put_failure;
  1587. if (ieee802154_llsec_send_key_id(msg, &key->id) < 0)
  1588. goto nla_put_failure;
  1589. nla_nest_end(msg, nl_key_id);
  1590. if (nla_put_u8(msg, NL802154_KEY_ATTR_USAGE_FRAMES,
  1591. key->key->frame_types))
  1592. goto nla_put_failure;
  1593. if (key->key->frame_types & BIT(NL802154_FRAME_CMD)) {
  1594. /* TODO for each nested */
  1595. memset(commands, 0, sizeof(commands));
  1596. commands[7] = key->key->cmd_frame_ids;
  1597. if (nla_put(msg, NL802154_KEY_ATTR_USAGE_CMDS,
  1598. sizeof(commands), commands))
  1599. goto nla_put_failure;
  1600. }
  1601. if (nla_put(msg, NL802154_KEY_ATTR_BYTES, NL802154_KEY_SIZE,
  1602. key->key->key))
  1603. goto nla_put_failure;
  1604. nla_nest_end(msg, nl_key);
  1605. genlmsg_end(msg, hdr);
  1606. return 0;
  1607. nla_put_failure:
  1608. genlmsg_cancel(msg, hdr);
  1609. return -EMSGSIZE;
  1610. }
  1611. static int
  1612. nl802154_dump_llsec_key(struct sk_buff *skb, struct netlink_callback *cb)
  1613. {
  1614. struct cfg802154_registered_device *rdev = NULL;
  1615. struct ieee802154_llsec_key_entry *key;
  1616. struct ieee802154_llsec_table *table;
  1617. struct wpan_dev *wpan_dev;
  1618. int err;
  1619. err = nl802154_prepare_wpan_dev_dump(skb, cb, &rdev, &wpan_dev);
  1620. if (err)
  1621. return err;
  1622. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR) {
  1623. err = skb->len;
  1624. goto out_err;
  1625. }
  1626. if (!wpan_dev->netdev) {
  1627. err = -EINVAL;
  1628. goto out_err;
  1629. }
  1630. rdev_lock_llsec_table(rdev, wpan_dev);
  1631. rdev_get_llsec_table(rdev, wpan_dev, &table);
  1632. /* TODO make it like station dump */
  1633. if (cb->args[2])
  1634. goto out;
  1635. list_for_each_entry(key, &table->keys, list) {
  1636. if (nl802154_send_key(skb, NL802154_CMD_NEW_SEC_KEY,
  1637. NETLINK_CB(cb->skb).portid,
  1638. cb->nlh->nlmsg_seq, NLM_F_MULTI,
  1639. rdev, wpan_dev->netdev, key) < 0) {
  1640. /* TODO */
  1641. err = -EIO;
  1642. rdev_unlock_llsec_table(rdev, wpan_dev);
  1643. goto out_err;
  1644. }
  1645. }
  1646. cb->args[2] = 1;
  1647. out:
  1648. rdev_unlock_llsec_table(rdev, wpan_dev);
  1649. err = skb->len;
  1650. out_err:
  1651. nl802154_finish_wpan_dev_dump(rdev);
  1652. return err;
  1653. }
  1654. static const struct nla_policy nl802154_key_policy[NL802154_KEY_ATTR_MAX + 1] = {
  1655. [NL802154_KEY_ATTR_ID] = { NLA_NESTED },
  1656. /* TODO handle it as for_each_nested and NLA_FLAG? */
  1657. [NL802154_KEY_ATTR_USAGE_FRAMES] = { NLA_U8 },
  1658. /* TODO handle it as for_each_nested, not static array? */
  1659. [NL802154_KEY_ATTR_USAGE_CMDS] = { .len = NL802154_CMD_FRAME_NR_IDS / 8 },
  1660. [NL802154_KEY_ATTR_BYTES] = { .len = NL802154_KEY_SIZE },
  1661. };
  1662. static int nl802154_add_llsec_key(struct sk_buff *skb, struct genl_info *info)
  1663. {
  1664. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1665. struct net_device *dev = info->user_ptr[1];
  1666. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1667. struct nlattr *attrs[NL802154_KEY_ATTR_MAX + 1];
  1668. struct ieee802154_llsec_key key = { };
  1669. struct ieee802154_llsec_key_id id = { };
  1670. u32 commands[NL802154_CMD_FRAME_NR_IDS / 32] = { };
  1671. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  1672. return -EOPNOTSUPP;
  1673. if (!info->attrs[NL802154_ATTR_SEC_KEY] ||
  1674. nla_parse_nested_deprecated(attrs, NL802154_KEY_ATTR_MAX, info->attrs[NL802154_ATTR_SEC_KEY], nl802154_key_policy, info->extack))
  1675. return -EINVAL;
  1676. if (!attrs[NL802154_KEY_ATTR_USAGE_FRAMES] ||
  1677. !attrs[NL802154_KEY_ATTR_BYTES])
  1678. return -EINVAL;
  1679. if (ieee802154_llsec_parse_key_id(attrs[NL802154_KEY_ATTR_ID], &id) < 0)
  1680. return -ENOBUFS;
  1681. key.frame_types = nla_get_u8(attrs[NL802154_KEY_ATTR_USAGE_FRAMES]);
  1682. if (key.frame_types > BIT(NL802154_FRAME_MAX) ||
  1683. ((key.frame_types & BIT(NL802154_FRAME_CMD)) &&
  1684. !attrs[NL802154_KEY_ATTR_USAGE_CMDS]))
  1685. return -EINVAL;
  1686. if (attrs[NL802154_KEY_ATTR_USAGE_CMDS]) {
  1687. /* TODO for each nested */
  1688. nla_memcpy(commands, attrs[NL802154_KEY_ATTR_USAGE_CMDS],
  1689. NL802154_CMD_FRAME_NR_IDS / 8);
  1690. /* TODO understand the -EINVAL logic here? last condition */
  1691. if (commands[0] || commands[1] || commands[2] || commands[3] ||
  1692. commands[4] || commands[5] || commands[6] ||
  1693. commands[7] > BIT(NL802154_CMD_FRAME_MAX))
  1694. return -EINVAL;
  1695. key.cmd_frame_ids = commands[7];
  1696. } else {
  1697. key.cmd_frame_ids = 0;
  1698. }
  1699. nla_memcpy(key.key, attrs[NL802154_KEY_ATTR_BYTES], NL802154_KEY_SIZE);
  1700. if (ieee802154_llsec_parse_key_id(attrs[NL802154_KEY_ATTR_ID], &id) < 0)
  1701. return -ENOBUFS;
  1702. return rdev_add_llsec_key(rdev, wpan_dev, &id, &key);
  1703. }
  1704. static int nl802154_del_llsec_key(struct sk_buff *skb, struct genl_info *info)
  1705. {
  1706. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1707. struct net_device *dev = info->user_ptr[1];
  1708. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1709. struct nlattr *attrs[NL802154_KEY_ATTR_MAX + 1];
  1710. struct ieee802154_llsec_key_id id;
  1711. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  1712. return -EOPNOTSUPP;
  1713. if (!info->attrs[NL802154_ATTR_SEC_KEY] ||
  1714. nla_parse_nested_deprecated(attrs, NL802154_KEY_ATTR_MAX, info->attrs[NL802154_ATTR_SEC_KEY], nl802154_key_policy, info->extack))
  1715. return -EINVAL;
  1716. if (ieee802154_llsec_parse_key_id(attrs[NL802154_KEY_ATTR_ID], &id) < 0)
  1717. return -ENOBUFS;
  1718. return rdev_del_llsec_key(rdev, wpan_dev, &id);
  1719. }
  1720. static int nl802154_send_device(struct sk_buff *msg, u32 cmd, u32 portid,
  1721. u32 seq, int flags,
  1722. struct cfg802154_registered_device *rdev,
  1723. struct net_device *dev,
  1724. const struct ieee802154_llsec_device *dev_desc)
  1725. {
  1726. void *hdr;
  1727. struct nlattr *nl_device;
  1728. hdr = nl802154hdr_put(msg, portid, seq, flags, cmd);
  1729. if (!hdr)
  1730. return -ENOBUFS;
  1731. if (nla_put_u32(msg, NL802154_ATTR_IFINDEX, dev->ifindex))
  1732. goto nla_put_failure;
  1733. nl_device = nla_nest_start_noflag(msg, NL802154_ATTR_SEC_DEVICE);
  1734. if (!nl_device)
  1735. goto nla_put_failure;
  1736. if (nla_put_u32(msg, NL802154_DEV_ATTR_FRAME_COUNTER,
  1737. dev_desc->frame_counter) ||
  1738. nla_put_le16(msg, NL802154_DEV_ATTR_PAN_ID, dev_desc->pan_id) ||
  1739. nla_put_le16(msg, NL802154_DEV_ATTR_SHORT_ADDR,
  1740. dev_desc->short_addr) ||
  1741. nla_put_le64(msg, NL802154_DEV_ATTR_EXTENDED_ADDR,
  1742. dev_desc->hwaddr, NL802154_DEV_ATTR_PAD) ||
  1743. nla_put_u8(msg, NL802154_DEV_ATTR_SECLEVEL_EXEMPT,
  1744. dev_desc->seclevel_exempt) ||
  1745. nla_put_u32(msg, NL802154_DEV_ATTR_KEY_MODE, dev_desc->key_mode))
  1746. goto nla_put_failure;
  1747. nla_nest_end(msg, nl_device);
  1748. genlmsg_end(msg, hdr);
  1749. return 0;
  1750. nla_put_failure:
  1751. genlmsg_cancel(msg, hdr);
  1752. return -EMSGSIZE;
  1753. }
  1754. static int
  1755. nl802154_dump_llsec_dev(struct sk_buff *skb, struct netlink_callback *cb)
  1756. {
  1757. struct cfg802154_registered_device *rdev = NULL;
  1758. struct ieee802154_llsec_device *dev;
  1759. struct ieee802154_llsec_table *table;
  1760. struct wpan_dev *wpan_dev;
  1761. int err;
  1762. err = nl802154_prepare_wpan_dev_dump(skb, cb, &rdev, &wpan_dev);
  1763. if (err)
  1764. return err;
  1765. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR) {
  1766. err = skb->len;
  1767. goto out_err;
  1768. }
  1769. if (!wpan_dev->netdev) {
  1770. err = -EINVAL;
  1771. goto out_err;
  1772. }
  1773. rdev_lock_llsec_table(rdev, wpan_dev);
  1774. rdev_get_llsec_table(rdev, wpan_dev, &table);
  1775. /* TODO make it like station dump */
  1776. if (cb->args[2])
  1777. goto out;
  1778. list_for_each_entry(dev, &table->devices, list) {
  1779. if (nl802154_send_device(skb, NL802154_CMD_NEW_SEC_LEVEL,
  1780. NETLINK_CB(cb->skb).portid,
  1781. cb->nlh->nlmsg_seq, NLM_F_MULTI,
  1782. rdev, wpan_dev->netdev, dev) < 0) {
  1783. /* TODO */
  1784. err = -EIO;
  1785. rdev_unlock_llsec_table(rdev, wpan_dev);
  1786. goto out_err;
  1787. }
  1788. }
  1789. cb->args[2] = 1;
  1790. out:
  1791. rdev_unlock_llsec_table(rdev, wpan_dev);
  1792. err = skb->len;
  1793. out_err:
  1794. nl802154_finish_wpan_dev_dump(rdev);
  1795. return err;
  1796. }
  1797. static const struct nla_policy nl802154_dev_policy[NL802154_DEV_ATTR_MAX + 1] = {
  1798. [NL802154_DEV_ATTR_FRAME_COUNTER] = { NLA_U32 },
  1799. [NL802154_DEV_ATTR_PAN_ID] = { .type = NLA_U16 },
  1800. [NL802154_DEV_ATTR_SHORT_ADDR] = { .type = NLA_U16 },
  1801. [NL802154_DEV_ATTR_EXTENDED_ADDR] = { .type = NLA_U64 },
  1802. [NL802154_DEV_ATTR_SECLEVEL_EXEMPT] = { NLA_U8 },
  1803. [NL802154_DEV_ATTR_KEY_MODE] = { NLA_U32 },
  1804. };
  1805. static int
  1806. ieee802154_llsec_parse_device(struct nlattr *nla,
  1807. struct ieee802154_llsec_device *dev)
  1808. {
  1809. struct nlattr *attrs[NL802154_DEV_ATTR_MAX + 1];
  1810. if (!nla || nla_parse_nested_deprecated(attrs, NL802154_DEV_ATTR_MAX, nla, nl802154_dev_policy, NULL))
  1811. return -EINVAL;
  1812. memset(dev, 0, sizeof(*dev));
  1813. if (!attrs[NL802154_DEV_ATTR_FRAME_COUNTER] ||
  1814. !attrs[NL802154_DEV_ATTR_PAN_ID] ||
  1815. !attrs[NL802154_DEV_ATTR_SHORT_ADDR] ||
  1816. !attrs[NL802154_DEV_ATTR_EXTENDED_ADDR] ||
  1817. !attrs[NL802154_DEV_ATTR_SECLEVEL_EXEMPT] ||
  1818. !attrs[NL802154_DEV_ATTR_KEY_MODE])
  1819. return -EINVAL;
  1820. /* TODO be32 */
  1821. dev->frame_counter = nla_get_u32(attrs[NL802154_DEV_ATTR_FRAME_COUNTER]);
  1822. dev->pan_id = nla_get_le16(attrs[NL802154_DEV_ATTR_PAN_ID]);
  1823. dev->short_addr = nla_get_le16(attrs[NL802154_DEV_ATTR_SHORT_ADDR]);
  1824. /* TODO rename hwaddr to extended_addr */
  1825. dev->hwaddr = nla_get_le64(attrs[NL802154_DEV_ATTR_EXTENDED_ADDR]);
  1826. dev->seclevel_exempt = nla_get_u8(attrs[NL802154_DEV_ATTR_SECLEVEL_EXEMPT]);
  1827. dev->key_mode = nla_get_u32(attrs[NL802154_DEV_ATTR_KEY_MODE]);
  1828. if (dev->key_mode > NL802154_DEVKEY_MAX ||
  1829. (dev->seclevel_exempt != 0 && dev->seclevel_exempt != 1))
  1830. return -EINVAL;
  1831. return 0;
  1832. }
  1833. static int nl802154_add_llsec_dev(struct sk_buff *skb, struct genl_info *info)
  1834. {
  1835. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1836. struct net_device *dev = info->user_ptr[1];
  1837. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1838. struct ieee802154_llsec_device dev_desc;
  1839. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  1840. return -EOPNOTSUPP;
  1841. if (ieee802154_llsec_parse_device(info->attrs[NL802154_ATTR_SEC_DEVICE],
  1842. &dev_desc) < 0)
  1843. return -EINVAL;
  1844. return rdev_add_device(rdev, wpan_dev, &dev_desc);
  1845. }
  1846. static int nl802154_del_llsec_dev(struct sk_buff *skb, struct genl_info *info)
  1847. {
  1848. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1849. struct net_device *dev = info->user_ptr[1];
  1850. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1851. struct nlattr *attrs[NL802154_DEV_ATTR_MAX + 1];
  1852. __le64 extended_addr;
  1853. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  1854. return -EOPNOTSUPP;
  1855. if (!info->attrs[NL802154_ATTR_SEC_DEVICE] ||
  1856. nla_parse_nested_deprecated(attrs, NL802154_DEV_ATTR_MAX, info->attrs[NL802154_ATTR_SEC_DEVICE], nl802154_dev_policy, info->extack))
  1857. return -EINVAL;
  1858. if (!attrs[NL802154_DEV_ATTR_EXTENDED_ADDR])
  1859. return -EINVAL;
  1860. extended_addr = nla_get_le64(attrs[NL802154_DEV_ATTR_EXTENDED_ADDR]);
  1861. return rdev_del_device(rdev, wpan_dev, extended_addr);
  1862. }
  1863. static int nl802154_send_devkey(struct sk_buff *msg, u32 cmd, u32 portid,
  1864. u32 seq, int flags,
  1865. struct cfg802154_registered_device *rdev,
  1866. struct net_device *dev, __le64 extended_addr,
  1867. const struct ieee802154_llsec_device_key *devkey)
  1868. {
  1869. void *hdr;
  1870. struct nlattr *nl_devkey, *nl_key_id;
  1871. hdr = nl802154hdr_put(msg, portid, seq, flags, cmd);
  1872. if (!hdr)
  1873. return -ENOBUFS;
  1874. if (nla_put_u32(msg, NL802154_ATTR_IFINDEX, dev->ifindex))
  1875. goto nla_put_failure;
  1876. nl_devkey = nla_nest_start_noflag(msg, NL802154_ATTR_SEC_DEVKEY);
  1877. if (!nl_devkey)
  1878. goto nla_put_failure;
  1879. if (nla_put_le64(msg, NL802154_DEVKEY_ATTR_EXTENDED_ADDR,
  1880. extended_addr, NL802154_DEVKEY_ATTR_PAD) ||
  1881. nla_put_u32(msg, NL802154_DEVKEY_ATTR_FRAME_COUNTER,
  1882. devkey->frame_counter))
  1883. goto nla_put_failure;
  1884. nl_key_id = nla_nest_start_noflag(msg, NL802154_DEVKEY_ATTR_ID);
  1885. if (!nl_key_id)
  1886. goto nla_put_failure;
  1887. if (ieee802154_llsec_send_key_id(msg, &devkey->key_id) < 0)
  1888. goto nla_put_failure;
  1889. nla_nest_end(msg, nl_key_id);
  1890. nla_nest_end(msg, nl_devkey);
  1891. genlmsg_end(msg, hdr);
  1892. return 0;
  1893. nla_put_failure:
  1894. genlmsg_cancel(msg, hdr);
  1895. return -EMSGSIZE;
  1896. }
  1897. static int
  1898. nl802154_dump_llsec_devkey(struct sk_buff *skb, struct netlink_callback *cb)
  1899. {
  1900. struct cfg802154_registered_device *rdev = NULL;
  1901. struct ieee802154_llsec_device_key *kpos;
  1902. struct ieee802154_llsec_device *dpos;
  1903. struct ieee802154_llsec_table *table;
  1904. struct wpan_dev *wpan_dev;
  1905. int err;
  1906. err = nl802154_prepare_wpan_dev_dump(skb, cb, &rdev, &wpan_dev);
  1907. if (err)
  1908. return err;
  1909. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR) {
  1910. err = skb->len;
  1911. goto out_err;
  1912. }
  1913. if (!wpan_dev->netdev) {
  1914. err = -EINVAL;
  1915. goto out_err;
  1916. }
  1917. rdev_lock_llsec_table(rdev, wpan_dev);
  1918. rdev_get_llsec_table(rdev, wpan_dev, &table);
  1919. /* TODO make it like station dump */
  1920. if (cb->args[2])
  1921. goto out;
  1922. /* TODO look if remove devkey and do some nested attribute */
  1923. list_for_each_entry(dpos, &table->devices, list) {
  1924. list_for_each_entry(kpos, &dpos->keys, list) {
  1925. if (nl802154_send_devkey(skb,
  1926. NL802154_CMD_NEW_SEC_LEVEL,
  1927. NETLINK_CB(cb->skb).portid,
  1928. cb->nlh->nlmsg_seq,
  1929. NLM_F_MULTI, rdev,
  1930. wpan_dev->netdev,
  1931. dpos->hwaddr,
  1932. kpos) < 0) {
  1933. /* TODO */
  1934. err = -EIO;
  1935. rdev_unlock_llsec_table(rdev, wpan_dev);
  1936. goto out_err;
  1937. }
  1938. }
  1939. }
  1940. cb->args[2] = 1;
  1941. out:
  1942. rdev_unlock_llsec_table(rdev, wpan_dev);
  1943. err = skb->len;
  1944. out_err:
  1945. nl802154_finish_wpan_dev_dump(rdev);
  1946. return err;
  1947. }
  1948. static const struct nla_policy nl802154_devkey_policy[NL802154_DEVKEY_ATTR_MAX + 1] = {
  1949. [NL802154_DEVKEY_ATTR_FRAME_COUNTER] = { NLA_U32 },
  1950. [NL802154_DEVKEY_ATTR_EXTENDED_ADDR] = { NLA_U64 },
  1951. [NL802154_DEVKEY_ATTR_ID] = { NLA_NESTED },
  1952. };
  1953. static int nl802154_add_llsec_devkey(struct sk_buff *skb, struct genl_info *info)
  1954. {
  1955. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1956. struct net_device *dev = info->user_ptr[1];
  1957. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1958. struct nlattr *attrs[NL802154_DEVKEY_ATTR_MAX + 1];
  1959. struct ieee802154_llsec_device_key key;
  1960. __le64 extended_addr;
  1961. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  1962. return -EOPNOTSUPP;
  1963. if (!info->attrs[NL802154_ATTR_SEC_DEVKEY] ||
  1964. nla_parse_nested_deprecated(attrs, NL802154_DEVKEY_ATTR_MAX, info->attrs[NL802154_ATTR_SEC_DEVKEY], nl802154_devkey_policy, info->extack) < 0)
  1965. return -EINVAL;
  1966. if (!attrs[NL802154_DEVKEY_ATTR_FRAME_COUNTER] ||
  1967. !attrs[NL802154_DEVKEY_ATTR_EXTENDED_ADDR])
  1968. return -EINVAL;
  1969. /* TODO change key.id ? */
  1970. if (ieee802154_llsec_parse_key_id(attrs[NL802154_DEVKEY_ATTR_ID],
  1971. &key.key_id) < 0)
  1972. return -ENOBUFS;
  1973. /* TODO be32 */
  1974. key.frame_counter = nla_get_u32(attrs[NL802154_DEVKEY_ATTR_FRAME_COUNTER]);
  1975. /* TODO change naming hwaddr -> extended_addr
  1976. * check unique identifier short+pan OR extended_addr
  1977. */
  1978. extended_addr = nla_get_le64(attrs[NL802154_DEVKEY_ATTR_EXTENDED_ADDR]);
  1979. return rdev_add_devkey(rdev, wpan_dev, extended_addr, &key);
  1980. }
  1981. static int nl802154_del_llsec_devkey(struct sk_buff *skb, struct genl_info *info)
  1982. {
  1983. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  1984. struct net_device *dev = info->user_ptr[1];
  1985. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  1986. struct nlattr *attrs[NL802154_DEVKEY_ATTR_MAX + 1];
  1987. struct ieee802154_llsec_device_key key;
  1988. __le64 extended_addr;
  1989. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  1990. return -EOPNOTSUPP;
  1991. if (!info->attrs[NL802154_ATTR_SEC_DEVKEY] ||
  1992. nla_parse_nested_deprecated(attrs, NL802154_DEVKEY_ATTR_MAX, info->attrs[NL802154_ATTR_SEC_DEVKEY], nl802154_devkey_policy, info->extack))
  1993. return -EINVAL;
  1994. if (!attrs[NL802154_DEVKEY_ATTR_EXTENDED_ADDR])
  1995. return -EINVAL;
  1996. /* TODO change key.id ? */
  1997. if (ieee802154_llsec_parse_key_id(attrs[NL802154_DEVKEY_ATTR_ID],
  1998. &key.key_id) < 0)
  1999. return -ENOBUFS;
  2000. /* TODO change naming hwaddr -> extended_addr
  2001. * check unique identifier short+pan OR extended_addr
  2002. */
  2003. extended_addr = nla_get_le64(attrs[NL802154_DEVKEY_ATTR_EXTENDED_ADDR]);
  2004. return rdev_del_devkey(rdev, wpan_dev, extended_addr, &key);
  2005. }
  2006. static int nl802154_send_seclevel(struct sk_buff *msg, u32 cmd, u32 portid,
  2007. u32 seq, int flags,
  2008. struct cfg802154_registered_device *rdev,
  2009. struct net_device *dev,
  2010. const struct ieee802154_llsec_seclevel *sl)
  2011. {
  2012. void *hdr;
  2013. struct nlattr *nl_seclevel;
  2014. hdr = nl802154hdr_put(msg, portid, seq, flags, cmd);
  2015. if (!hdr)
  2016. return -ENOBUFS;
  2017. if (nla_put_u32(msg, NL802154_ATTR_IFINDEX, dev->ifindex))
  2018. goto nla_put_failure;
  2019. nl_seclevel = nla_nest_start_noflag(msg, NL802154_ATTR_SEC_LEVEL);
  2020. if (!nl_seclevel)
  2021. goto nla_put_failure;
  2022. if (nla_put_u32(msg, NL802154_SECLEVEL_ATTR_FRAME, sl->frame_type) ||
  2023. nla_put_u32(msg, NL802154_SECLEVEL_ATTR_LEVELS, sl->sec_levels) ||
  2024. nla_put_u8(msg, NL802154_SECLEVEL_ATTR_DEV_OVERRIDE,
  2025. sl->device_override))
  2026. goto nla_put_failure;
  2027. if (sl->frame_type == NL802154_FRAME_CMD) {
  2028. if (nla_put_u32(msg, NL802154_SECLEVEL_ATTR_CMD_FRAME,
  2029. sl->cmd_frame_id))
  2030. goto nla_put_failure;
  2031. }
  2032. nla_nest_end(msg, nl_seclevel);
  2033. genlmsg_end(msg, hdr);
  2034. return 0;
  2035. nla_put_failure:
  2036. genlmsg_cancel(msg, hdr);
  2037. return -EMSGSIZE;
  2038. }
  2039. static int
  2040. nl802154_dump_llsec_seclevel(struct sk_buff *skb, struct netlink_callback *cb)
  2041. {
  2042. struct cfg802154_registered_device *rdev = NULL;
  2043. struct ieee802154_llsec_seclevel *sl;
  2044. struct ieee802154_llsec_table *table;
  2045. struct wpan_dev *wpan_dev;
  2046. int err;
  2047. err = nl802154_prepare_wpan_dev_dump(skb, cb, &rdev, &wpan_dev);
  2048. if (err)
  2049. return err;
  2050. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR) {
  2051. err = skb->len;
  2052. goto out_err;
  2053. }
  2054. if (!wpan_dev->netdev) {
  2055. err = -EINVAL;
  2056. goto out_err;
  2057. }
  2058. rdev_lock_llsec_table(rdev, wpan_dev);
  2059. rdev_get_llsec_table(rdev, wpan_dev, &table);
  2060. /* TODO make it like station dump */
  2061. if (cb->args[2])
  2062. goto out;
  2063. list_for_each_entry(sl, &table->security_levels, list) {
  2064. if (nl802154_send_seclevel(skb, NL802154_CMD_NEW_SEC_LEVEL,
  2065. NETLINK_CB(cb->skb).portid,
  2066. cb->nlh->nlmsg_seq, NLM_F_MULTI,
  2067. rdev, wpan_dev->netdev, sl) < 0) {
  2068. /* TODO */
  2069. err = -EIO;
  2070. rdev_unlock_llsec_table(rdev, wpan_dev);
  2071. goto out_err;
  2072. }
  2073. }
  2074. cb->args[2] = 1;
  2075. out:
  2076. rdev_unlock_llsec_table(rdev, wpan_dev);
  2077. err = skb->len;
  2078. out_err:
  2079. nl802154_finish_wpan_dev_dump(rdev);
  2080. return err;
  2081. }
  2082. static const struct nla_policy nl802154_seclevel_policy[NL802154_SECLEVEL_ATTR_MAX + 1] = {
  2083. [NL802154_SECLEVEL_ATTR_LEVELS] = { .type = NLA_U8 },
  2084. [NL802154_SECLEVEL_ATTR_FRAME] = { .type = NLA_U32 },
  2085. [NL802154_SECLEVEL_ATTR_CMD_FRAME] = { .type = NLA_U32 },
  2086. [NL802154_SECLEVEL_ATTR_DEV_OVERRIDE] = { .type = NLA_U8 },
  2087. };
  2088. static int
  2089. llsec_parse_seclevel(struct nlattr *nla, struct ieee802154_llsec_seclevel *sl)
  2090. {
  2091. struct nlattr *attrs[NL802154_SECLEVEL_ATTR_MAX + 1];
  2092. if (!nla || nla_parse_nested_deprecated(attrs, NL802154_SECLEVEL_ATTR_MAX, nla, nl802154_seclevel_policy, NULL))
  2093. return -EINVAL;
  2094. memset(sl, 0, sizeof(*sl));
  2095. if (!attrs[NL802154_SECLEVEL_ATTR_LEVELS] ||
  2096. !attrs[NL802154_SECLEVEL_ATTR_FRAME] ||
  2097. !attrs[NL802154_SECLEVEL_ATTR_DEV_OVERRIDE])
  2098. return -EINVAL;
  2099. sl->sec_levels = nla_get_u8(attrs[NL802154_SECLEVEL_ATTR_LEVELS]);
  2100. sl->frame_type = nla_get_u32(attrs[NL802154_SECLEVEL_ATTR_FRAME]);
  2101. sl->device_override = nla_get_u8(attrs[NL802154_SECLEVEL_ATTR_DEV_OVERRIDE]);
  2102. if (sl->frame_type > NL802154_FRAME_MAX ||
  2103. (sl->device_override != 0 && sl->device_override != 1))
  2104. return -EINVAL;
  2105. if (sl->frame_type == NL802154_FRAME_CMD) {
  2106. if (!attrs[NL802154_SECLEVEL_ATTR_CMD_FRAME])
  2107. return -EINVAL;
  2108. sl->cmd_frame_id = nla_get_u32(attrs[NL802154_SECLEVEL_ATTR_CMD_FRAME]);
  2109. if (sl->cmd_frame_id > NL802154_CMD_FRAME_MAX)
  2110. return -EINVAL;
  2111. }
  2112. return 0;
  2113. }
  2114. static int nl802154_add_llsec_seclevel(struct sk_buff *skb,
  2115. struct genl_info *info)
  2116. {
  2117. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  2118. struct net_device *dev = info->user_ptr[1];
  2119. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  2120. struct ieee802154_llsec_seclevel sl;
  2121. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  2122. return -EOPNOTSUPP;
  2123. if (llsec_parse_seclevel(info->attrs[NL802154_ATTR_SEC_LEVEL],
  2124. &sl) < 0)
  2125. return -EINVAL;
  2126. return rdev_add_seclevel(rdev, wpan_dev, &sl);
  2127. }
  2128. static int nl802154_del_llsec_seclevel(struct sk_buff *skb,
  2129. struct genl_info *info)
  2130. {
  2131. struct cfg802154_registered_device *rdev = info->user_ptr[0];
  2132. struct net_device *dev = info->user_ptr[1];
  2133. struct wpan_dev *wpan_dev = dev->ieee802154_ptr;
  2134. struct ieee802154_llsec_seclevel sl;
  2135. if (wpan_dev->iftype == NL802154_IFTYPE_MONITOR)
  2136. return -EOPNOTSUPP;
  2137. if (llsec_parse_seclevel(info->attrs[NL802154_ATTR_SEC_LEVEL],
  2138. &sl) < 0)
  2139. return -EINVAL;
  2140. return rdev_del_seclevel(rdev, wpan_dev, &sl);
  2141. }
  2142. #endif /* CONFIG_IEEE802154_NL802154_EXPERIMENTAL */
  2143. #define NL802154_FLAG_NEED_WPAN_PHY 0x01
  2144. #define NL802154_FLAG_NEED_NETDEV 0x02
  2145. #define NL802154_FLAG_NEED_RTNL 0x04
  2146. #define NL802154_FLAG_CHECK_NETDEV_UP 0x08
  2147. #define NL802154_FLAG_NEED_WPAN_DEV 0x10
  2148. static int nl802154_pre_doit(const struct genl_split_ops *ops,
  2149. struct sk_buff *skb,
  2150. struct genl_info *info)
  2151. {
  2152. struct cfg802154_registered_device *rdev;
  2153. struct wpan_dev *wpan_dev;
  2154. struct net_device *dev;
  2155. bool rtnl = ops->internal_flags & NL802154_FLAG_NEED_RTNL;
  2156. if (rtnl)
  2157. rtnl_lock();
  2158. if (ops->internal_flags & NL802154_FLAG_NEED_WPAN_PHY) {
  2159. rdev = cfg802154_get_dev_from_info(genl_info_net(info), info);
  2160. if (IS_ERR(rdev)) {
  2161. if (rtnl)
  2162. rtnl_unlock();
  2163. return PTR_ERR(rdev);
  2164. }
  2165. info->user_ptr[0] = rdev;
  2166. } else if (ops->internal_flags & NL802154_FLAG_NEED_NETDEV ||
  2167. ops->internal_flags & NL802154_FLAG_NEED_WPAN_DEV) {
  2168. ASSERT_RTNL();
  2169. wpan_dev = __cfg802154_wpan_dev_from_attrs(genl_info_net(info),
  2170. info->attrs);
  2171. if (IS_ERR(wpan_dev)) {
  2172. if (rtnl)
  2173. rtnl_unlock();
  2174. return PTR_ERR(wpan_dev);
  2175. }
  2176. dev = wpan_dev->netdev;
  2177. rdev = wpan_phy_to_rdev(wpan_dev->wpan_phy);
  2178. if (ops->internal_flags & NL802154_FLAG_NEED_NETDEV) {
  2179. if (!dev) {
  2180. if (rtnl)
  2181. rtnl_unlock();
  2182. return -EINVAL;
  2183. }
  2184. info->user_ptr[1] = dev;
  2185. } else {
  2186. info->user_ptr[1] = wpan_dev;
  2187. }
  2188. if (dev) {
  2189. if (ops->internal_flags & NL802154_FLAG_CHECK_NETDEV_UP &&
  2190. !netif_running(dev)) {
  2191. if (rtnl)
  2192. rtnl_unlock();
  2193. return -ENETDOWN;
  2194. }
  2195. dev_hold(dev);
  2196. }
  2197. info->user_ptr[0] = rdev;
  2198. }
  2199. return 0;
  2200. }
  2201. static void nl802154_post_doit(const struct genl_split_ops *ops,
  2202. struct sk_buff *skb,
  2203. struct genl_info *info)
  2204. {
  2205. if (info->user_ptr[1]) {
  2206. if (ops->internal_flags & NL802154_FLAG_NEED_WPAN_DEV) {
  2207. struct wpan_dev *wpan_dev = info->user_ptr[1];
  2208. dev_put(wpan_dev->netdev);
  2209. } else {
  2210. dev_put(info->user_ptr[1]);
  2211. }
  2212. }
  2213. if (ops->internal_flags & NL802154_FLAG_NEED_RTNL)
  2214. rtnl_unlock();
  2215. }
  2216. static const struct genl_ops nl802154_ops[] = {
  2217. {
  2218. .cmd = NL802154_CMD_GET_WPAN_PHY,
  2219. .validate = GENL_DONT_VALIDATE_STRICT |
  2220. GENL_DONT_VALIDATE_DUMP_STRICT,
  2221. .doit = nl802154_get_wpan_phy,
  2222. .dumpit = nl802154_dump_wpan_phy,
  2223. .done = nl802154_dump_wpan_phy_done,
  2224. /* can be retrieved by unprivileged users */
  2225. .internal_flags = NL802154_FLAG_NEED_WPAN_PHY |
  2226. NL802154_FLAG_NEED_RTNL,
  2227. },
  2228. {
  2229. .cmd = NL802154_CMD_GET_INTERFACE,
  2230. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2231. .doit = nl802154_get_interface,
  2232. .dumpit = nl802154_dump_interface,
  2233. /* can be retrieved by unprivileged users */
  2234. .internal_flags = NL802154_FLAG_NEED_WPAN_DEV |
  2235. NL802154_FLAG_NEED_RTNL,
  2236. },
  2237. {
  2238. .cmd = NL802154_CMD_NEW_INTERFACE,
  2239. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2240. .doit = nl802154_new_interface,
  2241. .flags = GENL_ADMIN_PERM,
  2242. .internal_flags = NL802154_FLAG_NEED_WPAN_PHY |
  2243. NL802154_FLAG_NEED_RTNL,
  2244. },
  2245. {
  2246. .cmd = NL802154_CMD_DEL_INTERFACE,
  2247. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2248. .doit = nl802154_del_interface,
  2249. .flags = GENL_ADMIN_PERM,
  2250. .internal_flags = NL802154_FLAG_NEED_WPAN_DEV |
  2251. NL802154_FLAG_NEED_RTNL,
  2252. },
  2253. {
  2254. .cmd = NL802154_CMD_SET_CHANNEL,
  2255. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2256. .doit = nl802154_set_channel,
  2257. .flags = GENL_ADMIN_PERM,
  2258. .internal_flags = NL802154_FLAG_NEED_WPAN_PHY |
  2259. NL802154_FLAG_NEED_RTNL,
  2260. },
  2261. {
  2262. .cmd = NL802154_CMD_SET_CCA_MODE,
  2263. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2264. .doit = nl802154_set_cca_mode,
  2265. .flags = GENL_ADMIN_PERM,
  2266. .internal_flags = NL802154_FLAG_NEED_WPAN_PHY |
  2267. NL802154_FLAG_NEED_RTNL,
  2268. },
  2269. {
  2270. .cmd = NL802154_CMD_SET_CCA_ED_LEVEL,
  2271. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2272. .doit = nl802154_set_cca_ed_level,
  2273. .flags = GENL_ADMIN_PERM,
  2274. .internal_flags = NL802154_FLAG_NEED_WPAN_PHY |
  2275. NL802154_FLAG_NEED_RTNL,
  2276. },
  2277. {
  2278. .cmd = NL802154_CMD_SET_TX_POWER,
  2279. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2280. .doit = nl802154_set_tx_power,
  2281. .flags = GENL_ADMIN_PERM,
  2282. .internal_flags = NL802154_FLAG_NEED_WPAN_PHY |
  2283. NL802154_FLAG_NEED_RTNL,
  2284. },
  2285. {
  2286. .cmd = NL802154_CMD_SET_WPAN_PHY_NETNS,
  2287. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2288. .doit = nl802154_wpan_phy_netns,
  2289. .flags = GENL_ADMIN_PERM,
  2290. .internal_flags = NL802154_FLAG_NEED_WPAN_PHY |
  2291. NL802154_FLAG_NEED_RTNL,
  2292. },
  2293. {
  2294. .cmd = NL802154_CMD_SET_PAN_ID,
  2295. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2296. .doit = nl802154_set_pan_id,
  2297. .flags = GENL_ADMIN_PERM,
  2298. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2299. NL802154_FLAG_NEED_RTNL,
  2300. },
  2301. {
  2302. .cmd = NL802154_CMD_SET_SHORT_ADDR,
  2303. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2304. .doit = nl802154_set_short_addr,
  2305. .flags = GENL_ADMIN_PERM,
  2306. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2307. NL802154_FLAG_NEED_RTNL,
  2308. },
  2309. {
  2310. .cmd = NL802154_CMD_SET_BACKOFF_EXPONENT,
  2311. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2312. .doit = nl802154_set_backoff_exponent,
  2313. .flags = GENL_ADMIN_PERM,
  2314. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2315. NL802154_FLAG_NEED_RTNL,
  2316. },
  2317. {
  2318. .cmd = NL802154_CMD_SET_MAX_CSMA_BACKOFFS,
  2319. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2320. .doit = nl802154_set_max_csma_backoffs,
  2321. .flags = GENL_ADMIN_PERM,
  2322. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2323. NL802154_FLAG_NEED_RTNL,
  2324. },
  2325. {
  2326. .cmd = NL802154_CMD_SET_MAX_FRAME_RETRIES,
  2327. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2328. .doit = nl802154_set_max_frame_retries,
  2329. .flags = GENL_ADMIN_PERM,
  2330. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2331. NL802154_FLAG_NEED_RTNL,
  2332. },
  2333. {
  2334. .cmd = NL802154_CMD_SET_LBT_MODE,
  2335. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2336. .doit = nl802154_set_lbt_mode,
  2337. .flags = GENL_ADMIN_PERM,
  2338. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2339. NL802154_FLAG_NEED_RTNL,
  2340. },
  2341. {
  2342. .cmd = NL802154_CMD_SET_ACKREQ_DEFAULT,
  2343. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2344. .doit = nl802154_set_ackreq_default,
  2345. .flags = GENL_ADMIN_PERM,
  2346. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2347. NL802154_FLAG_NEED_RTNL,
  2348. },
  2349. {
  2350. .cmd = NL802154_CMD_TRIGGER_SCAN,
  2351. .doit = nl802154_trigger_scan,
  2352. .flags = GENL_ADMIN_PERM,
  2353. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2354. NL802154_FLAG_CHECK_NETDEV_UP |
  2355. NL802154_FLAG_NEED_RTNL,
  2356. },
  2357. {
  2358. .cmd = NL802154_CMD_ABORT_SCAN,
  2359. .doit = nl802154_abort_scan,
  2360. .flags = GENL_ADMIN_PERM,
  2361. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2362. NL802154_FLAG_CHECK_NETDEV_UP |
  2363. NL802154_FLAG_NEED_RTNL,
  2364. },
  2365. {
  2366. .cmd = NL802154_CMD_SEND_BEACONS,
  2367. .doit = nl802154_send_beacons,
  2368. .flags = GENL_ADMIN_PERM,
  2369. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2370. NL802154_FLAG_CHECK_NETDEV_UP |
  2371. NL802154_FLAG_NEED_RTNL,
  2372. },
  2373. {
  2374. .cmd = NL802154_CMD_STOP_BEACONS,
  2375. .doit = nl802154_stop_beacons,
  2376. .flags = GENL_ADMIN_PERM,
  2377. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2378. NL802154_FLAG_CHECK_NETDEV_UP |
  2379. NL802154_FLAG_NEED_RTNL,
  2380. },
  2381. {
  2382. .cmd = NL802154_CMD_ASSOCIATE,
  2383. .doit = nl802154_associate,
  2384. .flags = GENL_ADMIN_PERM,
  2385. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2386. NL802154_FLAG_CHECK_NETDEV_UP |
  2387. NL802154_FLAG_NEED_RTNL,
  2388. },
  2389. {
  2390. .cmd = NL802154_CMD_DISASSOCIATE,
  2391. .doit = nl802154_disassociate,
  2392. .flags = GENL_ADMIN_PERM,
  2393. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2394. NL802154_FLAG_CHECK_NETDEV_UP |
  2395. NL802154_FLAG_NEED_RTNL,
  2396. },
  2397. {
  2398. .cmd = NL802154_CMD_SET_MAX_ASSOCIATIONS,
  2399. .doit = nl802154_set_max_associations,
  2400. .flags = GENL_ADMIN_PERM,
  2401. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2402. NL802154_FLAG_NEED_RTNL,
  2403. },
  2404. {
  2405. .cmd = NL802154_CMD_LIST_ASSOCIATIONS,
  2406. .dumpit = nl802154_list_associations,
  2407. /* can be retrieved by unprivileged users */
  2408. },
  2409. #ifdef CONFIG_IEEE802154_NL802154_EXPERIMENTAL
  2410. {
  2411. .cmd = NL802154_CMD_SET_SEC_PARAMS,
  2412. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2413. .doit = nl802154_set_llsec_params,
  2414. .flags = GENL_ADMIN_PERM,
  2415. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2416. NL802154_FLAG_NEED_RTNL,
  2417. },
  2418. {
  2419. .cmd = NL802154_CMD_GET_SEC_KEY,
  2420. .validate = GENL_DONT_VALIDATE_STRICT |
  2421. GENL_DONT_VALIDATE_DUMP_STRICT,
  2422. /* TODO .doit by matching key id? */
  2423. .dumpit = nl802154_dump_llsec_key,
  2424. .flags = GENL_ADMIN_PERM,
  2425. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2426. NL802154_FLAG_NEED_RTNL,
  2427. },
  2428. {
  2429. .cmd = NL802154_CMD_NEW_SEC_KEY,
  2430. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2431. .doit = nl802154_add_llsec_key,
  2432. .flags = GENL_ADMIN_PERM,
  2433. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2434. NL802154_FLAG_NEED_RTNL,
  2435. },
  2436. {
  2437. .cmd = NL802154_CMD_DEL_SEC_KEY,
  2438. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2439. .doit = nl802154_del_llsec_key,
  2440. .flags = GENL_ADMIN_PERM,
  2441. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2442. NL802154_FLAG_NEED_RTNL,
  2443. },
  2444. /* TODO unique identifier must short+pan OR extended_addr */
  2445. {
  2446. .cmd = NL802154_CMD_GET_SEC_DEV,
  2447. .validate = GENL_DONT_VALIDATE_STRICT |
  2448. GENL_DONT_VALIDATE_DUMP_STRICT,
  2449. /* TODO .doit by matching extended_addr? */
  2450. .dumpit = nl802154_dump_llsec_dev,
  2451. .flags = GENL_ADMIN_PERM,
  2452. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2453. NL802154_FLAG_NEED_RTNL,
  2454. },
  2455. {
  2456. .cmd = NL802154_CMD_NEW_SEC_DEV,
  2457. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2458. .doit = nl802154_add_llsec_dev,
  2459. .flags = GENL_ADMIN_PERM,
  2460. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2461. NL802154_FLAG_NEED_RTNL,
  2462. },
  2463. {
  2464. .cmd = NL802154_CMD_DEL_SEC_DEV,
  2465. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2466. .doit = nl802154_del_llsec_dev,
  2467. .flags = GENL_ADMIN_PERM,
  2468. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2469. NL802154_FLAG_NEED_RTNL,
  2470. },
  2471. /* TODO remove complete devkey, put it as nested? */
  2472. {
  2473. .cmd = NL802154_CMD_GET_SEC_DEVKEY,
  2474. .validate = GENL_DONT_VALIDATE_STRICT |
  2475. GENL_DONT_VALIDATE_DUMP_STRICT,
  2476. /* TODO doit by matching ??? */
  2477. .dumpit = nl802154_dump_llsec_devkey,
  2478. .flags = GENL_ADMIN_PERM,
  2479. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2480. NL802154_FLAG_NEED_RTNL,
  2481. },
  2482. {
  2483. .cmd = NL802154_CMD_NEW_SEC_DEVKEY,
  2484. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2485. .doit = nl802154_add_llsec_devkey,
  2486. .flags = GENL_ADMIN_PERM,
  2487. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2488. NL802154_FLAG_NEED_RTNL,
  2489. },
  2490. {
  2491. .cmd = NL802154_CMD_DEL_SEC_DEVKEY,
  2492. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2493. .doit = nl802154_del_llsec_devkey,
  2494. .flags = GENL_ADMIN_PERM,
  2495. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2496. NL802154_FLAG_NEED_RTNL,
  2497. },
  2498. {
  2499. .cmd = NL802154_CMD_GET_SEC_LEVEL,
  2500. .validate = GENL_DONT_VALIDATE_STRICT |
  2501. GENL_DONT_VALIDATE_DUMP_STRICT,
  2502. /* TODO .doit by matching frame_type? */
  2503. .dumpit = nl802154_dump_llsec_seclevel,
  2504. .flags = GENL_ADMIN_PERM,
  2505. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2506. NL802154_FLAG_NEED_RTNL,
  2507. },
  2508. {
  2509. .cmd = NL802154_CMD_NEW_SEC_LEVEL,
  2510. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2511. .doit = nl802154_add_llsec_seclevel,
  2512. .flags = GENL_ADMIN_PERM,
  2513. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2514. NL802154_FLAG_NEED_RTNL,
  2515. },
  2516. {
  2517. .cmd = NL802154_CMD_DEL_SEC_LEVEL,
  2518. .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
  2519. /* TODO match frame_type only? */
  2520. .doit = nl802154_del_llsec_seclevel,
  2521. .flags = GENL_ADMIN_PERM,
  2522. .internal_flags = NL802154_FLAG_NEED_NETDEV |
  2523. NL802154_FLAG_NEED_RTNL,
  2524. },
  2525. #endif /* CONFIG_IEEE802154_NL802154_EXPERIMENTAL */
  2526. };
  2527. static struct genl_family nl802154_fam __ro_after_init = {
  2528. .name = NL802154_GENL_NAME, /* have users key off the name instead */
  2529. .hdrsize = 0, /* no private header */
  2530. .version = 1, /* no particular meaning now */
  2531. .maxattr = NL802154_ATTR_MAX,
  2532. .policy = nl802154_policy,
  2533. .netnsok = true,
  2534. .pre_doit = nl802154_pre_doit,
  2535. .post_doit = nl802154_post_doit,
  2536. .module = THIS_MODULE,
  2537. .ops = nl802154_ops,
  2538. .n_ops = ARRAY_SIZE(nl802154_ops),
  2539. .resv_start_op = NL802154_CMD_DEL_SEC_LEVEL + 1,
  2540. .mcgrps = nl802154_mcgrps,
  2541. .n_mcgrps = ARRAY_SIZE(nl802154_mcgrps),
  2542. };
  2543. /* initialisation/exit functions */
  2544. int __init nl802154_init(void)
  2545. {
  2546. return genl_register_family(&nl802154_fam);
  2547. }
  2548. void nl802154_exit(void)
  2549. {
  2550. genl_unregister_family(&nl802154_fam);
  2551. }