2007-07-05 16:02:19 +00:00
|
|
|
/*
|
2009-12-15 08:11:09 +00:00
|
|
|
* devfs.c - a device file system for barebox
|
2007-07-05 16:02:19 +00:00
|
|
|
*
|
|
|
|
* Copyright (c) 2007 Sascha Hauer <s.hauer@pengutronix.de>, Pengutronix
|
|
|
|
*
|
|
|
|
* See file CREDITS for list of people who contributed to this
|
|
|
|
* project.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2
|
|
|
|
* as published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2007-07-05 16:01:39 +00:00
|
|
|
#include <common.h>
|
|
|
|
#include <driver.h>
|
|
|
|
#include <init.h>
|
|
|
|
#include <malloc.h>
|
|
|
|
#include <fs.h>
|
|
|
|
#include <command.h>
|
|
|
|
#include <errno.h>
|
2007-10-11 18:20:36 +00:00
|
|
|
#include <xfuncs.h>
|
2007-07-05 16:01:39 +00:00
|
|
|
#include <linux/stat.h>
|
2009-04-22 21:39:27 +00:00
|
|
|
#include <ioctl.h>
|
|
|
|
#include <nand.h>
|
2010-06-28 08:21:57 +00:00
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/mtd/mtd.h>
|
2009-04-22 21:39:27 +00:00
|
|
|
#include <linux/mtd/mtd-abi.h>
|
|
|
|
#include <partition.h>
|
2007-07-05 16:01:39 +00:00
|
|
|
|
2011-04-04 11:13:30 +00:00
|
|
|
extern struct list_head cdev_list;
|
2009-06-11 13:33:49 +00:00
|
|
|
|
|
|
|
static int devfs_read(struct device_d *_dev, FILE *f, void *buf, size_t size)
|
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2009-06-11 13:33:49 +00:00
|
|
|
|
|
|
|
return cdev_read(cdev, buf, size, f->pos, f->flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int devfs_write(struct device_d *_dev, FILE *f, const void *buf, size_t size)
|
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2009-06-11 13:33:49 +00:00
|
|
|
|
2012-06-25 12:03:27 +00:00
|
|
|
if (cdev->flags & DEVFS_PARTITION_READONLY)
|
|
|
|
return -EPERM;
|
|
|
|
|
2009-06-11 13:33:49 +00:00
|
|
|
return cdev_write(cdev, buf, size, f->pos, f->flags);
|
2009-04-22 21:39:27 +00:00
|
|
|
}
|
|
|
|
|
2011-10-14 11:46:09 +00:00
|
|
|
static loff_t devfs_lseek(struct device_d *_dev, FILE *f, loff_t pos)
|
2009-04-22 21:39:27 +00:00
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2011-10-19 07:27:47 +00:00
|
|
|
loff_t ret = -1;
|
2009-04-22 21:39:27 +00:00
|
|
|
|
|
|
|
if (cdev->ops->lseek)
|
|
|
|
ret = cdev->ops->lseek(cdev, pos + cdev->offset);
|
2008-06-06 07:25:13 +00:00
|
|
|
|
2008-08-01 06:59:27 +00:00
|
|
|
if (ret != -1)
|
2008-06-06 07:25:13 +00:00
|
|
|
f->pos = pos;
|
|
|
|
|
2009-09-25 10:17:21 +00:00
|
|
|
return ret - cdev->offset;
|
2008-06-06 07:25:13 +00:00
|
|
|
}
|
|
|
|
|
2016-02-08 13:23:12 +00:00
|
|
|
static int devfs_erase(struct device_d *_dev, FILE *f, loff_t count, loff_t offset)
|
2007-07-05 16:01:54 +00:00
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2009-04-22 21:39:27 +00:00
|
|
|
|
2012-06-25 12:03:27 +00:00
|
|
|
if (cdev->flags & DEVFS_PARTITION_READONLY)
|
|
|
|
return -EPERM;
|
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
if (!cdev->ops->erase)
|
|
|
|
return -ENOSYS;
|
2007-07-05 16:01:54 +00:00
|
|
|
|
2012-06-06 16:04:59 +00:00
|
|
|
if (count + offset > cdev->size)
|
|
|
|
count = cdev->size - offset;
|
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
return cdev->ops->erase(cdev, count, offset + cdev->offset);
|
2007-07-05 16:01:54 +00:00
|
|
|
}
|
|
|
|
|
2011-10-14 11:46:09 +00:00
|
|
|
static int devfs_protect(struct device_d *_dev, FILE *f, size_t count, loff_t offset, int prot)
|
2007-07-16 08:29:28 +00:00
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2007-07-16 08:29:28 +00:00
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
if (!cdev->ops->protect)
|
|
|
|
return -ENOSYS;
|
|
|
|
|
|
|
|
return cdev->ops->protect(cdev, count, offset + cdev->offset, prot);
|
2007-07-16 08:29:28 +00:00
|
|
|
}
|
|
|
|
|
2007-07-15 11:50:04 +00:00
|
|
|
static int devfs_memmap(struct device_d *_dev, FILE *f, void **map, int flags)
|
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2009-04-22 21:39:27 +00:00
|
|
|
int ret = -ENOSYS;
|
|
|
|
|
|
|
|
if (!cdev->ops->memmap)
|
|
|
|
return -EINVAL;
|
2007-07-15 11:50:04 +00:00
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
ret = cdev->ops->memmap(cdev, map, flags);
|
|
|
|
|
|
|
|
if (!ret)
|
2011-10-19 07:27:47 +00:00
|
|
|
*map = (void *)((unsigned long)*map + (unsigned long)cdev->offset);
|
2009-04-22 21:39:27 +00:00
|
|
|
|
|
|
|
return ret;
|
2007-07-15 11:50:04 +00:00
|
|
|
}
|
|
|
|
|
2008-06-06 07:25:13 +00:00
|
|
|
static int devfs_open(struct device_d *_dev, FILE *f, const char *filename)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
2009-04-22 21:39:27 +00:00
|
|
|
struct cdev *cdev;
|
2009-09-10 10:07:10 +00:00
|
|
|
int ret;
|
2007-07-05 16:01:39 +00:00
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
cdev = cdev_by_name(filename + 1);
|
|
|
|
|
|
|
|
if (!cdev)
|
2007-07-05 16:01:39 +00:00
|
|
|
return -ENOENT;
|
|
|
|
|
2012-07-03 08:12:40 +00:00
|
|
|
f->size = cdev->flags & DEVFS_IS_CHARACTER_DEV ?
|
|
|
|
FILE_SIZE_STREAM : cdev->size;
|
2014-12-02 16:29:05 +00:00
|
|
|
f->priv = cdev;
|
2009-04-22 21:39:27 +00:00
|
|
|
|
2009-09-10 10:07:10 +00:00
|
|
|
if (cdev->ops->open) {
|
2011-12-11 11:38:04 +00:00
|
|
|
ret = cdev->ops->open(cdev, f->flags);
|
2009-09-10 10:07:10 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
cdev->open++;
|
2009-04-22 21:39:27 +00:00
|
|
|
|
|
|
|
return 0;
|
2007-07-05 16:01:39 +00:00
|
|
|
}
|
|
|
|
|
2008-06-06 07:25:13 +00:00
|
|
|
static int devfs_close(struct device_d *_dev, FILE *f)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2009-09-10 10:07:10 +00:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (cdev->ops->close) {
|
2011-04-04 11:12:38 +00:00
|
|
|
ret = cdev->ops->close(cdev);
|
2009-09-10 10:07:10 +00:00
|
|
|
if (ret)
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
cdev->open--;
|
2009-04-22 21:39:27 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2008-06-06 07:25:13 +00:00
|
|
|
|
2011-03-25 10:04:53 +00:00
|
|
|
static int devfs_flush(struct device_d *_dev, FILE *f)
|
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2011-03-25 10:04:53 +00:00
|
|
|
|
|
|
|
if (cdev->ops->flush)
|
|
|
|
return cdev->ops->flush(cdev);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-06-06 07:25:13 +00:00
|
|
|
static int devfs_ioctl(struct device_d *_dev, FILE *f, int request, void *buf)
|
|
|
|
{
|
2014-12-02 16:29:05 +00:00
|
|
|
struct cdev *cdev = f->priv;
|
2010-06-29 13:56:36 +00:00
|
|
|
|
2011-04-04 11:13:30 +00:00
|
|
|
return cdev_ioctl(cdev, request, buf);
|
2007-07-05 16:01:39 +00:00
|
|
|
}
|
|
|
|
|
2007-07-05 19:46:42 +00:00
|
|
|
static int devfs_truncate(struct device_d *dev, FILE *f, ulong size)
|
|
|
|
{
|
2014-10-08 11:48:12 +00:00
|
|
|
if (f->fsdev->dev.num_resources < 1)
|
2011-07-29 07:15:38 +00:00
|
|
|
return -ENOSPC;
|
2014-10-08 11:48:12 +00:00
|
|
|
if (size > resource_size(&f->fsdev->dev.resource[0]))
|
2007-07-05 19:46:42 +00:00
|
|
|
return -ENOSPC;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-09-28 08:07:26 +00:00
|
|
|
static DIR* devfs_opendir(struct device_d *dev, const char *pathname)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
2007-07-05 16:01:52 +00:00
|
|
|
DIR *dir;
|
2007-07-05 16:01:39 +00:00
|
|
|
|
2007-10-11 18:20:36 +00:00
|
|
|
dir = xzalloc(sizeof(DIR));
|
2007-07-05 16:01:39 +00:00
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
if (!list_empty(&cdev_list))
|
|
|
|
dir->priv = list_first_entry(&cdev_list, struct cdev, list);
|
2007-07-05 16:01:39 +00:00
|
|
|
|
|
|
|
return dir;
|
|
|
|
}
|
|
|
|
|
2007-09-28 08:07:26 +00:00
|
|
|
static struct dirent* devfs_readdir(struct device_d *_dev, DIR *dir)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
2009-04-22 21:39:27 +00:00
|
|
|
struct cdev *cdev = dir->priv;
|
2007-07-05 16:01:39 +00:00
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
if (!cdev)
|
2007-10-11 18:20:36 +00:00
|
|
|
return NULL;
|
2007-07-05 16:01:39 +00:00
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
list_for_each_entry_from(cdev, &cdev_list, list) {
|
|
|
|
strcpy(dir->d.d_name, cdev->name);
|
|
|
|
dir->priv = list_entry(cdev->list.next, struct cdev, list);
|
2007-07-05 16:01:39 +00:00
|
|
|
return &dir->d;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2007-09-28 08:07:26 +00:00
|
|
|
static int devfs_closedir(struct device_d *dev, DIR *dir)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
|
|
|
free(dir);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-09-28 08:07:26 +00:00
|
|
|
static int devfs_stat(struct device_d *_dev, const char *filename, struct stat *s)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
2009-04-22 21:39:27 +00:00
|
|
|
struct cdev *cdev;
|
2007-07-05 16:01:39 +00:00
|
|
|
|
2015-12-10 08:31:02 +00:00
|
|
|
cdev = lcdev_by_name(filename + 1);
|
2009-04-22 21:39:27 +00:00
|
|
|
if (!cdev)
|
2007-07-05 16:01:39 +00:00
|
|
|
return -ENOENT;
|
|
|
|
|
|
|
|
s->st_mode = S_IFCHR;
|
2009-04-22 21:39:27 +00:00
|
|
|
s->st_size = cdev->size;
|
2015-12-10 08:31:02 +00:00
|
|
|
|
|
|
|
if (cdev->link)
|
|
|
|
s->st_mode |= S_IFLNK;
|
|
|
|
|
|
|
|
cdev = cdev_readlink(cdev);
|
|
|
|
|
2009-04-22 21:39:27 +00:00
|
|
|
if (cdev->ops->write)
|
2007-07-05 16:01:39 +00:00
|
|
|
s->st_mode |= S_IWUSR;
|
2009-04-22 21:39:27 +00:00
|
|
|
if (cdev->ops->read)
|
2007-07-05 16:01:39 +00:00
|
|
|
s->st_mode |= S_IRUSR;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-09-28 08:07:26 +00:00
|
|
|
static int devfs_probe(struct device_d *dev)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
2015-10-13 07:34:15 +00:00
|
|
|
struct fs_device_d *fsdev = dev_to_fs_device(dev);
|
|
|
|
|
|
|
|
if (strcmp(fsdev->path, "/dev")) {
|
|
|
|
dev_err(dev, "devfs can only be mounted on /dev/\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2007-07-05 16:01:39 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-10 19:11:45 +00:00
|
|
|
static void devfs_delete(struct device_d *dev)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-12-10 08:31:02 +00:00
|
|
|
static int devfs_readlink(struct device_d *dev, const char *pathname,
|
|
|
|
char *buf, size_t bufsz)
|
|
|
|
{
|
|
|
|
struct cdev *cdev;
|
|
|
|
|
|
|
|
cdev = cdev_by_name(pathname + 1);
|
|
|
|
if (!cdev)
|
|
|
|
return -ENOENT;
|
|
|
|
|
|
|
|
while (cdev->link)
|
|
|
|
cdev = cdev->link;
|
|
|
|
|
|
|
|
bufsz = min(bufsz, strlen(cdev->name));
|
|
|
|
memcpy(buf, cdev->name, bufsz);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-05 16:01:39 +00:00
|
|
|
static struct fs_driver_d devfs_driver = {
|
|
|
|
.read = devfs_read,
|
|
|
|
.write = devfs_write,
|
2008-06-06 07:25:13 +00:00
|
|
|
.lseek = devfs_lseek,
|
2007-07-05 16:01:39 +00:00
|
|
|
.open = devfs_open,
|
|
|
|
.close = devfs_close,
|
2011-03-25 10:04:53 +00:00
|
|
|
.flush = devfs_flush,
|
2008-06-06 07:25:13 +00:00
|
|
|
.ioctl = devfs_ioctl,
|
2007-07-05 16:01:39 +00:00
|
|
|
.opendir = devfs_opendir,
|
|
|
|
.readdir = devfs_readdir,
|
2007-07-05 19:46:42 +00:00
|
|
|
.truncate = devfs_truncate,
|
2007-07-05 16:01:39 +00:00
|
|
|
.closedir = devfs_closedir,
|
|
|
|
.stat = devfs_stat,
|
2007-07-05 16:01:54 +00:00
|
|
|
.erase = devfs_erase,
|
2007-07-16 08:29:28 +00:00
|
|
|
.protect = devfs_protect,
|
2007-07-15 11:50:04 +00:00
|
|
|
.memmap = devfs_memmap,
|
2015-12-10 08:31:02 +00:00
|
|
|
.readlink = devfs_readlink,
|
2007-07-05 16:01:39 +00:00
|
|
|
.flags = FS_DRIVER_NO_DEV,
|
|
|
|
.drv = {
|
|
|
|
.probe = devfs_probe,
|
2009-06-10 19:11:45 +00:00
|
|
|
.remove = devfs_delete,
|
2007-07-05 16:01:39 +00:00
|
|
|
.name = "devfs",
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2007-09-28 08:07:26 +00:00
|
|
|
static int devfs_init(void)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
2009-06-10 19:15:06 +00:00
|
|
|
return register_fs_driver(&devfs_driver);
|
2007-07-05 16:01:39 +00:00
|
|
|
}
|
|
|
|
|
2009-06-10 21:41:27 +00:00
|
|
|
coredevice_initcall(devfs_init);
|