2013-06-26 22:30:40 +08:00
|
|
|
/*
|
2021-03-08 18:19:04 +08:00
|
|
|
* Copyright (c) 2006-2021, RT-Thread Development Team
|
2013-06-26 22:30:40 +08:00
|
|
|
*
|
2018-10-15 01:46:07 +08:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
2013-06-26 22:30:40 +08:00
|
|
|
*
|
|
|
|
* Change Logs:
|
|
|
|
* Date Author Notes
|
2018-02-11 13:57:51 +08:00
|
|
|
* 2018-02-11 Bernard Ignore O_CREAT flag in open.
|
2013-06-26 22:30:40 +08:00
|
|
|
*/
|
2022-12-03 12:07:44 +08:00
|
|
|
#include <rthw.h>
|
2023-06-10 21:35:25 +08:00
|
|
|
#include <rtdbg.h>
|
2017-10-15 22:44:53 +08:00
|
|
|
#include <rtdevice.h>
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <errno.h>
|
|
|
|
|
2011-10-08 14:50:08 +00:00
|
|
|
#include <dfs.h>
|
|
|
|
#include <dfs_fs.h>
|
2017-10-15 22:44:53 +08:00
|
|
|
#include <dfs_file.h>
|
2023-06-10 21:35:25 +08:00
|
|
|
#include <dfs_dentry.h>
|
|
|
|
#include <dfs_mnt.h>
|
2011-10-08 14:50:08 +00:00
|
|
|
|
|
|
|
#include "devfs.h"
|
|
|
|
|
|
|
|
struct device_dirent
|
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
struct rt_device **devices;
|
|
|
|
uint32_t device_count;
|
|
|
|
};
|
|
|
|
|
|
|
|
int dfs_devfs_open(struct dfs_file *file);
|
|
|
|
int dfs_devfs_close(struct dfs_file *file);
|
2023-07-24 09:12:35 +08:00
|
|
|
off_t generic_dfs_lseek(struct dfs_file *file, off_t offset, int whence);
|
|
|
|
ssize_t dfs_devfs_read(struct dfs_file *file, void *buf, size_t count, off_t *pos);
|
|
|
|
ssize_t dfs_devfs_write(struct dfs_file *file, const void *buf, size_t count, off_t *pos);
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_ioctl(struct dfs_file *file, int cmd, void *args);
|
|
|
|
int dfs_devfs_getdents(struct dfs_file *file, struct dirent *dirp, uint32_t count);
|
|
|
|
static int dfs_devfs_poll(struct dfs_file *file, struct rt_pollreq *req);
|
2023-12-12 18:05:39 +08:00
|
|
|
int dfs_devfs_flush(struct dfs_file *file);
|
|
|
|
off_t dfs_devfs_lseek(struct dfs_file *file, off_t offset, int wherece);
|
|
|
|
int dfs_devfs_truncate(struct dfs_file *file, off_t offset);
|
|
|
|
int dfs_devfs_mmap(struct dfs_file *file, struct lwp_avl_struct *mmap);
|
|
|
|
int dfs_devfs_lock(struct dfs_file *file, struct file_lock *flock);
|
|
|
|
int dfs_devfs_flock(struct dfs_file *file, int operation, struct file_lock *flock);
|
2023-06-10 21:35:25 +08:00
|
|
|
|
|
|
|
int dfs_devfs_mount(struct dfs_mnt *mnt, unsigned long rwflag, const void *data);
|
|
|
|
int dfs_devfs_umount(struct dfs_mnt *mnt);
|
|
|
|
int dfs_devfs_unlink(struct dfs_dentry *dentry);
|
|
|
|
int dfs_devfs_stat(struct dfs_dentry *dentry, struct stat *st);
|
|
|
|
int dfs_devfs_statfs(struct dfs_mnt *mnt, struct statfs *buf);
|
|
|
|
static struct dfs_vnode *dfs_devfs_lookup(struct dfs_dentry *dentry);
|
|
|
|
struct dfs_vnode *dfs_devfs_create_vnode(struct dfs_dentry *dentry, int type, mode_t mode);
|
|
|
|
static int dfs_devfs_free_vnode(struct dfs_vnode *vnode);
|
|
|
|
|
|
|
|
static const struct dfs_file_ops _dev_fops =
|
|
|
|
{
|
2023-12-12 18:05:39 +08:00
|
|
|
.open = dfs_devfs_open,
|
|
|
|
.close = dfs_devfs_close,
|
|
|
|
.lseek = generic_dfs_lseek,
|
|
|
|
.read = dfs_devfs_read,
|
|
|
|
.write = dfs_devfs_write,
|
|
|
|
.ioctl = dfs_devfs_ioctl,
|
|
|
|
.getdents = dfs_devfs_getdents,
|
|
|
|
.poll = dfs_devfs_poll,
|
|
|
|
|
|
|
|
.flush = dfs_devfs_flush,
|
|
|
|
.lseek = dfs_devfs_lseek,
|
|
|
|
.truncate = dfs_devfs_truncate,
|
|
|
|
.mmap = dfs_devfs_mmap,
|
|
|
|
.lock = dfs_devfs_lock,
|
|
|
|
.flock = dfs_devfs_flock,
|
2023-06-10 21:35:25 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static const struct dfs_filesystem_ops _devfs_ops =
|
|
|
|
{
|
|
|
|
.name = "devfs",
|
|
|
|
|
|
|
|
.default_fops = &_dev_fops,
|
|
|
|
.mount = dfs_devfs_mount,
|
|
|
|
.umount = dfs_devfs_umount,
|
|
|
|
.unlink = dfs_devfs_unlink,
|
|
|
|
.stat = dfs_devfs_stat,
|
|
|
|
.statfs = dfs_devfs_statfs,
|
|
|
|
.lookup = dfs_devfs_lookup,
|
|
|
|
.create_vnode = dfs_devfs_create_vnode,
|
|
|
|
.free_vnode = dfs_devfs_free_vnode,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct dfs_filesystem_type _devfs =
|
|
|
|
{
|
|
|
|
.fs_ops = &_devfs_ops,
|
2011-10-08 14:50:08 +00:00
|
|
|
};
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
static int _device_to_mode(struct rt_device *device)
|
|
|
|
{
|
|
|
|
int mode = 0;
|
|
|
|
|
|
|
|
switch (device->type)
|
|
|
|
{
|
|
|
|
case RT_Device_Class_Char:
|
|
|
|
mode = S_IFCHR | 0777;
|
|
|
|
break;
|
|
|
|
case RT_Device_Class_Block:
|
|
|
|
mode = S_IFBLK | 0777;
|
|
|
|
break;
|
|
|
|
case RT_Device_Class_Pipe:
|
|
|
|
mode = S_IFIFO | 0777;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
mode = S_IFCHR | 0777;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return mode;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _devfs_root_dirent_update(struct dfs_vnode *vnode)
|
|
|
|
{
|
|
|
|
rt_err_t result = RT_EOK;
|
|
|
|
|
|
|
|
if (vnode)
|
|
|
|
{
|
|
|
|
// result = rt_mutex_take(&vnode->lock, RT_WAITING_FOREVER);
|
|
|
|
result = dfs_file_lock();
|
|
|
|
if (result == RT_EOK)
|
|
|
|
{
|
|
|
|
rt_uint32_t count = 0;
|
|
|
|
struct device_dirent *root_dirent = (struct device_dirent*) vnode->data;
|
|
|
|
if (root_dirent) rt_free(root_dirent);
|
|
|
|
|
|
|
|
count = rt_object_get_length(RT_Object_Class_Device);
|
|
|
|
root_dirent = (struct device_dirent *)rt_malloc(sizeof(struct device_dirent) + count * sizeof(rt_device_t));
|
|
|
|
if (root_dirent != RT_NULL)
|
|
|
|
{
|
|
|
|
root_dirent->device_count = count;
|
|
|
|
if (count != 0)
|
|
|
|
{
|
|
|
|
root_dirent->devices = (rt_device_t *)(root_dirent + 1);
|
|
|
|
rt_object_get_pointers(RT_Object_Class_Device, (rt_object_t *)root_dirent->devices, count);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
root_dirent->devices = RT_NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
vnode->data = root_dirent;
|
|
|
|
result = count;
|
|
|
|
dfs_file_unlock();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct dfs_vnode *dfs_devfs_lookup(struct dfs_dentry *dentry)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
rt_device_t device = RT_NULL;
|
|
|
|
struct dfs_vnode *vnode = RT_NULL;
|
|
|
|
const char *pathname = dentry->pathname;
|
|
|
|
|
|
|
|
DLOG(msg, "devfs", "vnode", DLOG_MSG, "dfs_vnode_create");
|
|
|
|
vnode = dfs_vnode_create();
|
|
|
|
if (vnode)
|
|
|
|
{
|
|
|
|
if (pathname[0] == '/' && pathname[1] == '\0')
|
|
|
|
{
|
|
|
|
int count = _devfs_root_dirent_update(vnode);
|
|
|
|
|
|
|
|
vnode->mode = S_IFDIR | 0644;
|
|
|
|
vnode->size = count;
|
|
|
|
vnode->nlink = 1;
|
|
|
|
vnode->fops = &_dev_fops;
|
|
|
|
vnode->mnt = dentry->mnt;
|
|
|
|
vnode->type = FT_DIRECTORY;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
device = rt_device_find(&pathname[1]);
|
|
|
|
if (!device)
|
|
|
|
{
|
|
|
|
DLOG(msg, "devfs", "vnode", DLOG_MSG, "dfs_vnode_destroy(vnode), no-device");
|
|
|
|
dfs_vnode_destroy(vnode);
|
|
|
|
vnode = RT_NULL;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
vnode->mode = _device_to_mode(device);
|
|
|
|
vnode->size = device->ref_count;
|
|
|
|
vnode->nlink = 1;
|
2023-12-12 18:05:39 +08:00
|
|
|
vnode->fops = &_dev_fops;
|
2023-06-10 21:35:25 +08:00
|
|
|
vnode->data = device;
|
|
|
|
vnode->mnt = dentry->mnt;
|
|
|
|
vnode->type = FT_DEVICE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return vnode;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct dfs_vnode *dfs_devfs_create_vnode(struct dfs_dentry *dentry, int type, mode_t mode)
|
|
|
|
{
|
|
|
|
#ifdef RT_USING_DEV_BUS
|
|
|
|
if (dentry && type == FT_DIRECTORY)
|
|
|
|
{
|
|
|
|
/* regester bus device */
|
|
|
|
if (rt_device_bus_create(&dentry->pathname[1], 0))
|
|
|
|
{
|
|
|
|
return dfs_devfs_lookup(dentry);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return RT_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dfs_devfs_free_vnode(struct dfs_vnode *vnode)
|
|
|
|
{
|
|
|
|
if (S_ISDIR(vnode->mode))
|
|
|
|
{
|
|
|
|
struct device_dirent *root_dirent;
|
|
|
|
|
|
|
|
root_dirent = (struct device_dirent *)vnode->data;
|
|
|
|
RT_ASSERT(root_dirent != RT_NULL);
|
|
|
|
|
|
|
|
/* release dirent */
|
|
|
|
DLOG(msg, "devfs", "devfs", DLOG_MSG, "free root_dirent");
|
|
|
|
rt_free(root_dirent);
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* which is a device */
|
|
|
|
vnode->data = RT_NULL;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dfs_devfs_mount(struct dfs_mnt *mnt, unsigned long rwflag, const void *data)
|
|
|
|
{
|
|
|
|
RT_ASSERT(mnt != RT_NULL);
|
2023-08-02 13:59:43 +08:00
|
|
|
|
|
|
|
rt_atomic_add(&(mnt->ref_count), 1);
|
|
|
|
mnt->flags |= MNT_IS_LOCKED;
|
|
|
|
|
2017-10-15 22:44:53 +08:00
|
|
|
return RT_EOK;
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_umount(struct dfs_mnt *mnt)
|
2023-03-29 14:36:36 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dfs_devfs_statfs(struct dfs_mnt *mnt, struct statfs *buf)
|
|
|
|
{
|
|
|
|
if (mnt && buf)
|
|
|
|
{
|
|
|
|
buf->f_bsize = 512;
|
|
|
|
buf->f_blocks = 2048 * 64; // 64M
|
|
|
|
buf->f_bfree = buf->f_blocks;
|
|
|
|
buf->f_bavail = buf->f_bfree;
|
|
|
|
}
|
2023-03-29 14:36:36 +08:00
|
|
|
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_ioctl(struct dfs_file *file, int cmd, void *args)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2023-12-12 18:05:39 +08:00
|
|
|
rt_err_t result = RT_EOK;
|
|
|
|
rt_device_t device;
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
RT_ASSERT(file != RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
/* get device handler */
|
2023-12-12 18:05:39 +08:00
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
if ((file->dentry->pathname[0] == '/') && (file->dentry->pathname[1] == '\0'))
|
2023-03-29 11:13:35 +08:00
|
|
|
return -RT_ENOSYS;
|
|
|
|
|
2023-12-12 18:05:39 +08:00
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->ioctl)
|
|
|
|
{
|
|
|
|
result = device->fops->ioctl(file, cmd, args);
|
|
|
|
}
|
|
|
|
else if (device->ops)
|
|
|
|
#else
|
|
|
|
if (device->ops)
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
{
|
|
|
|
result = rt_device_control(device, cmd, args);
|
|
|
|
}
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2017-10-15 22:44:53 +08:00
|
|
|
return result;
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
|
|
|
|
2023-07-24 09:12:35 +08:00
|
|
|
ssize_t dfs_devfs_read(struct dfs_file *file, void *buf, size_t count, off_t *pos)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2023-12-12 18:05:39 +08:00
|
|
|
int result = -RT_EIO;
|
|
|
|
rt_device_t device;
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
RT_ASSERT(file != RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
/* get device handler */
|
2023-12-12 18:05:39 +08:00
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
if ((file->dentry->pathname[0] == '/') && (file->dentry->pathname[1] == '\0'))
|
2023-03-29 11:13:35 +08:00
|
|
|
return -RT_ENOSYS;
|
|
|
|
|
2023-12-12 18:05:39 +08:00
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->read)
|
|
|
|
{
|
|
|
|
result = device->fops->read(file, buf, count, pos);
|
|
|
|
}
|
|
|
|
else if (device->ops)
|
|
|
|
#else
|
|
|
|
if (device->ops)
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
{
|
|
|
|
/* read device data */
|
|
|
|
result = rt_device_read(device, *pos, buf, count);
|
|
|
|
*pos += result;
|
|
|
|
}
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
return result;
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
|
|
|
|
2023-07-24 09:12:35 +08:00
|
|
|
ssize_t dfs_devfs_write(struct dfs_file *file, const void *buf, size_t count, off_t *pos)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2023-12-12 18:05:39 +08:00
|
|
|
int result = -RT_EIO;
|
|
|
|
rt_device_t device;
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
RT_ASSERT(file != RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
/* get device handler */
|
2023-12-12 18:05:39 +08:00
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
if ((file->dentry->pathname[0] == '/') && (file->dentry->pathname[1] == '\0'))
|
2023-03-29 11:13:35 +08:00
|
|
|
return -RT_ENOSYS;
|
|
|
|
|
2023-12-12 18:05:39 +08:00
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->write)
|
|
|
|
{
|
|
|
|
result = device->fops->write(file, buf, count, pos);
|
|
|
|
}
|
|
|
|
else if (device->ops)
|
|
|
|
#else
|
|
|
|
if (device->ops)
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
{
|
|
|
|
/* read device data */
|
|
|
|
result = rt_device_write(device, *pos, buf, count);
|
|
|
|
*pos += result;
|
|
|
|
}
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
return result;
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_close(struct dfs_file *file)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
rt_err_t result = RT_EOK;
|
|
|
|
rt_device_t device;
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
RT_ASSERT(file != RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
if (!S_ISDIR(file->vnode->mode))
|
2013-06-26 22:30:40 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
/* get device handler */
|
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2023-12-12 18:05:39 +08:00
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->close)
|
|
|
|
{
|
|
|
|
result = device->fops->close(file);
|
|
|
|
}
|
|
|
|
else if (device->ops)
|
|
|
|
#else
|
|
|
|
if (device->ops)
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
{
|
|
|
|
/* close device handler */
|
|
|
|
result = rt_device_close(device);
|
|
|
|
}
|
2013-06-26 22:30:40 +08:00
|
|
|
}
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
return result;
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_open(struct dfs_file *file)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
rt_err_t result = RT_EOK;
|
2013-06-26 22:30:40 +08:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
/* open root directory */
|
|
|
|
if ((file->dentry->pathname[0] == '/' && file->dentry->pathname[1] == '\0') ||
|
|
|
|
(S_ISDIR(file->vnode->mode)))
|
2022-12-03 12:07:44 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
/* re-open the root directory for re-scan devices */
|
|
|
|
_devfs_root_dirent_update(file->vnode);
|
|
|
|
return RT_EOK;
|
2022-12-03 12:07:44 +08:00
|
|
|
}
|
2023-06-10 21:35:25 +08:00
|
|
|
else
|
2013-06-26 22:30:40 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
rt_device_t device = RT_NULL;
|
2013-06-26 22:30:40 +08:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
device = (struct rt_device *)file->vnode->data;
|
|
|
|
if (device)
|
2013-06-26 22:30:40 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
2023-12-12 18:05:39 +08:00
|
|
|
if (device->fops && device->fops->open)
|
2013-06-26 22:30:40 +08:00
|
|
|
{
|
2023-12-12 18:05:39 +08:00
|
|
|
result = device->fops->open(file);
|
|
|
|
if (result == RT_EOK || result == -RT_ENOSYS)
|
2022-01-26 16:02:24 +08:00
|
|
|
{
|
2023-12-12 18:05:39 +08:00
|
|
|
file->fops = &_dev_fops;
|
|
|
|
return RT_EOK;
|
2022-01-26 16:02:24 +08:00
|
|
|
}
|
2013-06-26 22:30:40 +08:00
|
|
|
}
|
2023-12-12 18:05:39 +08:00
|
|
|
else if (device->ops)
|
|
|
|
#else
|
|
|
|
if (device->ops)
|
2023-06-10 21:35:25 +08:00
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
2017-10-15 22:44:53 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
result = rt_device_open(device, RT_DEVICE_OFLAG_RDWR);
|
|
|
|
if (result == RT_EOK || result == -RT_ENOSYS)
|
|
|
|
{
|
|
|
|
file->fops = &_dev_fops;
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
2017-10-15 22:44:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-01-28 14:13:44 +08:00
|
|
|
|
|
|
|
/* open device failed. */
|
2017-10-15 22:44:53 +08:00
|
|
|
return -EIO;
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_unlink(struct dfs_dentry *dentry)
|
2022-12-03 12:07:44 +08:00
|
|
|
{
|
|
|
|
#ifdef RT_USING_DEV_BUS
|
2023-12-12 18:05:39 +08:00
|
|
|
rt_device_t device;
|
2022-12-03 12:07:44 +08:00
|
|
|
|
2023-12-12 18:05:39 +08:00
|
|
|
device = rt_device_find(&dentry->pathname[1]);
|
|
|
|
if (device == RT_NULL)
|
2022-12-03 12:07:44 +08:00
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
2023-12-12 18:05:39 +08:00
|
|
|
if (device->type != RT_Device_Class_Bus)
|
2022-12-03 12:07:44 +08:00
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
2023-12-12 18:05:39 +08:00
|
|
|
rt_device_bus_destroy(device);
|
2022-12-03 12:07:44 +08:00
|
|
|
#endif
|
|
|
|
return RT_EOK;
|
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_stat(struct dfs_dentry *dentry, struct stat *st)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
int ret = RT_EOK;
|
|
|
|
const char *path = RT_NULL;
|
|
|
|
struct dfs_vnode *vnode = RT_NULL;
|
2013-06-26 22:30:40 +08:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
if (dentry && dentry->vnode)
|
2013-06-26 22:30:40 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
path = dentry->pathname;
|
|
|
|
vnode = dentry->vnode;
|
2013-06-26 22:30:40 +08:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
/* stat root directory */
|
|
|
|
if ((path[0] == '/') && (path[1] == '\0'))
|
2013-06-26 22:30:40 +08:00
|
|
|
{
|
|
|
|
st->st_dev = 0;
|
2023-06-10 21:35:25 +08:00
|
|
|
st->st_gid = vnode->gid;
|
|
|
|
st->st_uid = vnode->uid;
|
|
|
|
st->st_ino = 0;
|
|
|
|
st->st_mode = vnode->mode;
|
|
|
|
st->st_nlink = vnode->nlink;
|
|
|
|
st->st_size = vnode->size;
|
|
|
|
st->st_mtim.tv_nsec = vnode->mtime.tv_nsec;
|
|
|
|
st->st_mtim.tv_sec = vnode->mtime.tv_sec;
|
|
|
|
st->st_ctim.tv_nsec = vnode->ctime.tv_nsec;
|
|
|
|
st->st_ctim.tv_sec = vnode->ctime.tv_sec;
|
|
|
|
st->st_atim.tv_nsec = vnode->atime.tv_nsec;
|
|
|
|
st->st_atim.tv_sec = vnode->atime.tv_sec;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2023-12-12 18:05:39 +08:00
|
|
|
rt_device_t device;
|
2013-06-26 22:30:40 +08:00
|
|
|
|
2023-12-12 18:05:39 +08:00
|
|
|
device = rt_device_find(&path[1]);
|
|
|
|
if (device != RT_NULL)
|
2023-06-10 21:35:25 +08:00
|
|
|
{
|
|
|
|
st->st_dev = 0;
|
|
|
|
st->st_gid = vnode->gid;
|
|
|
|
st->st_uid = vnode->uid;
|
|
|
|
st->st_ino = 0;
|
|
|
|
st->st_mode = vnode->mode;
|
|
|
|
st->st_nlink = vnode->nlink;
|
|
|
|
st->st_size = vnode->size;
|
|
|
|
st->st_mtim.tv_nsec = vnode->mtime.tv_nsec;
|
|
|
|
st->st_mtim.tv_sec = vnode->mtime.tv_sec;
|
|
|
|
st->st_ctim.tv_nsec = vnode->ctime.tv_nsec;
|
|
|
|
st->st_ctim.tv_sec = vnode->ctime.tv_sec;
|
|
|
|
st->st_atim.tv_nsec = vnode->atime.tv_nsec;
|
|
|
|
st->st_atim.tv_sec = vnode->atime.tv_sec;
|
|
|
|
}
|
2013-06-26 22:30:40 +08:00
|
|
|
else
|
2023-06-10 21:35:25 +08:00
|
|
|
{
|
|
|
|
ret = -ENOENT;
|
|
|
|
}
|
2013-06-26 22:30:40 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
return ret;
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_getdents(struct dfs_file *file, struct dirent *dirp, uint32_t count)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2013-06-26 22:30:40 +08:00
|
|
|
rt_uint32_t index;
|
|
|
|
rt_object_t object;
|
|
|
|
struct dirent *d;
|
|
|
|
struct device_dirent *root_dirent;
|
|
|
|
|
2022-12-03 12:07:44 +08:00
|
|
|
root_dirent = (struct device_dirent *)file->vnode->data;
|
2013-06-26 22:30:40 +08:00
|
|
|
RT_ASSERT(root_dirent != RT_NULL);
|
|
|
|
|
|
|
|
/* make integer count */
|
|
|
|
count = (count / sizeof(struct dirent));
|
|
|
|
if (count == 0)
|
2017-10-15 22:44:53 +08:00
|
|
|
return -EINVAL;
|
2013-06-26 22:30:40 +08:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
for (index = 0; index < count && index + file->fpos < root_dirent->device_count; index ++)
|
2013-06-26 22:30:40 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
object = (rt_object_t)root_dirent->devices[file->fpos + index];
|
2013-06-26 22:30:40 +08:00
|
|
|
|
|
|
|
d = dirp + index;
|
2023-06-10 21:35:25 +08:00
|
|
|
d->d_type = DT_REG;
|
2013-06-26 22:30:40 +08:00
|
|
|
d->d_namlen = RT_NAME_MAX;
|
|
|
|
d->d_reclen = (rt_uint16_t)sizeof(struct dirent);
|
|
|
|
rt_strncpy(d->d_name, object->name, RT_NAME_MAX);
|
2023-06-10 21:35:25 +08:00
|
|
|
d->d_name[RT_NAME_MAX] = '\0';
|
2013-06-26 22:30:40 +08:00
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
file->fpos += index;
|
2013-06-26 22:30:40 +08:00
|
|
|
|
|
|
|
return index * sizeof(struct dirent);
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
static int dfs_devfs_poll(struct dfs_file *file, struct rt_pollreq *req)
|
2011-10-08 14:50:08 +00:00
|
|
|
{
|
2017-10-15 22:44:53 +08:00
|
|
|
int mask = 0;
|
2023-12-12 18:05:39 +08:00
|
|
|
rt_device_t device;
|
|
|
|
|
|
|
|
/* get device handler */
|
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
|
|
|
|
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->poll)
|
|
|
|
{
|
|
|
|
mask = device->fops->poll(file, req);
|
|
|
|
}
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
2017-10-15 22:44:53 +08:00
|
|
|
|
|
|
|
return mask;
|
|
|
|
}
|
2013-06-26 22:30:40 +08:00
|
|
|
|
2023-12-12 18:05:39 +08:00
|
|
|
int dfs_devfs_flush(struct dfs_file *file)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
rt_device_t device;
|
|
|
|
|
|
|
|
/* get device handler */
|
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
|
|
|
|
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->flush)
|
|
|
|
{
|
|
|
|
ret = device->fops->flush(file);
|
|
|
|
}
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
off_t dfs_devfs_lseek(struct dfs_file *file, off_t offset, int wherece)
|
|
|
|
{
|
|
|
|
off_t ret = 0;
|
|
|
|
rt_device_t device;
|
|
|
|
|
|
|
|
/* get device handler */
|
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
|
|
|
|
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->lseek)
|
|
|
|
{
|
|
|
|
ret = device->fops->lseek(file, offset, wherece);
|
|
|
|
}
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dfs_devfs_truncate(struct dfs_file *file, off_t offset)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
rt_device_t device;
|
|
|
|
|
|
|
|
/* get device handler */
|
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
|
|
|
|
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->truncate)
|
|
|
|
{
|
|
|
|
ret = device->fops->truncate(file, offset);
|
|
|
|
}
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dfs_devfs_mmap(struct dfs_file *file, struct lwp_avl_struct *mmap)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
rt_device_t device;
|
|
|
|
|
|
|
|
/* get device handler */
|
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
|
|
|
|
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->mmap)
|
|
|
|
{
|
|
|
|
ret = device->fops->mmap(file, mmap);
|
|
|
|
}
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dfs_devfs_lock(struct dfs_file *file, struct file_lock *flock)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
rt_device_t device;
|
|
|
|
|
|
|
|
/* get device handler */
|
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
|
|
|
|
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->lock)
|
|
|
|
{
|
|
|
|
ret = device->fops->lock(file, flock);
|
|
|
|
}
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
int dfs_devfs_flock(struct dfs_file *file, int operation, struct file_lock *flock)
|
|
|
|
{
|
|
|
|
int ret = 0;
|
|
|
|
rt_device_t device;
|
|
|
|
|
|
|
|
/* get device handler */
|
|
|
|
device = (rt_device_t)file->vnode->data;
|
|
|
|
RT_ASSERT(device != RT_NULL);
|
|
|
|
|
|
|
|
#ifdef RT_USING_POSIX_DEVIO
|
|
|
|
if (device->fops && device->fops->flock)
|
|
|
|
{
|
|
|
|
ret = device->fops->flock(file, operation, flock);
|
|
|
|
}
|
|
|
|
#endif /* RT_USING_POSIX_DEVIO */
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
int dfs_devfs_init(void)
|
2017-10-15 22:44:53 +08:00
|
|
|
{
|
2023-06-10 21:35:25 +08:00
|
|
|
/* register devfs file system */
|
|
|
|
dfs_register(&_devfs);
|
2017-10-15 22:44:53 +08:00
|
|
|
|
2023-06-10 21:35:25 +08:00
|
|
|
dfs_mount(RT_NULL, "/dev", "devfs", 0, RT_NULL);
|
2011-10-08 14:50:08 +00:00
|
|
|
|
2013-06-26 22:30:40 +08:00
|
|
|
return 0;
|
2011-10-08 14:50:08 +00:00
|
|
|
}
|
2023-06-10 21:35:25 +08:00
|
|
|
INIT_COMPONENT_EXPORT(dfs_devfs_init);
|
|
|
|
|