device_free_unused(NULL);
}
+static int set_device_state(struct device *dev, bool state)
+{
+ if (state)
+ system_if_up(dev);
+ else
+ system_if_down(dev);
+
+ return 0;
+}
+
+static int
+simple_device_set_state(struct device *dev, bool state)
+{
+ struct device *pdev;
+ int ret = 0;
+
+ pdev = dev->parent.dev;
+ if (state && !pdev) {
+ pdev = system_if_get_parent(dev);
+ if (pdev)
+ device_add_user(&dev->parent, pdev);
+ }
+
+ if (pdev) {
+ if (state)
+ ret = device_claim(&dev->parent);
+ else
+ device_release(&dev->parent);
+
+ if (ret < 0)
+ return ret;
+ }
+ return set_device_state(dev, state);
+}
+
static struct device *
simple_device_create(const char *name, struct blob_attr *attr)
{
if (!dev)
return NULL;
+ dev->set_state = simple_device_set_state;
device_init_settings(dev, tb);
return dev;
static void simple_device_free(struct device *dev)
{
+ if (dev->parent.dev)
+ device_remove_user(&dev->parent);
device_cleanup(dev);
free(dev);
}
device_unlock();
}
-static int set_device_state(struct device *dev, bool state)
-{
- if (state)
- system_if_up(dev);
- else
- system_if_down(dev);
-
- return 0;
-}
-
int device_claim(struct device_user *dep)
{
struct device *dev = dep->dev;
}
static struct device *
-device_create_default(const char *name)
+device_create_default(const char *name, bool external)
{
struct device *dev;
D(DEVICE, "Create simple device '%s'\n", name);
dev = calloc(1, sizeof(*dev));
+ dev->external = external;
+ dev->set_state = simple_device_set_state;
device_init(dev, &simple_device_type, name);
dev->default_config = true;
return dev;
}
struct device *
-device_get(const char *name, bool create)
+device_get(const char *name, int create)
{
struct device *dev;
if (!create)
return NULL;
- return device_create_default(name);
+ return device_create_default(name, create > 1);
}
static void
}
}
+void
+device_free(struct device *dev)
+{
+ __devlock++;
+ dev->type->free(dev);
+ __devlock--;
+}
+
static void
__device_free_unused(struct device *dev)
{
- if (!list_empty(&dev->users) || dev->current_config || config_init)
+ if (!list_empty(&dev->users) || dev->current_config || __devlock)
return;
device_free(dev);
if (dev->type != &simple_device_type)
continue;
- ndev = device_create_default(dev->ifname);
+ ndev = device_create_default(dev->ifname, dev->external);
device_replace(ndev, dev);
}
}