Line data Source code
1 : /*
2 : Authors:
3 : Pavel Březina <pbrezina@redhat.com>
4 :
5 : Copyright (C) 2014 Red Hat
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 as published by
9 : the Free Software Foundation; either version 3 of the License, or
10 : (at your option) any later version.
11 :
12 : This program is distributed in the hope that it will be useful,
13 : but WITHOUT ANY WARRANTY; without even the implied warranty of
14 : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 : GNU General Public License for more details.
16 :
17 : You should have received a copy of the GNU General Public License
18 : along with this program. If not, see <http://www.gnu.org/licenses/>.
19 : */
20 :
21 : #include <talloc.h>
22 : #include <dbus/dbus.h>
23 : #include <dhash.h>
24 :
25 : #include "util/util.h"
26 : #include "sbus/sssd_dbus.h"
27 : #include "sbus/sssd_dbus_meta.h"
28 : #include "sbus/sssd_dbus_private.h"
29 :
30 : static struct sbus_interface *
31 0 : sbus_iface_list_lookup(struct sbus_interface_list *list,
32 : const char *iface)
33 : {
34 : struct sbus_interface_list *item;
35 :
36 0 : DLIST_FOR_EACH(item, list) {
37 0 : if (strcmp(item->interface->vtable->meta->name, iface) == 0) {
38 0 : return item->interface;
39 : }
40 : }
41 :
42 0 : return NULL;
43 : }
44 :
45 : static errno_t
46 0 : sbus_iface_list_copy(TALLOC_CTX *mem_ctx,
47 : struct sbus_interface_list *list,
48 : struct sbus_interface_list **_copy)
49 : {
50 : TALLOC_CTX *list_ctx;
51 0 : struct sbus_interface_list *new_list = NULL;
52 : struct sbus_interface_list *new_item;
53 : struct sbus_interface_list *item;
54 : errno_t ret;
55 :
56 0 : if (list == NULL) {
57 0 : *_copy = NULL;
58 0 : return EOK;
59 : }
60 :
61 0 : list_ctx = talloc_new(mem_ctx);
62 0 : if (list_ctx == NULL) {
63 0 : return ENOMEM;
64 : }
65 :
66 0 : DLIST_FOR_EACH(item, list) {
67 0 : if (sbus_iface_list_lookup(new_list,
68 0 : item->interface->vtable->meta->name) != NULL) {
69 : /* already in list */
70 0 : continue;
71 : }
72 :
73 0 : new_item = talloc_zero(list_ctx, struct sbus_interface_list);
74 0 : if (new_item == NULL) {
75 0 : ret = ENOMEM;
76 0 : goto done;
77 : }
78 :
79 0 : new_item->interface = item->interface;
80 0 : DLIST_ADD(new_list, new_item);
81 : }
82 :
83 0 : *_copy = new_list;
84 0 : ret = EOK;
85 :
86 : done:
87 0 : if (ret != EOK) {
88 0 : talloc_free(list_ctx);
89 : }
90 :
91 0 : return ret;
92 : }
93 :
94 : /**
95 : * Object paths that represent all objects under the path:
96 : * /org/object/path/~* (without tilda)
97 : */
98 0 : static bool sbus_opath_is_subtree(const char *path)
99 : {
100 : size_t len;
101 :
102 0 : len = strlen(path);
103 :
104 0 : if (len < 2) {
105 0 : return false;
106 : }
107 :
108 0 : return path[len - 2] == '/' && path[len - 1] == '*';
109 : }
110 :
111 : /**
112 : * If the path represents a subtree object path, this function will
113 : * remove /~* from the end.
114 : */
115 0 : static char *sbus_opath_get_base_path(TALLOC_CTX *mem_ctx,
116 : const char *object_path)
117 : {
118 : char *tree_path;
119 : size_t len;
120 :
121 0 : tree_path = talloc_strdup(mem_ctx, object_path);
122 0 : if (tree_path == NULL) {
123 0 : return NULL;
124 : }
125 :
126 0 : if (!sbus_opath_is_subtree(tree_path)) {
127 0 : return tree_path;
128 : }
129 :
130 : /* replace / only if it is not a root path (only slash) */
131 0 : len = strlen(tree_path);
132 0 : tree_path[len - 1] = '\0';
133 0 : tree_path[len - 2] = (len - 2 != 0) ? '\0' : '/';
134 :
135 0 : return tree_path;
136 : }
137 :
138 0 : static char *sbus_opath_parent_subtree(TALLOC_CTX *mem_ctx,
139 : const char *path)
140 : {
141 : char *subtree;
142 : char *slash;
143 :
144 : /* first remove /~* from the end, stop when we have reached the root i.e.
145 : * subtree == "/" */
146 0 : subtree = sbus_opath_get_base_path(mem_ctx, path);
147 0 : if (subtree == NULL || subtree[1] == '\0') {
148 0 : return NULL;
149 : }
150 :
151 : /* Find the first separator and replace the part with asterisk. */
152 0 : slash = strrchr(subtree, '/');
153 0 : if (slash == NULL) {
154 : /* we cannot continue up */
155 0 : talloc_free(subtree);
156 0 : return NULL;
157 : }
158 :
159 0 : if (*(slash + 1) == '\0') {
160 : /* this object path is invalid since it cannot end with slash */
161 0 : DEBUG(SSSDBG_CRIT_FAILURE, "Invalid object path '%s'?\n", path);
162 0 : talloc_free(subtree);
163 0 : return NULL;
164 : }
165 :
166 : /* because object path cannot end with / there is enough space for
167 : * asterisk and terminating zero */
168 0 : *(slash + 1) = '*';
169 0 : *(slash + 2) = '\0';
170 :
171 0 : return subtree;
172 : }
173 :
174 : /**
175 : * The following path related functions are based on similar code in
176 : * storaged, just tailored to use talloc instead of glib
177 : */
178 : char *
179 7 : sbus_opath_escape_part(TALLOC_CTX *mem_ctx,
180 : const char *object_path_part)
181 : {
182 : size_t n;
183 7 : char *safe_path = NULL;
184 7 : TALLOC_CTX *tmp_ctx = NULL;
185 :
186 : /* The path must be valid */
187 7 : if (object_path_part == NULL) {
188 1 : return NULL;
189 : }
190 :
191 6 : tmp_ctx = talloc_new(NULL);
192 6 : if (tmp_ctx == NULL) {
193 0 : return NULL;
194 : }
195 :
196 6 : safe_path = talloc_strdup(tmp_ctx, "");
197 6 : if (safe_path == NULL) {
198 0 : goto done;
199 : }
200 :
201 : /* Special case for an empty string */
202 6 : if (strcmp(object_path_part, "") == 0) {
203 : /* the for loop would just fall through */
204 1 : safe_path = talloc_asprintf_append_buffer(safe_path, "_");
205 1 : if (safe_path == NULL) {
206 0 : goto done;
207 : }
208 : }
209 :
210 61 : for (n = 0; object_path_part[n]; n++) {
211 55 : int c = object_path_part[n];
212 : /* D-Bus spec says:
213 : * *
214 : * * Each element must only contain the ASCII characters
215 : * "[A-Z][a-z][0-9]_"
216 : * */
217 55 : if ((c >= 'A' && c <= 'Z')
218 55 : || (c >= 'a' && c <= 'z')
219 4 : || (c >= '0' && c <= '9')) {
220 51 : safe_path = talloc_asprintf_append_buffer(safe_path, "%c", c);
221 102 : if (safe_path == NULL) {
222 0 : goto done;
223 : }
224 : } else {
225 4 : safe_path = talloc_asprintf_append_buffer(safe_path, "_%02x", c);
226 4 : if (safe_path == NULL) {
227 0 : goto done;
228 : }
229 : }
230 : }
231 :
232 6 : safe_path = talloc_steal(mem_ctx, safe_path);
233 :
234 : done:
235 6 : talloc_free(tmp_ctx);
236 6 : return safe_path;
237 : }
238 :
239 12 : static inline int unhexchar(char c)
240 : {
241 12 : if (c >= '0' && c <= '9') {
242 6 : return c - '0';
243 : }
244 :
245 6 : if (c >= 'a' && c <= 'f') {
246 6 : return c - 'a' + 10;
247 : }
248 :
249 0 : if (c >= 'A' && c <= 'F') {
250 0 : return c - 'A' + 10;
251 : }
252 :
253 0 : return -1;
254 : }
255 :
256 : char *
257 20 : sbus_opath_unescape_part(TALLOC_CTX *mem_ctx,
258 : const char *object_path_part)
259 : {
260 : char *safe_path;
261 : const char *p;
262 : int a, b, c;
263 20 : TALLOC_CTX *tmp_ctx = NULL;
264 :
265 20 : tmp_ctx = talloc_new(NULL);
266 20 : if (tmp_ctx == NULL) {
267 0 : return NULL;
268 : }
269 :
270 20 : safe_path = talloc_strdup(tmp_ctx, "");
271 20 : if (safe_path == NULL) {
272 0 : goto done;
273 : }
274 :
275 : /* Special case for the empty string */
276 20 : if (strcmp(object_path_part, "_") == 0) {
277 1 : safe_path = talloc_steal(mem_ctx, safe_path);
278 1 : goto done;
279 : }
280 :
281 157 : for (p = object_path_part; *p; p++) {
282 139 : if (*p == '_') {
283 : /* There must be at least two more chars after underscore */
284 7 : if (p[1] == '\0' || p[2] == '\0') {
285 1 : safe_path = NULL;
286 1 : goto done;
287 : }
288 :
289 6 : if ((a = unhexchar(p[1])) < 0
290 6 : || (b = unhexchar(p[2])) < 0) {
291 : /* Invalid escape code, let's take it literal then */
292 0 : c = '_';
293 : } else {
294 6 : c = ((a << 4) | b);
295 6 : p += 2;
296 : }
297 : } else {
298 132 : c = *p;
299 : }
300 :
301 138 : safe_path = talloc_asprintf_append_buffer(safe_path, "%c", c);
302 138 : if (safe_path == NULL) {
303 0 : goto done;
304 : }
305 : }
306 :
307 18 : safe_path = talloc_steal(mem_ctx, safe_path);
308 :
309 : done:
310 20 : talloc_free(tmp_ctx);
311 20 : return safe_path;
312 : }
313 :
314 : char *
315 2 : _sbus_opath_compose(TALLOC_CTX *mem_ctx,
316 : const char *base,
317 : const char *part, ...)
318 : {
319 : char *safe_part;
320 2 : char *path = NULL;
321 : va_list va;
322 :
323 2 : if (base == NULL) {
324 0 : DEBUG(SSSDBG_OP_FAILURE, "Wrong object path base!\n");
325 0 : return NULL;
326 : }
327 :
328 2 : path = talloc_strdup(mem_ctx, base);
329 2 : if (path == NULL) return NULL;
330 :
331 2 : va_start(va, part);
332 6 : while (part != NULL) {
333 2 : safe_part = sbus_opath_escape_part(mem_ctx, part);
334 2 : if (safe_part == NULL) {
335 0 : DEBUG(SSSDBG_OP_FAILURE, "Could not add [%s] to objpath\n", part);
336 0 : goto fail;
337 : }
338 :
339 2 : path = talloc_asprintf_append(path, "/%s", safe_part);
340 2 : talloc_free(safe_part);
341 2 : if (path == NULL) {
342 0 : goto fail;
343 : }
344 :
345 2 : part = va_arg(va, const char *);
346 : }
347 2 : va_end(va);
348 :
349 2 : return path;
350 :
351 : fail:
352 0 : va_end(va);
353 0 : talloc_free(path);
354 0 : return NULL;
355 : }
356 :
357 : errno_t
358 7 : sbus_opath_decompose(TALLOC_CTX *mem_ctx,
359 : const char *object_path,
360 : const char *prefix,
361 : char ***_components,
362 : size_t *_len)
363 : {
364 : TALLOC_CTX *tmp_ctx;
365 : const char *path;
366 : char **decomposed;
367 : char **unescaped;
368 : errno_t ret;
369 : int len;
370 : int i;
371 :
372 7 : tmp_ctx = talloc_new(NULL);
373 7 : if (tmp_ctx == NULL) {
374 0 : return ENOMEM;
375 : }
376 :
377 : /* Strip prefix from the path. */
378 7 : if (prefix != NULL) {
379 3 : path = sbus_opath_strip_prefix(object_path, prefix);
380 3 : if (path == NULL) {
381 1 : ret = ERR_SBUS_INVALID_PATH;
382 1 : goto done;
383 : }
384 : } else {
385 4 : path = object_path;
386 : }
387 :
388 : /* Split the string using / as delimiter. */
389 6 : split_on_separator(tmp_ctx, path, '/', true, true, &decomposed, &len);
390 :
391 : /* Unescape parts. */
392 6 : unescaped = talloc_zero_array(tmp_ctx, char *, len + 1);
393 6 : if (unescaped == NULL) {
394 0 : ret = ENOMEM;
395 0 : goto done;
396 : }
397 :
398 19 : for (i = 0; i < len; i++) {
399 13 : unescaped[i] = sbus_opath_unescape_part(unescaped, decomposed[i]);
400 13 : if (unescaped[i] == NULL) {
401 0 : ret = ENOMEM;
402 0 : goto done;
403 : }
404 : }
405 :
406 6 : if (_components != NULL) {
407 6 : *_components = talloc_steal(mem_ctx, unescaped);
408 : }
409 :
410 6 : if (_len != NULL) {
411 6 : *_len = len;
412 : }
413 :
414 6 : ret = EOK;
415 :
416 : done:
417 7 : talloc_free(tmp_ctx);
418 7 : return ret;
419 : }
420 :
421 : errno_t
422 2 : sbus_opath_decompose_exact(TALLOC_CTX *mem_ctx,
423 : const char *object_path,
424 : const char *prefix,
425 : size_t expected,
426 : char ***_components)
427 : {
428 : char **components;
429 : size_t len;
430 : errno_t ret;
431 :
432 2 : ret = sbus_opath_decompose(mem_ctx, object_path, prefix,
433 : &components, &len);
434 2 : if (ret != EOK) {
435 0 : return ret;
436 : }
437 :
438 2 : if (len != expected) {
439 1 : talloc_free(components);
440 1 : return ERR_SBUS_INVALID_PATH;
441 : }
442 :
443 1 : if (_components != NULL) {
444 1 : *_components = components;
445 : }
446 :
447 1 : return EOK;
448 : }
449 :
450 : const char *
451 9 : sbus_opath_strip_prefix(const char *object_path,
452 : const char *prefix)
453 : {
454 9 : if (strncmp(object_path, prefix, strlen(prefix)) == 0) {
455 6 : return object_path + strlen(prefix);
456 : }
457 :
458 3 : return NULL;
459 : }
460 :
461 : char *
462 4 : sbus_opath_get_object_name(TALLOC_CTX *mem_ctx,
463 : const char *object_path,
464 : const char *base_path)
465 : {
466 : const char *name;
467 :
468 4 : name = sbus_opath_strip_prefix(object_path, base_path);
469 4 : if (name == NULL || name[0] == '\0') {
470 2 : return NULL;
471 : }
472 :
473 : /* if base_path did not end with / */
474 2 : if (name[0] == '/') {
475 1 : name = name + 1;
476 : }
477 :
478 2 : return sbus_opath_unescape_part(mem_ctx, name);
479 : }
480 :
481 : static void
482 0 : sbus_opath_hash_delete_cb(hash_entry_t *item,
483 : hash_destroy_enum deltype,
484 : void *pvt)
485 : {
486 : struct sbus_connection *conn;
487 : char *path;
488 :
489 0 : conn = talloc_get_type(pvt, struct sbus_connection);
490 0 : path = sbus_opath_get_base_path(NULL, item->key.str);
491 :
492 0 : dbus_connection_unregister_object_path(conn->dbus.conn, path);
493 0 : }
494 :
495 : errno_t
496 0 : sbus_opath_hash_init(TALLOC_CTX *mem_ctx,
497 : struct sbus_connection *conn,
498 : hash_table_t **_table)
499 : {
500 0 : return sss_hash_create_ex(mem_ctx, 10, _table, 0, 0, 0, 0,
501 : sbus_opath_hash_delete_cb, conn);
502 : }
503 :
504 : static errno_t
505 0 : sbus_opath_hash_add_iface(hash_table_t *table,
506 : const char *object_path,
507 : struct sbus_interface *iface,
508 : bool *_path_known)
509 : {
510 0 : TALLOC_CTX *tmp_ctx = NULL;
511 0 : struct sbus_interface_list *list = NULL;
512 0 : struct sbus_interface_list *item = NULL;
513 0 : const char *iface_name = iface->vtable->meta->name;
514 : hash_key_t key;
515 : hash_value_t value;
516 : bool path_known;
517 : errno_t ret;
518 : int hret;
519 :
520 0 : tmp_ctx = talloc_new(NULL);
521 0 : if (tmp_ctx == NULL) {
522 0 : return ENOMEM;
523 : }
524 :
525 0 : DEBUG(SSSDBG_TRACE_FUNC, "Registering interface %s with path %s\n",
526 : iface_name, object_path);
527 :
528 : /* create new list item */
529 :
530 0 : item = talloc_zero(tmp_ctx, struct sbus_interface_list);
531 0 : if (item == NULL) {
532 0 : return ENOMEM;
533 : }
534 :
535 0 : item->interface = iface;
536 :
537 : /* first lookup existing list in hash table */
538 :
539 0 : key.type = HASH_KEY_STRING;
540 0 : key.str = talloc_strdup(tmp_ctx, object_path);
541 0 : if (key.str == NULL) {
542 0 : ret = ENOMEM;
543 0 : goto done;
544 : }
545 :
546 0 : hret = hash_lookup(table, &key, &value);
547 0 : if (hret == HASH_SUCCESS) {
548 : /* This object path has already some interface registered. We will
549 : * check for existence of the interface currently being added and
550 : * add it if missing. */
551 :
552 0 : path_known = true;
553 :
554 0 : list = talloc_get_type(value.ptr, struct sbus_interface_list);
555 0 : if (sbus_iface_list_lookup(list, iface_name) != NULL) {
556 0 : DEBUG(SSSDBG_MINOR_FAILURE, "Trying to register the same interface"
557 : " twice: iface=%s, opath=%s\n", iface_name, object_path);
558 0 : ret = EEXIST;
559 0 : goto done;
560 : }
561 :
562 0 : DLIST_ADD_END(list, item, struct sbus_interface_list *);
563 0 : ret = EOK;
564 0 : goto done;
565 0 : } else if (hret != HASH_ERROR_KEY_NOT_FOUND) {
566 0 : ret = EIO;
567 0 : goto done;
568 : }
569 :
570 : /* otherwise create new hash entry and new list */
571 :
572 0 : path_known = false;
573 0 : list = item;
574 :
575 0 : value.type = HASH_VALUE_PTR;
576 0 : value.ptr = list;
577 :
578 0 : hret = hash_enter(table, &key, &value);
579 0 : if (hret != HASH_SUCCESS) {
580 0 : ret = EIO;
581 0 : goto done;
582 : }
583 :
584 0 : talloc_steal(table, key.str);
585 0 : ret = EOK;
586 :
587 : done:
588 0 : if (ret == EOK) {
589 0 : talloc_steal(item, iface);
590 0 : talloc_steal(table, item);
591 0 : *_path_known = path_known;
592 : }
593 :
594 0 : talloc_free(tmp_ctx);
595 0 : return ret;
596 : }
597 :
598 : static bool
599 0 : sbus_opath_hash_has_path(hash_table_t *table,
600 : const char *object_path)
601 : {
602 : hash_key_t key;
603 :
604 0 : key.type = HASH_KEY_STRING;
605 0 : key.str = discard_const(object_path);
606 :
607 0 : return hash_has_key(table, &key);
608 : }
609 :
610 : /**
611 : * First @object_path is looked up in @table, if it is not found it steps up
612 : * in the path hierarchy and try to lookup the parent node. This continues
613 : * until the root is reached.
614 : */
615 : struct sbus_interface *
616 0 : sbus_opath_hash_lookup_iface(hash_table_t *table,
617 : const char *object_path,
618 : const char *iface_name)
619 : {
620 0 : TALLOC_CTX *tmp_ctx = NULL;
621 0 : struct sbus_interface_list *list = NULL;
622 0 : struct sbus_interface *iface = NULL;
623 0 : char *lookup_path = NULL;
624 : hash_key_t key;
625 : hash_value_t value;
626 : int hret;
627 :
628 0 : tmp_ctx = talloc_new(NULL);
629 0 : if (tmp_ctx == NULL) {
630 0 : return NULL;
631 : }
632 :
633 0 : lookup_path = talloc_strdup(tmp_ctx, object_path);
634 0 : if (lookup_path == NULL) {
635 0 : goto done;
636 : }
637 :
638 0 : while (lookup_path != NULL) {
639 0 : key.type = HASH_KEY_STRING;
640 0 : key.str = lookup_path;
641 :
642 0 : hret = hash_lookup(table, &key, &value);
643 0 : if (hret == HASH_SUCCESS) {
644 0 : list = talloc_get_type(value.ptr, struct sbus_interface_list);
645 0 : iface = sbus_iface_list_lookup(list, iface_name);
646 0 : if (iface != NULL) {
647 0 : goto done;
648 : }
649 0 : } else if (hret != HASH_ERROR_KEY_NOT_FOUND) {
650 0 : DEBUG(SSSDBG_OP_FAILURE,
651 : "Unable to search hash table: hret=%d\n", hret);
652 0 : iface = NULL;
653 0 : goto done;
654 : }
655 :
656 : /* we will not free lookup path since it is freed with tmp_ctx
657 : * and the object paths are supposed to be small */
658 0 : lookup_path = sbus_opath_parent_subtree(tmp_ctx, lookup_path);
659 : }
660 :
661 : done:
662 0 : talloc_free(tmp_ctx);
663 0 : return iface;
664 : }
665 :
666 : /**
667 : * Acquire list of all interfaces that are supported on given object path.
668 : */
669 : errno_t
670 0 : sbus_opath_hash_lookup_supported(TALLOC_CTX *mem_ctx,
671 : hash_table_t *table,
672 : const char *object_path,
673 : struct sbus_interface_list **_list)
674 : {
675 0 : TALLOC_CTX *tmp_ctx = NULL;
676 0 : TALLOC_CTX *list_ctx = NULL;
677 0 : struct sbus_interface_list *copy = NULL;
678 0 : struct sbus_interface_list *list = NULL;
679 0 : char *lookup_path = NULL;
680 : hash_key_t key;
681 : hash_value_t value;
682 : errno_t ret;
683 : int hret;
684 :
685 0 : tmp_ctx = talloc_new(NULL);
686 0 : if (tmp_ctx == NULL) {
687 0 : return ENOMEM;
688 : }
689 :
690 0 : list_ctx = talloc_new(tmp_ctx);
691 0 : if (list_ctx == NULL) {
692 0 : ret = ENOMEM;
693 0 : goto done;
694 : }
695 :
696 0 : lookup_path = talloc_strdup(tmp_ctx, object_path);
697 0 : if (lookup_path == NULL) {
698 0 : ret = ENOMEM;
699 0 : goto done;
700 : }
701 :
702 0 : while (lookup_path != NULL) {
703 0 : key.type = HASH_KEY_STRING;
704 0 : key.str = lookup_path;
705 :
706 0 : hret = hash_lookup(table, &key, &value);
707 0 : if (hret == HASH_SUCCESS) {
708 0 : ret = sbus_iface_list_copy(list_ctx, value.ptr, ©);
709 0 : if (ret != EOK) {
710 0 : goto done;
711 : }
712 :
713 0 : DLIST_CONCATENATE(list, copy, struct sbus_interface_list *);
714 0 : } else if (hret != HASH_ERROR_KEY_NOT_FOUND) {
715 0 : DEBUG(SSSDBG_OP_FAILURE,
716 : "Unable to search hash table: hret=%d\n", hret);
717 0 : ret = EIO;
718 0 : goto done;
719 : }
720 :
721 : /* we will not free lookup path since it is freed with tmp_ctx
722 : * and the object paths are supposed to be small */
723 0 : lookup_path = sbus_opath_parent_subtree(tmp_ctx, lookup_path);
724 : }
725 :
726 0 : talloc_steal(mem_ctx, list_ctx);
727 0 : *_list = list;
728 0 : ret = EOK;
729 :
730 : done:
731 0 : talloc_free(tmp_ctx);
732 0 : return ret;
733 : }
734 :
735 : errno_t
736 0 : sbus_nodes_hash_init(TALLOC_CTX *mem_ctx,
737 : struct sbus_connection *conn,
738 : hash_table_t **_table)
739 : {
740 0 : return sss_hash_create_ex(mem_ctx, 10, _table, 0, 0, 0, 0,
741 : NULL, conn);
742 : }
743 :
744 : struct sbus_nodes_data {
745 : sbus_nodes_fn nodes_fn;
746 : void *handler_data;
747 : };
748 :
749 : static errno_t
750 0 : sbus_nodes_hash_add(hash_table_t *table,
751 : const char *object_path,
752 : sbus_nodes_fn nodes_fn,
753 : void *handler_data)
754 : {
755 : TALLOC_CTX *tmp_ctx;
756 : struct sbus_nodes_data *data;
757 : hash_key_t key;
758 : hash_value_t value;
759 : errno_t ret;
760 : bool has_key;
761 : int hret;
762 :
763 0 : tmp_ctx = talloc_new(NULL);
764 0 : if (tmp_ctx == NULL) {
765 0 : return ENOMEM;
766 : }
767 :
768 0 : key.type = HASH_KEY_STRING;
769 0 : key.str = talloc_strdup(tmp_ctx, object_path);
770 0 : if (key.str == NULL) {
771 0 : return ENOMEM;
772 : }
773 :
774 0 : has_key = hash_has_key(table, &key);
775 0 : if (has_key) {
776 0 : ret = EEXIST;
777 0 : goto done;
778 : }
779 :
780 0 : data = talloc_zero(tmp_ctx, struct sbus_nodes_data);
781 0 : if (data == NULL) {
782 0 : ret = ENOMEM;
783 0 : goto done;
784 : }
785 :
786 0 : data->handler_data = handler_data;
787 0 : data->nodes_fn = nodes_fn;
788 :
789 0 : value.type = HASH_VALUE_PTR;
790 0 : value.ptr = data;
791 :
792 0 : hret = hash_enter(table, &key, &value);
793 0 : if (hret != HASH_SUCCESS) {
794 0 : ret = EIO;
795 0 : goto done;
796 : }
797 :
798 0 : talloc_steal(table, key.str);
799 0 : talloc_steal(table, data);
800 :
801 0 : ret = EOK;
802 :
803 : done:
804 0 : talloc_free(tmp_ctx);
805 0 : return ret;
806 : }
807 :
808 : const char **
809 0 : sbus_nodes_hash_lookup(TALLOC_CTX *mem_ctx,
810 : hash_table_t *table,
811 : const char *object_path)
812 : {
813 : struct sbus_nodes_data *data;
814 : hash_key_t key;
815 : hash_value_t value;
816 : int hret;
817 :
818 0 : key.type = HASH_KEY_STRING;
819 0 : key.str = discard_const(object_path);
820 :
821 0 : hret = hash_lookup(table, &key, &value);
822 0 : if (hret == HASH_ERROR_KEY_NOT_FOUND) {
823 0 : return NULL;
824 0 : } else if (hret != HASH_SUCCESS) {
825 0 : DEBUG(SSSDBG_OP_FAILURE,
826 : "Unable to search hash table: hret=%d\n", hret);
827 0 : return NULL;
828 : }
829 :
830 0 : data = talloc_get_type(value.ptr, struct sbus_nodes_data);
831 :
832 0 : return data->nodes_fn(mem_ctx, object_path, data->handler_data);
833 : }
834 :
835 : static struct sbus_interface *
836 0 : sbus_new_interface(TALLOC_CTX *mem_ctx,
837 : const char *object_path,
838 : struct sbus_vtable *iface_vtable,
839 : void *handler_data)
840 : {
841 : struct sbus_interface *intf;
842 :
843 0 : intf = talloc_zero(mem_ctx, struct sbus_interface);
844 0 : if (intf == NULL) {
845 0 : DEBUG(SSSDBG_FATAL_FAILURE, "Cannot allocate a new sbus_interface.\n");
846 0 : return NULL;
847 : }
848 :
849 0 : intf->path = talloc_strdup(intf, object_path);
850 0 : if (intf->path == NULL) {
851 0 : DEBUG(SSSDBG_FATAL_FAILURE, "Cannot duplicate object path.\n");
852 0 : talloc_free(intf);
853 0 : return NULL;
854 : }
855 :
856 0 : intf->vtable = iface_vtable;
857 0 : intf->handler_data = handler_data;
858 0 : return intf;
859 : }
860 :
861 : static DBusHandlerResult
862 : sbus_message_handler(DBusConnection *dbus_conn,
863 : DBusMessage *message,
864 : void *user_data);
865 :
866 : static errno_t
867 0 : sbus_conn_register_path(struct sbus_connection *conn,
868 : const char *path)
869 : {
870 : static DBusObjectPathVTable vtable = {NULL, sbus_message_handler,
871 : NULL, NULL, NULL, NULL};
872 : DBusError error;
873 0 : char *reg_path = NULL;
874 : dbus_bool_t dbret;
875 :
876 0 : DEBUG(SSSDBG_TRACE_FUNC, "Registering object path %s with D-Bus "
877 : "connection\n", path);
878 :
879 0 : if (sbus_opath_is_subtree(path)) {
880 0 : reg_path = sbus_opath_get_base_path(conn, path);
881 0 : if (reg_path == NULL) {
882 0 : return ENOMEM;
883 : }
884 :
885 : /* D-Bus does not allow to have both object path and fallback
886 : * registered. Since we handle the real message handlers ourselves
887 : * we will register fallback only in this case. */
888 0 : if (sbus_opath_hash_has_path(conn->managed_paths, reg_path)) {
889 0 : dbus_connection_unregister_object_path(conn->dbus.conn, reg_path);
890 : }
891 :
892 0 : dbret = dbus_connection_register_fallback(conn->dbus.conn, reg_path,
893 : &vtable, conn);
894 0 : talloc_free(reg_path);
895 : } else {
896 0 : dbus_error_init(&error);
897 :
898 0 : dbret = dbus_connection_try_register_object_path(conn->dbus.conn, path,
899 : &vtable, conn, &error);
900 :
901 0 : if (dbus_error_is_set(&error) &&
902 0 : strcmp(error.name, DBUS_ERROR_OBJECT_PATH_IN_USE) == 0) {
903 : /* A fallback is probably already registered. Just return. */
904 0 : dbus_error_free(&error);
905 0 : return EOK;
906 : }
907 : }
908 :
909 0 : if (!dbret) {
910 0 : DEBUG(SSSDBG_FATAL_FAILURE, "Unable to register object path "
911 : "%s with D-Bus connection.\n", path);
912 0 : return ENOMEM;
913 : }
914 :
915 0 : return EOK;
916 : }
917 :
918 : errno_t
919 0 : sbus_conn_register_iface(struct sbus_connection *conn,
920 : struct sbus_vtable *iface_vtable,
921 : const char *object_path,
922 : void *handler_data)
923 : {
924 0 : struct sbus_interface *iface = NULL;
925 : bool path_known;
926 : errno_t ret;
927 :
928 0 : if (conn == NULL || iface_vtable == NULL || object_path == NULL) {
929 0 : return EINVAL;
930 : }
931 :
932 0 : iface = sbus_new_interface(conn, object_path, iface_vtable, handler_data);
933 0 : if (iface == NULL) {
934 0 : return ENOMEM;
935 : }
936 :
937 0 : ret = sbus_opath_hash_add_iface(conn->managed_paths, object_path, iface,
938 : &path_known);
939 0 : if (ret != EOK) {
940 0 : talloc_free(iface);
941 0 : return ret;
942 : }
943 :
944 0 : if (path_known) {
945 : /* this object path is already registered */
946 0 : return EOK;
947 : }
948 :
949 : /* if ret != EOK we will still leave iface in the table, since
950 : * we probably don't have enough memory to remove it correctly anyway */
951 :
952 0 : ret = sbus_conn_register_path(conn, object_path);
953 0 : if (ret != EOK) {
954 0 : return ret;
955 : }
956 :
957 : /* register standard interfaces with this object path as well */
958 0 : ret = sbus_conn_register_iface(conn, sbus_properties_vtable(),
959 : object_path, conn);
960 0 : if (ret != EOK) {
961 0 : return ret;
962 : }
963 :
964 0 : ret = sbus_conn_register_iface(conn, sbus_introspect_vtable(),
965 : object_path, conn);
966 0 : if (ret != EOK) {
967 0 : return ret;
968 : }
969 :
970 0 : return ret;
971 : }
972 :
973 : void
974 0 : sbus_conn_register_nodes(struct sbus_connection *conn,
975 : const char *path,
976 : sbus_nodes_fn nodes_fn,
977 : void *data)
978 : {
979 : errno_t ret;
980 :
981 0 : ret = sbus_nodes_hash_add(conn->nodes_fns, path, nodes_fn, data);
982 0 : if (ret != EOK && ret != EEXIST) {
983 0 : DEBUG(SSSDBG_MINOR_FAILURE, "Unable to register node function with "
984 : "%s. Introspection may not work correctly.\n", path);
985 : }
986 0 : }
987 :
988 : errno_t
989 0 : sbus_conn_reregister_paths(struct sbus_connection *conn)
990 : {
991 0 : hash_key_t *keys = NULL;
992 : unsigned long count;
993 : unsigned long i;
994 : errno_t ret;
995 : int hret;
996 :
997 0 : hret = hash_keys(conn->managed_paths, &count, &keys);
998 0 : if (hret != HASH_SUCCESS) {
999 0 : ret = ENOMEM;
1000 0 : goto done;
1001 : }
1002 :
1003 0 : for (i = 0; i < count; i++) {
1004 0 : ret = sbus_conn_register_path(conn, keys[i].str);
1005 0 : if (ret != EOK) {
1006 0 : goto done;
1007 : }
1008 : }
1009 :
1010 0 : ret = EOK;
1011 :
1012 : done:
1013 0 : talloc_free(keys);
1014 0 : return ret;
1015 : }
1016 :
1017 : static void
1018 : sbus_message_handler_got_caller_id(struct tevent_req *req);
1019 :
1020 : static DBusHandlerResult
1021 0 : sbus_message_handler(DBusConnection *dbus_conn,
1022 : DBusMessage *message,
1023 : void *handler_data)
1024 : {
1025 : struct tevent_req *req;
1026 : struct sbus_connection *conn;
1027 : struct sbus_interface *iface;
1028 : struct sbus_request *sbus_req;
1029 : const struct sbus_method_meta *method;
1030 : const char *iface_name;
1031 : const char *method_name;
1032 : const char *path;
1033 : const char *sender;
1034 :
1035 0 : conn = talloc_get_type(handler_data, struct sbus_connection);
1036 :
1037 : /* header information */
1038 0 : iface_name = dbus_message_get_interface(message);
1039 0 : method_name = dbus_message_get_member(message);
1040 0 : path = dbus_message_get_path(message);
1041 0 : sender = dbus_message_get_sender(message);
1042 :
1043 0 : DEBUG(SSSDBG_TRACE_INTERNAL, "Received SBUS method %s.%s on path %s\n",
1044 : iface_name, method_name, path);
1045 :
1046 : /* try to find the interface */
1047 0 : iface = sbus_opath_hash_lookup_iface(conn->managed_paths,
1048 : path, iface_name);
1049 0 : if (iface == NULL) {
1050 0 : goto fail;
1051 : }
1052 :
1053 0 : method = sbus_meta_find_method(iface->vtable->meta, method_name);
1054 0 : if (method == NULL || method->vtable_offset == 0) {
1055 : goto fail;
1056 : }
1057 :
1058 : /* we have a valid handler, create D-Bus request */
1059 0 : sbus_req = sbus_new_request(conn, iface, message);
1060 0 : if (sbus_req == NULL) {
1061 0 : return DBUS_HANDLER_RESULT_NEED_MEMORY;
1062 : }
1063 :
1064 0 : sbus_req->method = method;
1065 :
1066 : /* now get the sender ID */
1067 0 : req = sbus_get_sender_id_send(sbus_req, conn->ev, conn, sender);
1068 0 : if (req == NULL) {
1069 0 : talloc_free(sbus_req);
1070 0 : return DBUS_HANDLER_RESULT_NEED_MEMORY;
1071 : }
1072 0 : tevent_req_set_callback(req, sbus_message_handler_got_caller_id, sbus_req);
1073 :
1074 0 : return DBUS_HANDLER_RESULT_HANDLED;
1075 :
1076 : fail: ;
1077 : DBusMessage *reply;
1078 :
1079 0 : DEBUG(SSSDBG_CRIT_FAILURE, "No matching handler found for method %s.%s "
1080 : "on path %s\n", iface_name, method_name, path);
1081 :
1082 0 : reply = dbus_message_new_error(message, DBUS_ERROR_UNKNOWN_METHOD, NULL);
1083 0 : sbus_conn_send_reply(conn, reply);
1084 :
1085 0 : return DBUS_HANDLER_RESULT_HANDLED;
1086 : }
1087 :
1088 : static void
1089 0 : sbus_message_handler_got_caller_id(struct tevent_req *req)
1090 : {
1091 : struct sbus_request *sbus_req;
1092 : const struct sbus_method_meta *method;
1093 : sbus_msg_handler_fn handler;
1094 : sbus_method_invoker_fn invoker;
1095 : void *pvt;
1096 : DBusError *error;
1097 : errno_t ret;
1098 :
1099 0 : sbus_req = tevent_req_callback_data(req, struct sbus_request);
1100 0 : method = sbus_req->method;
1101 :
1102 0 : ret = sbus_get_sender_id_recv(req, &sbus_req->client);
1103 0 : if (ret != EOK) {
1104 0 : error = sbus_error_new(sbus_req, DBUS_ERROR_FAILED, "Failed to "
1105 : "resolve caller's ID: %s\n", sss_strerror(ret));
1106 0 : sbus_request_fail_and_finish(sbus_req, error);
1107 0 : return;
1108 : }
1109 :
1110 0 : handler = VTABLE_FUNC(sbus_req->intf->vtable, method->vtable_offset);
1111 0 : invoker = method->invoker;
1112 0 : pvt = sbus_req->intf->handler_data;
1113 :
1114 0 : sbus_request_invoke_or_finish(sbus_req, handler, pvt, invoker);
1115 0 : return;
1116 : }
|