summaryrefslogtreecommitdiffstats
path: root/fs
diff options
context:
space:
mode:
authorSascha Hauer <s.hauer@pengutronix.de>2012-02-14 23:07:57 +0100
committerSascha Hauer <s.hauer@pengutronix.de>2012-02-17 10:27:15 +0100
commit9bf08d582a5016ed7c74bffe23901fa06ed27afa (patch)
tree0c2d16ad2b55b19d3a53624c71920f26a2f0ed5d /fs
parent33b5ead7e63a8051805adf38b3c9728283e18f2b (diff)
downloadbarebox-9bf08d582a5016ed7c74bffe23901fa06ed27afa.tar.gz
barebox-9bf08d582a5016ed7c74bffe23901fa06ed27afa.tar.xz
fs: put fs devices on its own bus
By putting the fs devices/drivers on a bus on its own we can hook into the bus remove function to cleanup the fs device. This way we can unmount a device by simply unregistering the device which is useful for for example USB mass storage devices. These can now unregister the assoiated filesystems by unregistering their child devices. Signed-off-by: Sascha Hauer <s.hauer@pengutronix.de>
Diffstat (limited to 'fs')
-rw-r--r--fs/fs.c122
1 files changed, 76 insertions, 46 deletions
diff --git a/fs/fs.c b/fs/fs.c
index 3e65c64209..637a01f8c5 100644
--- a/fs/fs.c
+++ b/fs/fs.c
@@ -697,11 +697,67 @@ int close(int fd)
}
EXPORT_SYMBOL(close);
-static LIST_HEAD(fs_driver_list);
+static int fs_match(struct device_d *dev, struct driver_d *drv)
+{
+ return strcmp(dev->name, drv->name) ? -1 : 0;
+}
+
+static int fs_probe(struct device_d *dev)
+{
+ struct fs_device_d *fsdev = container_of(dev, struct fs_device_d, dev);
+ struct mtab_entry *entry = &fsdev->mtab;
+ int ret;
+
+ ret = dev->driver->probe(dev);
+ if (ret)
+ return ret;
+
+ if (fsdev->cdev) {
+ dev_add_child(fsdev->cdev->dev, &fsdev->dev);
+ entry->parent_device = fsdev->cdev->dev;
+ }
+
+ entry->dev = &fsdev->dev;
+
+ list_add_tail(&entry->list, &mtab_list);
+
+ if (!mtab_root)
+ mtab_root = entry;
+
+ return 0;
+}
+
+static void fs_remove(struct device_d *dev)
+{
+ struct fs_device_d *fsdev = container_of(dev, struct fs_device_d, dev);
+ struct mtab_entry *entry = &fsdev->mtab;
+
+ if (fsdev->dev.driver) {
+ dev->driver->remove(dev);
+ list_del(&entry->list);
+ }
+
+ free(entry->path);
+
+ if (entry == mtab_root)
+ mtab_root = NULL;
+
+ free(fsdev->backingstore);
+ free(fsdev);
+}
+
+struct bus_type fs_bus = {
+ .name = "fs",
+ .match = fs_match,
+ .probe = fs_probe,
+ .remove = fs_remove,
+};
int register_fs_driver(struct fs_driver_d *fsdrv)
{
+ fsdrv->drv.bus = &fs_bus;
register_driver(&fsdrv->drv);
+
return 0;
}
EXPORT_SYMBOL(register_fs_driver);
@@ -714,10 +770,7 @@ EXPORT_SYMBOL(register_fs_driver);
*/
int mount(const char *device, const char *fsname, const char *_path)
{
- struct mtab_entry *entry;
struct fs_device_d *fsdev;
- struct device_d *parent_device = NULL;
- struct cdev *cdev = NULL;
int ret;
char *path = normalise_path(_path);
@@ -728,17 +781,17 @@ int mount(const char *device, const char *fsname, const char *_path)
if (strchr(path + 1, '/')) {
printf("mounting allowed on first directory level only\n");
errno = -EBUSY;
- goto out;
+ goto err_free_path;
}
if (mtab_root) {
if (path_check_prereq(path, S_IFDIR))
- goto out;
+ goto err_free_path;
} else {
/* no mtab, so we only allow to mount on '/' */
if (*path != '/' || *(path + 1)) {
errno = -ENOTDIR;
- goto out;
+ goto err_free_path;
}
}
@@ -747,51 +800,37 @@ int mount(const char *device, const char *fsname, const char *_path)
safe_strncpy(fsdev->dev.name, fsname, MAX_DRIVER_NAME);
fsdev->dev.type_data = fsdev;
fsdev->dev.id = get_free_deviceid(fsdev->dev.name);
+ fsdev->mtab.path = xstrdup(path);
+ fsdev->dev.bus = &fs_bus;
+
+ if (!strncmp(device, "/dev/", 5))
+ fsdev->cdev = cdev_by_name(device + 5);
if ((ret = register_device(&fsdev->dev))) {
errno = ret;
- goto out1;
+ goto err_register;
}
if (!fsdev->dev.driver) {
- /* driver didn't accept the device. Bail out */
+ /*
+ * Driver didn't accept the device or no driver for this
+ * device. Bail out
+ */
errno = -EINVAL;
- goto out2;
+ goto err_no_driver;
}
- if (!strncmp(device, "/dev/", 5)) {
- cdev = cdev_by_name(device + 5);
- if(cdev)
- parent_device = cdev->dev;
- }
-
- if (parent_device)
- dev_add_child(parent_device, &fsdev->dev);
-
- /* add mtab entry */
- entry = &fsdev->mtab;
- entry->path = xstrdup(path);
- entry->dev = &fsdev->dev;
- entry->parent_device = parent_device;
-
- list_add_tail(&entry->list, &mtab_list);
-
- if (!mtab_root)
- mtab_root = entry;
-
errno = 0;
- free(path);
return 0;
-out2:
+err_no_driver:
unregister_device(&fsdev->dev);
-out1:
- if (fsdev->backingstore)
- free(fsdev->backingstore);
- free(fsdev);
-out:
+err_register:
+ fs_remove(&fsdev->dev);
+err_free_path:
free(path);
+
return errno;
}
EXPORT_SYMBOL(mount);
@@ -800,7 +839,6 @@ int umount(const char *pathname)
{
struct mtab_entry *entry = NULL, *e;
char *p = normalise_path(pathname);
- struct fs_device_d *fsdev;
for_each_mtab_entry(e) {
if (!strcmp(p, e->path)) {
@@ -821,15 +859,7 @@ int umount(const char *pathname)
return errno;
}
- free(entry->path);
- list_del(&entry->list);
- if (entry == mtab_root)
- mtab_root = NULL;
-
unregister_device(entry->dev);
- fsdev = entry->dev->type_data;
- free(fsdev->backingstore);
- free(fsdev);
return 0;
}