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 <linux/stat.h>
|
|
|
|
|
|
|
|
static int devfs_read(struct device_d *_dev, FILE *f, void *buf, size_t size)
|
|
|
|
{
|
|
|
|
struct device_d *dev = f->inode;
|
2007-07-05 16:01:53 +00:00
|
|
|
|
|
|
|
return dev_read(dev, buf, size, f->pos, f->flags);
|
2007-07-05 16:01:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int devfs_write(struct device_d *_dev, FILE *f, const void *buf, size_t size)
|
|
|
|
{
|
|
|
|
struct device_d *dev = f->inode;
|
2007-07-05 16:01:53 +00:00
|
|
|
|
|
|
|
return dev_write(dev, buf, size, f->pos, f->flags);
|
2007-07-05 16:01:39 +00:00
|
|
|
}
|
|
|
|
|
2007-07-05 16:01:54 +00:00
|
|
|
static int devfs_erase(struct device_d *_dev, FILE *f, size_t count, unsigned long offset)
|
|
|
|
{
|
|
|
|
struct device_d *dev = f->inode;
|
|
|
|
|
|
|
|
return dev_erase(dev, count, offset);
|
|
|
|
}
|
|
|
|
|
2007-07-05 16:01:39 +00:00
|
|
|
static int devfs_open(struct device_d *_dev, FILE *file, const char *filename)
|
|
|
|
{
|
|
|
|
struct device_d *dev = get_device_by_id(filename + 1);
|
|
|
|
|
|
|
|
if (!dev)
|
|
|
|
return -ENOENT;
|
|
|
|
|
|
|
|
file->size = dev->size;
|
|
|
|
file->inode = dev;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int devfs_close(struct device_d *dev, FILE *f)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2007-07-05 16:01:52 +00:00
|
|
|
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-07-05 16:01:52 +00:00
|
|
|
dir = malloc(sizeof(DIR));
|
2007-07-05 16:01:39 +00:00
|
|
|
if (!dir)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
dir->priv = get_first_device();
|
|
|
|
|
|
|
|
return dir;
|
|
|
|
}
|
|
|
|
|
2007-07-05 16:01:52 +00:00
|
|
|
struct dirent* devfs_readdir(struct device_d *_dev, DIR *dir)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
|
|
|
struct device_d *dev = dir->priv;
|
|
|
|
|
2007-07-05 16:01:44 +00:00
|
|
|
while (dev && (!strlen(dev->id) || !dev->driver))
|
2007-07-05 16:01:39 +00:00
|
|
|
dev = dev->next;
|
|
|
|
|
|
|
|
if (dev) {
|
2007-07-05 16:01:52 +00:00
|
|
|
strcpy(dir->d.d_name, dev->id);
|
2007-07-05 16:01:39 +00:00
|
|
|
dir->priv = dev->next;
|
|
|
|
return &dir->d;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2007-07-05 16:01:52 +00:00
|
|
|
int devfs_closedir(struct device_d *dev, DIR *dir)
|
2007-07-05 16:01:39 +00:00
|
|
|
{
|
|
|
|
free(dir);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int devfs_stat(struct device_d *_dev, const char *filename, struct stat *s)
|
|
|
|
{
|
|
|
|
struct device_d *dev;
|
|
|
|
|
|
|
|
dev = get_device_by_id(filename + 1);
|
|
|
|
if (!dev)
|
|
|
|
return -ENOENT;
|
|
|
|
|
2007-07-05 16:01:44 +00:00
|
|
|
if (!dev->driver)
|
|
|
|
return -ENXIO;
|
|
|
|
|
2007-07-05 16:01:39 +00:00
|
|
|
s->st_mode = S_IFCHR;
|
|
|
|
s->st_size = dev->size;
|
|
|
|
if (dev->driver->write)
|
|
|
|
s->st_mode |= S_IWUSR;
|
|
|
|
if (dev->driver->read)
|
|
|
|
s->st_mode |= S_IRUSR;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int devfs_probe(struct device_d *dev)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct fs_driver_d devfs_driver = {
|
|
|
|
.type = FS_TYPE_DEVFS,
|
|
|
|
.read = devfs_read,
|
|
|
|
.write = devfs_write,
|
|
|
|
.open = devfs_open,
|
|
|
|
.close = devfs_close,
|
|
|
|
.opendir = devfs_opendir,
|
|
|
|
.readdir = devfs_readdir,
|
|
|
|
.closedir = devfs_closedir,
|
|
|
|
.stat = devfs_stat,
|
2007-07-05 16:01:54 +00:00
|
|
|
.erase = devfs_erase,
|
2007-07-05 16:01:39 +00:00
|
|
|
.flags = FS_DRIVER_NO_DEV,
|
|
|
|
.drv = {
|
|
|
|
.type = DEVICE_TYPE_FS,
|
|
|
|
.probe = devfs_probe,
|
|
|
|
.name = "devfs",
|
|
|
|
.type_data = &devfs_driver,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
int devfs_init(void)
|
|
|
|
{
|
|
|
|
return register_driver(&devfs_driver.drv);
|
|
|
|
}
|
|
|
|
|
|
|
|
device_initcall(devfs_init);
|
|
|
|
|