source: mainline/uspace/lib/c/generic/loc.c@ 9359aae

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 9359aae was 3e6a98c5, checked in by Jiri Svoboda <jiri@…>, 13 years ago

Standards-compliant boolean type.

  • Property mode set to 100644
File size: 20.1 KB
RevLine 
[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
41static FIBRIL_MUTEX_INITIALIZE(loc_supp_block_mutex);
42static FIBRIL_MUTEX_INITIALIZE(loc_cons_block_mutex);
43
44static FIBRIL_MUTEX_INITIALIZE(loc_supplier_mutex);
45static FIBRIL_MUTEX_INITIALIZE(loc_consumer_mutex);
46
[12f9f0d0]47static FIBRIL_MUTEX_INITIALIZE(loc_callback_mutex);
48static bool loc_callback_created = false;
[a6240a31]49static loc_cat_change_cb_t cat_change_cb = NULL;
[12f9f0d0]50
[15f3c3f]51static async_sess_t *loc_supp_block_sess = NULL;
52static async_sess_t *loc_cons_block_sess = NULL;
53
54static async_sess_t *loc_supplier_sess = NULL;
55static async_sess_t *loc_consumer_sess = NULL;
56
[12f9f0d0]57static 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]87static 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]105static 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 */
137async_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 */
190async_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 */
231void loc_exchange_end(async_exch_t *exch)
232{
233 async_exchange_end(exch);
234}
235
236/** Register new driver with loc. */
[f302586]237int 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]274int 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]312int 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 */
321int 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
333int 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]383static 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 */
433int 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 */
447int 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 */
461int 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]466int 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 */
515int 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]556loc_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
571async_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
586int 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
601void 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
608static 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 */
624int 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]636static 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
648size_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
657size_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
666size_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
715size_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]764static 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]801static 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 */
847int 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 */
862int 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
868int 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}
Note: See TracBrowser for help on using the repository browser.