123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846 |
- /*
- * Copyright (c) 2006-2018, RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- * 2005-02-22 Bernard The first version.
- * 2011-12-08 Bernard Merges rename patch from iamcacy.
- * 2015-05-27 Bernard Fix the fd clear issue.
- * 2019-01-24 Bernard Remove file repeatedly open check.
- */
- #include <dfs.h>
- #include <dfs_file.h>
- #include <dfs_private.h>
- /**
- * @addtogroup FileApi
- */
- /*@{*/
- /**
- * this function will open a file which specified by path with specified flags.
- *
- * @param fd the file descriptor pointer to return the corresponding result.
- * @param path the specified file path.
- * @param flags the flags for open operator.
- *
- * @return 0 on successful, -1 on failed.
- */
- int dfs_file_open(struct dfs_fd *fd, const char *path, int flags)
- {
- struct dfs_filesystem *fs;
- char *fullpath;
- int result;
- /* parameter check */
- if (fd == NULL)
- return -EINVAL;
- /* make sure we have an absolute path */
- fullpath = dfs_normalize_path(NULL, path);
- if (fullpath == NULL)
- {
- return -ENOMEM;
- }
- LOG_D("open file:%s", fullpath);
- /* find filesystem */
- fs = dfs_filesystem_lookup(fullpath);
- if (fs == NULL)
- {
- rt_free(fullpath); /* release path */
- return -ENOENT;
- }
- LOG_D("open in filesystem:%s", fs->ops->name);
- fd->fs = fs; /* set file system */
- fd->fops = fs->ops->fops; /* set file ops */
- /* initialize the fd item */
- fd->type = FT_REGULAR;
- fd->flags = flags;
- fd->size = 0;
- fd->pos = 0;
- fd->data = fs;
- if (!(fs->ops->flags & DFS_FS_FLAG_FULLPATH))
- {
- if (dfs_subdir(fs->path, fullpath) == NULL)
- fd->path = rt_strdup("/");
- else
- fd->path = rt_strdup(dfs_subdir(fs->path, fullpath));
- rt_free(fullpath);
- LOG_D("Actual file path: %s", fd->path);
- }
- else
- {
- fd->path = fullpath;
- }
- /* specific file system open routine */
- if (fd->fops->open == NULL)
- {
- /* clear fd */
- rt_free(fd->path);
- fd->path = NULL;
- return -ENOSYS;
- }
- if ((result = fd->fops->open(fd)) < 0)
- {
- /* clear fd */
- rt_free(fd->path);
- fd->path = NULL;
- LOG_D("%s open failed", fullpath);
- return result;
- }
- fd->flags |= DFS_F_OPEN;
- if (flags & O_DIRECTORY)
- {
- fd->type = FT_DIRECTORY;
- fd->flags |= DFS_F_DIRECTORY;
- }
- LOG_D("open successful");
- return 0;
- }
- /**
- * this function will close a file descriptor.
- *
- * @param fd the file descriptor to be closed.
- *
- * @return 0 on successful, -1 on failed.
- */
- int dfs_file_close(struct dfs_fd *fd)
- {
- int result = 0;
- if (fd == NULL)
- return -ENXIO;
- if (fd->fops->close != NULL)
- result = fd->fops->close(fd);
- /* close fd error, return */
- if (result < 0)
- return result;
- rt_free(fd->path);
- fd->path = NULL;
- return result;
- }
- /**
- * this function will perform a io control on a file descriptor.
- *
- * @param fd the file descriptor.
- * @param cmd the command to send to file descriptor.
- * @param args the argument to send to file descriptor.
- *
- * @return 0 on successful, -1 on failed.
- */
- int dfs_file_ioctl(struct dfs_fd *fd, int cmd, void *args)
- {
- if (fd == NULL)
- return -EINVAL;
- /* regular file system fd */
- if (fd->type == FT_REGULAR || fd->type == FT_DEVICE)
- {
- switch (cmd)
- {
- case F_GETFL:
- return fd->flags; /* return flags */
- case F_SETFL:
- {
- int flags = (int)(rt_base_t)args;
- int mask = O_NONBLOCK | O_APPEND;
- flags &= mask;
- fd->flags &= ~mask;
- fd->flags |= flags;
- }
- return 0;
- }
- }
- if (fd->fops->ioctl != NULL)
- return fd->fops->ioctl(fd, cmd, args);
- return -ENOSYS;
- }
- /**
- * this function will read specified length data from a file descriptor to a
- * buffer.
- *
- * @param fd the file descriptor.
- * @param buf the buffer to save the read data.
- * @param len the length of data buffer to be read.
- *
- * @return the actual read data bytes or 0 on end of file or failed.
- */
- int dfs_file_read(struct dfs_fd *fd, void *buf, size_t len)
- {
- int result = 0;
- if (fd == NULL)
- return -EINVAL;
- if (fd->fops->read == NULL)
- return -ENOSYS;
- if ((result = fd->fops->read(fd, buf, len)) < 0)
- fd->flags |= DFS_F_EOF;
- return result;
- }
- /**
- * this function will fetch directory entries from a directory descriptor.
- *
- * @param fd the directory descriptor.
- * @param dirp the dirent buffer to save result.
- * @param nbytes the available room in the buffer.
- *
- * @return the read dirent, others on failed.
- */
- int dfs_file_getdents(struct dfs_fd *fd, struct dirent *dirp, size_t nbytes)
- {
- /* parameter check */
- if (fd == NULL || fd->type != FT_DIRECTORY)
- return -EINVAL;
- if (fd->fops->getdents != NULL)
- return fd->fops->getdents(fd, dirp, nbytes);
- return -ENOSYS;
- }
- /**
- * this function will unlink (remove) a specified path file from file system.
- *
- * @param path the specified path file to be unlinked.
- *
- * @return 0 on successful, -1 on failed.
- */
- int dfs_file_unlink(const char *path)
- {
- int result;
- char *fullpath;
- struct dfs_filesystem *fs;
- /* Make sure we have an absolute path */
- fullpath = dfs_normalize_path(NULL, path);
- if (fullpath == NULL)
- {
- return -EINVAL;
- }
- /* get filesystem */
- if ((fs = dfs_filesystem_lookup(fullpath)) == NULL)
- {
- result = -ENOENT;
- goto __exit;
- }
- /* Check whether file is already open */
- if (fd_is_open(fullpath) == 0)
- {
- result = -EBUSY;
- goto __exit;
- }
- if (fs->ops->unlink != NULL)
- {
- if (!(fs->ops->flags & DFS_FS_FLAG_FULLPATH))
- {
- if (dfs_subdir(fs->path, fullpath) == NULL)
- result = fs->ops->unlink(fs, "/");
- else
- result = fs->ops->unlink(fs, dfs_subdir(fs->path, fullpath));
- }
- else
- result = fs->ops->unlink(fs, fullpath);
- }
- else result = -ENOSYS;
- __exit:
- rt_free(fullpath);
- return result;
- }
- /**
- * this function will write some specified length data to file system.
- *
- * @param fd the file descriptor.
- * @param buf the data buffer to be written.
- * @param len the data buffer length
- *
- * @return the actual written data length.
- */
- int dfs_file_write(struct dfs_fd *fd, const void *buf, size_t len)
- {
- if (fd == NULL)
- return -EINVAL;
- if (fd->fops->write == NULL)
- return -ENOSYS;
- return fd->fops->write(fd, buf, len);
- }
- /**
- * this function will flush buffer on a file descriptor.
- *
- * @param fd the file descriptor.
- *
- * @return 0 on successful, -1 on failed.
- */
- int dfs_file_flush(struct dfs_fd *fd)
- {
- if (fd == NULL)
- return -EINVAL;
- if (fd->fops->flush == NULL)
- return -ENOSYS;
- return fd->fops->flush(fd);
- }
- /**
- * this function will seek the offset for specified file descriptor.
- *
- * @param fd the file descriptor.
- * @param offset the offset to be sought.
- *
- * @return the current position after seek.
- */
- int dfs_file_lseek(struct dfs_fd *fd, off_t offset)
- {
- int result;
- if (fd == NULL)
- return -EINVAL;
- if (fd->fops->lseek == NULL)
- return -ENOSYS;
- result = fd->fops->lseek(fd, offset);
- /* update current position */
- if (result >= 0)
- fd->pos = result;
- return result;
- }
- /**
- * this function will get file information.
- *
- * @param path the file path.
- * @param buf the data buffer to save stat description.
- *
- * @return 0 on successful, -1 on failed.
- */
- int dfs_file_stat(const char *path, struct stat *buf)
- {
- int result;
- char *fullpath;
- struct dfs_filesystem *fs;
- fullpath = dfs_normalize_path(NULL, path);
- if (fullpath == NULL)
- {
- return -1;
- }
- if ((fs = dfs_filesystem_lookup(fullpath)) == NULL)
- {
- LOG_E("can't find mounted filesystem on this path:%s", fullpath);
- rt_free(fullpath);
- return -ENOENT;
- }
- if ((fullpath[0] == '/' && fullpath[1] == '\0') ||
- (dfs_subdir(fs->path, fullpath) == NULL))
- {
- /* it's the root directory */
- buf->st_dev = 0;
- buf->st_mode = S_IRUSR | S_IRGRP | S_IROTH |
- S_IWUSR | S_IWGRP | S_IWOTH;
- buf->st_mode |= S_IFDIR | S_IXUSR | S_IXGRP | S_IXOTH;
- buf->st_size = 0;
- buf->st_mtime = 0;
- /* release full path */
- rt_free(fullpath);
- return RT_EOK;
- }
- else
- {
- if (fs->ops->stat == NULL)
- {
- rt_free(fullpath);
- LOG_E("the filesystem didn't implement this function");
- return -ENOSYS;
- }
- /* get the real file path and get file stat */
- if (fs->ops->flags & DFS_FS_FLAG_FULLPATH)
- result = fs->ops->stat(fs, fullpath, buf);
- else
- result = fs->ops->stat(fs, dfs_subdir(fs->path, fullpath), buf);
- }
- rt_free(fullpath);
- return result;
- }
- /**
- * this function will rename an old path name to a new path name.
- *
- * @param oldpath the old path name.
- * @param newpath the new path name.
- *
- * @return 0 on successful, -1 on failed.
- */
- int dfs_file_rename(const char *oldpath, const char *newpath)
- {
- int result;
- struct dfs_filesystem *oldfs, *newfs;
- char *oldfullpath, *newfullpath;
- result = RT_EOK;
- newfullpath = NULL;
- oldfullpath = NULL;
- oldfullpath = dfs_normalize_path(NULL, oldpath);
- if (oldfullpath == NULL)
- {
- result = -ENOENT;
- goto __exit;
- }
- newfullpath = dfs_normalize_path(NULL, newpath);
- if (newfullpath == NULL)
- {
- result = -ENOENT;
- goto __exit;
- }
- oldfs = dfs_filesystem_lookup(oldfullpath);
- newfs = dfs_filesystem_lookup(newfullpath);
- if (oldfs == newfs)
- {
- if (oldfs->ops->rename == NULL)
- {
- result = -ENOSYS;
- }
- else
- {
- if (oldfs->ops->flags & DFS_FS_FLAG_FULLPATH)
- result = oldfs->ops->rename(oldfs, oldfullpath, newfullpath);
- else
- /* use sub directory to rename in file system */
- result = oldfs->ops->rename(oldfs,
- dfs_subdir(oldfs->path, oldfullpath),
- dfs_subdir(newfs->path, newfullpath));
- }
- }
- else
- {
- result = -EXDEV;
- }
- __exit:
- rt_free(oldfullpath);
- rt_free(newfullpath);
- /* not at same file system, return EXDEV */
- return result;
- }
- /**
- * this function is will cause the regular file referenced by fd
- * to be truncated to a size of precisely length bytes.
- *
- * @param fd the file descriptor.
- * @param length the length to be truncated.
- *
- * @return the status of truncated.
- */
- int dfs_file_ftruncate(struct dfs_fd *fd, off_t length)
- {
- int result;
- /* fd is null or not a regular file system fd, or length is invalid */
- if (fd == NULL || fd->type != FT_REGULAR || length < 0)
- return -EINVAL;
- if (fd->fops->ioctl == NULL)
- return -ENOSYS;
- result = fd->fops->ioctl(fd, RT_FIOFTRUNCATE, (void*)&length);
- /* update current size */
- if (result == 0)
- fd->size = length;
- return result;
- }
- #ifdef RT_USING_FINSH
- #include <finsh.h>
- static struct dfs_fd fd;
- static struct dirent dirent;
- void ls(const char *pathname)
- {
- struct stat stat;
- int length;
- char *fullpath, *path;
- fullpath = NULL;
- if (pathname == NULL)
- {
- #ifdef DFS_USING_WORKDIR
- /* open current working directory */
- path = rt_strdup(working_directory);
- #else
- path = rt_strdup("/");
- #endif
- if (path == NULL)
- return ; /* out of memory */
- }
- else
- {
- path = (char *)pathname;
- }
- /* list directory */
- if (dfs_file_open(&fd, path, O_DIRECTORY) == 0)
- {
- rt_kprintf("Directory %s:\n", path);
- do
- {
- memset(&dirent, 0, sizeof(struct dirent));
- length = dfs_file_getdents(&fd, &dirent, sizeof(struct dirent));
- if (length > 0)
- {
- memset(&stat, 0, sizeof(struct stat));
- /* build full path for each file */
- fullpath = dfs_normalize_path(path, dirent.d_name);
- if (fullpath == NULL)
- break;
- if (dfs_file_stat(fullpath, &stat) == 0)
- {
- rt_kprintf("%-20s", dirent.d_name);
- if (S_ISDIR(stat.st_mode))
- {
- rt_kprintf("%-25s\n", "<DIR>");
- }
- else
- {
- rt_kprintf("%-25lu\n", (unsigned long)stat.st_size);
- }
- }
- else
- rt_kprintf("BAD file: %s\n", dirent.d_name);
- rt_free(fullpath);
- }
- }
- while (length > 0);
- dfs_file_close(&fd);
- }
- else
- {
- rt_kprintf("No such directory\n");
- }
- if (pathname == NULL)
- rt_free(path);
- }
- FINSH_FUNCTION_EXPORT(ls, list directory contents);
- void rm(const char *filename)
- {
- if (dfs_file_unlink(filename) < 0)
- {
- rt_kprintf("Delete %s failed\n", filename);
- }
- }
- FINSH_FUNCTION_EXPORT(rm, remove files or directories);
- void cat(const char *filename)
- {
- uint32_t length;
- char buffer[81];
- if (dfs_file_open(&fd, filename, O_RDONLY) < 0)
- {
- rt_kprintf("Open %s failed\n", filename);
- return;
- }
- do
- {
- memset(buffer, 0, sizeof(buffer));
- length = dfs_file_read(&fd, buffer, sizeof(buffer) - 1);
- if (length > 0)
- {
- rt_kprintf("%s", buffer);
- }
- }
- while (length > 0);
- dfs_file_close(&fd);
- }
- FINSH_FUNCTION_EXPORT(cat, print file);
- #define BUF_SZ 4096
- static void copyfile(const char *src, const char *dst)
- {
- struct dfs_fd src_fd;
- rt_uint8_t *block_ptr;
- rt_int32_t read_bytes;
- block_ptr = (rt_uint8_t *)rt_malloc(BUF_SZ);
- if (block_ptr == NULL)
- {
- rt_kprintf("out of memory\n");
- return;
- }
- if (dfs_file_open(&src_fd, src, O_RDONLY) < 0)
- {
- rt_free(block_ptr);
- rt_kprintf("Read %s failed\n", src);
- return;
- }
- if (dfs_file_open(&fd, dst, O_WRONLY | O_CREAT) < 0)
- {
- rt_free(block_ptr);
- dfs_file_close(&src_fd);
- rt_kprintf("Write %s failed\n", dst);
- return;
- }
- do
- {
- read_bytes = dfs_file_read(&src_fd, block_ptr, BUF_SZ);
- if (read_bytes > 0)
- {
- int length;
- length = dfs_file_write(&fd, block_ptr, read_bytes);
- if (length != read_bytes)
- {
- /* write failed. */
- rt_kprintf("Write file data failed, errno=%d\n", length);
- break;
- }
- }
- }
- while (read_bytes > 0);
- dfs_file_close(&src_fd);
- dfs_file_close(&fd);
- rt_free(block_ptr);
- }
- extern int mkdir(const char *path, mode_t mode);
- static void copydir(const char *src, const char *dst)
- {
- struct dirent dirent;
- struct stat stat;
- int length;
- struct dfs_fd cpfd;
- if (dfs_file_open(&cpfd, src, O_DIRECTORY) < 0)
- {
- rt_kprintf("open %s failed\n", src);
- return ;
- }
- do
- {
- memset(&dirent, 0, sizeof(struct dirent));
- length = dfs_file_getdents(&cpfd, &dirent, sizeof(struct dirent));
- if (length > 0)
- {
- char *src_entry_full = NULL;
- char *dst_entry_full = NULL;
- if (strcmp(dirent.d_name, "..") == 0 || strcmp(dirent.d_name, ".") == 0)
- continue;
- /* build full path for each file */
- if ((src_entry_full = dfs_normalize_path(src, dirent.d_name)) == NULL)
- {
- rt_kprintf("out of memory!\n");
- break;
- }
- if ((dst_entry_full = dfs_normalize_path(dst, dirent.d_name)) == NULL)
- {
- rt_kprintf("out of memory!\n");
- rt_free(src_entry_full);
- break;
- }
- memset(&stat, 0, sizeof(struct stat));
- if (dfs_file_stat(src_entry_full, &stat) != 0)
- {
- rt_kprintf("open file: %s failed\n", dirent.d_name);
- continue;
- }
- if (S_ISDIR(stat.st_mode))
- {
- mkdir(dst_entry_full, 0);
- copydir(src_entry_full, dst_entry_full);
- }
- else
- {
- copyfile(src_entry_full, dst_entry_full);
- }
- rt_free(src_entry_full);
- rt_free(dst_entry_full);
- }
- }
- while (length > 0);
- dfs_file_close(&cpfd);
- }
- static const char *_get_path_lastname(const char *path)
- {
- char *ptr;
- if ((ptr = (char *)strrchr(path, '/')) == NULL)
- return path;
- /* skip the '/' then return */
- return ++ptr;
- }
- void copy(const char *src, const char *dst)
- {
- #define FLAG_SRC_TYPE 0x03
- #define FLAG_SRC_IS_DIR 0x01
- #define FLAG_SRC_IS_FILE 0x02
- #define FLAG_SRC_NON_EXSIT 0x00
- #define FLAG_DST_TYPE 0x0C
- #define FLAG_DST_IS_DIR 0x04
- #define FLAG_DST_IS_FILE 0x08
- #define FLAG_DST_NON_EXSIT 0x00
- struct stat stat;
- uint32_t flag = 0;
- /* check the staus of src and dst */
- if (dfs_file_stat(src, &stat) < 0)
- {
- rt_kprintf("copy failed, bad %s\n", src);
- return;
- }
- if (S_ISDIR(stat.st_mode))
- flag |= FLAG_SRC_IS_DIR;
- else
- flag |= FLAG_SRC_IS_FILE;
- if (dfs_file_stat(dst, &stat) < 0)
- {
- flag |= FLAG_DST_NON_EXSIT;
- }
- else
- {
- if (S_ISDIR(stat.st_mode))
- flag |= FLAG_DST_IS_DIR;
- else
- flag |= FLAG_DST_IS_FILE;
- }
- //2. check status
- if ((flag & FLAG_SRC_IS_DIR) && (flag & FLAG_DST_IS_FILE))
- {
- rt_kprintf("cp faild, cp dir to file is not permitted!\n");
- return ;
- }
- //3. do copy
- if (flag & FLAG_SRC_IS_FILE)
- {
- if (flag & FLAG_DST_IS_DIR)
- {
- char *fdst;
- fdst = dfs_normalize_path(dst, _get_path_lastname(src));
- if (fdst == NULL)
- {
- rt_kprintf("out of memory\n");
- return;
- }
- copyfile(src, fdst);
- rt_free(fdst);
- }
- else
- {
- copyfile(src, dst);
- }
- }
- else //flag & FLAG_SRC_IS_DIR
- {
- if (flag & FLAG_DST_IS_DIR)
- {
- char *fdst;
- fdst = dfs_normalize_path(dst, _get_path_lastname(src));
- if (fdst == NULL)
- {
- rt_kprintf("out of memory\n");
- return;
- }
- mkdir(fdst, 0);
- copydir(src, fdst);
- rt_free(fdst);
- }
- else if ((flag & FLAG_DST_TYPE) == FLAG_DST_NON_EXSIT)
- {
- mkdir(dst, 0);
- copydir(src, dst);
- }
- else
- {
- copydir(src, dst);
- }
- }
- }
- FINSH_FUNCTION_EXPORT(copy, copy file or dir)
- #endif
- /* @} */
|