dbus_new.c 87 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378
  1. /*
  2. * WPA Supplicant / dbus-based control interface
  3. * Copyright (c) 2006, Dan Williams <dcbw@redhat.com> and Red Hat, Inc.
  4. * Copyright (c) 2009-2010, Witold Sowa <witold.sowa@gmail.com>
  5. * Copyright (c) 2009, Jouni Malinen <j@w1.fi>
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License version 2 as
  9. * published by the Free Software Foundation.
  10. *
  11. * Alternatively, this software may be distributed under the terms of BSD
  12. * license.
  13. *
  14. * See README and COPYING for more details.
  15. */
  16. #include "includes.h"
  17. #include "common.h"
  18. #include "common/ieee802_11_defs.h"
  19. #include "wps/wps.h"
  20. #include "../config.h"
  21. #include "../wpa_supplicant_i.h"
  22. #include "../bss.h"
  23. #include "../wpas_glue.h"
  24. #include "dbus_new_helpers.h"
  25. #include "dbus_dict_helpers.h"
  26. #include "dbus_new.h"
  27. #include "dbus_new_handlers.h"
  28. #include "dbus_common_i.h"
  29. #include "dbus_new_handlers_p2p.h"
  30. #include "p2p/p2p.h"
  31. /**
  32. * wpas_dbus_signal_interface - Send a interface related event signal
  33. * @wpa_s: %wpa_supplicant network interface data
  34. * @sig_name: signal name - InterfaceAdded or InterfaceRemoved
  35. * @properties: Whether to add second argument with object properties
  36. *
  37. * Notify listeners about event related with interface
  38. */
  39. static void wpas_dbus_signal_interface(struct wpa_supplicant *wpa_s,
  40. const char *sig_name, int properties)
  41. {
  42. struct wpas_dbus_priv *iface;
  43. DBusMessage *msg;
  44. DBusMessageIter iter;
  45. iface = wpa_s->global->dbus;
  46. /* Do nothing if the control interface is not turned on */
  47. if (iface == NULL)
  48. return;
  49. msg = dbus_message_new_signal(WPAS_DBUS_NEW_PATH,
  50. WPAS_DBUS_NEW_INTERFACE, sig_name);
  51. if (msg == NULL)
  52. return;
  53. dbus_message_iter_init_append(msg, &iter);
  54. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  55. &wpa_s->dbus_new_path))
  56. goto err;
  57. if (properties) {
  58. if (!wpa_dbus_get_object_properties(
  59. iface, wpa_s->dbus_new_path,
  60. WPAS_DBUS_NEW_IFACE_INTERFACE, &iter))
  61. goto err;
  62. }
  63. dbus_connection_send(iface->con, msg, NULL);
  64. dbus_message_unref(msg);
  65. return;
  66. err:
  67. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  68. dbus_message_unref(msg);
  69. }
  70. /**
  71. * wpas_dbus_signal_interface_added - Send a interface created signal
  72. * @wpa_s: %wpa_supplicant network interface data
  73. *
  74. * Notify listeners about creating new interface
  75. */
  76. static void wpas_dbus_signal_interface_added(struct wpa_supplicant *wpa_s)
  77. {
  78. wpas_dbus_signal_interface(wpa_s, "InterfaceAdded", TRUE);
  79. }
  80. /**
  81. * wpas_dbus_signal_interface_removed - Send a interface removed signal
  82. * @wpa_s: %wpa_supplicant network interface data
  83. *
  84. * Notify listeners about removing interface
  85. */
  86. static void wpas_dbus_signal_interface_removed(struct wpa_supplicant *wpa_s)
  87. {
  88. wpas_dbus_signal_interface(wpa_s, "InterfaceRemoved", FALSE);
  89. }
  90. /**
  91. * wpas_dbus_signal_scan_done - send scan done signal
  92. * @wpa_s: %wpa_supplicant network interface data
  93. * @success: indicates if scanning succeed or failed
  94. *
  95. * Notify listeners about finishing a scan
  96. */
  97. void wpas_dbus_signal_scan_done(struct wpa_supplicant *wpa_s, int success)
  98. {
  99. struct wpas_dbus_priv *iface;
  100. DBusMessage *msg;
  101. dbus_bool_t succ;
  102. iface = wpa_s->global->dbus;
  103. /* Do nothing if the control interface is not turned on */
  104. if (iface == NULL)
  105. return;
  106. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  107. WPAS_DBUS_NEW_IFACE_INTERFACE,
  108. "ScanDone");
  109. if (msg == NULL)
  110. return;
  111. succ = success ? TRUE : FALSE;
  112. if (dbus_message_append_args(msg, DBUS_TYPE_BOOLEAN, &succ,
  113. DBUS_TYPE_INVALID))
  114. dbus_connection_send(iface->con, msg, NULL);
  115. else
  116. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  117. dbus_message_unref(msg);
  118. }
  119. /**
  120. * wpas_dbus_signal_blob - Send a BSS related event signal
  121. * @wpa_s: %wpa_supplicant network interface data
  122. * @bss_obj_path: BSS object path
  123. * @sig_name: signal name - BSSAdded or BSSRemoved
  124. * @properties: Whether to add second argument with object properties
  125. *
  126. * Notify listeners about event related with BSS
  127. */
  128. static void wpas_dbus_signal_bss(struct wpa_supplicant *wpa_s,
  129. const char *bss_obj_path,
  130. const char *sig_name, int properties)
  131. {
  132. struct wpas_dbus_priv *iface;
  133. DBusMessage *msg;
  134. DBusMessageIter iter;
  135. iface = wpa_s->global->dbus;
  136. /* Do nothing if the control interface is not turned on */
  137. if (iface == NULL)
  138. return;
  139. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  140. WPAS_DBUS_NEW_IFACE_INTERFACE,
  141. sig_name);
  142. if (msg == NULL)
  143. return;
  144. dbus_message_iter_init_append(msg, &iter);
  145. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  146. &bss_obj_path))
  147. goto err;
  148. if (properties) {
  149. if (!wpa_dbus_get_object_properties(iface, bss_obj_path,
  150. WPAS_DBUS_NEW_IFACE_BSS,
  151. &iter))
  152. goto err;
  153. }
  154. dbus_connection_send(iface->con, msg, NULL);
  155. dbus_message_unref(msg);
  156. return;
  157. err:
  158. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  159. dbus_message_unref(msg);
  160. }
  161. /**
  162. * wpas_dbus_signal_bss_added - Send a BSS added signal
  163. * @wpa_s: %wpa_supplicant network interface data
  164. * @bss_obj_path: new BSS object path
  165. *
  166. * Notify listeners about adding new BSS
  167. */
  168. static void wpas_dbus_signal_bss_added(struct wpa_supplicant *wpa_s,
  169. const char *bss_obj_path)
  170. {
  171. wpas_dbus_signal_bss(wpa_s, bss_obj_path, "BSSAdded", TRUE);
  172. }
  173. /**
  174. * wpas_dbus_signal_bss_removed - Send a BSS removed signal
  175. * @wpa_s: %wpa_supplicant network interface data
  176. * @bss_obj_path: BSS object path
  177. *
  178. * Notify listeners about removing BSS
  179. */
  180. static void wpas_dbus_signal_bss_removed(struct wpa_supplicant *wpa_s,
  181. const char *bss_obj_path)
  182. {
  183. wpas_dbus_signal_bss(wpa_s, bss_obj_path, "BSSRemoved", FALSE);
  184. }
  185. /**
  186. * wpas_dbus_signal_blob - Send a blob related event signal
  187. * @wpa_s: %wpa_supplicant network interface data
  188. * @name: blob name
  189. * @sig_name: signal name - BlobAdded or BlobRemoved
  190. *
  191. * Notify listeners about event related with blob
  192. */
  193. static void wpas_dbus_signal_blob(struct wpa_supplicant *wpa_s,
  194. const char *name, const char *sig_name)
  195. {
  196. struct wpas_dbus_priv *iface;
  197. DBusMessage *msg;
  198. iface = wpa_s->global->dbus;
  199. /* Do nothing if the control interface is not turned on */
  200. if (iface == NULL)
  201. return;
  202. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  203. WPAS_DBUS_NEW_IFACE_INTERFACE,
  204. sig_name);
  205. if (msg == NULL)
  206. return;
  207. if (dbus_message_append_args(msg, DBUS_TYPE_STRING, &name,
  208. DBUS_TYPE_INVALID))
  209. dbus_connection_send(iface->con, msg, NULL);
  210. else
  211. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  212. dbus_message_unref(msg);
  213. }
  214. /**
  215. * wpas_dbus_signal_blob_added - Send a blob added signal
  216. * @wpa_s: %wpa_supplicant network interface data
  217. * @name: blob name
  218. *
  219. * Notify listeners about adding a new blob
  220. */
  221. void wpas_dbus_signal_blob_added(struct wpa_supplicant *wpa_s,
  222. const char *name)
  223. {
  224. wpas_dbus_signal_blob(wpa_s, name, "BlobAdded");
  225. }
  226. /**
  227. * wpas_dbus_signal_blob_removed - Send a blob removed signal
  228. * @wpa_s: %wpa_supplicant network interface data
  229. * @name: blob name
  230. *
  231. * Notify listeners about removing blob
  232. */
  233. void wpas_dbus_signal_blob_removed(struct wpa_supplicant *wpa_s,
  234. const char *name)
  235. {
  236. wpas_dbus_signal_blob(wpa_s, name, "BlobRemoved");
  237. }
  238. /**
  239. * wpas_dbus_signal_network - Send a network related event signal
  240. * @wpa_s: %wpa_supplicant network interface data
  241. * @id: new network id
  242. * @sig_name: signal name - NetworkAdded, NetworkRemoved or NetworkSelected
  243. * @properties: determines if add second argument with object properties
  244. *
  245. * Notify listeners about event related with configured network
  246. */
  247. static void wpas_dbus_signal_network(struct wpa_supplicant *wpa_s,
  248. int id, const char *sig_name,
  249. int properties)
  250. {
  251. struct wpas_dbus_priv *iface;
  252. DBusMessage *msg;
  253. DBusMessageIter iter;
  254. char net_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  255. iface = wpa_s->global->dbus;
  256. /* Do nothing if the control interface is not turned on */
  257. if (iface == NULL)
  258. return;
  259. os_snprintf(net_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  260. "%s/" WPAS_DBUS_NEW_NETWORKS_PART "/%u",
  261. wpa_s->dbus_new_path, id);
  262. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  263. WPAS_DBUS_NEW_IFACE_INTERFACE,
  264. sig_name);
  265. if (msg == NULL)
  266. return;
  267. dbus_message_iter_init_append(msg, &iter);
  268. path = net_obj_path;
  269. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  270. &path))
  271. goto err;
  272. if (properties) {
  273. if (!wpa_dbus_get_object_properties(
  274. iface, net_obj_path, WPAS_DBUS_NEW_IFACE_NETWORK,
  275. &iter))
  276. goto err;
  277. }
  278. dbus_connection_send(iface->con, msg, NULL);
  279. dbus_message_unref(msg);
  280. return;
  281. err:
  282. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  283. dbus_message_unref(msg);
  284. }
  285. /**
  286. * wpas_dbus_signal_network_added - Send a network added signal
  287. * @wpa_s: %wpa_supplicant network interface data
  288. * @id: new network id
  289. *
  290. * Notify listeners about adding new network
  291. */
  292. static void wpas_dbus_signal_network_added(struct wpa_supplicant *wpa_s,
  293. int id)
  294. {
  295. wpas_dbus_signal_network(wpa_s, id, "NetworkAdded", TRUE);
  296. }
  297. /**
  298. * wpas_dbus_signal_network_removed - Send a network removed signal
  299. * @wpa_s: %wpa_supplicant network interface data
  300. * @id: network id
  301. *
  302. * Notify listeners about removing a network
  303. */
  304. static void wpas_dbus_signal_network_removed(struct wpa_supplicant *wpa_s,
  305. int id)
  306. {
  307. wpas_dbus_signal_network(wpa_s, id, "NetworkRemoved", FALSE);
  308. }
  309. /**
  310. * wpas_dbus_signal_network_selected - Send a network selected signal
  311. * @wpa_s: %wpa_supplicant network interface data
  312. * @id: network id
  313. *
  314. * Notify listeners about selecting a network
  315. */
  316. void wpas_dbus_signal_network_selected(struct wpa_supplicant *wpa_s, int id)
  317. {
  318. wpas_dbus_signal_network(wpa_s, id, "NetworkSelected", FALSE);
  319. }
  320. /**
  321. * wpas_dbus_signal_network_request - Indicate that additional information
  322. * (EAP password, etc.) is required to complete the association to this SSID
  323. * @wpa_s: %wpa_supplicant network interface data
  324. * @rtype: The specific additional information required
  325. * @default_text: Optional description of required information
  326. *
  327. * Request additional information or passwords to complete an association
  328. * request.
  329. */
  330. void wpas_dbus_signal_network_request(struct wpa_supplicant *wpa_s,
  331. struct wpa_ssid *ssid,
  332. enum wpa_ctrl_req_type rtype,
  333. const char *default_txt)
  334. {
  335. struct wpas_dbus_priv *iface;
  336. DBusMessage *msg;
  337. DBusMessageIter iter;
  338. char net_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  339. const char *field, *txt = NULL, *net_ptr;
  340. iface = wpa_s->global->dbus;
  341. /* Do nothing if the control interface is not turned on */
  342. if (iface == NULL)
  343. return;
  344. field = wpa_supplicant_ctrl_req_to_string(rtype, default_txt, &txt);
  345. if (field == NULL)
  346. return;
  347. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  348. WPAS_DBUS_NEW_IFACE_INTERFACE,
  349. "NetworkRequest");
  350. if (msg == NULL)
  351. return;
  352. os_snprintf(net_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  353. "%s/" WPAS_DBUS_NEW_NETWORKS_PART "/%u",
  354. wpa_s->dbus_new_path, ssid->id);
  355. net_ptr = &net_obj_path[0];
  356. dbus_message_iter_init_append(msg, &iter);
  357. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  358. &net_ptr))
  359. goto err;
  360. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &field))
  361. goto err;
  362. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &txt))
  363. goto err;
  364. dbus_connection_send(iface->con, msg, NULL);
  365. dbus_message_unref(msg);
  366. return;
  367. err:
  368. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  369. dbus_message_unref(msg);
  370. }
  371. /**
  372. * wpas_dbus_signal_network_enabled_changed - Signals Enabled property changes
  373. * @wpa_s: %wpa_supplicant network interface data
  374. * @ssid: configured network which Enabled property has changed
  375. *
  376. * Sends PropertyChanged signals containing new value of Enabled property
  377. * for specified network
  378. */
  379. void wpas_dbus_signal_network_enabled_changed(struct wpa_supplicant *wpa_s,
  380. struct wpa_ssid *ssid)
  381. {
  382. char path[WPAS_DBUS_OBJECT_PATH_MAX];
  383. os_snprintf(path, WPAS_DBUS_OBJECT_PATH_MAX,
  384. "%s/" WPAS_DBUS_NEW_NETWORKS_PART "/%d",
  385. wpa_s->dbus_new_path, ssid->id);
  386. wpa_dbus_mark_property_changed(wpa_s->global->dbus, path,
  387. WPAS_DBUS_NEW_IFACE_NETWORK, "Enabled");
  388. }
  389. #ifdef CONFIG_WPS
  390. /**
  391. * wpas_dbus_signal_wps_event_success - Signals Success WPS event
  392. * @wpa_s: %wpa_supplicant network interface data
  393. *
  394. * Sends Event dbus signal with name "success" and empty dict as arguments
  395. */
  396. void wpas_dbus_signal_wps_event_success(struct wpa_supplicant *wpa_s)
  397. {
  398. DBusMessage *msg;
  399. DBusMessageIter iter, dict_iter;
  400. struct wpas_dbus_priv *iface;
  401. char *key = "success";
  402. iface = wpa_s->global->dbus;
  403. /* Do nothing if the control interface is not turned on */
  404. if (iface == NULL)
  405. return;
  406. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  407. WPAS_DBUS_NEW_IFACE_WPS, "Event");
  408. if (msg == NULL)
  409. return;
  410. dbus_message_iter_init_append(msg, &iter);
  411. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &key) ||
  412. !wpa_dbus_dict_open_write(&iter, &dict_iter) ||
  413. !wpa_dbus_dict_close_write(&iter, &dict_iter))
  414. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  415. else
  416. dbus_connection_send(iface->con, msg, NULL);
  417. dbus_message_unref(msg);
  418. }
  419. /**
  420. * wpas_dbus_signal_wps_event_fail - Signals Fail WPS event
  421. * @wpa_s: %wpa_supplicant network interface data
  422. *
  423. * Sends Event dbus signal with name "fail" and dictionary containing
  424. * "msg field with fail message number (int32) as arguments
  425. */
  426. void wpas_dbus_signal_wps_event_fail(struct wpa_supplicant *wpa_s,
  427. struct wps_event_fail *fail)
  428. {
  429. DBusMessage *msg;
  430. DBusMessageIter iter, dict_iter;
  431. struct wpas_dbus_priv *iface;
  432. char *key = "fail";
  433. iface = wpa_s->global->dbus;
  434. /* Do nothing if the control interface is not turned on */
  435. if (iface == NULL)
  436. return;
  437. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  438. WPAS_DBUS_NEW_IFACE_WPS, "Event");
  439. if (msg == NULL)
  440. return;
  441. dbus_message_iter_init_append(msg, &iter);
  442. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &key) ||
  443. !wpa_dbus_dict_open_write(&iter, &dict_iter) ||
  444. !wpa_dbus_dict_append_int32(&dict_iter, "msg", fail->msg) ||
  445. !wpa_dbus_dict_close_write(&iter, &dict_iter))
  446. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  447. else
  448. dbus_connection_send(iface->con, msg, NULL);
  449. dbus_message_unref(msg);
  450. }
  451. /**
  452. * wpas_dbus_signal_wps_event_m2d - Signals M2D WPS event
  453. * @wpa_s: %wpa_supplicant network interface data
  454. *
  455. * Sends Event dbus signal with name "m2d" and dictionary containing
  456. * fields of wps_event_m2d structure.
  457. */
  458. void wpas_dbus_signal_wps_event_m2d(struct wpa_supplicant *wpa_s,
  459. struct wps_event_m2d *m2d)
  460. {
  461. DBusMessage *msg;
  462. DBusMessageIter iter, dict_iter;
  463. struct wpas_dbus_priv *iface;
  464. char *key = "m2d";
  465. iface = wpa_s->global->dbus;
  466. /* Do nothing if the control interface is not turned on */
  467. if (iface == NULL)
  468. return;
  469. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  470. WPAS_DBUS_NEW_IFACE_WPS, "Event");
  471. if (msg == NULL)
  472. return;
  473. dbus_message_iter_init_append(msg, &iter);
  474. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &key) ||
  475. !wpa_dbus_dict_open_write(&iter, &dict_iter) ||
  476. !wpa_dbus_dict_append_uint16(&dict_iter, "config_methods",
  477. m2d->config_methods) ||
  478. !wpa_dbus_dict_append_byte_array(&dict_iter, "manufacturer",
  479. (const char *) m2d->manufacturer,
  480. m2d->manufacturer_len) ||
  481. !wpa_dbus_dict_append_byte_array(&dict_iter, "model_name",
  482. (const char *) m2d->model_name,
  483. m2d->model_name_len) ||
  484. !wpa_dbus_dict_append_byte_array(&dict_iter, "model_number",
  485. (const char *) m2d->model_number,
  486. m2d->model_number_len) ||
  487. !wpa_dbus_dict_append_byte_array(&dict_iter, "serial_number",
  488. (const char *)
  489. m2d->serial_number,
  490. m2d->serial_number_len) ||
  491. !wpa_dbus_dict_append_byte_array(&dict_iter, "dev_name",
  492. (const char *) m2d->dev_name,
  493. m2d->dev_name_len) ||
  494. !wpa_dbus_dict_append_byte_array(&dict_iter, "primary_dev_type",
  495. (const char *)
  496. m2d->primary_dev_type, 8) ||
  497. !wpa_dbus_dict_append_uint16(&dict_iter, "config_error",
  498. m2d->config_error) ||
  499. !wpa_dbus_dict_append_uint16(&dict_iter, "dev_password_id",
  500. m2d->dev_password_id) ||
  501. !wpa_dbus_dict_close_write(&iter, &dict_iter))
  502. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  503. else
  504. dbus_connection_send(iface->con, msg, NULL);
  505. dbus_message_unref(msg);
  506. }
  507. /**
  508. * wpas_dbus_signal_wps_cred - Signals new credentials
  509. * @wpa_s: %wpa_supplicant network interface data
  510. *
  511. * Sends signal with credentials in directory argument
  512. */
  513. void wpas_dbus_signal_wps_cred(struct wpa_supplicant *wpa_s,
  514. const struct wps_credential *cred)
  515. {
  516. DBusMessage *msg;
  517. DBusMessageIter iter, dict_iter;
  518. struct wpas_dbus_priv *iface;
  519. char *auth_type[6]; /* we have six possible authorization types */
  520. int at_num = 0;
  521. char *encr_type[4]; /* we have four possible encryption types */
  522. int et_num = 0;
  523. iface = wpa_s->global->dbus;
  524. /* Do nothing if the control interface is not turned on */
  525. if (iface == NULL)
  526. return;
  527. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  528. WPAS_DBUS_NEW_IFACE_WPS,
  529. "Credentials");
  530. if (msg == NULL)
  531. return;
  532. dbus_message_iter_init_append(msg, &iter);
  533. if (!wpa_dbus_dict_open_write(&iter, &dict_iter))
  534. goto nomem;
  535. if (cred->auth_type & WPS_AUTH_OPEN)
  536. auth_type[at_num++] = "open";
  537. if (cred->auth_type & WPS_AUTH_WPAPSK)
  538. auth_type[at_num++] = "wpa-psk";
  539. if (cred->auth_type & WPS_AUTH_SHARED)
  540. auth_type[at_num++] = "shared";
  541. if (cred->auth_type & WPS_AUTH_WPA)
  542. auth_type[at_num++] = "wpa-eap";
  543. if (cred->auth_type & WPS_AUTH_WPA2)
  544. auth_type[at_num++] = "wpa2-eap";
  545. if (cred->auth_type & WPS_AUTH_WPA2PSK)
  546. auth_type[at_num++] =
  547. "wpa2-psk";
  548. if (cred->encr_type & WPS_ENCR_NONE)
  549. encr_type[et_num++] = "none";
  550. if (cred->encr_type & WPS_ENCR_WEP)
  551. encr_type[et_num++] = "wep";
  552. if (cred->encr_type & WPS_ENCR_TKIP)
  553. encr_type[et_num++] = "tkip";
  554. if (cred->encr_type & WPS_ENCR_AES)
  555. encr_type[et_num++] = "aes";
  556. if (wpa_s->current_ssid) {
  557. if (!wpa_dbus_dict_append_byte_array(
  558. &dict_iter, "BSSID",
  559. (const char *) wpa_s->current_ssid->bssid,
  560. ETH_ALEN))
  561. goto nomem;
  562. }
  563. if (!wpa_dbus_dict_append_byte_array(&dict_iter, "SSID",
  564. (const char *) cred->ssid,
  565. cred->ssid_len) ||
  566. !wpa_dbus_dict_append_string_array(&dict_iter, "AuthType",
  567. (const char **) auth_type,
  568. at_num) ||
  569. !wpa_dbus_dict_append_string_array(&dict_iter, "EncrType",
  570. (const char **) encr_type,
  571. et_num) ||
  572. !wpa_dbus_dict_append_byte_array(&dict_iter, "Key",
  573. (const char *) cred->key,
  574. cred->key_len) ||
  575. !wpa_dbus_dict_append_uint32(&dict_iter, "KeyIndex",
  576. cred->key_idx) ||
  577. !wpa_dbus_dict_close_write(&iter, &dict_iter))
  578. goto nomem;
  579. dbus_connection_send(iface->con, msg, NULL);
  580. nomem:
  581. dbus_message_unref(msg);
  582. }
  583. #endif /* CONFIG_WPS */
  584. void wpas_dbus_signal_certification(struct wpa_supplicant *wpa_s,
  585. int depth, const char *subject,
  586. const char *cert_hash,
  587. const struct wpabuf *cert)
  588. {
  589. struct wpas_dbus_priv *iface;
  590. DBusMessage *msg;
  591. DBusMessageIter iter, dict_iter;
  592. iface = wpa_s->global->dbus;
  593. /* Do nothing if the control interface is not turned on */
  594. if (iface == NULL)
  595. return;
  596. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  597. WPAS_DBUS_NEW_IFACE_INTERFACE,
  598. "Certification");
  599. if (msg == NULL)
  600. return;
  601. dbus_message_iter_init_append(msg, &iter);
  602. if (!wpa_dbus_dict_open_write(&iter, &dict_iter))
  603. goto nomem;
  604. if (!wpa_dbus_dict_append_uint32(&dict_iter, "depth", depth) ||
  605. !wpa_dbus_dict_append_string(&dict_iter, "subject", subject))
  606. goto nomem;
  607. if (cert_hash &&
  608. !wpa_dbus_dict_append_string(&dict_iter, "cert_hash", cert_hash))
  609. goto nomem;
  610. if (cert &&
  611. !wpa_dbus_dict_append_byte_array(&dict_iter, "cert",
  612. wpabuf_head(cert),
  613. wpabuf_len(cert)))
  614. goto nomem;
  615. if (!wpa_dbus_dict_close_write(&iter, &dict_iter))
  616. goto nomem;
  617. dbus_connection_send(iface->con, msg, NULL);
  618. nomem:
  619. dbus_message_unref(msg);
  620. }
  621. #ifdef CONFIG_P2P
  622. /**
  623. * wpas_dbus_signal_p2p_group_removed - Signals P2P group was removed
  624. * @wpa_s: %wpa_supplicant network interface data
  625. * @role: role of this device (client or GO)
  626. * Sends signal with i/f name and role as string arguments
  627. */
  628. void wpas_dbus_signal_p2p_group_removed(struct wpa_supplicant *wpa_s,
  629. const char *role)
  630. {
  631. DBusMessage *msg;
  632. DBusMessageIter iter;
  633. struct wpas_dbus_priv *iface = wpa_s->global->dbus;
  634. char *ifname = wpa_s->ifname;
  635. /* Do nothing if the control interface is not turned on */
  636. if (iface == NULL)
  637. return;
  638. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  639. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  640. "GroupFinished");
  641. if (msg == NULL)
  642. return;
  643. dbus_message_iter_init_append(msg, &iter);
  644. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &ifname)) {
  645. wpa_printf(MSG_ERROR, "dbus: Failed to construct GroupFinished"
  646. "signal -not enough memory for ifname ");
  647. goto err;
  648. }
  649. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &role))
  650. wpa_printf(MSG_ERROR, "dbus: Failed to construct GroupFinished"
  651. "signal -not enough memory for role ");
  652. else
  653. dbus_connection_send(iface->con, msg, NULL);
  654. err:
  655. dbus_message_unref(msg);
  656. }
  657. /**
  658. * wpas_dbus_signal_p2p_provision_discovery - Signals various PD events
  659. *
  660. * @dev_addr - who sent the request or responded to our request.
  661. * @request - Will be 1 if request, 0 for response.
  662. * @status - valid only in case of response
  663. * @config_methods - wps config methods
  664. * @generated_pin - pin to be displayed in case of WPS_CONFIG_DISPLAY method
  665. *
  666. * Sends following provision discovery related events:
  667. * ProvisionDiscoveryRequestDisplayPin
  668. * ProvisionDiscoveryResponseDisplayPin
  669. * ProvisionDiscoveryRequestEnterPin
  670. * ProvisionDiscoveryResponseEnterPin
  671. * ProvisionDiscoveryPBCRequest
  672. * ProvisionDiscoveryPBCResponse
  673. *
  674. * TODO::
  675. * ProvisionDiscoveryFailure (timeout case)
  676. */
  677. void wpas_dbus_signal_p2p_provision_discovery(struct wpa_supplicant *wpa_s,
  678. const u8 *dev_addr, int request,
  679. enum p2p_prov_disc_status status,
  680. u16 config_methods,
  681. unsigned int generated_pin)
  682. {
  683. DBusMessage *msg;
  684. DBusMessageIter iter;
  685. struct wpas_dbus_priv *iface;
  686. char *_signal;
  687. int add_pin = 0;
  688. char peer_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  689. int error_ret = 1;
  690. char pin[9], *p_pin = NULL;
  691. iface = wpa_s->global->dbus;
  692. /* Do nothing if the control interface is not turned on */
  693. if (iface == NULL)
  694. return;
  695. if (request || !status) {
  696. if (config_methods & WPS_CONFIG_DISPLAY)
  697. _signal = request ?
  698. "ProvisionDiscoveryRequestDisplayPin" :
  699. "ProvisionDiscoveryResponseEnterPin";
  700. else if (config_methods & WPS_CONFIG_KEYPAD)
  701. _signal = request ?
  702. "ProvisionDiscoveryRequestEnterPin" :
  703. "ProvisionDiscoveryResponseDisplayPin";
  704. else if (config_methods & WPS_CONFIG_PUSHBUTTON)
  705. _signal = request ? "ProvisionDiscoveryPBCRequest" :
  706. "ProvisionDiscoveryPBCResponse";
  707. else
  708. return; /* Unknown or un-supported method */
  709. } else if (!request && status)
  710. /* Explicit check for failure response */
  711. _signal = "ProvisionDiscoveryFailure";
  712. add_pin = ((request && (config_methods & WPS_CONFIG_DISPLAY)) ||
  713. (!request && !status &&
  714. (config_methods & WPS_CONFIG_KEYPAD)));
  715. if (add_pin) {
  716. os_snprintf(pin, sizeof(pin), "%08d", generated_pin);
  717. p_pin = pin;
  718. }
  719. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  720. WPAS_DBUS_NEW_IFACE_P2PDEVICE, _signal);
  721. if (msg == NULL)
  722. return;
  723. /* Check if this is a known peer */
  724. if (p2p_get_peer_info(wpa_s->global->p2p, dev_addr, 0, NULL, 0) < 0)
  725. goto error;
  726. os_snprintf(peer_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  727. "%s/" WPAS_DBUS_NEW_P2P_PEERS_PART "/"
  728. COMPACT_MACSTR,
  729. wpa_s->dbus_new_path, MAC2STR(dev_addr));
  730. path = peer_obj_path;
  731. dbus_message_iter_init_append(msg, &iter);
  732. if (!dbus_message_iter_append_basic(&iter,
  733. DBUS_TYPE_OBJECT_PATH,
  734. &path))
  735. goto error;
  736. if (!request && status)
  737. /* Attach status to ProvisionDiscoveryFailure */
  738. error_ret = !dbus_message_iter_append_basic(&iter,
  739. DBUS_TYPE_INT32,
  740. &status);
  741. else
  742. error_ret = (add_pin &&
  743. !dbus_message_iter_append_basic(&iter,
  744. DBUS_TYPE_STRING,
  745. &p_pin));
  746. error:
  747. if (!error_ret)
  748. dbus_connection_send(iface->con, msg, NULL);
  749. else
  750. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  751. dbus_message_unref(msg);
  752. }
  753. void wpas_dbus_signal_p2p_go_neg_req(struct wpa_supplicant *wpa_s,
  754. const u8 *src, u16 dev_passwd_id)
  755. {
  756. DBusMessage *msg;
  757. DBusMessageIter iter;
  758. struct wpas_dbus_priv *iface;
  759. char peer_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  760. iface = wpa_s->global->dbus;
  761. /* Do nothing if the control interface is not turned on */
  762. if (iface == NULL)
  763. return;
  764. os_snprintf(peer_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  765. "%s/" WPAS_DBUS_NEW_P2P_PEERS_PART "/" COMPACT_MACSTR,
  766. wpa_s->dbus_new_path, MAC2STR(src));
  767. path = peer_obj_path;
  768. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  769. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  770. "GONegotiationRequest");
  771. if (msg == NULL)
  772. return;
  773. dbus_message_iter_init_append(msg, &iter);
  774. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  775. &path) ||
  776. !dbus_message_iter_append_basic(&iter, DBUS_TYPE_UINT16,
  777. &dev_passwd_id))
  778. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  779. else
  780. dbus_connection_send(iface->con, msg, NULL);
  781. dbus_message_unref(msg);
  782. }
  783. static int wpas_dbus_get_group_obj_path(struct wpa_supplicant *wpa_s,
  784. const struct wpa_ssid *ssid,
  785. char *group_obj_path)
  786. {
  787. char group_name[3];
  788. if (os_memcmp(ssid->ssid, P2P_WILDCARD_SSID, P2P_WILDCARD_SSID_LEN))
  789. return -1;
  790. memcpy(group_name, ssid->ssid + P2P_WILDCARD_SSID_LEN, 2);
  791. group_name[2] = '\0';
  792. os_snprintf(group_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  793. "%s/" WPAS_DBUS_NEW_P2P_GROUPS_PART "/%s",
  794. wpa_s->dbus_new_path, group_name);
  795. return 0;
  796. }
  797. /**
  798. * wpas_dbus_signal_p2p_group_started - Signals P2P group has
  799. * started. Emitted when a group is successfully started
  800. * irrespective of the role (client/GO) of the current device
  801. *
  802. * @wpa_s: %wpa_supplicant network interface data
  803. * @ssid: SSID object
  804. * @client: this device is P2P client
  805. * @network_id: network id of the group started, use instead of ssid->id
  806. * to account for persistent groups
  807. */
  808. void wpas_dbus_signal_p2p_group_started(struct wpa_supplicant *wpa_s,
  809. const struct wpa_ssid *ssid,
  810. int client, int network_id)
  811. {
  812. DBusMessage *msg;
  813. DBusMessageIter iter, dict_iter;
  814. struct wpas_dbus_priv *iface;
  815. char net_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  816. char group_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  817. iface = wpa_s->parent->global->dbus;
  818. /* Do nothing if the control interface is not turned on */
  819. if (iface == NULL)
  820. return;
  821. if (wpas_dbus_get_group_obj_path(wpa_s, ssid, group_obj_path) < 0)
  822. return;
  823. /* New interface has been created for this group */
  824. msg = dbus_message_new_signal(wpa_s->parent->dbus_new_path,
  825. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  826. "GroupStarted");
  827. if (msg == NULL)
  828. return;
  829. dbus_message_iter_init_append(msg, &iter);
  830. if (!wpa_dbus_dict_open_write(&iter, &dict_iter))
  831. goto nomem;
  832. /*
  833. * In case the device supports creating a separate interface the
  834. * DBus client will need to know the object path for the interface
  835. * object this group was created on, so include it here.
  836. */
  837. if (!wpa_dbus_dict_append_object_path(&dict_iter,
  838. "interface_object",
  839. wpa_s->dbus_new_path))
  840. goto nomem;
  841. if (!wpa_dbus_dict_append_string(&dict_iter, "role",
  842. client ? "client" : "GO"))
  843. goto nomem;
  844. os_snprintf(net_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  845. "%s/" WPAS_DBUS_NEW_NETWORKS_PART "/%u",
  846. wpa_s->parent->dbus_new_path, network_id);
  847. if (!wpa_dbus_dict_append_object_path(&dict_iter, "group_object",
  848. group_obj_path) ||
  849. !wpa_dbus_dict_append_object_path(&dict_iter, "network_object",
  850. net_obj_path) ||
  851. !wpa_dbus_dict_close_write(&iter, &dict_iter))
  852. goto nomem;
  853. dbus_connection_send(iface->con, msg, NULL);
  854. nomem:
  855. dbus_message_unref(msg);
  856. }
  857. /**
  858. *
  859. * Method to emit GONeogtiation Success or Failure signals based
  860. * on status.
  861. * @status: Status of the GO neg request. 0 for success, other for errors.
  862. */
  863. void wpas_dbus_signal_p2p_go_neg_resp(struct wpa_supplicant *wpa_s, int status)
  864. {
  865. DBusMessage *msg;
  866. DBusMessageIter iter;
  867. struct wpas_dbus_priv *iface;
  868. iface = wpa_s->global->dbus;
  869. /* Do nothing if the control interface is not turned on */
  870. if (iface == NULL)
  871. return;
  872. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  873. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  874. status ? "GONegotiationFailure" :
  875. "GONegotiationSuccess");
  876. if (msg == NULL)
  877. return;
  878. if (status) {
  879. dbus_message_iter_init_append(msg, &iter);
  880. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_INT32,
  881. &status)) {
  882. wpa_printf(MSG_ERROR,
  883. "dbus: Failed to construct signal");
  884. goto err;
  885. }
  886. }
  887. dbus_connection_send(iface->con, msg, NULL);
  888. err:
  889. dbus_message_unref(msg);
  890. }
  891. /**
  892. *
  893. * Method to emit Invitation Result signal based on status and
  894. * bssid
  895. * @status: Status of the Invite request. 0 for success, other
  896. * for errors
  897. * @bssid : Basic Service Set Identifier
  898. */
  899. void wpas_dbus_signal_p2p_invitation_result(struct wpa_supplicant *wpa_s,
  900. int status, const u8 *bssid)
  901. {
  902. DBusMessage *msg;
  903. DBusMessageIter iter, dict_iter;
  904. struct wpas_dbus_priv *iface;
  905. wpa_printf(MSG_INFO, "%s\n", __func__);
  906. iface = wpa_s->global->dbus;
  907. /* Do nothing if the control interface is not turned on */
  908. if (iface == NULL)
  909. return;
  910. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  911. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  912. "InvitationResult");
  913. if (msg == NULL)
  914. return;
  915. dbus_message_iter_init_append(msg, &iter);
  916. if (!wpa_dbus_dict_open_write(&iter, &dict_iter))
  917. goto nomem;
  918. if (!wpa_dbus_dict_append_int32(&dict_iter, "status", status))
  919. goto nomem;
  920. if (bssid) {
  921. if (!wpa_dbus_dict_append_byte_array(&dict_iter, "BSSID",
  922. (const char *) bssid,
  923. ETH_ALEN))
  924. goto nomem;
  925. }
  926. if (!wpa_dbus_dict_close_write(&iter, &dict_iter))
  927. goto nomem;
  928. dbus_connection_send(iface->con, msg, NULL);
  929. nomem:
  930. dbus_message_unref(msg);
  931. }
  932. /**
  933. *
  934. * Method to emit a signal for a peer joining the group.
  935. * The signal will carry path to the group member object
  936. * constructed using p2p i/f addr used for connecting.
  937. *
  938. * @wpa_s: %wpa_supplicant network interface data
  939. * @member_addr: addr (p2p i/f) of the peer joining the group
  940. */
  941. void wpas_dbus_signal_p2p_peer_joined(struct wpa_supplicant *wpa_s,
  942. const u8 *member)
  943. {
  944. struct wpas_dbus_priv *iface;
  945. DBusMessage *msg;
  946. DBusMessageIter iter;
  947. char groupmember_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  948. iface = wpa_s->global->dbus;
  949. /* Do nothing if the control interface is not turned on */
  950. if (iface == NULL)
  951. return;
  952. if (!wpa_s->dbus_groupobj_path)
  953. return;
  954. os_snprintf(groupmember_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  955. "%s/" WPAS_DBUS_NEW_P2P_GROUPMEMBERS_PART "/"
  956. COMPACT_MACSTR,
  957. wpa_s->dbus_groupobj_path, MAC2STR(member));
  958. msg = dbus_message_new_signal(wpa_s->dbus_groupobj_path,
  959. WPAS_DBUS_NEW_IFACE_P2P_GROUP,
  960. "PeerJoined");
  961. if (msg == NULL)
  962. return;
  963. dbus_message_iter_init_append(msg, &iter);
  964. path = groupmember_obj_path;
  965. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  966. &path))
  967. goto err;
  968. dbus_connection_send(iface->con, msg, NULL);
  969. dbus_message_unref(msg);
  970. return;
  971. err:
  972. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  973. dbus_message_unref(msg);
  974. }
  975. /**
  976. *
  977. * Method to emit a signal for a peer disconnecting the group.
  978. * The signal will carry path to the group member object
  979. * constructed using p2p i/f addr used for connecting.
  980. *
  981. * @wpa_s: %wpa_supplicant network interface data
  982. * @member_addr: addr (p2p i/f) of the peer joining the group
  983. */
  984. void wpas_dbus_signal_p2p_peer_disconnected(struct wpa_supplicant *wpa_s,
  985. const u8 *member)
  986. {
  987. struct wpas_dbus_priv *iface;
  988. DBusMessage *msg;
  989. DBusMessageIter iter;
  990. char groupmember_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  991. iface = wpa_s->global->dbus;
  992. /* Do nothing if the control interface is not turned on */
  993. if (iface == NULL)
  994. return;
  995. if (!wpa_s->dbus_groupobj_path)
  996. return;
  997. os_snprintf(groupmember_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  998. "%s/" WPAS_DBUS_NEW_P2P_GROUPMEMBERS_PART "/"
  999. COMPACT_MACSTR,
  1000. wpa_s->dbus_groupobj_path, MAC2STR(member));
  1001. msg = dbus_message_new_signal(wpa_s->dbus_groupobj_path,
  1002. WPAS_DBUS_NEW_IFACE_P2P_GROUP,
  1003. "PeerDisconnected");
  1004. if (msg == NULL)
  1005. return;
  1006. dbus_message_iter_init_append(msg, &iter);
  1007. path = groupmember_obj_path;
  1008. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  1009. &path))
  1010. goto err;
  1011. dbus_connection_send(iface->con, msg, NULL);
  1012. dbus_message_unref(msg);
  1013. return;
  1014. err:
  1015. wpa_printf(MSG_ERROR, "dbus: Failed to construct PeerDisconnected "
  1016. "signal");
  1017. dbus_message_unref(msg);
  1018. }
  1019. /**
  1020. *
  1021. * Method to emit a signal for a service discovery request.
  1022. * The signal will carry station address, frequency, dialog token,
  1023. * update indicator and it tlvs
  1024. *
  1025. * @wpa_s: %wpa_supplicant network interface data
  1026. * @sa: station addr (p2p i/f) of the peer
  1027. * @dialog_token: service discovery request dialog token
  1028. * @update_indic: service discovery request update indicator
  1029. * @tlvs: service discovery request genrated byte array of tlvs
  1030. * @tlvs_len: service discovery request tlvs length
  1031. */
  1032. void wpas_dbus_signal_p2p_sd_request(struct wpa_supplicant *wpa_s,
  1033. int freq, const u8 *sa, u8 dialog_token,
  1034. u16 update_indic, const u8 *tlvs,
  1035. size_t tlvs_len)
  1036. {
  1037. DBusMessage *msg;
  1038. DBusMessageIter iter, dict_iter;
  1039. struct wpas_dbus_priv *iface;
  1040. char peer_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  1041. iface = wpa_s->global->dbus;
  1042. /* Do nothing if the control interface is not turned on */
  1043. if (iface == NULL)
  1044. return;
  1045. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  1046. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1047. "ServiceDiscoveryRequest");
  1048. if (msg == NULL)
  1049. return;
  1050. /* Check if this is a known peer */
  1051. if (p2p_get_peer_info(wpa_s->global->p2p, sa, 0, NULL, 0) < 0)
  1052. goto error;
  1053. os_snprintf(peer_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  1054. "%s/" WPAS_DBUS_NEW_P2P_PEERS_PART "/"
  1055. COMPACT_MACSTR, wpa_s->dbus_new_path, MAC2STR(sa));
  1056. path = peer_obj_path;
  1057. dbus_message_iter_init_append(msg, &iter);
  1058. if (!wpa_dbus_dict_open_write(&iter, &dict_iter))
  1059. goto error;
  1060. if (!wpa_dbus_dict_append_object_path(&dict_iter, "peer_object",
  1061. path) ||
  1062. !wpa_dbus_dict_append_int32(&dict_iter, "frequency", freq) ||
  1063. !wpa_dbus_dict_append_int32(&dict_iter, "dialog_token",
  1064. dialog_token) ||
  1065. !wpa_dbus_dict_append_uint16(&dict_iter, "update_indicator",
  1066. update_indic) ||
  1067. !wpa_dbus_dict_append_byte_array(&dict_iter, "tlvs",
  1068. (const char *) tlvs,
  1069. tlvs_len) ||
  1070. !wpa_dbus_dict_close_write(&iter, &dict_iter))
  1071. goto error;
  1072. dbus_connection_send(iface->con, msg, NULL);
  1073. dbus_message_unref(msg);
  1074. return;
  1075. error:
  1076. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  1077. dbus_message_unref(msg);
  1078. }
  1079. /**
  1080. *
  1081. * Method to emit a signal for a service discovery response.
  1082. * The signal will carry station address, update indicator and it
  1083. * tlvs
  1084. *
  1085. * @wpa_s: %wpa_supplicant network interface data
  1086. * @sa: station addr (p2p i/f) of the peer
  1087. * @update_indic: service discovery request update indicator
  1088. * @tlvs: service discovery request genrated byte array of tlvs
  1089. * @tlvs_len: service discovery request tlvs length
  1090. */
  1091. void wpas_dbus_signal_p2p_sd_response(struct wpa_supplicant *wpa_s,
  1092. const u8 *sa, u16 update_indic,
  1093. const u8 *tlvs, size_t tlvs_len)
  1094. {
  1095. DBusMessage *msg;
  1096. DBusMessageIter iter, dict_iter;
  1097. struct wpas_dbus_priv *iface;
  1098. char peer_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  1099. iface = wpa_s->global->dbus;
  1100. /* Do nothing if the control interface is not turned on */
  1101. if (iface == NULL)
  1102. return;
  1103. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  1104. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1105. "ServiceDiscoveryResponse");
  1106. if (msg == NULL)
  1107. return;
  1108. /* Check if this is a known peer */
  1109. if (p2p_get_peer_info(wpa_s->global->p2p, sa, 0, NULL, 0) < 0)
  1110. goto error;
  1111. os_snprintf(peer_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  1112. "%s/" WPAS_DBUS_NEW_P2P_PEERS_PART "/"
  1113. COMPACT_MACSTR, wpa_s->dbus_new_path, MAC2STR(sa));
  1114. path = peer_obj_path;
  1115. dbus_message_iter_init_append(msg, &iter);
  1116. if (!wpa_dbus_dict_open_write(&iter, &dict_iter))
  1117. goto error;
  1118. if (!wpa_dbus_dict_append_object_path(&dict_iter, "peer_object",
  1119. path) ||
  1120. !wpa_dbus_dict_append_uint16(&dict_iter, "update_indicator",
  1121. update_indic) ||
  1122. !wpa_dbus_dict_append_byte_array(&dict_iter, "tlvs",
  1123. (const char *) tlvs,
  1124. tlvs_len) ||
  1125. !wpa_dbus_dict_close_write(&iter, &dict_iter))
  1126. goto error;
  1127. dbus_connection_send(iface->con, msg, NULL);
  1128. dbus_message_unref(msg);
  1129. return;
  1130. error:
  1131. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  1132. dbus_message_unref(msg);
  1133. }
  1134. /**
  1135. * wpas_dbus_signal_persistent_group - Send a persistent group related
  1136. * event signal
  1137. * @wpa_s: %wpa_supplicant network interface data
  1138. * @id: new persistent group id
  1139. * @sig_name: signal name - PersistentGroupAdded, PersistentGroupRemoved
  1140. * @properties: determines if add second argument with object properties
  1141. *
  1142. * Notify listeners about an event related to persistent groups.
  1143. */
  1144. static void wpas_dbus_signal_persistent_group(struct wpa_supplicant *wpa_s,
  1145. int id, const char *sig_name,
  1146. int properties)
  1147. {
  1148. struct wpas_dbus_priv *iface;
  1149. DBusMessage *msg;
  1150. DBusMessageIter iter;
  1151. char pgrp_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  1152. iface = wpa_s->global->dbus;
  1153. /* Do nothing if the control interface is not turned on */
  1154. if (iface == NULL)
  1155. return;
  1156. os_snprintf(pgrp_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  1157. "%s/" WPAS_DBUS_NEW_PERSISTENT_GROUPS_PART "/%u",
  1158. wpa_s->dbus_new_path, id);
  1159. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  1160. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1161. sig_name);
  1162. if (msg == NULL)
  1163. return;
  1164. dbus_message_iter_init_append(msg, &iter);
  1165. path = pgrp_obj_path;
  1166. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  1167. &path))
  1168. goto err;
  1169. if (properties) {
  1170. if (!wpa_dbus_get_object_properties(
  1171. iface, pgrp_obj_path,
  1172. WPAS_DBUS_NEW_IFACE_PERSISTENT_GROUP, &iter))
  1173. goto err;
  1174. }
  1175. dbus_connection_send(iface->con, msg, NULL);
  1176. dbus_message_unref(msg);
  1177. return;
  1178. err:
  1179. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  1180. dbus_message_unref(msg);
  1181. }
  1182. /**
  1183. * wpas_dbus_signal_persistent_group_added - Send a persistent_group
  1184. * added signal
  1185. * @wpa_s: %wpa_supplicant network interface data
  1186. * @id: new persistent group id
  1187. *
  1188. * Notify listeners about addition of a new persistent group.
  1189. */
  1190. static void wpas_dbus_signal_persistent_group_added(
  1191. struct wpa_supplicant *wpa_s, int id)
  1192. {
  1193. wpas_dbus_signal_persistent_group(wpa_s, id, "PersistentGroupAdded",
  1194. TRUE);
  1195. }
  1196. /**
  1197. * wpas_dbus_signal_persistent_group_removed - Send a persistent_group
  1198. * removed signal
  1199. * @wpa_s: %wpa_supplicant network interface data
  1200. * @id: persistent group id
  1201. *
  1202. * Notify listeners about removal of a persistent group.
  1203. */
  1204. static void wpas_dbus_signal_persistent_group_removed(
  1205. struct wpa_supplicant *wpa_s, int id)
  1206. {
  1207. wpas_dbus_signal_persistent_group(wpa_s, id, "PersistentGroupRemoved",
  1208. FALSE);
  1209. }
  1210. /**
  1211. * wpas_dbus_signal_p2p_wps_failed - Signals WpsFailed event
  1212. * @wpa_s: %wpa_supplicant network interface data
  1213. *
  1214. * Sends Event dbus signal with name "fail" and dictionary containing
  1215. * "msg" field with fail message number (int32) as arguments
  1216. */
  1217. void wpas_dbus_signal_p2p_wps_failed(struct wpa_supplicant *wpa_s,
  1218. struct wps_event_fail *fail)
  1219. {
  1220. DBusMessage *msg;
  1221. DBusMessageIter iter, dict_iter;
  1222. struct wpas_dbus_priv *iface;
  1223. char *key = "fail";
  1224. iface = wpa_s->global->dbus;
  1225. /* Do nothing if the control interface is not turned on */
  1226. if (iface == NULL)
  1227. return;
  1228. msg = dbus_message_new_signal(wpa_s->dbus_new_path,
  1229. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1230. "WpsFailed");
  1231. if (msg == NULL)
  1232. return;
  1233. dbus_message_iter_init_append(msg, &iter);
  1234. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_STRING, &key) ||
  1235. !wpa_dbus_dict_open_write(&iter, &dict_iter) ||
  1236. !wpa_dbus_dict_append_int32(&dict_iter, "msg", fail->msg) ||
  1237. !wpa_dbus_dict_append_int16(&dict_iter, "config_error",
  1238. fail->config_error) ||
  1239. !wpa_dbus_dict_close_write(&iter, &dict_iter))
  1240. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  1241. else
  1242. dbus_connection_send(iface->con, msg, NULL);
  1243. dbus_message_unref(msg);
  1244. }
  1245. #endif /*CONFIG_P2P*/
  1246. /**
  1247. * wpas_dbus_signal_prop_changed - Signals change of property
  1248. * @wpa_s: %wpa_supplicant network interface data
  1249. * @property: indicates which property has changed
  1250. *
  1251. * Sends PropertyChanged signals with path, interface and arguments
  1252. * depending on which property has changed.
  1253. */
  1254. void wpas_dbus_signal_prop_changed(struct wpa_supplicant *wpa_s,
  1255. enum wpas_dbus_prop property)
  1256. {
  1257. char *prop;
  1258. if (wpa_s->dbus_new_path == NULL)
  1259. return; /* Skip signal since D-Bus setup is not yet ready */
  1260. switch (property) {
  1261. case WPAS_DBUS_PROP_AP_SCAN:
  1262. prop = "ApScan";
  1263. break;
  1264. case WPAS_DBUS_PROP_SCANNING:
  1265. prop = "Scanning";
  1266. break;
  1267. case WPAS_DBUS_PROP_STATE:
  1268. prop = "State";
  1269. break;
  1270. case WPAS_DBUS_PROP_CURRENT_BSS:
  1271. prop = "CurrentBSS";
  1272. break;
  1273. case WPAS_DBUS_PROP_CURRENT_NETWORK:
  1274. prop = "CurrentNetwork";
  1275. break;
  1276. case WPAS_DBUS_PROP_BSSS:
  1277. prop = "BSSs";
  1278. break;
  1279. case WPAS_DBUS_PROP_CURRENT_AUTH_MODE:
  1280. prop = "CurrentAuthMode";
  1281. break;
  1282. default:
  1283. wpa_printf(MSG_ERROR, "dbus: %s: Unknown Property value %d",
  1284. __func__, property);
  1285. return;
  1286. }
  1287. wpa_dbus_mark_property_changed(wpa_s->global->dbus,
  1288. wpa_s->dbus_new_path,
  1289. WPAS_DBUS_NEW_IFACE_INTERFACE, prop);
  1290. }
  1291. /**
  1292. * wpas_dbus_bss_signal_prop_changed - Signals change of BSS property
  1293. * @wpa_s: %wpa_supplicant network interface data
  1294. * @property: indicates which property has changed
  1295. * @id: unique BSS identifier
  1296. *
  1297. * Sends PropertyChanged signals with path, interface, and arguments depending
  1298. * on which property has changed.
  1299. */
  1300. void wpas_dbus_bss_signal_prop_changed(struct wpa_supplicant *wpa_s,
  1301. enum wpas_dbus_bss_prop property,
  1302. unsigned int id)
  1303. {
  1304. char path[WPAS_DBUS_OBJECT_PATH_MAX];
  1305. char *prop;
  1306. switch (property) {
  1307. case WPAS_DBUS_BSS_PROP_SIGNAL:
  1308. prop = "Signal";
  1309. break;
  1310. case WPAS_DBUS_BSS_PROP_FREQ:
  1311. prop = "Frequency";
  1312. break;
  1313. case WPAS_DBUS_BSS_PROP_MODE:
  1314. prop = "Mode";
  1315. break;
  1316. case WPAS_DBUS_BSS_PROP_PRIVACY:
  1317. prop = "Privacy";
  1318. break;
  1319. case WPAS_DBUS_BSS_PROP_RATES:
  1320. prop = "Rates";
  1321. break;
  1322. case WPAS_DBUS_BSS_PROP_WPA:
  1323. prop = "WPA";
  1324. break;
  1325. case WPAS_DBUS_BSS_PROP_RSN:
  1326. prop = "RSN";
  1327. break;
  1328. case WPAS_DBUS_BSS_PROP_IES:
  1329. prop = "IEs";
  1330. break;
  1331. default:
  1332. wpa_printf(MSG_ERROR, "dbus: %s: Unknown Property value %d",
  1333. __func__, property);
  1334. return;
  1335. }
  1336. os_snprintf(path, WPAS_DBUS_OBJECT_PATH_MAX,
  1337. "%s/" WPAS_DBUS_NEW_BSSIDS_PART "/%u",
  1338. wpa_s->dbus_new_path, id);
  1339. wpa_dbus_mark_property_changed(wpa_s->global->dbus, path,
  1340. WPAS_DBUS_NEW_IFACE_BSS, prop);
  1341. }
  1342. /**
  1343. * wpas_dbus_signal_debug_level_changed - Signals change of debug param
  1344. * @global: wpa_global structure
  1345. *
  1346. * Sends PropertyChanged signals informing that debug level has changed.
  1347. */
  1348. void wpas_dbus_signal_debug_level_changed(struct wpa_global *global)
  1349. {
  1350. wpa_dbus_mark_property_changed(global->dbus, WPAS_DBUS_NEW_PATH,
  1351. WPAS_DBUS_NEW_INTERFACE,
  1352. "DebugLevel");
  1353. }
  1354. /**
  1355. * wpas_dbus_signal_debug_timestamp_changed - Signals change of debug param
  1356. * @global: wpa_global structure
  1357. *
  1358. * Sends PropertyChanged signals informing that debug timestamp has changed.
  1359. */
  1360. void wpas_dbus_signal_debug_timestamp_changed(struct wpa_global *global)
  1361. {
  1362. wpa_dbus_mark_property_changed(global->dbus, WPAS_DBUS_NEW_PATH,
  1363. WPAS_DBUS_NEW_INTERFACE,
  1364. "DebugTimestamp");
  1365. }
  1366. /**
  1367. * wpas_dbus_signal_debug_show_keys_changed - Signals change of debug param
  1368. * @global: wpa_global structure
  1369. *
  1370. * Sends PropertyChanged signals informing that debug show_keys has changed.
  1371. */
  1372. void wpas_dbus_signal_debug_show_keys_changed(struct wpa_global *global)
  1373. {
  1374. wpa_dbus_mark_property_changed(global->dbus, WPAS_DBUS_NEW_PATH,
  1375. WPAS_DBUS_NEW_INTERFACE,
  1376. "DebugShowKeys");
  1377. }
  1378. static void wpas_dbus_register(struct wpa_dbus_object_desc *obj_desc,
  1379. void *priv,
  1380. WPADBusArgumentFreeFunction priv_free,
  1381. const struct wpa_dbus_method_desc *methods,
  1382. const struct wpa_dbus_property_desc *properties,
  1383. const struct wpa_dbus_signal_desc *signals)
  1384. {
  1385. int n;
  1386. obj_desc->user_data = priv;
  1387. obj_desc->user_data_free_func = priv_free;
  1388. obj_desc->methods = methods;
  1389. obj_desc->properties = properties;
  1390. obj_desc->signals = signals;
  1391. for (n = 0; properties && properties->dbus_property; properties++)
  1392. n++;
  1393. obj_desc->prop_changed_flags = os_zalloc(n);
  1394. if (!obj_desc->prop_changed_flags)
  1395. wpa_printf(MSG_DEBUG, "dbus: %s: can't register handlers",
  1396. __func__);
  1397. }
  1398. static const struct wpa_dbus_method_desc wpas_dbus_global_methods[] = {
  1399. { "CreateInterface", WPAS_DBUS_NEW_INTERFACE,
  1400. (WPADBusMethodHandler) &wpas_dbus_handler_create_interface,
  1401. {
  1402. { "args", "a{sv}", ARG_IN },
  1403. { "path", "o", ARG_OUT },
  1404. END_ARGS
  1405. }
  1406. },
  1407. { "RemoveInterface", WPAS_DBUS_NEW_INTERFACE,
  1408. (WPADBusMethodHandler) &wpas_dbus_handler_remove_interface,
  1409. {
  1410. { "path", "o", ARG_IN },
  1411. END_ARGS
  1412. }
  1413. },
  1414. { "GetInterface", WPAS_DBUS_NEW_INTERFACE,
  1415. (WPADBusMethodHandler) &wpas_dbus_handler_get_interface,
  1416. {
  1417. { "ifname", "s", ARG_IN },
  1418. { "path", "o", ARG_OUT },
  1419. END_ARGS
  1420. }
  1421. },
  1422. { NULL, NULL, NULL, { END_ARGS } }
  1423. };
  1424. static const struct wpa_dbus_property_desc wpas_dbus_global_properties[] = {
  1425. { "DebugLevel", WPAS_DBUS_NEW_INTERFACE, "s",
  1426. wpas_dbus_getter_debug_level,
  1427. wpas_dbus_setter_debug_level
  1428. },
  1429. { "DebugTimestamp", WPAS_DBUS_NEW_INTERFACE, "b",
  1430. wpas_dbus_getter_debug_timestamp,
  1431. wpas_dbus_setter_debug_timestamp
  1432. },
  1433. { "DebugShowKeys", WPAS_DBUS_NEW_INTERFACE, "b",
  1434. wpas_dbus_getter_debug_show_keys,
  1435. wpas_dbus_setter_debug_show_keys
  1436. },
  1437. { "Interfaces", WPAS_DBUS_NEW_INTERFACE, "ao",
  1438. wpas_dbus_getter_interfaces,
  1439. NULL
  1440. },
  1441. { "EapMethods", WPAS_DBUS_NEW_INTERFACE, "as",
  1442. wpas_dbus_getter_eap_methods,
  1443. NULL
  1444. },
  1445. { NULL, NULL, NULL, NULL, NULL }
  1446. };
  1447. static const struct wpa_dbus_signal_desc wpas_dbus_global_signals[] = {
  1448. { "InterfaceAdded", WPAS_DBUS_NEW_INTERFACE,
  1449. {
  1450. { "path", "o", ARG_OUT },
  1451. { "properties", "a{sv}", ARG_OUT },
  1452. END_ARGS
  1453. }
  1454. },
  1455. { "InterfaceRemoved", WPAS_DBUS_NEW_INTERFACE,
  1456. {
  1457. { "path", "o", ARG_OUT },
  1458. END_ARGS
  1459. }
  1460. },
  1461. { "NetworkRequest", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1462. {
  1463. { "path", "o", ARG_OUT },
  1464. { "field", "s", ARG_OUT },
  1465. { "text", "s", ARG_OUT },
  1466. END_ARGS
  1467. }
  1468. },
  1469. /* Deprecated: use org.freedesktop.DBus.Properties.PropertiesChanged */
  1470. { "PropertiesChanged", WPAS_DBUS_NEW_INTERFACE,
  1471. {
  1472. { "properties", "a{sv}", ARG_OUT },
  1473. END_ARGS
  1474. }
  1475. },
  1476. { NULL, NULL, { END_ARGS } }
  1477. };
  1478. /**
  1479. * wpas_dbus_ctrl_iface_init - Initialize dbus control interface
  1480. * @global: Pointer to global data from wpa_supplicant_init()
  1481. * Returns: 0 on success or -1 on failure
  1482. *
  1483. * Initialize the dbus control interface for wpa_supplicantand and start
  1484. * receiving commands from external programs over the bus.
  1485. */
  1486. int wpas_dbus_ctrl_iface_init(struct wpas_dbus_priv *priv)
  1487. {
  1488. struct wpa_dbus_object_desc *obj_desc;
  1489. int ret;
  1490. obj_desc = os_zalloc(sizeof(struct wpa_dbus_object_desc));
  1491. if (!obj_desc) {
  1492. wpa_printf(MSG_ERROR, "Not enough memory "
  1493. "to create object description");
  1494. return -1;
  1495. }
  1496. wpas_dbus_register(obj_desc, priv->global, NULL,
  1497. wpas_dbus_global_methods,
  1498. wpas_dbus_global_properties,
  1499. wpas_dbus_global_signals);
  1500. wpa_printf(MSG_DEBUG, "dbus: Register D-Bus object '%s'",
  1501. WPAS_DBUS_NEW_PATH);
  1502. ret = wpa_dbus_ctrl_iface_init(priv, WPAS_DBUS_NEW_PATH,
  1503. WPAS_DBUS_NEW_SERVICE,
  1504. obj_desc);
  1505. if (ret < 0)
  1506. free_dbus_object_desc(obj_desc);
  1507. else
  1508. priv->dbus_new_initialized = 1;
  1509. return ret;
  1510. }
  1511. /**
  1512. * wpas_dbus_ctrl_iface_deinit - Deinitialize dbus ctrl interface for
  1513. * wpa_supplicant
  1514. * @iface: Pointer to dbus private data from wpas_dbus_init()
  1515. *
  1516. * Deinitialize the dbus control interface that was initialized with
  1517. * wpas_dbus_ctrl_iface_init().
  1518. */
  1519. void wpas_dbus_ctrl_iface_deinit(struct wpas_dbus_priv *iface)
  1520. {
  1521. if (!iface->dbus_new_initialized)
  1522. return;
  1523. wpa_printf(MSG_DEBUG, "dbus: Unregister D-Bus object '%s'",
  1524. WPAS_DBUS_NEW_PATH);
  1525. dbus_connection_unregister_object_path(iface->con,
  1526. WPAS_DBUS_NEW_PATH);
  1527. }
  1528. static void wpa_dbus_free(void *ptr)
  1529. {
  1530. os_free(ptr);
  1531. }
  1532. static const struct wpa_dbus_property_desc wpas_dbus_network_properties[] = {
  1533. { "Properties", WPAS_DBUS_NEW_IFACE_NETWORK, "a{sv}",
  1534. wpas_dbus_getter_network_properties,
  1535. wpas_dbus_setter_network_properties
  1536. },
  1537. { "Enabled", WPAS_DBUS_NEW_IFACE_NETWORK, "b",
  1538. wpas_dbus_getter_enabled,
  1539. wpas_dbus_setter_enabled
  1540. },
  1541. { NULL, NULL, NULL, NULL, NULL }
  1542. };
  1543. static const struct wpa_dbus_signal_desc wpas_dbus_network_signals[] = {
  1544. /* Deprecated: use org.freedesktop.DBus.Properties.PropertiesChanged */
  1545. { "PropertiesChanged", WPAS_DBUS_NEW_IFACE_NETWORK,
  1546. {
  1547. { "properties", "a{sv}", ARG_OUT },
  1548. END_ARGS
  1549. }
  1550. },
  1551. { NULL, NULL, { END_ARGS } }
  1552. };
  1553. /**
  1554. * wpas_dbus_register_network - Register a configured network with dbus
  1555. * @wpa_s: wpa_supplicant interface structure
  1556. * @ssid: network configuration data
  1557. * Returns: 0 on success, -1 on failure
  1558. *
  1559. * Registers network representing object with dbus
  1560. */
  1561. int wpas_dbus_register_network(struct wpa_supplicant *wpa_s,
  1562. struct wpa_ssid *ssid)
  1563. {
  1564. struct wpas_dbus_priv *ctrl_iface;
  1565. struct wpa_dbus_object_desc *obj_desc;
  1566. struct network_handler_args *arg;
  1567. char net_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  1568. #ifdef CONFIG_P2P
  1569. /*
  1570. * If it is a persistent group register it as such.
  1571. * This is to handle cases where an interface is being initialized
  1572. * with a list of networks read from config.
  1573. */
  1574. if (network_is_persistent_group(ssid))
  1575. return wpas_dbus_register_persistent_group(wpa_s, ssid);
  1576. #endif /* CONFIG_P2P */
  1577. /* Do nothing if the control interface is not turned on */
  1578. if (wpa_s == NULL || wpa_s->global == NULL)
  1579. return 0;
  1580. ctrl_iface = wpa_s->global->dbus;
  1581. if (ctrl_iface == NULL)
  1582. return 0;
  1583. os_snprintf(net_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  1584. "%s/" WPAS_DBUS_NEW_NETWORKS_PART "/%u",
  1585. wpa_s->dbus_new_path, ssid->id);
  1586. wpa_printf(MSG_DEBUG, "dbus: Register network object '%s'",
  1587. net_obj_path);
  1588. obj_desc = os_zalloc(sizeof(struct wpa_dbus_object_desc));
  1589. if (!obj_desc) {
  1590. wpa_printf(MSG_ERROR, "Not enough memory "
  1591. "to create object description");
  1592. goto err;
  1593. }
  1594. /* allocate memory for handlers arguments */
  1595. arg = os_zalloc(sizeof(struct network_handler_args));
  1596. if (!arg) {
  1597. wpa_printf(MSG_ERROR, "Not enough memory "
  1598. "to create arguments for method");
  1599. goto err;
  1600. }
  1601. arg->wpa_s = wpa_s;
  1602. arg->ssid = ssid;
  1603. wpas_dbus_register(obj_desc, arg, wpa_dbus_free, NULL,
  1604. wpas_dbus_network_properties,
  1605. wpas_dbus_network_signals);
  1606. if (wpa_dbus_register_object_per_iface(ctrl_iface, net_obj_path,
  1607. wpa_s->ifname, obj_desc))
  1608. goto err;
  1609. wpas_dbus_signal_network_added(wpa_s, ssid->id);
  1610. return 0;
  1611. err:
  1612. free_dbus_object_desc(obj_desc);
  1613. return -1;
  1614. }
  1615. /**
  1616. * wpas_dbus_unregister_network - Unregister a configured network from dbus
  1617. * @wpa_s: wpa_supplicant interface structure
  1618. * @nid: network id
  1619. * Returns: 0 on success, -1 on failure
  1620. *
  1621. * Unregisters network representing object from dbus
  1622. */
  1623. int wpas_dbus_unregister_network(struct wpa_supplicant *wpa_s, int nid)
  1624. {
  1625. struct wpas_dbus_priv *ctrl_iface;
  1626. char net_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  1627. int ret;
  1628. struct wpa_ssid *ssid;
  1629. ssid = wpa_config_get_network(wpa_s->conf, nid);
  1630. #ifdef CONFIG_P2P
  1631. /* If it is a persistent group unregister it as such */
  1632. if (ssid && network_is_persistent_group(ssid))
  1633. return wpas_dbus_unregister_persistent_group(wpa_s, nid);
  1634. #endif /* CONFIG_P2P */
  1635. /* Do nothing if the control interface is not turned on */
  1636. if (wpa_s->global == NULL || wpa_s->dbus_new_path == NULL)
  1637. return 0;
  1638. ctrl_iface = wpa_s->global->dbus;
  1639. if (ctrl_iface == NULL)
  1640. return 0;
  1641. os_snprintf(net_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  1642. "%s/" WPAS_DBUS_NEW_NETWORKS_PART "/%u",
  1643. wpa_s->dbus_new_path, nid);
  1644. wpa_printf(MSG_DEBUG, "dbus: Unregister network object '%s'",
  1645. net_obj_path);
  1646. ret = wpa_dbus_unregister_object_per_iface(ctrl_iface, net_obj_path);
  1647. if (!ret)
  1648. wpas_dbus_signal_network_removed(wpa_s, nid);
  1649. return ret;
  1650. }
  1651. static const struct wpa_dbus_property_desc wpas_dbus_bss_properties[] = {
  1652. { "SSID", WPAS_DBUS_NEW_IFACE_BSS, "ay",
  1653. wpas_dbus_getter_bss_ssid,
  1654. NULL
  1655. },
  1656. { "BSSID", WPAS_DBUS_NEW_IFACE_BSS, "ay",
  1657. wpas_dbus_getter_bss_bssid,
  1658. NULL
  1659. },
  1660. { "Privacy", WPAS_DBUS_NEW_IFACE_BSS, "b",
  1661. wpas_dbus_getter_bss_privacy,
  1662. NULL
  1663. },
  1664. { "Mode", WPAS_DBUS_NEW_IFACE_BSS, "s",
  1665. wpas_dbus_getter_bss_mode,
  1666. NULL
  1667. },
  1668. { "Signal", WPAS_DBUS_NEW_IFACE_BSS, "n",
  1669. wpas_dbus_getter_bss_signal,
  1670. NULL
  1671. },
  1672. { "Frequency", WPAS_DBUS_NEW_IFACE_BSS, "q",
  1673. wpas_dbus_getter_bss_frequency,
  1674. NULL
  1675. },
  1676. { "Rates", WPAS_DBUS_NEW_IFACE_BSS, "au",
  1677. wpas_dbus_getter_bss_rates,
  1678. NULL
  1679. },
  1680. { "WPA", WPAS_DBUS_NEW_IFACE_BSS, "a{sv}",
  1681. wpas_dbus_getter_bss_wpa,
  1682. NULL
  1683. },
  1684. { "RSN", WPAS_DBUS_NEW_IFACE_BSS, "a{sv}",
  1685. wpas_dbus_getter_bss_rsn,
  1686. NULL
  1687. },
  1688. { "IEs", WPAS_DBUS_NEW_IFACE_BSS, "ay",
  1689. wpas_dbus_getter_bss_ies,
  1690. NULL
  1691. },
  1692. { NULL, NULL, NULL, NULL, NULL }
  1693. };
  1694. static const struct wpa_dbus_signal_desc wpas_dbus_bss_signals[] = {
  1695. /* Deprecated: use org.freedesktop.DBus.Properties.PropertiesChanged */
  1696. { "PropertiesChanged", WPAS_DBUS_NEW_IFACE_BSS,
  1697. {
  1698. { "properties", "a{sv}", ARG_OUT },
  1699. END_ARGS
  1700. }
  1701. },
  1702. { NULL, NULL, { END_ARGS } }
  1703. };
  1704. /**
  1705. * wpas_dbus_unregister_bss - Unregister a scanned BSS from dbus
  1706. * @wpa_s: wpa_supplicant interface structure
  1707. * @bssid: scanned network bssid
  1708. * @id: unique BSS identifier
  1709. * Returns: 0 on success, -1 on failure
  1710. *
  1711. * Unregisters BSS representing object from dbus
  1712. */
  1713. int wpas_dbus_unregister_bss(struct wpa_supplicant *wpa_s,
  1714. u8 bssid[ETH_ALEN], unsigned int id)
  1715. {
  1716. struct wpas_dbus_priv *ctrl_iface;
  1717. char bss_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  1718. /* Do nothing if the control interface is not turned on */
  1719. if (wpa_s == NULL || wpa_s->global == NULL)
  1720. return 0;
  1721. ctrl_iface = wpa_s->global->dbus;
  1722. if (ctrl_iface == NULL)
  1723. return 0;
  1724. os_snprintf(bss_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  1725. "%s/" WPAS_DBUS_NEW_BSSIDS_PART "/%u",
  1726. wpa_s->dbus_new_path, id);
  1727. wpa_printf(MSG_DEBUG, "dbus: Unregister BSS object '%s'",
  1728. bss_obj_path);
  1729. if (wpa_dbus_unregister_object_per_iface(ctrl_iface, bss_obj_path)) {
  1730. wpa_printf(MSG_ERROR, "dbus: Cannot unregister BSS object %s",
  1731. bss_obj_path);
  1732. return -1;
  1733. }
  1734. wpas_dbus_signal_bss_removed(wpa_s, bss_obj_path);
  1735. wpas_dbus_signal_prop_changed(wpa_s, WPAS_DBUS_PROP_BSSS);
  1736. return 0;
  1737. }
  1738. /**
  1739. * wpas_dbus_register_bss - Register a scanned BSS with dbus
  1740. * @wpa_s: wpa_supplicant interface structure
  1741. * @bssid: scanned network bssid
  1742. * @id: unique BSS identifier
  1743. * Returns: 0 on success, -1 on failure
  1744. *
  1745. * Registers BSS representing object with dbus
  1746. */
  1747. int wpas_dbus_register_bss(struct wpa_supplicant *wpa_s,
  1748. u8 bssid[ETH_ALEN], unsigned int id)
  1749. {
  1750. struct wpas_dbus_priv *ctrl_iface;
  1751. struct wpa_dbus_object_desc *obj_desc;
  1752. char bss_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  1753. struct bss_handler_args *arg;
  1754. /* Do nothing if the control interface is not turned on */
  1755. if (wpa_s == NULL || wpa_s->global == NULL)
  1756. return 0;
  1757. ctrl_iface = wpa_s->global->dbus;
  1758. if (ctrl_iface == NULL)
  1759. return 0;
  1760. os_snprintf(bss_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  1761. "%s/" WPAS_DBUS_NEW_BSSIDS_PART "/%u",
  1762. wpa_s->dbus_new_path, id);
  1763. obj_desc = os_zalloc(sizeof(struct wpa_dbus_object_desc));
  1764. if (!obj_desc) {
  1765. wpa_printf(MSG_ERROR, "Not enough memory "
  1766. "to create object description");
  1767. goto err;
  1768. }
  1769. arg = os_zalloc(sizeof(struct bss_handler_args));
  1770. if (!arg) {
  1771. wpa_printf(MSG_ERROR, "Not enough memory "
  1772. "to create arguments for handler");
  1773. goto err;
  1774. }
  1775. arg->wpa_s = wpa_s;
  1776. arg->id = id;
  1777. wpas_dbus_register(obj_desc, arg, wpa_dbus_free, NULL,
  1778. wpas_dbus_bss_properties,
  1779. wpas_dbus_bss_signals);
  1780. wpa_printf(MSG_DEBUG, "dbus: Register BSS object '%s'",
  1781. bss_obj_path);
  1782. if (wpa_dbus_register_object_per_iface(ctrl_iface, bss_obj_path,
  1783. wpa_s->ifname, obj_desc)) {
  1784. wpa_printf(MSG_ERROR,
  1785. "Cannot register BSSID dbus object %s.",
  1786. bss_obj_path);
  1787. goto err;
  1788. }
  1789. wpas_dbus_signal_bss_added(wpa_s, bss_obj_path);
  1790. wpas_dbus_signal_prop_changed(wpa_s, WPAS_DBUS_PROP_BSSS);
  1791. return 0;
  1792. err:
  1793. free_dbus_object_desc(obj_desc);
  1794. return -1;
  1795. }
  1796. static const struct wpa_dbus_method_desc wpas_dbus_interface_methods[] = {
  1797. { "Scan", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1798. (WPADBusMethodHandler) &wpas_dbus_handler_scan,
  1799. {
  1800. { "args", "a{sv}", ARG_IN },
  1801. END_ARGS
  1802. }
  1803. },
  1804. { "Disconnect", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1805. (WPADBusMethodHandler) &wpas_dbus_handler_disconnect,
  1806. {
  1807. END_ARGS
  1808. }
  1809. },
  1810. { "AddNetwork", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1811. (WPADBusMethodHandler) &wpas_dbus_handler_add_network,
  1812. {
  1813. { "args", "a{sv}", ARG_IN },
  1814. { "path", "o", ARG_OUT },
  1815. END_ARGS
  1816. }
  1817. },
  1818. { "RemoveNetwork", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1819. (WPADBusMethodHandler) &wpas_dbus_handler_remove_network,
  1820. {
  1821. { "path", "o", ARG_IN },
  1822. END_ARGS
  1823. }
  1824. },
  1825. { "RemoveAllNetworks", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1826. (WPADBusMethodHandler) &wpas_dbus_handler_remove_all_networks,
  1827. {
  1828. END_ARGS
  1829. }
  1830. },
  1831. { "SelectNetwork", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1832. (WPADBusMethodHandler) &wpas_dbus_handler_select_network,
  1833. {
  1834. { "path", "o", ARG_IN },
  1835. END_ARGS
  1836. }
  1837. },
  1838. { "NetworkReply", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1839. (WPADBusMethodHandler) &wpas_dbus_handler_network_reply,
  1840. {
  1841. { "path", "o", ARG_IN },
  1842. { "field", "s", ARG_IN },
  1843. { "value", "s", ARG_IN },
  1844. END_ARGS
  1845. }
  1846. },
  1847. { "AddBlob", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1848. (WPADBusMethodHandler) &wpas_dbus_handler_add_blob,
  1849. {
  1850. { "name", "s", ARG_IN },
  1851. { "data", "ay", ARG_IN },
  1852. END_ARGS
  1853. }
  1854. },
  1855. { "GetBlob", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1856. (WPADBusMethodHandler) &wpas_dbus_handler_get_blob,
  1857. {
  1858. { "name", "s", ARG_IN },
  1859. { "data", "ay", ARG_OUT },
  1860. END_ARGS
  1861. }
  1862. },
  1863. { "RemoveBlob", WPAS_DBUS_NEW_IFACE_INTERFACE,
  1864. (WPADBusMethodHandler) &wpas_dbus_handler_remove_blob,
  1865. {
  1866. { "name", "s", ARG_IN },
  1867. END_ARGS
  1868. }
  1869. },
  1870. #ifdef CONFIG_WPS
  1871. { "Start", WPAS_DBUS_NEW_IFACE_WPS,
  1872. (WPADBusMethodHandler) &wpas_dbus_handler_wps_start,
  1873. {
  1874. { "args", "a{sv}", ARG_IN },
  1875. { "output", "a{sv}", ARG_OUT },
  1876. END_ARGS
  1877. }
  1878. },
  1879. #endif /* CONFIG_WPS */
  1880. #ifdef CONFIG_P2P
  1881. { "Find", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1882. (WPADBusMethodHandler)wpas_dbus_handler_p2p_find,
  1883. {
  1884. { "args", "a{sv}", ARG_IN },
  1885. END_ARGS
  1886. }
  1887. },
  1888. { "StopFind", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1889. (WPADBusMethodHandler)wpas_dbus_handler_p2p_stop_find,
  1890. {
  1891. END_ARGS
  1892. }
  1893. },
  1894. { "Listen", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1895. (WPADBusMethodHandler)wpas_dbus_handler_p2p_listen,
  1896. {
  1897. { "timeout", "i", ARG_IN },
  1898. END_ARGS
  1899. }
  1900. },
  1901. { "ExtendedListen", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1902. (WPADBusMethodHandler)wpas_dbus_handler_p2p_extendedlisten,
  1903. {
  1904. { "args", "a{sv}", ARG_IN },
  1905. END_ARGS
  1906. }
  1907. },
  1908. { "PresenceRequest", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1909. (WPADBusMethodHandler)wpas_dbus_handler_p2p_presence_request,
  1910. {
  1911. { "args", "a{sv}", ARG_IN },
  1912. END_ARGS
  1913. }
  1914. },
  1915. { "ProvisionDiscoveryRequest", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1916. (WPADBusMethodHandler)wpas_dbus_handler_p2p_prov_disc_req,
  1917. {
  1918. { "peer", "o", ARG_IN },
  1919. { "config_method", "s", ARG_IN },
  1920. END_ARGS
  1921. }
  1922. },
  1923. { "Connect", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1924. (WPADBusMethodHandler)wpas_dbus_handler_p2p_connect,
  1925. {
  1926. { "args", "a{sv}", ARG_IN },
  1927. { "generated_pin", "s", ARG_OUT },
  1928. END_ARGS
  1929. }
  1930. },
  1931. { "GroupAdd", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1932. (WPADBusMethodHandler)wpas_dbus_handler_p2p_group_add,
  1933. {
  1934. { "args", "a{sv}", ARG_IN },
  1935. END_ARGS
  1936. }
  1937. },
  1938. { "Invite", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1939. (WPADBusMethodHandler)wpas_dbus_handler_p2p_invite,
  1940. {
  1941. { "args", "a{sv}", ARG_IN },
  1942. END_ARGS
  1943. }
  1944. },
  1945. { "Disconnect", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1946. (WPADBusMethodHandler)wpas_dbus_handler_p2p_disconnect,
  1947. {
  1948. END_ARGS
  1949. }
  1950. },
  1951. { "RejectPeer", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1952. (WPADBusMethodHandler)wpas_dbus_handler_p2p_rejectpeer,
  1953. {
  1954. { "peer", "o", ARG_IN },
  1955. END_ARGS
  1956. }
  1957. },
  1958. { "Flush", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1959. (WPADBusMethodHandler)wpas_dbus_handler_p2p_flush,
  1960. {
  1961. END_ARGS
  1962. }
  1963. },
  1964. { "AddService", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1965. (WPADBusMethodHandler)wpas_dbus_handler_p2p_add_service,
  1966. {
  1967. { "args", "a{sv}", ARG_IN },
  1968. END_ARGS
  1969. }
  1970. },
  1971. { "DeleteService", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1972. (WPADBusMethodHandler)wpas_dbus_handler_p2p_delete_service,
  1973. {
  1974. { "args", "a{sv}", ARG_IN },
  1975. END_ARGS
  1976. }
  1977. },
  1978. { "FlushService", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1979. (WPADBusMethodHandler)wpas_dbus_handler_p2p_flush_service,
  1980. {
  1981. END_ARGS
  1982. }
  1983. },
  1984. { "ServiceDiscoveryRequest", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1985. (WPADBusMethodHandler)wpas_dbus_handler_p2p_service_sd_req,
  1986. {
  1987. { "args", "a{sv}", ARG_IN },
  1988. END_ARGS
  1989. }
  1990. },
  1991. { "ServiceDiscoveryResponse", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1992. (WPADBusMethodHandler)wpas_dbus_handler_p2p_service_sd_res,
  1993. {
  1994. { "args", "a{sv}", ARG_IN },
  1995. END_ARGS
  1996. }
  1997. },
  1998. { "ServiceDiscoveryCancelRequest", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  1999. (WPADBusMethodHandler)wpas_dbus_handler_p2p_service_sd_cancel_req,
  2000. {
  2001. { "args", "t", ARG_IN },
  2002. END_ARGS
  2003. }
  2004. },
  2005. { "ServiceUpdate", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2006. (WPADBusMethodHandler)wpas_dbus_handler_p2p_service_update,
  2007. {
  2008. END_ARGS
  2009. }
  2010. },
  2011. { "ServiceDiscoveryExternal", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2012. (WPADBusMethodHandler)wpas_dbus_handler_p2p_serv_disc_external,
  2013. {
  2014. { "arg", "i", ARG_IN },
  2015. END_ARGS
  2016. }
  2017. },
  2018. { "ServiceDiscoveryExternal", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2019. (WPADBusMethodHandler)wpas_dbus_handler_p2p_serv_disc_external,
  2020. {
  2021. { "arg", "i", ARG_IN },
  2022. END_ARGS
  2023. }
  2024. },
  2025. { "AddPersistentGroup", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2026. (WPADBusMethodHandler) wpas_dbus_handler_add_persistent_group,
  2027. {
  2028. { "args", "a{sv}", ARG_IN },
  2029. { "path", "o", ARG_OUT },
  2030. END_ARGS
  2031. }
  2032. },
  2033. { "RemovePersistentGroup", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2034. (WPADBusMethodHandler) wpas_dbus_handler_remove_persistent_group,
  2035. {
  2036. { "path", "o", ARG_IN },
  2037. END_ARGS
  2038. }
  2039. },
  2040. { "RemoveAllPersistentGroups", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2041. (WPADBusMethodHandler)
  2042. wpas_dbus_handler_remove_all_persistent_groups,
  2043. {
  2044. END_ARGS
  2045. }
  2046. },
  2047. #endif /* CONFIG_P2P */
  2048. { "FlushBSS", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2049. (WPADBusMethodHandler) &wpas_dbus_handler_flush_bss,
  2050. {
  2051. { "age", "u", ARG_IN },
  2052. END_ARGS
  2053. }
  2054. },
  2055. { NULL, NULL, NULL, { END_ARGS } }
  2056. };
  2057. static const struct wpa_dbus_property_desc wpas_dbus_interface_properties[] = {
  2058. { "Capabilities", WPAS_DBUS_NEW_IFACE_INTERFACE, "a{sv}",
  2059. wpas_dbus_getter_capabilities,
  2060. NULL
  2061. },
  2062. { "State", WPAS_DBUS_NEW_IFACE_INTERFACE, "s",
  2063. wpas_dbus_getter_state,
  2064. NULL
  2065. },
  2066. { "Scanning", WPAS_DBUS_NEW_IFACE_INTERFACE, "b",
  2067. wpas_dbus_getter_scanning,
  2068. NULL
  2069. },
  2070. { "ApScan", WPAS_DBUS_NEW_IFACE_INTERFACE, "u",
  2071. wpas_dbus_getter_ap_scan,
  2072. wpas_dbus_setter_ap_scan
  2073. },
  2074. { "BSSExpireAge", WPAS_DBUS_NEW_IFACE_INTERFACE, "u",
  2075. wpas_dbus_getter_bss_expire_age,
  2076. wpas_dbus_setter_bss_expire_age
  2077. },
  2078. { "BSSExpireCount", WPAS_DBUS_NEW_IFACE_INTERFACE, "u",
  2079. wpas_dbus_getter_bss_expire_count,
  2080. wpas_dbus_setter_bss_expire_count
  2081. },
  2082. { "Country", WPAS_DBUS_NEW_IFACE_INTERFACE, "s",
  2083. wpas_dbus_getter_country,
  2084. wpas_dbus_setter_country
  2085. },
  2086. { "Ifname", WPAS_DBUS_NEW_IFACE_INTERFACE, "s",
  2087. wpas_dbus_getter_ifname,
  2088. NULL
  2089. },
  2090. { "Driver", WPAS_DBUS_NEW_IFACE_INTERFACE, "s",
  2091. wpas_dbus_getter_driver,
  2092. NULL
  2093. },
  2094. { "BridgeIfname", WPAS_DBUS_NEW_IFACE_INTERFACE, "s",
  2095. wpas_dbus_getter_bridge_ifname,
  2096. NULL
  2097. },
  2098. { "CurrentBSS", WPAS_DBUS_NEW_IFACE_INTERFACE, "o",
  2099. wpas_dbus_getter_current_bss,
  2100. NULL
  2101. },
  2102. { "CurrentNetwork", WPAS_DBUS_NEW_IFACE_INTERFACE, "o",
  2103. wpas_dbus_getter_current_network,
  2104. NULL
  2105. },
  2106. { "CurrentAuthMode", WPAS_DBUS_NEW_IFACE_INTERFACE, "s",
  2107. wpas_dbus_getter_current_auth_mode,
  2108. NULL
  2109. },
  2110. { "Blobs", WPAS_DBUS_NEW_IFACE_INTERFACE, "a{say}",
  2111. wpas_dbus_getter_blobs,
  2112. NULL
  2113. },
  2114. { "BSSs", WPAS_DBUS_NEW_IFACE_INTERFACE, "ao",
  2115. wpas_dbus_getter_bsss,
  2116. NULL
  2117. },
  2118. { "Networks", WPAS_DBUS_NEW_IFACE_INTERFACE, "ao",
  2119. wpas_dbus_getter_networks,
  2120. NULL
  2121. },
  2122. #ifdef CONFIG_WPS
  2123. { "ProcessCredentials", WPAS_DBUS_NEW_IFACE_WPS, "b",
  2124. wpas_dbus_getter_process_credentials,
  2125. wpas_dbus_setter_process_credentials
  2126. },
  2127. #endif /* CONFIG_WPS */
  2128. #ifdef CONFIG_P2P
  2129. { "P2PDeviceProperties", WPAS_DBUS_NEW_IFACE_P2PDEVICE, "a{sv}",
  2130. wpas_dbus_getter_p2p_device_properties,
  2131. wpas_dbus_setter_p2p_device_properties
  2132. },
  2133. { "Peers", WPAS_DBUS_NEW_IFACE_P2PDEVICE, "ao",
  2134. wpas_dbus_getter_p2p_peers,
  2135. NULL
  2136. },
  2137. { "Role", WPAS_DBUS_NEW_IFACE_P2PDEVICE, "s",
  2138. wpas_dbus_getter_p2p_role,
  2139. NULL
  2140. },
  2141. { "Group", WPAS_DBUS_NEW_IFACE_P2PDEVICE, "o",
  2142. wpas_dbus_getter_p2p_group,
  2143. NULL
  2144. },
  2145. { "PeerGO", WPAS_DBUS_NEW_IFACE_P2PDEVICE, "o",
  2146. wpas_dbus_getter_p2p_peergo,
  2147. NULL
  2148. },
  2149. { "PersistentGroups", WPAS_DBUS_NEW_IFACE_P2PDEVICE, "ao",
  2150. wpas_dbus_getter_persistent_groups,
  2151. NULL
  2152. },
  2153. #endif /* CONFIG_P2P */
  2154. { NULL, NULL, NULL, NULL, NULL }
  2155. };
  2156. static const struct wpa_dbus_signal_desc wpas_dbus_interface_signals[] = {
  2157. { "ScanDone", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2158. {
  2159. { "success", "b", ARG_OUT },
  2160. END_ARGS
  2161. }
  2162. },
  2163. { "BSSAdded", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2164. {
  2165. { "path", "o", ARG_OUT },
  2166. { "properties", "a{sv}", ARG_OUT },
  2167. END_ARGS
  2168. }
  2169. },
  2170. { "BSSRemoved", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2171. {
  2172. { "path", "o", ARG_OUT },
  2173. END_ARGS
  2174. }
  2175. },
  2176. { "BlobAdded", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2177. {
  2178. { "name", "s", ARG_OUT },
  2179. END_ARGS
  2180. }
  2181. },
  2182. { "BlobRemoved", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2183. {
  2184. { "name", "s", ARG_OUT },
  2185. END_ARGS
  2186. }
  2187. },
  2188. { "NetworkAdded", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2189. {
  2190. { "path", "o", ARG_OUT },
  2191. { "properties", "a{sv}", ARG_OUT },
  2192. END_ARGS
  2193. }
  2194. },
  2195. { "NetworkRemoved", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2196. {
  2197. { "path", "o", ARG_OUT },
  2198. END_ARGS
  2199. }
  2200. },
  2201. { "NetworkSelected", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2202. {
  2203. { "path", "o", ARG_OUT },
  2204. END_ARGS
  2205. }
  2206. },
  2207. /* Deprecated: use org.freedesktop.DBus.Properties.PropertiesChanged */
  2208. { "PropertiesChanged", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2209. {
  2210. { "properties", "a{sv}", ARG_OUT },
  2211. END_ARGS
  2212. }
  2213. },
  2214. #ifdef CONFIG_WPS
  2215. { "Event", WPAS_DBUS_NEW_IFACE_WPS,
  2216. {
  2217. { "name", "s", ARG_OUT },
  2218. { "args", "a{sv}", ARG_OUT },
  2219. END_ARGS
  2220. }
  2221. },
  2222. { "Credentials", WPAS_DBUS_NEW_IFACE_WPS,
  2223. {
  2224. { "credentials", "a{sv}", ARG_OUT },
  2225. END_ARGS
  2226. }
  2227. },
  2228. /* Deprecated: use org.freedesktop.DBus.Properties.PropertiesChanged */
  2229. { "PropertiesChanged", WPAS_DBUS_NEW_IFACE_WPS,
  2230. {
  2231. { "properties", "a{sv}", ARG_OUT },
  2232. END_ARGS
  2233. }
  2234. },
  2235. #endif /* CONFIG_WPS */
  2236. #ifdef CONFIG_P2P
  2237. { "P2PStateChanged", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2238. {
  2239. { "states", "a{ss}", ARG_OUT },
  2240. END_ARGS
  2241. }
  2242. },
  2243. { "DeviceFound", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2244. {
  2245. { "path", "o", ARG_OUT },
  2246. { "properties", "a{sv}", ARG_OUT },
  2247. END_ARGS
  2248. }
  2249. },
  2250. { "DeviceLost", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2251. {
  2252. { "path", "o", ARG_OUT },
  2253. END_ARGS
  2254. }
  2255. },
  2256. { "ProvisionDiscoveryRequestDisplayPin", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2257. {
  2258. { "peer_object", "o", ARG_OUT },
  2259. { "pin", "s", ARG_OUT },
  2260. END_ARGS
  2261. }
  2262. },
  2263. { "ProvisionDiscoveryResponseDisplayPin", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2264. {
  2265. { "peer_object", "o", ARG_OUT },
  2266. { "pin", "s", ARG_OUT },
  2267. END_ARGS
  2268. }
  2269. },
  2270. { "ProvisionDiscoveryRequestEnterPin", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2271. {
  2272. { "peer_object", "o", ARG_OUT },
  2273. END_ARGS
  2274. }
  2275. },
  2276. { "ProvisionDiscoveryResponseEnterPin", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2277. {
  2278. { "peer_object", "o", ARG_OUT },
  2279. END_ARGS
  2280. }
  2281. },
  2282. { "ProvisionDiscoveryPBCRequest", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2283. {
  2284. { "peer_object", "o", ARG_OUT },
  2285. END_ARGS
  2286. }
  2287. },
  2288. { "ProvisionDiscoveryPBCResponse", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2289. {
  2290. { "peer_object", "o", ARG_OUT },
  2291. END_ARGS
  2292. }
  2293. },
  2294. { "ProvisionDiscoveryFailure", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2295. {
  2296. { "peer_object", "o", ARG_OUT },
  2297. { "status", "i", ARG_OUT },
  2298. END_ARGS
  2299. }
  2300. },
  2301. { "GroupStarted", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2302. {
  2303. { "properties", "a{sv}", ARG_OUT },
  2304. END_ARGS
  2305. }
  2306. },
  2307. { "GONegotiationSuccess", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2308. {
  2309. END_ARGS
  2310. }
  2311. },
  2312. { "GONegotiationFailure", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2313. {
  2314. { "status", "i", ARG_OUT },
  2315. END_ARGS
  2316. }
  2317. },
  2318. { "GONegotiationRequest", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2319. {
  2320. { "path", "o", ARG_OUT },
  2321. { "dev_passwd_id", "i", ARG_OUT },
  2322. END_ARGS
  2323. }
  2324. },
  2325. { "InvitationResult", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2326. {
  2327. { "invite_result", "a{sv}", ARG_OUT },
  2328. END_ARGS
  2329. }
  2330. },
  2331. { "GroupFinished", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2332. {
  2333. { "ifname", "s", ARG_OUT },
  2334. { "role", "s", ARG_OUT },
  2335. END_ARGS
  2336. }
  2337. },
  2338. { "ServiceDiscoveryRequest", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2339. {
  2340. { "sd_request", "a{sv}", ARG_OUT },
  2341. END_ARGS
  2342. }
  2343. },
  2344. { "ServiceDiscoveryResponse", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2345. {
  2346. { "sd_response", "a{sv}", ARG_OUT },
  2347. END_ARGS
  2348. }
  2349. },
  2350. { "PersistentGroupAdded", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2351. {
  2352. { "path", "o", ARG_OUT },
  2353. { "properties", "a{sv}", ARG_OUT },
  2354. END_ARGS
  2355. }
  2356. },
  2357. { "PersistentGroupRemoved", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2358. {
  2359. { "path", "o", ARG_OUT },
  2360. END_ARGS
  2361. }
  2362. },
  2363. { "WpsFailed", WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2364. {
  2365. { "name", "s", ARG_OUT },
  2366. { "args", "a{sv}", ARG_OUT },
  2367. END_ARGS
  2368. }
  2369. },
  2370. #endif /* CONFIG_P2P */
  2371. { "Certification", WPAS_DBUS_NEW_IFACE_INTERFACE,
  2372. {
  2373. { "certification", "a{sv}", ARG_OUT },
  2374. END_ARGS
  2375. }
  2376. },
  2377. { NULL, NULL, { END_ARGS } }
  2378. };
  2379. int wpas_dbus_register_interface(struct wpa_supplicant *wpa_s)
  2380. {
  2381. struct wpa_dbus_object_desc *obj_desc = NULL;
  2382. struct wpas_dbus_priv *ctrl_iface = wpa_s->global->dbus;
  2383. int next;
  2384. /* Do nothing if the control interface is not turned on */
  2385. if (ctrl_iface == NULL)
  2386. return 0;
  2387. /* Create and set the interface's object path */
  2388. wpa_s->dbus_new_path = os_zalloc(WPAS_DBUS_OBJECT_PATH_MAX);
  2389. if (wpa_s->dbus_new_path == NULL)
  2390. return -1;
  2391. next = ctrl_iface->next_objid++;
  2392. os_snprintf(wpa_s->dbus_new_path, WPAS_DBUS_OBJECT_PATH_MAX,
  2393. WPAS_DBUS_NEW_PATH_INTERFACES "/%u",
  2394. next);
  2395. obj_desc = os_zalloc(sizeof(struct wpa_dbus_object_desc));
  2396. if (!obj_desc) {
  2397. wpa_printf(MSG_ERROR, "Not enough memory "
  2398. "to create object description");
  2399. goto err;
  2400. }
  2401. wpas_dbus_register(obj_desc, wpa_s, NULL, wpas_dbus_interface_methods,
  2402. wpas_dbus_interface_properties,
  2403. wpas_dbus_interface_signals);
  2404. wpa_printf(MSG_DEBUG, "dbus: Register interface object '%s'",
  2405. wpa_s->dbus_new_path);
  2406. if (wpa_dbus_register_object_per_iface(ctrl_iface,
  2407. wpa_s->dbus_new_path,
  2408. wpa_s->ifname, obj_desc))
  2409. goto err;
  2410. wpas_dbus_signal_interface_added(wpa_s);
  2411. return 0;
  2412. err:
  2413. os_free(wpa_s->dbus_new_path);
  2414. wpa_s->dbus_new_path = NULL;
  2415. free_dbus_object_desc(obj_desc);
  2416. return -1;
  2417. }
  2418. int wpas_dbus_unregister_interface(struct wpa_supplicant *wpa_s)
  2419. {
  2420. struct wpas_dbus_priv *ctrl_iface;
  2421. /* Do nothing if the control interface is not turned on */
  2422. if (wpa_s == NULL || wpa_s->global == NULL)
  2423. return 0;
  2424. ctrl_iface = wpa_s->global->dbus;
  2425. if (ctrl_iface == NULL)
  2426. return 0;
  2427. wpa_printf(MSG_DEBUG, "dbus: Unregister interface object '%s'",
  2428. wpa_s->dbus_new_path);
  2429. if (wpa_dbus_unregister_object_per_iface(ctrl_iface,
  2430. wpa_s->dbus_new_path))
  2431. return -1;
  2432. wpas_dbus_signal_interface_removed(wpa_s);
  2433. os_free(wpa_s->dbus_new_path);
  2434. wpa_s->dbus_new_path = NULL;
  2435. return 0;
  2436. }
  2437. #ifdef CONFIG_P2P
  2438. static const struct wpa_dbus_property_desc wpas_dbus_p2p_peer_properties[] = {
  2439. { "Properties", WPAS_DBUS_NEW_IFACE_P2P_PEER, "a{sv}",
  2440. wpas_dbus_getter_p2p_peer_properties,
  2441. NULL
  2442. },
  2443. { "IEs", WPAS_DBUS_NEW_IFACE_P2P_PEER, "ay",
  2444. wpas_dbus_getter_p2p_peer_ies,
  2445. NULL
  2446. },
  2447. { NULL, NULL, NULL, NULL, NULL }
  2448. };
  2449. static const struct wpa_dbus_signal_desc wpas_dbus_p2p_peer_signals[] = {
  2450. { NULL, NULL, { END_ARGS } }
  2451. };
  2452. /**
  2453. * wpas_dbus_signal_peer - Send a peer related event signal
  2454. * @wpa_s: %wpa_supplicant network interface data
  2455. * @dev: peer device object
  2456. * @interface: name of the interface emitting this signal.
  2457. * In case of peer objects, it would be emitted by either
  2458. * the "interface object" or by "peer objects"
  2459. * @sig_name: signal name - DeviceFound
  2460. *
  2461. * Notify listeners about event related with newly found p2p peer device
  2462. */
  2463. static void wpas_dbus_signal_peer(struct wpa_supplicant *wpa_s,
  2464. const u8 *dev_addr, const char *interface,
  2465. const char *sig_name)
  2466. {
  2467. struct wpas_dbus_priv *iface;
  2468. DBusMessage *msg;
  2469. DBusMessageIter iter;
  2470. char peer_obj_path[WPAS_DBUS_OBJECT_PATH_MAX], *path;
  2471. iface = wpa_s->global->dbus;
  2472. /* Do nothing if the control interface is not turned on */
  2473. if (iface == NULL)
  2474. return;
  2475. os_snprintf(peer_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  2476. "%s/" WPAS_DBUS_NEW_P2P_PEERS_PART "/" COMPACT_MACSTR,
  2477. wpa_s->dbus_new_path, MAC2STR(dev_addr));
  2478. msg = dbus_message_new_signal(wpa_s->dbus_new_path, interface,
  2479. sig_name);
  2480. if (msg == NULL)
  2481. return;
  2482. dbus_message_iter_init_append(msg, &iter);
  2483. path = peer_obj_path;
  2484. if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_OBJECT_PATH,
  2485. &path))
  2486. goto err;
  2487. dbus_connection_send(iface->con, msg, NULL);
  2488. dbus_message_unref(msg);
  2489. return;
  2490. err:
  2491. wpa_printf(MSG_ERROR, "dbus: Failed to construct signal");
  2492. dbus_message_unref(msg);
  2493. }
  2494. /**
  2495. * wpas_dbus_signal_peer_found - Send a peer found signal
  2496. * @wpa_s: %wpa_supplicant network interface data
  2497. * @dev: peer device object
  2498. *
  2499. * Notify listeners about find a p2p peer device found
  2500. */
  2501. void wpas_dbus_signal_peer_device_found(struct wpa_supplicant *wpa_s,
  2502. const u8 *dev_addr)
  2503. {
  2504. wpas_dbus_signal_peer(wpa_s, dev_addr,
  2505. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2506. "DeviceFound");
  2507. }
  2508. /**
  2509. * wpas_dbus_signal_peer_lost - Send a peer lost signal
  2510. * @wpa_s: %wpa_supplicant network interface data
  2511. * @dev: peer device object
  2512. *
  2513. * Notify listeners about lost a p2p peer device
  2514. */
  2515. void wpas_dbus_signal_peer_device_lost(struct wpa_supplicant *wpa_s,
  2516. const u8 *dev_addr)
  2517. {
  2518. wpas_dbus_signal_peer(wpa_s, dev_addr,
  2519. WPAS_DBUS_NEW_IFACE_P2PDEVICE,
  2520. "DeviceLost");
  2521. }
  2522. /**
  2523. * wpas_dbus_register_peer - Register a discovered peer object with dbus
  2524. * @wpa_s: wpa_supplicant interface structure
  2525. * @ssid: network configuration data
  2526. * Returns: 0 on success, -1 on failure
  2527. *
  2528. * Registers network representing object with dbus
  2529. */
  2530. int wpas_dbus_register_peer(struct wpa_supplicant *wpa_s, const u8 *dev_addr)
  2531. {
  2532. struct wpas_dbus_priv *ctrl_iface;
  2533. struct wpa_dbus_object_desc *obj_desc;
  2534. struct peer_handler_args *arg;
  2535. char peer_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  2536. /* Do nothing if the control interface is not turned on */
  2537. if (wpa_s == NULL || wpa_s->global == NULL)
  2538. return 0;
  2539. ctrl_iface = wpa_s->global->dbus;
  2540. if (ctrl_iface == NULL)
  2541. return 0;
  2542. os_snprintf(peer_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  2543. "%s/" WPAS_DBUS_NEW_P2P_PEERS_PART "/" COMPACT_MACSTR,
  2544. wpa_s->dbus_new_path, MAC2STR(dev_addr));
  2545. wpa_printf(MSG_INFO, "dbus: Register peer object '%s'",
  2546. peer_obj_path);
  2547. obj_desc = os_zalloc(sizeof(struct wpa_dbus_object_desc));
  2548. if (!obj_desc) {
  2549. wpa_printf(MSG_ERROR, "Not enough memory "
  2550. "to create object description");
  2551. goto err;
  2552. }
  2553. /* allocate memory for handlers arguments */
  2554. arg = os_zalloc(sizeof(struct peer_handler_args));
  2555. if (!arg) {
  2556. wpa_printf(MSG_ERROR, "Not enough memory "
  2557. "to create arguments for method");
  2558. goto err;
  2559. }
  2560. arg->wpa_s = wpa_s;
  2561. os_memcpy(arg->p2p_device_addr, dev_addr, ETH_ALEN);
  2562. wpas_dbus_register(obj_desc, arg, wpa_dbus_free,
  2563. NULL,
  2564. wpas_dbus_p2p_peer_properties,
  2565. wpas_dbus_p2p_peer_signals);
  2566. if (wpa_dbus_register_object_per_iface(ctrl_iface, peer_obj_path,
  2567. wpa_s->ifname, obj_desc))
  2568. goto err;
  2569. return 0;
  2570. err:
  2571. free_dbus_object_desc(obj_desc);
  2572. return -1;
  2573. }
  2574. /**
  2575. * wpas_dbus_unregister_peer - Unregister a peer object with dbus
  2576. * @wpa_s: wpa_supplicant interface structure
  2577. * @dev_addr: p2p device addr
  2578. * Returns: 0 on success, -1 on failure
  2579. *
  2580. * Registers network representing object with dbus
  2581. */
  2582. int wpas_dbus_unregister_peer(struct wpa_supplicant *wpa_s,
  2583. const u8 *dev_addr)
  2584. {
  2585. struct wpas_dbus_priv *ctrl_iface;
  2586. char peer_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  2587. int ret;
  2588. /* Do nothing if the control interface is not turned on */
  2589. if (wpa_s == NULL || wpa_s->global == NULL ||
  2590. wpa_s->dbus_new_path == NULL)
  2591. return 0;
  2592. ctrl_iface = wpa_s->global->dbus;
  2593. if (ctrl_iface == NULL)
  2594. return 0;
  2595. os_snprintf(peer_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  2596. "%s/" WPAS_DBUS_NEW_P2P_PEERS_PART "/" COMPACT_MACSTR,
  2597. wpa_s->dbus_new_path, MAC2STR(dev_addr));
  2598. wpa_printf(MSG_INFO, "dbus: Unregister peer object '%s'",
  2599. peer_obj_path);
  2600. ret = wpa_dbus_unregister_object_per_iface(ctrl_iface, peer_obj_path);
  2601. return ret;
  2602. }
  2603. static const struct wpa_dbus_property_desc wpas_dbus_p2p_group_properties[] = {
  2604. { "Members", WPAS_DBUS_NEW_IFACE_P2P_GROUP, "ao",
  2605. wpas_dbus_getter_p2p_group_members,
  2606. NULL
  2607. },
  2608. { "Properties",
  2609. WPAS_DBUS_NEW_IFACE_P2P_GROUP, "a{sv}",
  2610. wpas_dbus_getter_p2p_group_properties,
  2611. wpas_dbus_setter_p2p_group_properties
  2612. },
  2613. { NULL, NULL, NULL, NULL, NULL }
  2614. };
  2615. static const struct wpa_dbus_signal_desc wpas_dbus_p2p_group_signals[] = {
  2616. { "PeerJoined", WPAS_DBUS_NEW_IFACE_P2P_GROUP,
  2617. {
  2618. { "peer", "o", ARG_OUT },
  2619. END_ARGS
  2620. }
  2621. },
  2622. { "PeerDisconnected", WPAS_DBUS_NEW_IFACE_P2P_GROUP,
  2623. {
  2624. { "peer", "o", ARG_OUT },
  2625. END_ARGS
  2626. }
  2627. },
  2628. { NULL, NULL, { END_ARGS } }
  2629. };
  2630. /**
  2631. * wpas_dbus_register_p2p_group - Register a p2p group object with dbus
  2632. * @wpa_s: wpa_supplicant interface structure
  2633. * @ssid: SSID struct
  2634. * Returns: 0 on success, -1 on failure
  2635. *
  2636. * Registers p2p group representing object with dbus
  2637. */
  2638. void wpas_dbus_register_p2p_group(struct wpa_supplicant *wpa_s,
  2639. struct wpa_ssid *ssid)
  2640. {
  2641. struct wpas_dbus_priv *ctrl_iface;
  2642. struct wpa_dbus_object_desc *obj_desc;
  2643. char group_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  2644. /* Do nothing if the control interface is not turned on */
  2645. if (wpa_s == NULL || wpa_s->global == NULL)
  2646. return;
  2647. ctrl_iface = wpa_s->global->dbus;
  2648. if (ctrl_iface == NULL)
  2649. return;
  2650. if (wpa_s->dbus_groupobj_path) {
  2651. wpa_printf(MSG_INFO, "%s: Group object '%s' already exists",
  2652. __func__, wpa_s->dbus_groupobj_path);
  2653. return;
  2654. }
  2655. if (wpas_dbus_get_group_obj_path(wpa_s, ssid, group_obj_path) < 0)
  2656. return;
  2657. wpa_s->dbus_groupobj_path = os_strdup(group_obj_path);
  2658. if (wpa_s->dbus_groupobj_path == NULL)
  2659. return;
  2660. wpa_printf(MSG_INFO, "dbus: Register group object '%s'",
  2661. group_obj_path);
  2662. obj_desc = os_zalloc(sizeof(struct wpa_dbus_object_desc));
  2663. if (!obj_desc) {
  2664. wpa_printf(MSG_ERROR, "Not enough memory "
  2665. "to create object description");
  2666. goto err;
  2667. }
  2668. wpas_dbus_register(obj_desc, wpa_s, NULL, NULL,
  2669. wpas_dbus_p2p_group_properties,
  2670. wpas_dbus_p2p_group_signals);
  2671. if (wpa_dbus_register_object_per_iface(ctrl_iface, group_obj_path,
  2672. wpa_s->ifname, obj_desc))
  2673. goto err;
  2674. return;
  2675. err:
  2676. if (wpa_s->dbus_groupobj_path) {
  2677. os_free(wpa_s->dbus_groupobj_path);
  2678. wpa_s->dbus_groupobj_path = NULL;
  2679. }
  2680. free_dbus_object_desc(obj_desc);
  2681. }
  2682. /**
  2683. * wpas_dbus_unregister_p2p_group - Unregister a p2p group object from dbus
  2684. * @wpa_s: wpa_supplicant interface structure
  2685. * @ssid: network name of the p2p group started
  2686. */
  2687. void wpas_dbus_unregister_p2p_group(struct wpa_supplicant *wpa_s,
  2688. const struct wpa_ssid *ssid)
  2689. {
  2690. struct wpas_dbus_priv *ctrl_iface;
  2691. /* Do nothing if the control interface is not turned on */
  2692. if (wpa_s == NULL || wpa_s->global == NULL)
  2693. return;
  2694. ctrl_iface = wpa_s->global->dbus;
  2695. if (ctrl_iface == NULL)
  2696. return;
  2697. if (!wpa_s->dbus_groupobj_path) {
  2698. wpa_printf(MSG_DEBUG,
  2699. "%s: Group object '%s' already unregistered",
  2700. __func__, wpa_s->dbus_groupobj_path);
  2701. return;
  2702. }
  2703. wpa_printf(MSG_DEBUG, "dbus: Unregister group object '%s'",
  2704. wpa_s->dbus_groupobj_path);
  2705. wpa_dbus_unregister_object_per_iface(ctrl_iface,
  2706. wpa_s->dbus_groupobj_path);
  2707. os_free(wpa_s->dbus_groupobj_path);
  2708. wpa_s->dbus_groupobj_path = NULL;
  2709. }
  2710. static const struct wpa_dbus_property_desc
  2711. wpas_dbus_p2p_groupmember_properties[] = {
  2712. { "Properties", WPAS_DBUS_NEW_IFACE_P2P_GROUPMEMBER, "a{sv}",
  2713. wpas_dbus_getter_p2p_group_properties,
  2714. NULL
  2715. },
  2716. { NULL, NULL, NULL, NULL, NULL }
  2717. };
  2718. /**
  2719. * wpas_dbus_register_p2p_groupmember - Register a p2p groupmember
  2720. * object with dbus
  2721. * @wpa_s: wpa_supplicant interface structure
  2722. * @p2p_if_addr: i/f addr of the device joining this group
  2723. *
  2724. * Registers p2p groupmember representing object with dbus
  2725. */
  2726. void wpas_dbus_register_p2p_groupmember(struct wpa_supplicant *wpa_s,
  2727. const u8 *p2p_if_addr)
  2728. {
  2729. struct wpas_dbus_priv *ctrl_iface;
  2730. struct wpa_dbus_object_desc *obj_desc = NULL;
  2731. struct groupmember_handler_args *arg;
  2732. char groupmember_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  2733. /* Do nothing if the control interface is not turned on */
  2734. if (wpa_s == NULL || wpa_s->global == NULL)
  2735. return;
  2736. ctrl_iface = wpa_s->global->dbus;
  2737. if (ctrl_iface == NULL)
  2738. return;
  2739. if (!wpa_s->dbus_groupobj_path)
  2740. return;
  2741. os_snprintf(groupmember_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  2742. "%s/" WPAS_DBUS_NEW_P2P_GROUPMEMBERS_PART "/" COMPACT_MACSTR,
  2743. wpa_s->dbus_groupobj_path, MAC2STR(p2p_if_addr));
  2744. obj_desc = os_zalloc(sizeof(struct wpa_dbus_object_desc));
  2745. if (!obj_desc) {
  2746. wpa_printf(MSG_ERROR, "Not enough memory "
  2747. "to create object description");
  2748. goto err;
  2749. }
  2750. /* allocate memory for handlers arguments */
  2751. arg = os_zalloc(sizeof(struct groupmember_handler_args));
  2752. if (!arg) {
  2753. wpa_printf(MSG_ERROR, "Not enough memory "
  2754. "to create arguments for method");
  2755. goto err;
  2756. }
  2757. arg->wpa_s = wpa_s;
  2758. os_memcpy(arg->member_addr, p2p_if_addr, ETH_ALEN);
  2759. wpas_dbus_register(obj_desc, arg, wpa_dbus_free, NULL,
  2760. wpas_dbus_p2p_groupmember_properties, NULL);
  2761. if (wpa_dbus_register_object_per_iface(ctrl_iface, groupmember_obj_path,
  2762. wpa_s->ifname, obj_desc))
  2763. goto err;
  2764. wpa_printf(MSG_INFO,
  2765. "dbus: Registered group member object '%s' successfully",
  2766. groupmember_obj_path);
  2767. return;
  2768. err:
  2769. free_dbus_object_desc(obj_desc);
  2770. }
  2771. /**
  2772. * wpas_dbus_unregister_p2p_groupmember - Unregister a p2p groupmember
  2773. * object with dbus
  2774. * @wpa_s: wpa_supplicant interface structure
  2775. * @p2p_if_addr: i/f addr of the device joining this group
  2776. *
  2777. * Unregisters p2p groupmember representing object with dbus
  2778. */
  2779. void wpas_dbus_unregister_p2p_groupmember(struct wpa_supplicant *wpa_s,
  2780. const u8 *p2p_if_addr)
  2781. {
  2782. struct wpas_dbus_priv *ctrl_iface;
  2783. char groupmember_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  2784. /* Do nothing if the control interface is not turned on */
  2785. if (wpa_s == NULL || wpa_s->global == NULL)
  2786. return;
  2787. ctrl_iface = wpa_s->global->dbus;
  2788. if (ctrl_iface == NULL)
  2789. return;
  2790. if (!wpa_s->dbus_groupobj_path)
  2791. return;
  2792. os_snprintf(groupmember_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  2793. "%s/" WPAS_DBUS_NEW_P2P_GROUPMEMBERS_PART "/" COMPACT_MACSTR,
  2794. wpa_s->dbus_groupobj_path, MAC2STR(p2p_if_addr));
  2795. wpa_dbus_unregister_object_per_iface(ctrl_iface, groupmember_obj_path);
  2796. }
  2797. static const struct wpa_dbus_property_desc
  2798. wpas_dbus_persistent_group_properties[] = {
  2799. { "Properties", WPAS_DBUS_NEW_IFACE_PERSISTENT_GROUP, "a{sv}",
  2800. wpas_dbus_getter_persistent_group_properties,
  2801. wpas_dbus_setter_persistent_group_properties
  2802. },
  2803. { NULL, NULL, NULL, NULL, NULL }
  2804. };
  2805. /* No signals intended for persistent group objects */
  2806. /**
  2807. * wpas_dbus_register_persistent_group - Register a configured(saved)
  2808. * persistent group with dbus
  2809. * @wpa_s: wpa_supplicant interface structure
  2810. * @ssid: persistent group (still represented as a network within wpa)
  2811. * configuration data
  2812. * Returns: 0 on success, -1 on failure
  2813. *
  2814. * Registers a persistent group representing object with dbus.
  2815. */
  2816. int wpas_dbus_register_persistent_group(struct wpa_supplicant *wpa_s,
  2817. struct wpa_ssid *ssid)
  2818. {
  2819. struct wpas_dbus_priv *ctrl_iface;
  2820. struct wpa_dbus_object_desc *obj_desc;
  2821. struct network_handler_args *arg;
  2822. char pgrp_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  2823. /* Do nothing if the control interface is not turned on */
  2824. if (wpa_s == NULL || wpa_s->global == NULL)
  2825. return 0;
  2826. /* Make sure ssid is a persistent group */
  2827. if (ssid->disabled != 2 && !ssid->p2p_persistent_group)
  2828. return -1; /* should we return w/o complaining? */
  2829. ctrl_iface = wpa_s->global->dbus;
  2830. if (ctrl_iface == NULL)
  2831. return 0;
  2832. /*
  2833. * Intentionally not coming up with different numbering scheme
  2834. * for persistent groups.
  2835. */
  2836. os_snprintf(pgrp_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  2837. "%s/" WPAS_DBUS_NEW_PERSISTENT_GROUPS_PART "/%u",
  2838. wpa_s->dbus_new_path, ssid->id);
  2839. wpa_printf(MSG_DEBUG, "dbus: Register persistent group object '%s'",
  2840. pgrp_obj_path);
  2841. obj_desc = os_zalloc(sizeof(struct wpa_dbus_object_desc));
  2842. if (!obj_desc) {
  2843. wpa_printf(MSG_ERROR, "dbus: Not enough memory to create "
  2844. "object description");
  2845. goto err;
  2846. }
  2847. /*
  2848. * Reusing the same context structure as that for networks
  2849. * since these are represented using same data structure.
  2850. */
  2851. /* allocate memory for handlers arguments */
  2852. arg = os_zalloc(sizeof(struct network_handler_args));
  2853. if (!arg) {
  2854. wpa_printf(MSG_ERROR, "dbus: Not enough memory to create "
  2855. "arguments for method");
  2856. goto err;
  2857. }
  2858. arg->wpa_s = wpa_s;
  2859. arg->ssid = ssid;
  2860. wpas_dbus_register(obj_desc, arg, wpa_dbus_free, NULL,
  2861. wpas_dbus_persistent_group_properties,
  2862. NULL);
  2863. if (wpa_dbus_register_object_per_iface(ctrl_iface, pgrp_obj_path,
  2864. wpa_s->ifname, obj_desc))
  2865. goto err;
  2866. wpas_dbus_signal_persistent_group_added(wpa_s, ssid->id);
  2867. return 0;
  2868. err:
  2869. free_dbus_object_desc(obj_desc);
  2870. return -1;
  2871. }
  2872. /**
  2873. * wpas_dbus_unregister_persistent_group - Unregister a persistent_group
  2874. * from dbus
  2875. * @wpa_s: wpa_supplicant interface structure
  2876. * @nid: network id
  2877. * Returns: 0 on success, -1 on failure
  2878. *
  2879. * Unregisters persistent group representing object from dbus
  2880. *
  2881. * NOTE: There is a slight issue with the semantics here. While the
  2882. * implementation simply means the persistent group is unloaded from memory,
  2883. * it should not get interpreted as the group is actually being erased/removed
  2884. * from persistent storage as well.
  2885. */
  2886. int wpas_dbus_unregister_persistent_group(struct wpa_supplicant *wpa_s,
  2887. int nid)
  2888. {
  2889. struct wpas_dbus_priv *ctrl_iface;
  2890. char pgrp_obj_path[WPAS_DBUS_OBJECT_PATH_MAX];
  2891. int ret;
  2892. /* Do nothing if the control interface is not turned on */
  2893. if (wpa_s == NULL || wpa_s->global == NULL ||
  2894. wpa_s->dbus_new_path == NULL)
  2895. return 0;
  2896. ctrl_iface = wpa_s->global->dbus;
  2897. if (ctrl_iface == NULL)
  2898. return 0;
  2899. os_snprintf(pgrp_obj_path, WPAS_DBUS_OBJECT_PATH_MAX,
  2900. "%s/" WPAS_DBUS_NEW_PERSISTENT_GROUPS_PART "/%u",
  2901. wpa_s->dbus_new_path, nid);
  2902. wpa_printf(MSG_DEBUG, "dbus: Unregister persistent group object '%s'",
  2903. pgrp_obj_path);
  2904. ret = wpa_dbus_unregister_object_per_iface(ctrl_iface, pgrp_obj_path);
  2905. if (!ret)
  2906. wpas_dbus_signal_persistent_group_removed(wpa_s, nid);
  2907. return ret;
  2908. }
  2909. #endif /* CONFIG_P2P */