nuttx-apps/examples/userfs/userfs_main.c

608 lines
16 KiB
C

/****************************************************************************
* examples/userfs/userfs_main.c
*
* Copyright (C) 2017 Gregory Nutt. All rights reserved.
* Author: Gregory Nutt <gnutt@nuttx.org>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
* 3. Neither the name NuttX nor the names of its contributors may be
* used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*
****************************************************************************/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <fcntl.h>
#include <dirent.h>
#include <errno.h>
#include <nuttx/fs/userfs.h>
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
#define UFSTEST_NFILES 3
#define UFSTEST_FS_BLOCKSIZE 32
#define UFSTEST_FILE1_BLOCKS (2)
#define UFSTEST_FILE1_MXSIZE (UFSTEST_FILE1_BLOCKS * UFSTEST_FS_BLOCKSIZE)
#define UFSTEST_FILE2_BLOCKS (3)
#define UFSTEST_FILE2_MXSIZE (UFSTEST_FILE2_BLOCKS * UFSTEST_FS_BLOCKSIZE)
#define UFSTEST_FILE3_BLOCKS (4)
#define UFSTEST_FILE3_MXSIZE (UFSTEST_FILE3_BLOCKS * UFSTEST_FS_BLOCKSIZE)
#define UFSTEST_FS_NBLOCKS (UFSTEST_FILE1_BLOCKS + UFSTEST_FILE2_BLOCKS + UFSTEST_FILE3_BLOCKS)
#define UFSTEST_FS_NBYTES (UFSTEST_FILE1_MXSIZE + UFSTEST_FILE2_MXSIZE + UFSTEST_FILE3_MXSIZE)
#define UFSTEST_MXWRITE UFSTEST_FILE3_MXSIZE
#define UFSTEST_MOUNTPOUNT "/mnt/ufstest"
/****************************************************************************
* Private Types
****************************************************************************/
struct ufstest_file_s
{
struct dirent entry;
uint16_t inuse;
uint16_t mxsize;
FAR char *data;
};
struct ufstest_openfile_s
{
int16_t pos;
int16_t crefs;
FAR struct ufstest_file_s *file;
};
struct ufstest_opendir_s
{
int16_t index;
};
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
/* UserFS methods */
static int ufstest_open(FAR void *volinfo, FAR const char *relpath,
int oflags, mode_t mode, FAR void **openinfo);
static int ufstest_close(FAR void *volinfo, FAR void *openinfo);
static ssize_t ufstest_read(FAR void *volinfo, FAR void *openinfo,
FAR char *buffer, size_t buflen);
static ssize_t ufstest_write(FAR void *volinfo, FAR void *openinfo,
FAR const char *buffer, size_t buflen);
static off_t ufstest_seek(FAR void *volinfo, FAR void *openinfo,
off_t offset, int whence);
static int ufstest_ioctl(FAR void *volinfo, FAR void *openinfo, int cmd,
unsigned long arg);
static int ufstest_sync(FAR void *volinfo, FAR void *openinfo);
static int ufstest_dup(FAR void *volinfo, FAR void *oldinfo,
FAR void **newinfo);
static int ufstest_fstat(FAR void *volinfo, FAR void *openinfo,
FAR struct stat *buf);
static int ufstest_opendir(FAR void *volinfo, FAR const char *relpath,
FAR void **dir);
static int ufstest_closedir(FAR void *volinfo, FAR void *dir);
static int ufstest_readdir(FAR void *volinfo, FAR void *dir,
FAR struct dirent *entry);
static int ufstest_rewinddir(FAR void *volinfo, FAR void *dir);
static int ufstest_statfs(FAR void *volinfo, FAR struct statfs *buf);
static int ufstest_unlink(FAR void *volinfo, FAR const char *relpath);
static int ufstest_mkdir(FAR void *volinfo, FAR const char *relpath,
mode_t mode);
static int ufstest_rmdir(FAR void *volinfo, FAR const char *relpath);
static int ufstest_rename(FAR void *volinfo, FAR const char *oldrelpath,
FAR const char *newrelpath);
static int ufstest_stat(FAR void *volinfo, FAR const char *relpath,
FAR struct stat *buf);
static int ufstest_destroy(FAR void *volinfo);
/****************************************************************************
* Private Data
****************************************************************************/
static char g_file1_data[UFSTEST_FILE1_MXSIZE] = "This is file 1";
#define UFSTEST_INIT_FILE1_SIZE 14
static char g_file2_data[UFSTEST_FILE2_MXSIZE] = "This is file 2";
#define UFSTEST_INIT_FILE2_SIZE 14
static char g_file3_data[UFSTEST_FILE3_MXSIZE] = "This is file 3";
#define UFSTEST_INIT_FILE3_SIZE 14
static struct ufstest_file_s g_rootdir[UFSTEST_NFILES] =
{
{
{ DTYPE_FILE, "File1" },
UFSTEST_INIT_FILE1_SIZE,
UFSTEST_FILE1_MXSIZE,
g_file1_data
},
{
{ DTYPE_FILE, "File2" },
UFSTEST_INIT_FILE2_SIZE,
UFSTEST_FILE2_MXSIZE,
g_file2_data
},
{
{ DTYPE_FILE, "File3" },
UFSTEST_INIT_FILE3_SIZE,
UFSTEST_FILE3_MXSIZE,
g_file3_data
}
};
static const struct userfs_operations_s g_ufstest_ops =
{
ufstest_open,
ufstest_close,
ufstest_read,
ufstest_write,
ufstest_seek,
ufstest_ioctl,
ufstest_sync,
ufstest_dup,
ufstest_fstat,
ufstest_opendir,
ufstest_closedir,
ufstest_readdir,
ufstest_rewinddir,
ufstest_statfs,
ufstest_unlink,
ufstest_mkdir,
ufstest_rmdir,
ufstest_rename,
ufstest_stat,
ufstest_destroy
};
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: ufstest_findbyname
****************************************************************************/
static FAR struct ufstest_file_s *ufstest_findbyname(FAR const char *relpath)
{
int i;
for (i = 0; i < UFSTEST_NFILES; i++)
{
if (strcmp(relpath, g_rootdir[i].entry.d_name) == 0)
{
return &g_rootdir[i];
}
}
return NULL;
}
/****************************************************************************
* UserFS methods
****************************************************************************/
static int ufstest_open(FAR void *volinfo, FAR const char *relpath,
int oflags, mode_t mode, FAR void **openinfo)
{
FAR struct ufstest_openfile_s *opriv;
FAR struct ufstest_file_s *file;
file = ufstest_findbyname(relpath);
if (file != NULL)
{
opriv = (FAR struct ufstest_openfile_s *)
malloc(sizeof(struct ufstest_openfile_s ));
if (opriv == NULL)
{
return -ENOMEM;
}
if ((oflags & O_TRUNC) != 0)
{
file->inuse = 0;
}
if ((oflags & (O_WROK | O_APPEND)) == (O_WROK | O_APPEND))
{
opriv->pos = file->inuse;
}
else
{
opriv->pos = 0;
}
opriv->file = file;
/* Initiallly, there is one refernce count on the open data. This may
* be incremented in the event that the file is dup'ed.
*/
opriv->crefs = 1;
/* Return the opaque reference to the open data */
*openinfo = opriv;
return OK;
}
return -ENOENT;
}
static int ufstest_close(FAR void *volinfo, FAR void *openinfo)
{
FAR struct ufstest_openfile_s *opriv =
(FAR struct ufstest_openfile_s *)openinfo;
if (opriv != NULL)
{
if (opriv->crefs <= 1)
{
free(opriv);
}
else
{
opriv->crefs--;
}
}
return OK;
}
static ssize_t ufstest_read(FAR void *volinfo, FAR void *openinfo,
FAR char *buffer, size_t buflen)
{
FAR struct ufstest_openfile_s *opriv =
(FAR struct ufstest_openfile_s *)openinfo;
ssize_t readsize;
readsize = opriv->file->inuse - opriv->pos;
if (readsize > buflen)
{
readsize = buflen;
}
if (readsize > 0)
{
memcpy(buffer, &opriv->file->data[opriv->pos], readsize);
opriv->pos += readsize;
}
return readsize <= 0 ? 0 : readsize;
}
static ssize_t ufstest_write(FAR void *volinfo, FAR void *openinfo,
FAR const char *buffer, size_t buflen)
{
FAR struct ufstest_openfile_s *opriv =
(FAR struct ufstest_openfile_s *)openinfo;
ssize_t writesize;
writesize = opriv->file->mxsize - opriv->pos;
if (writesize > buflen)
{
writesize = buflen;
}
memcpy(&opriv->file->data[opriv->pos], buffer, writesize);
opriv->pos += writesize;
if (opriv->pos > opriv->file->inuse)
{
opriv->file->inuse = opriv->pos;
}
return writesize;
}
static off_t ufstest_seek(FAR void *volinfo, FAR void *openinfo,
off_t offset, int whence)
{
FAR struct ufstest_openfile_s *opriv =
(FAR struct ufstest_openfile_s *)openinfo;
off_t newpos;
switch (whence)
{
case SEEK_SET: /* The offset is set to offset bytes. */
newpos = offset;
break;
case SEEK_CUR: /* The offset is set to its current location plus
* offset bytes. */
newpos = offset + opriv->pos;
break;
case SEEK_END: /* The offset is set to the size of the file plus
* offset bytes. */
newpos = offset + opriv->file->inuse;
break;
default:
fprintf(stderr, "ERROR: Whence is invalid: %d\n", whence);
return -EINVAL;
}
if (newpos < 0)
{
newpos = 0;
}
else if (newpos > opriv->file->mxsize)
{
newpos = opriv->file->mxsize;
}
opriv->pos = newpos;
return newpos;
}
static int ufstest_ioctl(FAR void *volinfo, FAR void *openinfo, int cmd,
unsigned long arg)
{
return -ENOTTY;
}
static int ufstest_sync(FAR void *volinfo, FAR void *openinfo)
{
return OK;
}
static int ufstest_dup(FAR void *volinfo, FAR void *oldinfo,
FAR void **newinfo)
{
FAR struct ufstest_openfile_s *opriv =
(FAR struct ufstest_openfile_s *)oldinfo;
/* Increment the reference count on the open info */
opriv->crefs++;
/* And just copy the openinfo reference */
*newinfo = oldinfo;
return OK;
}
static int ufstest_fstat(FAR void *volinfo, FAR void *openinfo,
FAR struct stat *buf)
{
FAR struct ufstest_openfile_s *opriv =
(FAR struct ufstest_openfile_s *)openinfo;
buf->st_mode = (S_IFREG | S_IWUSR | S_IRUSR | S_IRGRP | S_IROTH);
buf->st_size = opriv->file->inuse;
buf->st_blksize = UFSTEST_FS_BLOCKSIZE;
buf->st_blocks = (opriv->file->inuse + UFSTEST_FS_BLOCKSIZE - 1) /
UFSTEST_FS_BLOCKSIZE;
buf->st_atime = 0;
buf->st_mtime = 0;
buf->st_ctime = 0;
return OK;
}
static int ufstest_opendir(FAR void *volinfo, FAR const char *relpath,
FAR void **dir)
{
FAR struct ufstest_opendir_s *odir;
if (!relpath || relpath[0] == '\0')
{
/* The path refers to the top level directory. */
odir = (FAR struct ufstest_opendir_s *)
malloc(sizeof(struct ufstest_opendir_s ));
if (odir == NULL)
{
return -ENOMEM;
}
odir->index = 0;
*dir = odir;
return OK;
}
return -ENOENT;
}
static int ufstest_closedir(FAR void *volinfo, FAR void *dir)
{
if (dir != NULL)
{
free(dir);
}
return OK;
}
static int ufstest_readdir(FAR void *volinfo, FAR void *dir,
FAR struct dirent *entry)
{
FAR struct ufstest_file_s *priv = (FAR struct ufstest_file_s *)volinfo;
FAR struct ufstest_opendir_s *odir = (FAR struct ufstest_opendir_s *)dir;
if (odir->index < UFSTEST_NFILES)
{
memcpy(entry, &priv[odir->index].entry, sizeof(struct dirent));
odir->index++;
return OK;
}
return -ENOENT;
}
static int ufstest_rewinddir(FAR void *volinfo, FAR void *dir)
{
FAR struct ufstest_opendir_s *odir = (FAR struct ufstest_opendir_s *)dir;
odir->index = 0;
return OK;
}
static int ufstest_statfs(FAR void *volinfo, FAR struct statfs *buf)
{
int inuse = 0;
int i;
for (i = 0; i < UFSTEST_NFILES; i++)
{
inuse += (g_rootdir[i].inuse + UFSTEST_FS_BLOCKSIZE - 1) /
UFSTEST_FS_BLOCKSIZE;
}
buf->f_type = USERFS_MAGIC;
buf->f_namelen = NAME_MAX;
buf->f_bsize = UFSTEST_FS_BLOCKSIZE;
buf->f_blocks = UFSTEST_FS_NBLOCKS;
buf->f_bfree = UFSTEST_FS_NBLOCKS - inuse;
buf->f_bavail = UFSTEST_FS_NBLOCKS - inuse;
buf->f_files = UFSTEST_NFILES;
buf->f_ffree = 0;
return OK;
}
static int ufstest_unlink(FAR void *volinfo, FAR const char *relpath)
{
return -ENOSYS;
}
static int ufstest_mkdir(FAR void *volinfo, FAR const char *relpath,
mode_t mode)
{
return -ENOSYS;
}
static int ufstest_rmdir(FAR void *volinfo, FAR const char *relpath)
{
return -ENOSYS;
}
static int ufstest_rename(FAR void *volinfo, FAR const char *oldrelpath,
FAR const char *newrelpath)
{
FAR struct ufstest_file_s *file;
file = ufstest_findbyname(oldrelpath);
if (file != NULL)
{
strncpy(file->entry.d_name, newrelpath, NAME_MAX + 1);
return OK;
}
return -ENOENT;
}
static int ufstest_stat(FAR void *volinfo, FAR const char *relpath,
FAR struct stat *buf)
{
FAR void *openinfo;
int ret;
/* Are we stat'ing the directory? Of one of the files in the directory? */
if (*relpath == '\0')
{
memset(buf, 0, sizeof(struct stat));
buf->st_mode = (S_IFDIR | S_IRWXU | S_IRUSR | S_IRGRP | S_IRWXG |
S_IROTH | S_IRWXO);
buf->st_blksize = UFSTEST_FS_BLOCKSIZE;
}
else
{
ret = ufstest_open(volinfo, relpath, O_RDWR, 0644, &openinfo);
if (ret >= 0)
{
ret = ufstest_fstat(volinfo, openinfo, buf);
(void)ufstest_close(volinfo, openinfo);
}
}
return ret;
}
static int ufstest_destroy(FAR void *volinfo)
{
return OK;
}
/****************************************************************************
* ufstest_daemon
****************************************************************************/
int ufstest_daemon(int argc, char *argv[])
{
int ret;
ret = userfs_run(UFSTEST_MOUNTPOUNT, &g_ufstest_ops, g_rootdir,
UFSTEST_MXWRITE);
fprintf(stderr, "ERROR: userfs_run() returned: %d\n", ret);
return EXIT_FAILURE;
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* userfs_main
****************************************************************************/
#ifdef CONFIG_BUILD_KERNEL
int main(int argc, FAR char *argv[])
#else
int userfs_main(int argc, char *argv[])
#endif
{
FAR char *nshargv[1];
int pid;
/* Spawn the UserFS test daemon */
nshargv[0] = NULL;
pid = task_create("UserFS", CONFIG_EXAMPLES_USERFS_PRIORITY,
CONFIG_EXAMPLES_USERFS_STACKSIZE, ufstest_daemon,
(FAR char * const *)nshargv);
if (pid < 0)
{
return EXIT_FAILURE;
}
return EXIT_SUCCESS;
}