avl_init(&devices, avl_strcmp, false, NULL);
}
-static void free_device(struct device *dev)
+static void free_simple_device(struct device *dev)
{
cleanup_device(dev);
free(dev);
static int set_device_state(struct device *dev, bool state)
{
- if (state) {
- broadcast_device_event(dev, DEV_EVENT_SETUP);
+ if (state)
system_if_up(dev);
- broadcast_device_event(dev, DEV_EVENT_UP);
- } else {
- broadcast_device_event(dev, DEV_EVENT_TEARDOWN);
+ else
system_if_down(dev);
- broadcast_device_event(dev, DEV_EVENT_DOWN);
- }
+
return 0;
}
if (++dev->active != 1)
return 0;
+ broadcast_device_event(dev, DEV_EVENT_SETUP);
ret = dev->set_state(dev, true);
- if (ret != 0)
+ if (ret == 0)
+ broadcast_device_event(dev, DEV_EVENT_UP);
+ else
dev->active = 0;
return ret;
DPRINTF("release device %s, new refcount: %d\n", dev->ifname, dev->active);
assert(dev->active >= 0);
- if (!dev->active)
- dev->set_state(dev, false);
+ if (dev->active)
+ return;
+
+ broadcast_device_event(dev, DEV_EVENT_TEARDOWN);
+ dev->set_state(dev, false);
+ broadcast_device_event(dev, DEV_EVENT_DOWN);
}
int check_device_state(struct device *dev)
return dev->type->check_state(dev);
}
-int init_device(struct device *dev, const struct device_type *type, const char *ifname)
+void init_virtual_device(struct device *dev, const struct device_type *type, const char *name)
{
- int ret;
-
assert(dev);
assert(type);
- if (ifname)
- strncpy(dev->ifname, ifname, IFNAMSIZ);
+ if (name)
+ strncpy(dev->ifname, name, IFNAMSIZ);
+
+ fprintf(stderr, "Initialize device '%s'\n", dev->ifname);
+ INIT_LIST_HEAD(&dev->users);
+ dev->type = type;
+}
+
+int init_device(struct device *dev, const struct device_type *type, const char *ifname)
+{
+ int ret;
+
+ init_virtual_device(dev, type, ifname);
if (!dev->set_state)
dev->set_state = set_device_state;
- fprintf(stderr, "Initialize interface '%s'\n", dev->ifname);
- INIT_LIST_HEAD(&dev->users);
dev->avl.key = dev->ifname;
- dev->type = type;
ret = avl_insert(&devices, &dev->avl);
if (ret < 0)
return ret;
check_device_state(dev);
+
return 0;
}
static const struct device_type simple_type = {
.name = "Device",
.check_state = system_if_check,
- .free = free_device,
+ .free = free_simple_device,
};
struct device *dev;
{
struct device_user *dep, *tmp;
- fprintf(stderr, "Clean up interface '%s'\n", dev->ifname);
+ fprintf(stderr, "Clean up device '%s'\n", dev->ifname);
list_for_each_entry_safe(dep, tmp, &dev->users, list) {
if (!dep->cb)
continue;
dep->cb(dep, DEV_EVENT_REMOVE);
}
- avl_delete(&devices, &dev->avl);
+ if (dev->avl.key)
+ avl_delete(&devices, &dev->avl);
}
void set_device_present(struct device *dev, bool state)
list_del(&dep->list);
if (list_empty(&dev->users)) {
- /* all references have gone away, remove this interface */
- dev->type->free(dev);
+ /* all references have gone away, remove this device */
+ free_device(dev);
}
dep->dev = NULL;