2 * Copyright (C) 2011 Felix Fietkau <nbd@openwrt.org>
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU Lesser General Public License version 2.1
6 * as published by the Free Software Foundation
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
15 #include "ubusd_obj.h"
17 struct avl_tree obj_types
;
18 struct avl_tree objects
;
21 static void ubus_unref_object_type(struct ubus_object_type
*type
)
23 struct ubus_method
*m
;
25 if (--type
->refcount
> 0)
28 while (!list_empty(&type
->methods
)) {
29 m
= list_first_entry(&type
->methods
, struct ubus_method
, list
);
34 ubus_free_id(&obj_types
, &type
->id
);
38 static bool ubus_create_obj_method(struct ubus_object_type
*type
, struct blob_attr
*attr
)
40 struct ubus_method
*m
;
41 int bloblen
= blob_raw_len(attr
);
43 m
= calloc(1, sizeof(*m
) + bloblen
);
47 list_add_tail(&m
->list
, &type
->methods
);
48 memcpy(m
->data
, attr
, bloblen
);
49 m
->name
= blobmsg_name(m
->data
);
54 static struct ubus_object_type
*ubus_create_obj_type(struct blob_attr
*sig
)
56 struct ubus_object_type
*type
;
57 struct blob_attr
*pos
;
60 type
= calloc(1, sizeof(*type
));
63 if (!ubus_alloc_id(&obj_types
, &type
->id
, 0))
66 INIT_LIST_HEAD(&type
->methods
);
68 blob_for_each_attr(pos
, sig
, rem
) {
69 if (!blobmsg_check_attr(pos
, true))
72 if (!ubus_create_obj_method(type
, pos
))
79 ubus_unref_object_type(type
);
87 static struct ubus_object_type
*ubus_get_obj_type(uint32_t obj_id
)
89 struct ubus_object_type
*type
;
92 id
= ubus_find_id(&obj_types
, obj_id
);
96 type
= container_of(id
, struct ubus_object_type
, id
);
101 struct ubus_object
*ubusd_create_object_internal(struct ubus_object_type
*type
, uint32_t id
)
103 struct ubus_object
*obj
;
105 obj
= calloc(1, sizeof(*obj
));
109 if (!ubus_alloc_id(&objects
, &obj
->id
, id
))
113 INIT_LIST_HEAD(&obj
->list
);
114 INIT_LIST_HEAD(&obj
->events
);
125 struct ubus_object
*ubusd_create_object(struct ubus_client
*cl
, struct blob_attr
**attr
)
127 struct ubus_object
*obj
;
128 struct ubus_object_type
*type
= NULL
;
130 if (attr
[UBUS_ATTR_OBJTYPE
])
131 type
= ubus_get_obj_type(blob_get_u32(attr
[UBUS_ATTR_OBJTYPE
]));
132 else if (attr
[UBUS_ATTR_SIGNATURE
])
133 type
= ubus_create_obj_type(attr
[UBUS_ATTR_SIGNATURE
]);
135 if (!!type
^ !!attr
[UBUS_ATTR_OBJPATH
])
138 obj
= ubusd_create_object_internal(type
, 0);
140 ubus_unref_object_type(type
);
145 if (attr
[UBUS_ATTR_OBJPATH
]) {
146 obj
->path
.key
= strdup(blob_data(attr
[UBUS_ATTR_OBJPATH
]));
150 if (avl_insert(&path
, &obj
->path
) != 0) {
151 free((void *) obj
->path
.key
);
152 obj
->path
.key
= NULL
;
155 ubusd_send_obj_event(obj
, true);
159 list_add(&obj
->list
, &cl
->objects
);
164 ubusd_free_object(obj
);
168 void ubusd_free_object(struct ubus_object
*obj
)
170 ubusd_event_cleanup_object(obj
);
172 ubusd_send_obj_event(obj
, false);
173 avl_delete(&path
, &obj
->path
);
174 free((void *) obj
->path
.key
);
176 if (!list_empty(&obj
->list
))
177 list_del(&obj
->list
);
178 ubus_free_id(&objects
, &obj
->id
);
180 ubus_unref_object_type(obj
->type
);
184 static void __init
ubusd_obj_init(void)
186 ubus_init_id_tree(&objects
);
187 ubus_init_id_tree(&obj_types
);
188 ubus_init_string_tree(&path
, false);