| [15f3c3f] | 1 | /* | 
|---|
|  | 2 | * Copyright (c) 2007 Josef Cejka | 
|---|
| [cc574511] | 3 | * Copyright (c) 2011 Jiri Svoboda | 
|---|
| [15f3c3f] | 4 | * All rights reserved. | 
|---|
|  | 5 | * | 
|---|
|  | 6 | * Redistribution and use in source and binary forms, with or without | 
|---|
|  | 7 | * modification, are permitted provided that the following conditions | 
|---|
|  | 8 | * are met: | 
|---|
|  | 9 | * | 
|---|
|  | 10 | * - Redistributions of source code must retain the above copyright | 
|---|
|  | 11 | *   notice, this list of conditions and the following disclaimer. | 
|---|
|  | 12 | * - Redistributions in binary form must reproduce the above copyright | 
|---|
|  | 13 | *   notice, this list of conditions and the following disclaimer in the | 
|---|
|  | 14 | *   documentation and/or other materials provided with the distribution. | 
|---|
|  | 15 | * - The name of the author may not be used to endorse or promote products | 
|---|
|  | 16 | *   derived from this software without specific prior written permission. | 
|---|
|  | 17 | * | 
|---|
|  | 18 | * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR | 
|---|
|  | 19 | * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES | 
|---|
|  | 20 | * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. | 
|---|
|  | 21 | * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, | 
|---|
|  | 22 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT | 
|---|
|  | 23 | * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, | 
|---|
|  | 24 | * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY | 
|---|
|  | 25 | * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
|---|
|  | 26 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | 
|---|
|  | 27 | * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|---|
|  | 28 | */ | 
|---|
|  | 29 |  | 
|---|
|  | 30 | #include <str.h> | 
|---|
|  | 31 | #include <ipc/services.h> | 
|---|
|  | 32 | #include <ns.h> | 
|---|
|  | 33 | #include <ipc/loc.h> | 
|---|
|  | 34 | #include <loc.h> | 
|---|
|  | 35 | #include <fibril_synch.h> | 
|---|
|  | 36 | #include <async.h> | 
|---|
|  | 37 | #include <errno.h> | 
|---|
|  | 38 | #include <malloc.h> | 
|---|
| [3e6a98c5] | 39 | #include <stdbool.h> | 
|---|
| [15f3c3f] | 40 |  | 
|---|
|  | 41 | static FIBRIL_MUTEX_INITIALIZE(loc_supp_block_mutex); | 
|---|
|  | 42 | static FIBRIL_MUTEX_INITIALIZE(loc_cons_block_mutex); | 
|---|
|  | 43 |  | 
|---|
|  | 44 | static FIBRIL_MUTEX_INITIALIZE(loc_supplier_mutex); | 
|---|
|  | 45 | static FIBRIL_MUTEX_INITIALIZE(loc_consumer_mutex); | 
|---|
|  | 46 |  | 
|---|
| [12f9f0d0] | 47 | static FIBRIL_MUTEX_INITIALIZE(loc_callback_mutex); | 
|---|
|  | 48 | static bool loc_callback_created = false; | 
|---|
| [a6240a31] | 49 | static loc_cat_change_cb_t cat_change_cb = NULL; | 
|---|
| [12f9f0d0] | 50 |  | 
|---|
| [15f3c3f] | 51 | static async_sess_t *loc_supp_block_sess = NULL; | 
|---|
|  | 52 | static async_sess_t *loc_cons_block_sess = NULL; | 
|---|
|  | 53 |  | 
|---|
|  | 54 | static async_sess_t *loc_supplier_sess = NULL; | 
|---|
|  | 55 | static async_sess_t *loc_consumer_sess = NULL; | 
|---|
|  | 56 |  | 
|---|
| [12f9f0d0] | 57 | static void loc_cb_conn(ipc_callid_t iid, ipc_call_t *icall, void *arg) | 
|---|
|  | 58 | { | 
|---|
|  | 59 | while (true) { | 
|---|
|  | 60 | ipc_call_t call; | 
|---|
|  | 61 | ipc_callid_t callid = async_get_call(&call); | 
|---|
|  | 62 |  | 
|---|
|  | 63 | if (!IPC_GET_IMETHOD(call)) { | 
|---|
|  | 64 | /* TODO: Handle hangup */ | 
|---|
|  | 65 | return; | 
|---|
|  | 66 | } | 
|---|
|  | 67 |  | 
|---|
|  | 68 | switch (IPC_GET_IMETHOD(call)) { | 
|---|
|  | 69 | case LOC_EVENT_CAT_CHANGE: | 
|---|
|  | 70 | fibril_mutex_lock(&loc_callback_mutex); | 
|---|
| [a6240a31] | 71 | loc_cat_change_cb_t cb_fun = cat_change_cb; | 
|---|
| [12f9f0d0] | 72 | fibril_mutex_unlock(&loc_callback_mutex); | 
|---|
| [a6240a31] | 73 |  | 
|---|
|  | 74 | async_answer_0(callid, EOK); | 
|---|
|  | 75 |  | 
|---|
|  | 76 | if (cb_fun != NULL) | 
|---|
|  | 77 | (*cb_fun)(); | 
|---|
|  | 78 |  | 
|---|
| [12f9f0d0] | 79 | break; | 
|---|
|  | 80 | default: | 
|---|
| [a6240a31] | 81 | async_answer_0(callid, ENOTSUP); | 
|---|
| [12f9f0d0] | 82 | } | 
|---|
|  | 83 | } | 
|---|
|  | 84 | } | 
|---|
|  | 85 |  | 
|---|
|  | 86 |  | 
|---|
| [15f3c3f] | 87 | static void clone_session(fibril_mutex_t *mtx, async_sess_t *src, | 
|---|
|  | 88 | async_sess_t **dst) | 
|---|
|  | 89 | { | 
|---|
|  | 90 | fibril_mutex_lock(mtx); | 
|---|
|  | 91 |  | 
|---|
|  | 92 | if ((*dst == NULL) && (src != NULL)) | 
|---|
|  | 93 | *dst = src; | 
|---|
|  | 94 |  | 
|---|
|  | 95 | fibril_mutex_unlock(mtx); | 
|---|
|  | 96 | } | 
|---|
|  | 97 |  | 
|---|
| [a6240a31] | 98 | /** Create callback | 
|---|
|  | 99 | * | 
|---|
|  | 100 | * Must be called with loc_callback_mutex locked. | 
|---|
|  | 101 | * | 
|---|
|  | 102 | * @return EOK on success. | 
|---|
|  | 103 | * | 
|---|
|  | 104 | */ | 
|---|
| [12f9f0d0] | 105 | static int loc_callback_create(void) | 
|---|
|  | 106 | { | 
|---|
|  | 107 | if (!loc_callback_created) { | 
|---|
| [a6240a31] | 108 | async_exch_t *exch = | 
|---|
|  | 109 | loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
| [12f9f0d0] | 110 |  | 
|---|
|  | 111 | ipc_call_t answer; | 
|---|
|  | 112 | aid_t req = async_send_0(exch, LOC_CALLBACK_CREATE, &answer); | 
|---|
| [cd66f3c] | 113 | int rc = async_connect_to_me(exch, 0, 0, 0, loc_cb_conn, NULL); | 
|---|
| [12f9f0d0] | 114 | loc_exchange_end(exch); | 
|---|
|  | 115 |  | 
|---|
|  | 116 | if (rc != EOK) | 
|---|
| [a6240a31] | 117 | return rc; | 
|---|
| [12f9f0d0] | 118 |  | 
|---|
| [cd66f3c] | 119 | sysarg_t retval; | 
|---|
|  | 120 | async_wait_for(req, &retval); | 
|---|
|  | 121 | if (retval != EOK) | 
|---|
|  | 122 | return retval; | 
|---|
| [12f9f0d0] | 123 |  | 
|---|
|  | 124 | loc_callback_created = true; | 
|---|
|  | 125 | } | 
|---|
|  | 126 |  | 
|---|
| [a6240a31] | 127 | return EOK; | 
|---|
| [12f9f0d0] | 128 | } | 
|---|
|  | 129 |  | 
|---|
| [15f3c3f] | 130 | /** Start an async exchange on the loc session (blocking). | 
|---|
|  | 131 | * | 
|---|
|  | 132 | * @param iface Location service interface to choose | 
|---|
|  | 133 | * | 
|---|
|  | 134 | * @return New exchange. | 
|---|
|  | 135 | * | 
|---|
|  | 136 | */ | 
|---|
|  | 137 | async_exch_t *loc_exchange_begin_blocking(loc_interface_t iface) | 
|---|
|  | 138 | { | 
|---|
|  | 139 | switch (iface) { | 
|---|
|  | 140 | case LOC_PORT_SUPPLIER: | 
|---|
|  | 141 | fibril_mutex_lock(&loc_supp_block_mutex); | 
|---|
|  | 142 |  | 
|---|
|  | 143 | while (loc_supp_block_sess == NULL) { | 
|---|
|  | 144 | clone_session(&loc_supplier_mutex, loc_supplier_sess, | 
|---|
|  | 145 | &loc_supp_block_sess); | 
|---|
|  | 146 |  | 
|---|
|  | 147 | if (loc_supp_block_sess == NULL) | 
|---|
|  | 148 | loc_supp_block_sess = | 
|---|
|  | 149 | service_connect_blocking(EXCHANGE_SERIALIZE, | 
|---|
|  | 150 | SERVICE_LOC, LOC_PORT_SUPPLIER, 0); | 
|---|
|  | 151 | } | 
|---|
|  | 152 |  | 
|---|
|  | 153 | fibril_mutex_unlock(&loc_supp_block_mutex); | 
|---|
|  | 154 |  | 
|---|
|  | 155 | clone_session(&loc_supplier_mutex, loc_supp_block_sess, | 
|---|
|  | 156 | &loc_supplier_sess); | 
|---|
|  | 157 |  | 
|---|
|  | 158 | return async_exchange_begin(loc_supp_block_sess); | 
|---|
|  | 159 | case LOC_PORT_CONSUMER: | 
|---|
|  | 160 | fibril_mutex_lock(&loc_cons_block_mutex); | 
|---|
|  | 161 |  | 
|---|
|  | 162 | while (loc_cons_block_sess == NULL) { | 
|---|
|  | 163 | clone_session(&loc_consumer_mutex, loc_consumer_sess, | 
|---|
|  | 164 | &loc_cons_block_sess); | 
|---|
|  | 165 |  | 
|---|
|  | 166 | if (loc_cons_block_sess == NULL) | 
|---|
|  | 167 | loc_cons_block_sess = | 
|---|
|  | 168 | service_connect_blocking(EXCHANGE_SERIALIZE, | 
|---|
|  | 169 | SERVICE_LOC, LOC_PORT_CONSUMER, 0); | 
|---|
|  | 170 | } | 
|---|
|  | 171 |  | 
|---|
|  | 172 | fibril_mutex_unlock(&loc_cons_block_mutex); | 
|---|
|  | 173 |  | 
|---|
|  | 174 | clone_session(&loc_consumer_mutex, loc_cons_block_sess, | 
|---|
|  | 175 | &loc_consumer_sess); | 
|---|
|  | 176 |  | 
|---|
|  | 177 | return async_exchange_begin(loc_cons_block_sess); | 
|---|
|  | 178 | default: | 
|---|
|  | 179 | return NULL; | 
|---|
|  | 180 | } | 
|---|
|  | 181 | } | 
|---|
|  | 182 |  | 
|---|
|  | 183 | /** Start an async exchange on the loc session. | 
|---|
|  | 184 | * | 
|---|
|  | 185 | * @param iface Location service interface to choose | 
|---|
|  | 186 | * | 
|---|
|  | 187 | * @return New exchange. | 
|---|
|  | 188 | * | 
|---|
|  | 189 | */ | 
|---|
|  | 190 | async_exch_t *loc_exchange_begin(loc_interface_t iface) | 
|---|
|  | 191 | { | 
|---|
|  | 192 | switch (iface) { | 
|---|
|  | 193 | case LOC_PORT_SUPPLIER: | 
|---|
|  | 194 | fibril_mutex_lock(&loc_supplier_mutex); | 
|---|
|  | 195 |  | 
|---|
|  | 196 | if (loc_supplier_sess == NULL) | 
|---|
|  | 197 | loc_supplier_sess = | 
|---|
|  | 198 | service_connect(EXCHANGE_SERIALIZE, SERVICE_LOC, | 
|---|
|  | 199 | LOC_PORT_SUPPLIER, 0); | 
|---|
|  | 200 |  | 
|---|
|  | 201 | fibril_mutex_unlock(&loc_supplier_mutex); | 
|---|
|  | 202 |  | 
|---|
|  | 203 | if (loc_supplier_sess == NULL) | 
|---|
|  | 204 | return NULL; | 
|---|
|  | 205 |  | 
|---|
|  | 206 | return async_exchange_begin(loc_supplier_sess); | 
|---|
|  | 207 | case LOC_PORT_CONSUMER: | 
|---|
|  | 208 | fibril_mutex_lock(&loc_consumer_mutex); | 
|---|
|  | 209 |  | 
|---|
|  | 210 | if (loc_consumer_sess == NULL) | 
|---|
|  | 211 | loc_consumer_sess = | 
|---|
|  | 212 | service_connect(EXCHANGE_SERIALIZE, SERVICE_LOC, | 
|---|
|  | 213 | LOC_PORT_CONSUMER, 0); | 
|---|
|  | 214 |  | 
|---|
|  | 215 | fibril_mutex_unlock(&loc_consumer_mutex); | 
|---|
|  | 216 |  | 
|---|
|  | 217 | if (loc_consumer_sess == NULL) | 
|---|
|  | 218 | return NULL; | 
|---|
|  | 219 |  | 
|---|
|  | 220 | return async_exchange_begin(loc_consumer_sess); | 
|---|
|  | 221 | default: | 
|---|
|  | 222 | return NULL; | 
|---|
|  | 223 | } | 
|---|
|  | 224 | } | 
|---|
|  | 225 |  | 
|---|
|  | 226 | /** Finish an async exchange on the loc session. | 
|---|
|  | 227 | * | 
|---|
|  | 228 | * @param exch Exchange to be finished. | 
|---|
|  | 229 | * | 
|---|
|  | 230 | */ | 
|---|
|  | 231 | void loc_exchange_end(async_exch_t *exch) | 
|---|
|  | 232 | { | 
|---|
|  | 233 | async_exchange_end(exch); | 
|---|
|  | 234 | } | 
|---|
|  | 235 |  | 
|---|
|  | 236 | /** Register new driver with loc. */ | 
|---|
| [f302586] | 237 | int loc_server_register(const char *name) | 
|---|
| [15f3c3f] | 238 | { | 
|---|
|  | 239 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_SUPPLIER); | 
|---|
|  | 240 |  | 
|---|
|  | 241 | ipc_call_t answer; | 
|---|
|  | 242 | aid_t req = async_send_2(exch, LOC_SERVER_REGISTER, 0, 0, &answer); | 
|---|
|  | 243 | sysarg_t retval = async_data_write_start(exch, name, str_size(name)); | 
|---|
|  | 244 |  | 
|---|
|  | 245 | loc_exchange_end(exch); | 
|---|
|  | 246 |  | 
|---|
|  | 247 | if (retval != EOK) { | 
|---|
| [50b581d] | 248 | async_forget(req); | 
|---|
| [15f3c3f] | 249 | return retval; | 
|---|
|  | 250 | } | 
|---|
|  | 251 |  | 
|---|
|  | 252 | exch = loc_exchange_begin(LOC_PORT_SUPPLIER); | 
|---|
|  | 253 | async_connect_to_me(exch, 0, 0, 0, NULL, NULL); | 
|---|
|  | 254 | loc_exchange_end(exch); | 
|---|
|  | 255 |  | 
|---|
|  | 256 | async_wait_for(req, &retval); | 
|---|
|  | 257 | return retval; | 
|---|
|  | 258 | } | 
|---|
|  | 259 |  | 
|---|
| [d0dd7b5] | 260 | /** Register new service. | 
|---|
| [15f3c3f] | 261 | * | 
|---|
| [d0dd7b5] | 262 | * The @p interface is used when forwarding connection to the server. | 
|---|
| [15f3c3f] | 263 | * If not 0, the first argument is the interface and the second argument | 
|---|
|  | 264 | * is the service ID. | 
|---|
|  | 265 | * | 
|---|
|  | 266 | * When the interface is zero (default), the first argument is directly | 
|---|
|  | 267 | * the handle (to ensure backward compatibility). | 
|---|
|  | 268 | * | 
|---|
| [d0dd7b5] | 269 | * @param      fqsn      Fully qualified service name | 
|---|
|  | 270 | * @param[out] sid       Service ID of new service | 
|---|
|  | 271 | * @param      interface Interface when forwarding | 
|---|
| [15f3c3f] | 272 | * | 
|---|
|  | 273 | */ | 
|---|
| [d0dd7b5] | 274 | int loc_service_register_with_iface(const char *fqsn, | 
|---|
|  | 275 | service_id_t *sid, sysarg_t interface) | 
|---|
| [15f3c3f] | 276 | { | 
|---|
|  | 277 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_SUPPLIER); | 
|---|
|  | 278 |  | 
|---|
|  | 279 | ipc_call_t answer; | 
|---|
|  | 280 | aid_t req = async_send_2(exch, LOC_SERVICE_REGISTER, interface, 0, | 
|---|
|  | 281 | &answer); | 
|---|
| [d0dd7b5] | 282 | sysarg_t retval = async_data_write_start(exch, fqsn, str_size(fqsn)); | 
|---|
| [15f3c3f] | 283 |  | 
|---|
|  | 284 | loc_exchange_end(exch); | 
|---|
|  | 285 |  | 
|---|
|  | 286 | if (retval != EOK) { | 
|---|
| [50b581d] | 287 | async_forget(req); | 
|---|
| [15f3c3f] | 288 | return retval; | 
|---|
|  | 289 | } | 
|---|
|  | 290 |  | 
|---|
|  | 291 | async_wait_for(req, &retval); | 
|---|
|  | 292 |  | 
|---|
|  | 293 | if (retval != EOK) { | 
|---|
| [d0dd7b5] | 294 | if (sid != NULL) | 
|---|
|  | 295 | *sid = -1; | 
|---|
| [15f3c3f] | 296 |  | 
|---|
|  | 297 | return retval; | 
|---|
|  | 298 | } | 
|---|
|  | 299 |  | 
|---|
| [d0dd7b5] | 300 | if (sid != NULL) | 
|---|
|  | 301 | *sid = (service_id_t) IPC_GET_ARG1(answer); | 
|---|
| [15f3c3f] | 302 |  | 
|---|
|  | 303 | return retval; | 
|---|
|  | 304 | } | 
|---|
|  | 305 |  | 
|---|
| [d0dd7b5] | 306 | /** Register new service. | 
|---|
| [15f3c3f] | 307 | * | 
|---|
| [0fe52ef] | 308 | * @param fqsn Fully qualified service name | 
|---|
|  | 309 | * @param sid  Output: ID of new service | 
|---|
| [15f3c3f] | 310 | * | 
|---|
|  | 311 | */ | 
|---|
| [d0dd7b5] | 312 | int loc_service_register(const char *fqdn, service_id_t *sid) | 
|---|
| [15f3c3f] | 313 | { | 
|---|
| [d0dd7b5] | 314 | return loc_service_register_with_iface(fqdn, sid, 0); | 
|---|
|  | 315 | } | 
|---|
|  | 316 |  | 
|---|
|  | 317 | /** Unregister service. | 
|---|
|  | 318 | * | 
|---|
|  | 319 | * @param sid   Service ID | 
|---|
|  | 320 | */ | 
|---|
|  | 321 | int loc_service_unregister(service_id_t sid) | 
|---|
|  | 322 | { | 
|---|
|  | 323 | async_exch_t *exch; | 
|---|
|  | 324 | sysarg_t retval; | 
|---|
|  | 325 |  | 
|---|
|  | 326 | exch = loc_exchange_begin_blocking(LOC_PORT_SUPPLIER); | 
|---|
|  | 327 | retval = async_req_1_0(exch, LOC_SERVICE_UNREGISTER, sid); | 
|---|
|  | 328 | loc_exchange_end(exch); | 
|---|
|  | 329 |  | 
|---|
|  | 330 | return (int)retval; | 
|---|
| [15f3c3f] | 331 | } | 
|---|
|  | 332 |  | 
|---|
|  | 333 | int loc_service_get_id(const char *fqdn, service_id_t *handle, | 
|---|
|  | 334 | unsigned int flags) | 
|---|
|  | 335 | { | 
|---|
|  | 336 | async_exch_t *exch; | 
|---|
|  | 337 |  | 
|---|
|  | 338 | if (flags & IPC_FLAG_BLOCKING) | 
|---|
|  | 339 | exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 340 | else { | 
|---|
|  | 341 | exch = loc_exchange_begin(LOC_PORT_CONSUMER); | 
|---|
|  | 342 | if (exch == NULL) | 
|---|
|  | 343 | return errno; | 
|---|
|  | 344 | } | 
|---|
|  | 345 |  | 
|---|
|  | 346 | ipc_call_t answer; | 
|---|
|  | 347 | aid_t req = async_send_2(exch, LOC_SERVICE_GET_ID, flags, 0, | 
|---|
|  | 348 | &answer); | 
|---|
|  | 349 | sysarg_t retval = async_data_write_start(exch, fqdn, str_size(fqdn)); | 
|---|
|  | 350 |  | 
|---|
|  | 351 | loc_exchange_end(exch); | 
|---|
|  | 352 |  | 
|---|
|  | 353 | if (retval != EOK) { | 
|---|
| [50b581d] | 354 | async_forget(req); | 
|---|
| [15f3c3f] | 355 | return retval; | 
|---|
|  | 356 | } | 
|---|
|  | 357 |  | 
|---|
|  | 358 | async_wait_for(req, &retval); | 
|---|
|  | 359 |  | 
|---|
|  | 360 | if (retval != EOK) { | 
|---|
|  | 361 | if (handle != NULL) | 
|---|
|  | 362 | *handle = (service_id_t) -1; | 
|---|
|  | 363 |  | 
|---|
|  | 364 | return retval; | 
|---|
|  | 365 | } | 
|---|
|  | 366 |  | 
|---|
|  | 367 | if (handle != NULL) | 
|---|
|  | 368 | *handle = (service_id_t) IPC_GET_ARG1(answer); | 
|---|
|  | 369 |  | 
|---|
|  | 370 | return retval; | 
|---|
|  | 371 | } | 
|---|
|  | 372 |  | 
|---|
| [763e0cd] | 373 | /** Get object name. | 
|---|
| [cce8a83] | 374 | * | 
|---|
| [763e0cd] | 375 | * Provided ID of an object, return its name. | 
|---|
| [cce8a83] | 376 | * | 
|---|
| [763e0cd] | 377 | * @param method        IPC method | 
|---|
|  | 378 | * @param id            Object ID | 
|---|
| [cce8a83] | 379 | * @param name          Place to store pointer to new string. Caller should | 
|---|
|  | 380 | *                      free it using free(). | 
|---|
|  | 381 | * @return              EOK on success or negative error code | 
|---|
|  | 382 | */ | 
|---|
| [763e0cd] | 383 | static int loc_get_name_internal(sysarg_t method, sysarg_t id, char **name) | 
|---|
| [cce8a83] | 384 | { | 
|---|
|  | 385 | async_exch_t *exch; | 
|---|
|  | 386 | char name_buf[LOC_NAME_MAXLEN + 1]; | 
|---|
| [45058baa] | 387 | ipc_call_t dreply; | 
|---|
|  | 388 | size_t act_size; | 
|---|
|  | 389 | sysarg_t dretval; | 
|---|
| [cce8a83] | 390 |  | 
|---|
|  | 391 | *name = NULL; | 
|---|
|  | 392 | exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 393 |  | 
|---|
|  | 394 | ipc_call_t answer; | 
|---|
| [763e0cd] | 395 | aid_t req = async_send_1(exch, method, id, &answer); | 
|---|
| [45058baa] | 396 | aid_t dreq = async_data_read(exch, name_buf, LOC_NAME_MAXLEN, | 
|---|
|  | 397 | &dreply); | 
|---|
|  | 398 | async_wait_for(dreq, &dretval); | 
|---|
| [cce8a83] | 399 |  | 
|---|
|  | 400 | loc_exchange_end(exch); | 
|---|
|  | 401 |  | 
|---|
| [45058baa] | 402 | if (dretval != EOK) { | 
|---|
| [50b581d] | 403 | async_forget(req); | 
|---|
| [45058baa] | 404 | return dretval; | 
|---|
| [cce8a83] | 405 | } | 
|---|
|  | 406 |  | 
|---|
|  | 407 | sysarg_t retval; | 
|---|
|  | 408 | async_wait_for(req, &retval); | 
|---|
|  | 409 |  | 
|---|
|  | 410 | if (retval != EOK) | 
|---|
|  | 411 | return retval; | 
|---|
|  | 412 |  | 
|---|
| [45058baa] | 413 | act_size = IPC_GET_ARG2(dreply); | 
|---|
|  | 414 | assert(act_size <= LOC_NAME_MAXLEN); | 
|---|
|  | 415 | name_buf[act_size] = '\0'; | 
|---|
|  | 416 |  | 
|---|
| [cce8a83] | 417 | *name = str_dup(name_buf); | 
|---|
|  | 418 | if (*name == NULL) | 
|---|
|  | 419 | return ENOMEM; | 
|---|
|  | 420 |  | 
|---|
|  | 421 | return EOK; | 
|---|
|  | 422 | } | 
|---|
|  | 423 |  | 
|---|
| [763e0cd] | 424 | /** Get category name. | 
|---|
|  | 425 | * | 
|---|
|  | 426 | * Provided ID of a service, return its name. | 
|---|
|  | 427 | * | 
|---|
|  | 428 | * @param cat_id        Category ID | 
|---|
|  | 429 | * @param name          Place to store pointer to new string. Caller should | 
|---|
|  | 430 | *                      free it using free(). | 
|---|
|  | 431 | * @return              EOK on success or negative error code | 
|---|
|  | 432 | */ | 
|---|
|  | 433 | int loc_category_get_name(category_id_t cat_id, char **name) | 
|---|
|  | 434 | { | 
|---|
|  | 435 | return loc_get_name_internal(LOC_CATEGORY_GET_NAME, cat_id, name); | 
|---|
|  | 436 | } | 
|---|
|  | 437 |  | 
|---|
|  | 438 | /** Get service name. | 
|---|
|  | 439 | * | 
|---|
|  | 440 | * Provided ID of a service, return its name. | 
|---|
|  | 441 | * | 
|---|
|  | 442 | * @param svc_id        Service ID | 
|---|
|  | 443 | * @param name          Place to store pointer to new string. Caller should | 
|---|
|  | 444 | *                      free it using free(). | 
|---|
|  | 445 | * @return              EOK on success or negative error code | 
|---|
|  | 446 | */ | 
|---|
|  | 447 | int loc_service_get_name(service_id_t svc_id, char **name) | 
|---|
|  | 448 | { | 
|---|
|  | 449 | return loc_get_name_internal(LOC_SERVICE_GET_NAME, svc_id, name); | 
|---|
|  | 450 | } | 
|---|
| [cce8a83] | 451 |  | 
|---|
| [a3fcfba] | 452 | /** Get service server name. | 
|---|
|  | 453 | * | 
|---|
|  | 454 | * Provided ID of a service, return the name of its server. | 
|---|
|  | 455 | * | 
|---|
|  | 456 | * @param svc_id        Service ID | 
|---|
|  | 457 | * @param name          Place to store pointer to new string. Caller should | 
|---|
|  | 458 | *                      free it using free(). | 
|---|
|  | 459 | * @return              EOK on success or negative error code | 
|---|
|  | 460 | */ | 
|---|
|  | 461 | int loc_service_get_server_name(service_id_t svc_id, char **name) | 
|---|
|  | 462 | { | 
|---|
|  | 463 | return loc_get_name_internal(LOC_SERVICE_GET_SERVER_NAME, svc_id, name); | 
|---|
|  | 464 | } | 
|---|
|  | 465 |  | 
|---|
| [15f3c3f] | 466 | int loc_namespace_get_id(const char *name, service_id_t *handle, | 
|---|
|  | 467 | unsigned int flags) | 
|---|
|  | 468 | { | 
|---|
|  | 469 | async_exch_t *exch; | 
|---|
|  | 470 |  | 
|---|
|  | 471 | if (flags & IPC_FLAG_BLOCKING) | 
|---|
|  | 472 | exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 473 | else { | 
|---|
|  | 474 | exch = loc_exchange_begin(LOC_PORT_CONSUMER); | 
|---|
|  | 475 | if (exch == NULL) | 
|---|
|  | 476 | return errno; | 
|---|
|  | 477 | } | 
|---|
|  | 478 |  | 
|---|
|  | 479 | ipc_call_t answer; | 
|---|
|  | 480 | aid_t req = async_send_2(exch, LOC_NAMESPACE_GET_ID, flags, 0, | 
|---|
|  | 481 | &answer); | 
|---|
|  | 482 | sysarg_t retval = async_data_write_start(exch, name, str_size(name)); | 
|---|
|  | 483 |  | 
|---|
|  | 484 | loc_exchange_end(exch); | 
|---|
|  | 485 |  | 
|---|
|  | 486 | if (retval != EOK) { | 
|---|
| [50b581d] | 487 | async_forget(req); | 
|---|
| [15f3c3f] | 488 | return retval; | 
|---|
|  | 489 | } | 
|---|
|  | 490 |  | 
|---|
|  | 491 | async_wait_for(req, &retval); | 
|---|
|  | 492 |  | 
|---|
|  | 493 | if (retval != EOK) { | 
|---|
|  | 494 | if (handle != NULL) | 
|---|
|  | 495 | *handle = (service_id_t) -1; | 
|---|
|  | 496 |  | 
|---|
|  | 497 | return retval; | 
|---|
|  | 498 | } | 
|---|
|  | 499 |  | 
|---|
|  | 500 | if (handle != NULL) | 
|---|
|  | 501 | *handle = (service_id_t) IPC_GET_ARG1(answer); | 
|---|
|  | 502 |  | 
|---|
|  | 503 | return retval; | 
|---|
|  | 504 | } | 
|---|
|  | 505 |  | 
|---|
| [cc574511] | 506 | /** Get category ID. | 
|---|
|  | 507 | * | 
|---|
|  | 508 | * Provided name of a category, return its ID. | 
|---|
|  | 509 | * | 
|---|
|  | 510 | * @param name          Category name | 
|---|
|  | 511 | * @param cat_id        Place to store ID | 
|---|
|  | 512 | * @param flags         IPC_FLAG_BLOCKING to wait for location service to start | 
|---|
|  | 513 | * @return              EOK on success or negative error code | 
|---|
|  | 514 | */ | 
|---|
|  | 515 | int loc_category_get_id(const char *name, category_id_t *cat_id, | 
|---|
|  | 516 | unsigned int flags) | 
|---|
|  | 517 | { | 
|---|
|  | 518 | async_exch_t *exch; | 
|---|
|  | 519 |  | 
|---|
|  | 520 | if (flags & IPC_FLAG_BLOCKING) | 
|---|
|  | 521 | exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 522 | else { | 
|---|
|  | 523 | exch = loc_exchange_begin(LOC_PORT_CONSUMER); | 
|---|
|  | 524 | if (exch == NULL) | 
|---|
|  | 525 | return errno; | 
|---|
|  | 526 | } | 
|---|
|  | 527 |  | 
|---|
|  | 528 | ipc_call_t answer; | 
|---|
|  | 529 | aid_t req = async_send_0(exch, LOC_CATEGORY_GET_ID, | 
|---|
|  | 530 | &answer); | 
|---|
|  | 531 | sysarg_t retval = async_data_write_start(exch, name, str_size(name)); | 
|---|
|  | 532 |  | 
|---|
|  | 533 | loc_exchange_end(exch); | 
|---|
|  | 534 |  | 
|---|
|  | 535 | if (retval != EOK) { | 
|---|
| [50b581d] | 536 | async_forget(req); | 
|---|
| [cc574511] | 537 | return retval; | 
|---|
|  | 538 | } | 
|---|
|  | 539 |  | 
|---|
|  | 540 | async_wait_for(req, &retval); | 
|---|
|  | 541 |  | 
|---|
|  | 542 | if (retval != EOK) { | 
|---|
|  | 543 | if (cat_id != NULL) | 
|---|
|  | 544 | *cat_id = (category_id_t) -1; | 
|---|
|  | 545 |  | 
|---|
|  | 546 | return retval; | 
|---|
|  | 547 | } | 
|---|
|  | 548 |  | 
|---|
|  | 549 | if (cat_id != NULL) | 
|---|
|  | 550 | *cat_id = (category_id_t) IPC_GET_ARG1(answer); | 
|---|
|  | 551 |  | 
|---|
|  | 552 | return retval; | 
|---|
|  | 553 | } | 
|---|
|  | 554 |  | 
|---|
|  | 555 |  | 
|---|
| [15f3c3f] | 556 | loc_object_type_t loc_id_probe(service_id_t handle) | 
|---|
|  | 557 | { | 
|---|
|  | 558 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 559 |  | 
|---|
|  | 560 | sysarg_t type; | 
|---|
|  | 561 | int retval = async_req_1_1(exch, LOC_ID_PROBE, handle, &type); | 
|---|
|  | 562 |  | 
|---|
|  | 563 | loc_exchange_end(exch); | 
|---|
|  | 564 |  | 
|---|
|  | 565 | if (retval != EOK) | 
|---|
|  | 566 | return LOC_OBJECT_NONE; | 
|---|
|  | 567 |  | 
|---|
|  | 568 | return (loc_object_type_t) type; | 
|---|
|  | 569 | } | 
|---|
|  | 570 |  | 
|---|
|  | 571 | async_sess_t *loc_service_connect(exch_mgmt_t mgmt, service_id_t handle, | 
|---|
|  | 572 | unsigned int flags) | 
|---|
|  | 573 | { | 
|---|
|  | 574 | async_sess_t *sess; | 
|---|
|  | 575 |  | 
|---|
|  | 576 | if (flags & IPC_FLAG_BLOCKING) | 
|---|
|  | 577 | sess = service_connect_blocking(mgmt, SERVICE_LOC, | 
|---|
|  | 578 | LOC_CONNECT_TO_SERVICE, handle); | 
|---|
|  | 579 | else | 
|---|
|  | 580 | sess = service_connect(mgmt, SERVICE_LOC, | 
|---|
|  | 581 | LOC_CONNECT_TO_SERVICE, handle); | 
|---|
|  | 582 |  | 
|---|
|  | 583 | return sess; | 
|---|
|  | 584 | } | 
|---|
|  | 585 |  | 
|---|
|  | 586 | int loc_null_create(void) | 
|---|
|  | 587 | { | 
|---|
|  | 588 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 589 |  | 
|---|
|  | 590 | sysarg_t null_id; | 
|---|
|  | 591 | int retval = async_req_0_1(exch, LOC_NULL_CREATE, &null_id); | 
|---|
|  | 592 |  | 
|---|
|  | 593 | loc_exchange_end(exch); | 
|---|
|  | 594 |  | 
|---|
|  | 595 | if (retval != EOK) | 
|---|
|  | 596 | return -1; | 
|---|
|  | 597 |  | 
|---|
|  | 598 | return (int) null_id; | 
|---|
|  | 599 | } | 
|---|
|  | 600 |  | 
|---|
|  | 601 | void loc_null_destroy(int null_id) | 
|---|
|  | 602 | { | 
|---|
|  | 603 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 604 | async_req_1_0(exch, LOC_NULL_DESTROY, (sysarg_t) null_id); | 
|---|
|  | 605 | loc_exchange_end(exch); | 
|---|
|  | 606 | } | 
|---|
|  | 607 |  | 
|---|
|  | 608 | static size_t loc_count_namespaces_internal(async_exch_t *exch) | 
|---|
|  | 609 | { | 
|---|
|  | 610 | sysarg_t count; | 
|---|
|  | 611 | int retval = async_req_0_1(exch, LOC_GET_NAMESPACE_COUNT, &count); | 
|---|
|  | 612 | if (retval != EOK) | 
|---|
|  | 613 | return 0; | 
|---|
|  | 614 |  | 
|---|
|  | 615 | return count; | 
|---|
|  | 616 | } | 
|---|
|  | 617 |  | 
|---|
| [cc574511] | 618 | /** Add service to category. | 
|---|
|  | 619 | * | 
|---|
|  | 620 | * @param svc_id        Service ID | 
|---|
|  | 621 | * @param cat_id        Category ID | 
|---|
|  | 622 | * @return              EOK on success or negative error code | 
|---|
|  | 623 | */ | 
|---|
|  | 624 | int loc_service_add_to_cat(service_id_t svc_id, service_id_t cat_id) | 
|---|
|  | 625 | { | 
|---|
|  | 626 | async_exch_t *exch; | 
|---|
|  | 627 | sysarg_t retval; | 
|---|
|  | 628 |  | 
|---|
|  | 629 | exch = loc_exchange_begin_blocking(LOC_PORT_SUPPLIER); | 
|---|
|  | 630 | retval = async_req_2_0(exch, LOC_SERVICE_ADD_TO_CAT, svc_id, cat_id); | 
|---|
|  | 631 | loc_exchange_end(exch); | 
|---|
|  | 632 |  | 
|---|
|  | 633 | return retval; | 
|---|
|  | 634 | } | 
|---|
|  | 635 |  | 
|---|
| [15f3c3f] | 636 | static size_t loc_count_services_internal(async_exch_t *exch, | 
|---|
|  | 637 | service_id_t ns_handle) | 
|---|
|  | 638 | { | 
|---|
|  | 639 | sysarg_t count; | 
|---|
|  | 640 | int retval = async_req_1_1(exch, LOC_GET_SERVICE_COUNT, ns_handle, | 
|---|
|  | 641 | &count); | 
|---|
|  | 642 | if (retval != EOK) | 
|---|
|  | 643 | return 0; | 
|---|
|  | 644 |  | 
|---|
|  | 645 | return count; | 
|---|
|  | 646 | } | 
|---|
|  | 647 |  | 
|---|
|  | 648 | size_t loc_count_namespaces(void) | 
|---|
|  | 649 | { | 
|---|
|  | 650 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 651 | size_t size = loc_count_namespaces_internal(exch); | 
|---|
|  | 652 | loc_exchange_end(exch); | 
|---|
|  | 653 |  | 
|---|
|  | 654 | return size; | 
|---|
|  | 655 | } | 
|---|
|  | 656 |  | 
|---|
|  | 657 | size_t loc_count_services(service_id_t ns_handle) | 
|---|
|  | 658 | { | 
|---|
|  | 659 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 660 | size_t size = loc_count_services_internal(exch, ns_handle); | 
|---|
|  | 661 | loc_exchange_end(exch); | 
|---|
|  | 662 |  | 
|---|
|  | 663 | return size; | 
|---|
|  | 664 | } | 
|---|
|  | 665 |  | 
|---|
|  | 666 | size_t loc_get_namespaces(loc_sdesc_t **data) | 
|---|
|  | 667 | { | 
|---|
| [cc574511] | 668 | /* Loop until read is succesful */ | 
|---|
| [15f3c3f] | 669 | while (true) { | 
|---|
|  | 670 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 671 | size_t count = loc_count_namespaces_internal(exch); | 
|---|
|  | 672 | loc_exchange_end(exch); | 
|---|
|  | 673 |  | 
|---|
|  | 674 | if (count == 0) | 
|---|
|  | 675 | return 0; | 
|---|
|  | 676 |  | 
|---|
|  | 677 | loc_sdesc_t *devs = (loc_sdesc_t *) calloc(count, sizeof(loc_sdesc_t)); | 
|---|
|  | 678 | if (devs == NULL) | 
|---|
|  | 679 | return 0; | 
|---|
|  | 680 |  | 
|---|
|  | 681 | exch = loc_exchange_begin(LOC_PORT_CONSUMER); | 
|---|
|  | 682 |  | 
|---|
|  | 683 | ipc_call_t answer; | 
|---|
|  | 684 | aid_t req = async_send_0(exch, LOC_GET_NAMESPACES, &answer); | 
|---|
|  | 685 | int rc = async_data_read_start(exch, devs, count * sizeof(loc_sdesc_t)); | 
|---|
|  | 686 |  | 
|---|
|  | 687 | loc_exchange_end(exch); | 
|---|
|  | 688 |  | 
|---|
|  | 689 | if (rc == EOVERFLOW) { | 
|---|
|  | 690 | /* | 
|---|
|  | 691 | * Number of namespaces has changed since | 
|---|
|  | 692 | * the last call of LOC_GET_NAMESPACE_COUNT | 
|---|
|  | 693 | */ | 
|---|
|  | 694 | free(devs); | 
|---|
|  | 695 | continue; | 
|---|
|  | 696 | } | 
|---|
|  | 697 |  | 
|---|
|  | 698 | if (rc != EOK) { | 
|---|
| [50b581d] | 699 | async_forget(req); | 
|---|
| [15f3c3f] | 700 | free(devs); | 
|---|
|  | 701 | return 0; | 
|---|
|  | 702 | } | 
|---|
|  | 703 |  | 
|---|
|  | 704 | sysarg_t retval; | 
|---|
|  | 705 | async_wait_for(req, &retval); | 
|---|
|  | 706 |  | 
|---|
|  | 707 | if (retval != EOK) | 
|---|
|  | 708 | return 0; | 
|---|
|  | 709 |  | 
|---|
|  | 710 | *data = devs; | 
|---|
|  | 711 | return count; | 
|---|
|  | 712 | } | 
|---|
|  | 713 | } | 
|---|
|  | 714 |  | 
|---|
|  | 715 | size_t loc_get_services(service_id_t ns_handle, loc_sdesc_t **data) | 
|---|
|  | 716 | { | 
|---|
| [cc574511] | 717 | /* Loop until read is succesful */ | 
|---|
| [15f3c3f] | 718 | while (true) { | 
|---|
|  | 719 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 720 | size_t count = loc_count_services_internal(exch, ns_handle); | 
|---|
|  | 721 | loc_exchange_end(exch); | 
|---|
|  | 722 |  | 
|---|
|  | 723 | if (count == 0) | 
|---|
|  | 724 | return 0; | 
|---|
|  | 725 |  | 
|---|
|  | 726 | loc_sdesc_t *devs = (loc_sdesc_t *) calloc(count, sizeof(loc_sdesc_t)); | 
|---|
|  | 727 | if (devs == NULL) | 
|---|
|  | 728 | return 0; | 
|---|
|  | 729 |  | 
|---|
|  | 730 | exch = loc_exchange_begin(LOC_PORT_CONSUMER); | 
|---|
|  | 731 |  | 
|---|
|  | 732 | ipc_call_t answer; | 
|---|
|  | 733 | aid_t req = async_send_1(exch, LOC_GET_SERVICES, ns_handle, &answer); | 
|---|
|  | 734 | int rc = async_data_read_start(exch, devs, count * sizeof(loc_sdesc_t)); | 
|---|
|  | 735 |  | 
|---|
|  | 736 | loc_exchange_end(exch); | 
|---|
|  | 737 |  | 
|---|
|  | 738 | if (rc == EOVERFLOW) { | 
|---|
|  | 739 | /* | 
|---|
|  | 740 | * Number of services has changed since | 
|---|
|  | 741 | * the last call of LOC_GET_SERVICE_COUNT | 
|---|
|  | 742 | */ | 
|---|
|  | 743 | free(devs); | 
|---|
|  | 744 | continue; | 
|---|
|  | 745 | } | 
|---|
|  | 746 |  | 
|---|
|  | 747 | if (rc != EOK) { | 
|---|
| [50b581d] | 748 | async_forget(req); | 
|---|
| [15f3c3f] | 749 | free(devs); | 
|---|
|  | 750 | return 0; | 
|---|
|  | 751 | } | 
|---|
|  | 752 |  | 
|---|
|  | 753 | sysarg_t retval; | 
|---|
|  | 754 | async_wait_for(req, &retval); | 
|---|
|  | 755 |  | 
|---|
|  | 756 | if (retval != EOK) | 
|---|
|  | 757 | return 0; | 
|---|
|  | 758 |  | 
|---|
|  | 759 | *data = devs; | 
|---|
|  | 760 | return count; | 
|---|
|  | 761 | } | 
|---|
|  | 762 | } | 
|---|
| [cc574511] | 763 |  | 
|---|
| [278ac72] | 764 | static int loc_category_get_ids_once(sysarg_t method, sysarg_t arg1, | 
|---|
|  | 765 | sysarg_t *id_buf, size_t buf_size, size_t *act_size) | 
|---|
| [cc574511] | 766 | { | 
|---|
|  | 767 | async_exch_t *exch = loc_exchange_begin_blocking(LOC_PORT_CONSUMER); | 
|---|
|  | 768 |  | 
|---|
|  | 769 | ipc_call_t answer; | 
|---|
| [278ac72] | 770 | aid_t req = async_send_1(exch, method, arg1, &answer); | 
|---|
| [cc574511] | 771 | int rc = async_data_read_start(exch, id_buf, buf_size); | 
|---|
|  | 772 |  | 
|---|
|  | 773 | loc_exchange_end(exch); | 
|---|
|  | 774 |  | 
|---|
|  | 775 | if (rc != EOK) { | 
|---|
| [50b581d] | 776 | async_forget(req); | 
|---|
| [cc574511] | 777 | return rc; | 
|---|
|  | 778 | } | 
|---|
|  | 779 |  | 
|---|
|  | 780 | sysarg_t retval; | 
|---|
|  | 781 | async_wait_for(req, &retval); | 
|---|
|  | 782 |  | 
|---|
|  | 783 | if (retval != EOK) { | 
|---|
|  | 784 | return retval; | 
|---|
|  | 785 | } | 
|---|
|  | 786 |  | 
|---|
|  | 787 | *act_size = IPC_GET_ARG1(answer); | 
|---|
|  | 788 | return EOK; | 
|---|
|  | 789 | } | 
|---|
|  | 790 |  | 
|---|
| [278ac72] | 791 | /** Get list of IDs. | 
|---|
| [cc574511] | 792 | * | 
|---|
|  | 793 | * Returns an allocated array of service IDs. | 
|---|
|  | 794 | * | 
|---|
| [278ac72] | 795 | * @param method        IPC method | 
|---|
|  | 796 | * @param arg1          IPC argument 1 | 
|---|
| [cc574511] | 797 | * @param data          Place to store pointer to array of IDs | 
|---|
|  | 798 | * @param count         Place to store number of IDs | 
|---|
|  | 799 | * @return              EOK on success or negative error code | 
|---|
|  | 800 | */ | 
|---|
| [278ac72] | 801 | static int loc_get_ids_internal(sysarg_t method, sysarg_t arg1, | 
|---|
|  | 802 | sysarg_t **data, size_t *count) | 
|---|
| [cc574511] | 803 | { | 
|---|
|  | 804 | *data = NULL; | 
|---|
| [88057e3] | 805 | *count = 0; | 
|---|
|  | 806 |  | 
|---|
|  | 807 | size_t act_size = 0; | 
|---|
|  | 808 | int rc = loc_category_get_ids_once(method, arg1, NULL, 0, | 
|---|
| [278ac72] | 809 | &act_size); | 
|---|
| [cc574511] | 810 | if (rc != EOK) | 
|---|
|  | 811 | return rc; | 
|---|
| [88057e3] | 812 |  | 
|---|
|  | 813 | size_t alloc_size = act_size; | 
|---|
|  | 814 | service_id_t *ids = malloc(alloc_size); | 
|---|
| [cc574511] | 815 | if (ids == NULL) | 
|---|
|  | 816 | return ENOMEM; | 
|---|
| [88057e3] | 817 |  | 
|---|
| [cc574511] | 818 | while (true) { | 
|---|
| [278ac72] | 819 | rc = loc_category_get_ids_once(method, arg1, ids, alloc_size, | 
|---|
| [cc574511] | 820 | &act_size); | 
|---|
|  | 821 | if (rc != EOK) | 
|---|
|  | 822 | return rc; | 
|---|
| [88057e3] | 823 |  | 
|---|
| [c1f27f1d] | 824 | if (act_size <= alloc_size) | 
|---|
| [cc574511] | 825 | break; | 
|---|
| [88057e3] | 826 |  | 
|---|
|  | 827 | alloc_size = act_size; | 
|---|
|  | 828 | ids = realloc(ids, alloc_size); | 
|---|
| [cc574511] | 829 | if (ids == NULL) | 
|---|
|  | 830 | return ENOMEM; | 
|---|
|  | 831 | } | 
|---|
| [88057e3] | 832 |  | 
|---|
| [cc574511] | 833 | *count = act_size / sizeof(category_id_t); | 
|---|
|  | 834 | *data = ids; | 
|---|
|  | 835 | return EOK; | 
|---|
|  | 836 | } | 
|---|
| [278ac72] | 837 |  | 
|---|
|  | 838 | /** Get list of services in category. | 
|---|
|  | 839 | * | 
|---|
|  | 840 | * Returns an allocated array of service IDs. | 
|---|
|  | 841 | * | 
|---|
|  | 842 | * @param cat_id        Category ID | 
|---|
|  | 843 | * @param data          Place to store pointer to array of IDs | 
|---|
|  | 844 | * @param count         Place to store number of IDs | 
|---|
|  | 845 | * @return              EOK on success or negative error code | 
|---|
|  | 846 | */ | 
|---|
|  | 847 | int loc_category_get_svcs(category_id_t cat_id, service_id_t **data, | 
|---|
|  | 848 | size_t *count) | 
|---|
|  | 849 | { | 
|---|
|  | 850 | return loc_get_ids_internal(LOC_CATEGORY_GET_SVCS, cat_id, | 
|---|
|  | 851 | data, count); | 
|---|
|  | 852 | } | 
|---|
|  | 853 |  | 
|---|
|  | 854 | /** Get list of categories. | 
|---|
|  | 855 | * | 
|---|
|  | 856 | * Returns an allocated array of category IDs. | 
|---|
|  | 857 | * | 
|---|
|  | 858 | * @param data          Place to store pointer to array of IDs | 
|---|
|  | 859 | * @param count         Place to store number of IDs | 
|---|
|  | 860 | * @return              EOK on success or negative error code | 
|---|
|  | 861 | */ | 
|---|
|  | 862 | int loc_get_categories(category_id_t **data, size_t *count) | 
|---|
|  | 863 | { | 
|---|
|  | 864 | return loc_get_ids_internal(LOC_GET_CATEGORIES, 0, | 
|---|
|  | 865 | data, count); | 
|---|
|  | 866 | } | 
|---|
| [12f9f0d0] | 867 |  | 
|---|
|  | 868 | int loc_register_cat_change_cb(loc_cat_change_cb_t cb_fun) | 
|---|
|  | 869 | { | 
|---|
| [a6240a31] | 870 | fibril_mutex_lock(&loc_callback_mutex); | 
|---|
|  | 871 | if (loc_callback_create() != EOK) { | 
|---|
|  | 872 | fibril_mutex_unlock(&loc_callback_mutex); | 
|---|
| [12f9f0d0] | 873 | return EIO; | 
|---|
| [a6240a31] | 874 | } | 
|---|
|  | 875 |  | 
|---|
| [12f9f0d0] | 876 | cat_change_cb = cb_fun; | 
|---|
| [a6240a31] | 877 | fibril_mutex_unlock(&loc_callback_mutex); | 
|---|
|  | 878 |  | 
|---|
| [12f9f0d0] | 879 | return EOK; | 
|---|
|  | 880 | } | 
|---|