9
0
Fork 0
barebox/fs/devfs.c

350 lines
6.9 KiB
C
Raw Normal View History

2007-07-05 16:02:19 +00:00
/*
* devfs.c - a device file system for U-Boot
*
* 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.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
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>
#include <xfuncs.h>
2007-07-05 16:01:39 +00:00
#include <linux/stat.h>
#include <ioctl.h>
#include <nand.h>
#include <linux/mtd/mtd-abi.h>
#include <partition.h>
2007-07-05 16:01:39 +00:00
static LIST_HEAD(cdev_list);
struct cdev *cdev_by_name(const char *filename)
2007-07-05 16:01:39 +00:00
{
struct cdev *cdev;
2007-07-05 16:01:53 +00:00
list_for_each_entry(cdev, &cdev_list, list) {
if (!strcmp(cdev->name, filename))
return cdev;
}
return NULL;
2007-07-05 16:01:39 +00:00
}
int devfs_read(struct device_d *_dev, FILE *f, void *buf, size_t size)
2007-07-05 16:01:39 +00:00
{
struct cdev *cdev = f->inode;
if (!cdev->ops->read)
return -ENOSYS;
2007-07-05 16:01:53 +00:00
return cdev->ops->read(cdev, buf, size,
f->pos + cdev->offset, f->flags);
2007-07-05 16:01:39 +00:00
}
int devfs_write(struct device_d *_dev, FILE *f, const void *buf, size_t size)
{
struct cdev *cdev = f->inode;
if (!cdev->ops->write)
return -ENOSYS;
return cdev->ops->write(cdev, buf, size,
f->pos + cdev->offset, f->flags);
}
off_t devfs_lseek(struct device_d *_dev, FILE *f, off_t pos)
{
struct cdev *cdev = f->inode;
off_t ret = -1;
if (cdev->ops->lseek)
ret = cdev->ops->lseek(cdev, pos + cdev->offset);
if (ret != -1)
f->pos = pos;
return ret;
}
int devfs_erase(struct device_d *_dev, FILE *f, size_t count, unsigned long offset)
2007-07-05 16:01:54 +00:00
{
struct cdev *cdev = f->inode;
if (!cdev->ops->erase)
return -ENOSYS;
2007-07-05 16:01:54 +00:00
return cdev->ops->erase(cdev, count, offset + cdev->offset);
2007-07-05 16:01:54 +00:00
}
int devfs_protect(struct device_d *_dev, FILE *f, size_t count, unsigned long offset, int prot)
2007-07-16 08:29:28 +00:00
{
struct cdev *cdev = f->inode;
2007-07-16 08:29:28 +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
}
static int devfs_memmap(struct device_d *_dev, FILE *f, void **map, int flags)
{
struct cdev *cdev = f->inode;
int ret = -ENOSYS;
if (!cdev->ops->memmap)
return -EINVAL;
ret = cdev->ops->memmap(cdev, map, flags);
if (!ret)
*map = (void *)((unsigned long)*map + cdev->offset);
return ret;
}
static int devfs_open(struct device_d *_dev, FILE *f, const char *filename)
2007-07-05 16:01:39 +00:00
{
struct cdev *cdev;
2007-07-05 16:01:39 +00:00
cdev = cdev_by_name(filename + 1);
if (!cdev)
2007-07-05 16:01:39 +00:00
return -ENOENT;
f->size = cdev->size;
f->inode = cdev;
if (cdev->ops->open)
return cdev->ops->open(cdev, f);
return 0;
2007-07-05 16:01:39 +00:00
}
static int devfs_close(struct device_d *_dev, FILE *f)
2007-07-05 16:01:39 +00:00
{
struct cdev *cdev = f->inode;
if (cdev->ops->close)
return cdev->ops->close(cdev, f);
return 0;
}
static int partition_ioctl(struct cdev *cdev, int request, void *buf)
{
size_t offset;
switch (request) {
case MEMSETBADBLOCK:
case MEMGETBADBLOCK:
offset = (off_t)buf;
offset += cdev->offset;
return cdev->ops->ioctl(cdev, request, (void *)offset);
case MEMGETINFO:
return cdev->ops->ioctl(cdev, request, buf);
default:
return -EINVAL;
}
}
static int devfs_ioctl(struct device_d *_dev, FILE *f, int request, void *buf)
{
struct cdev *cdev = f->inode;
int ret = -EINVAL;
if (!cdev->ops->ioctl)
goto out;
if (cdev->flags & DEVFS_IS_PARTITION)
ret = partition_ioctl(cdev, request, buf);
else
ret = cdev->ops->ioctl(cdev, request, buf);
out:
return ret;
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)
{
if (size > f->dev->size)
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
{
DIR *dir;
2007-07-05 16:01:39 +00:00
dir = xzalloc(sizeof(DIR));
2007-07-05 16:01:39 +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
{
struct cdev *cdev = dir->priv;
2007-07-05 16:01:39 +00:00
if (!cdev)
return NULL;
2007-07-05 16:01:39 +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
{
struct cdev *cdev;
2007-07-05 16:01:39 +00:00
cdev = cdev_by_name(filename + 1);
if (!cdev)
2007-07-05 16:01:39 +00:00
return -ENOENT;
s->st_mode = S_IFCHR;
s->st_size = cdev->size;
if (cdev->ops->write)
2007-07-05 16:01:39 +00:00
s->st_mode |= S_IWUSR;
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
{
return 0;
}
static void devfs_delete(struct device_d *dev)
{
}
2007-07-05 16:01:39 +00:00
static struct fs_driver_d devfs_driver = {
.type = FS_TYPE_DEVFS,
.read = devfs_read,
.write = devfs_write,
.lseek = devfs_lseek,
2007-07-05 16:01:39 +00:00
.open = devfs_open,
.close = devfs_close,
.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,
.memmap = devfs_memmap,
2007-07-05 16:01:39 +00:00
.flags = FS_DRIVER_NO_DEV,
.drv = {
.type = DEVICE_TYPE_FS,
.probe = devfs_probe,
.remove = devfs_delete,
2007-07-05 16:01:39 +00:00
.name = "devfs",
.type_data = &devfs_driver,
}
};
2007-09-28 08:07:26 +00:00
static int devfs_init(void)
2007-07-05 16:01:39 +00:00
{
return register_driver(&devfs_driver.drv);
}
device_initcall(devfs_init);
int devfs_create(struct cdev *new)
{
struct cdev *cdev;
cdev = cdev_by_name(new->name);
if (cdev)
return -EEXIST;
list_add_tail(&new->list, &cdev_list);
return 0;
}
void devfs_remove(struct cdev *cdev)
{
list_del(&cdev->list);
}
int devfs_add_partition(const char *devname, unsigned long offset, size_t size,
int flags, const char *name)
{
struct cdev *cdev, *new;
cdev = cdev_by_name(name);
if (cdev)
return -EEXIST;
cdev = cdev_by_name(devname);
if (!cdev)
return -ENOENT;
if (offset + size > cdev->size)
return -EINVAL;
new = xzalloc(sizeof (*new));
new->name = strdup(name);
new->ops = cdev->ops;
new->priv = cdev->priv;
new->size = size;
new->offset = offset + cdev->offset;
new->dev = cdev->dev;
new->flags = flags | DEVFS_IS_PARTITION;
list_add_tail(&new->list, &cdev_list);
return 0;
}
int devfs_del_partition(const char *name)
{
struct cdev *cdev;
cdev = cdev_by_name(name);
if (!cdev)
return -ENOENT;
if (!(cdev->flags & DEVFS_IS_PARTITION))
return -EINVAL;
if (cdev->flags & DEVFS_PARTITION_FIXED)
return -EPERM;
devfs_remove(cdev);
free(cdev->name);
free(cdev);
return 0;
}