summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--nuttx/fs/fat/Make.defs2
-rw-r--r--nuttx/fs/fat/fs_fat32.c97
-rw-r--r--nuttx/fs/fat/fs_fat32.h7
-rw-r--r--nuttx/fs/fat/fs_fat32dirent.c964
-rw-r--r--nuttx/fs/fat/fs_fat32util.c785
5 files changed, 1016 insertions, 839 deletions
diff --git a/nuttx/fs/fat/Make.defs b/nuttx/fs/fat/Make.defs
index 111217efc..96be43961 100644
--- a/nuttx/fs/fat/Make.defs
+++ b/nuttx/fs/fat/Make.defs
@@ -37,7 +37,7 @@ ifeq ($(CONFIG_FS_FAT),y)
# Files required for FAT file system support
ASRCS +=
-CSRCS += fs_fat32.c fs_fat32attrib.c fs_fat32util.c
+CSRCS += fs_fat32.c fs_fat32dirent.c fs_fat32attrib.c fs_fat32util.c
# Files required for mkfatfs utility function
diff --git a/nuttx/fs/fat/fs_fat32.c b/nuttx/fs/fat/fs_fat32.c
index a572f4ac7..098f3be31 100644
--- a/nuttx/fs/fat/fs_fat32.c
+++ b/nuttx/fs/fat/fs_fat32.c
@@ -1,5 +1,5 @@
/****************************************************************************
- * fs_fat32.c
+ * fs/fat/fs_fat32.c
*
* Copyright (C) 2007-2009, 2011 Gregory Nutt. All rights reserved.
* Author: Gregory Nutt <spudmonkey@racsa.co.cr>
@@ -1179,7 +1179,7 @@ static int fat_sync(FAR struct file *filep)
* in the sector using the saved directory index.
*/
- direntry = &fs->fs_buffer[(ff->ff_dirindex & DIRSEC_NDXMASK(fs)) * 32];
+ direntry = &fs->fs_buffer[(ff->ff_dirindex & DIRSEC_NDXMASK(fs)) * DIR_SIZE];
/* Set the archive bit, set the write time, and update
* anything that may have* changed in the directory
@@ -1339,7 +1339,7 @@ static int fat_readdir(struct inode *mountpt, struct fs_dirent_s *dir)
/* Get a reference to the current directory entry */
- dirindex = (dir->u.fat.fd_index & DIRSEC_NDXMASK(fs)) * 32;
+ dirindex = (dir->u.fat.fd_index & DIRSEC_NDXMASK(fs)) * DIR_SIZE;
direntry = &fs->fs_buffer[dirindex];
/* Has it reached to end of the directory */
@@ -1740,7 +1740,8 @@ static int fat_mkdir(struct inode *mountpt, const char *relpath, mode_t mode)
/* What we want to see is for fat_finddirentry to fail with -ENOENT.
* This error means that no failure occurred but that nothing exists
- * with this name.
+ * with this name. NOTE: The name has already been set in dirinfo
+ * structure.
*/
if (ret != -ENOENT)
@@ -1813,7 +1814,10 @@ static int fat_mkdir(struct inode *mountpt, const char *relpath, mode_t mode)
}
}
- /* Now create the "." directory entry in the first directory slot */
+ /* Now create the "." directory entry in the first directory slot. These
+ * are special directory entries and are not handled by the normal directory
+ * management routines.
+ */
memset(&direntry[DIR_NAME], ' ', 8+3);
direntry[DIR_NAME] = '.';
@@ -1827,11 +1831,11 @@ static int fat_mkdir(struct inode *mountpt, const char *relpath, mode_t mode)
/* Create ".." directory entry in the second directory slot */
- direntry2 = direntry + 32;
+ direntry2 = direntry + DIR_SIZE;
/* So far, the two entries are nearly the same */
- memcpy(direntry2, direntry, 32);
+ memcpy(direntry2, direntry, DIR_SIZE);
direntry2[DIR_NAME+1] = '.';
/* Now add the cluster information to both directory entries */
@@ -1849,7 +1853,7 @@ static int fat_mkdir(struct inode *mountpt, const char *relpath, mode_t mode)
DIR_PUTFSTCLUSTLO(direntry2, parentcluster);
/* Save the first sector of the directory cluster and re-read
- * the parentsector
+ * the parentsector
*/
fs->fs_dirty = true;
@@ -1859,32 +1863,24 @@ static int fat_mkdir(struct inode *mountpt, const char *relpath, mode_t mode)
goto errout_with_semaphore;
}
- /* Initialize the new entry directory entry in the parent directory */
-
- direntry = dirinfo.fd_entry;
- memset(direntry, 0, 32);
-
- memcpy(direntry, dirinfo.fd_name, 8+3);
-#ifdef CONFIG_FLAT_LCNAMES
- DIR_PUTNTRES(direntry, dirinfo.fd_ntflags);
-#endif
- DIR_PUTATTRIBUTES(dirinfo.fd_entry, FATATTR_DIRECTORY);
-
- /* Same creation time as for . and .. */
+ /* Write the new entry directory entry in the parent directory */
- DIR_PUTCRTIME(dirinfo.fd_entry, crtime & 0xffff);
- DIR_PUTWRTTIME(dirinfo.fd_entry, crtime & 0xffff);
- DIR_PUTCRDATE(dirinfo.fd_entry, crtime >> 16);
- DIR_PUTWRTDATE(dirinfo.fd_entry, crtime >> 16);
+ ret = fat_dirwrite(fs, &dirinfo, FATATTR_DIRECTORY, crtime);
+ if (ret < 0)
+ {
+ goto errout_with_semaphore;
+ }
- /* Set subdirectory start cluster */
+ /* Set subdirectory start cluster. We assume that fat_dirwrite() did not
+ * change the sector in the cache.
+ */
DIR_PUTFSTCLUSTLO(dirinfo.fd_entry, dircluster);
DIR_PUTFSTCLUSTHI(dirinfo.fd_entry, dircluster >> 16);
+ fs->fs_dirty = true;
/* Now update the FAT32 FSINFO sector */
- fs->fs_dirty = true;
ret = fat_updatefsinfo(fs);
if (ret < 0)
{
@@ -1958,7 +1954,7 @@ int fat_rename(struct inode *mountpt, const char *oldrelpath,
off_t oldsector;
uint8_t *olddirentry;
uint8_t *newdirentry;
- uint8_t dirstate[32-11];
+ uint8_t dirstate[DIR_SIZE-DIR_ATTRIBUTES];
int ret;
/* Sanity checks */
@@ -1988,26 +1984,23 @@ int fat_rename(struct inode *mountpt, const char *oldrelpath,
goto errout_with_semaphore;
}
- /* Save the information that will need to recover the
- * directory sector and directory entry offset to the
- * old directory.
- */
-
- olddirentry = dirinfo.fd_entry;
-
- /* One more check: Make sure that the oldrelpath does
- * not refer to the root directory. We can't rename the
- * root directory.
+ /* One more check: Make sure that the oldrelpath does not refer to the
+ * root directory. We can't rename the root directory.
*/
- if (!olddirentry)
+ if (!dirinfo.fd_entry)
{
ret = -EXDEV;
goto errout_with_semaphore;
}
+ /* Save the information that will need to recover the directory sector and
+ * directory entry offset to the old directory.
+ */
+
+ olddirentry = dirinfo.fd_entry;
oldsector = fs->fs_currentsector;
- memcpy(dirstate, &olddirentry[DIR_ATTRIBUTES], 32-11);
+ memcpy(dirstate, &olddirentry[DIR_ATTRIBUTES], DIR_SIZE-DIR_ATTRIBUTES);
/* No find the directory where we should create the newpath object */
@@ -2037,18 +2030,17 @@ int fat_rename(struct inode *mountpt, const char *oldrelpath,
goto errout_with_semaphore;
}
- /* Create the new directory entry */
+ /* Write the new directory entry */
newdirentry = dirinfo.fd_entry;
-
- memcpy(&newdirentry[DIR_ATTRIBUTES], dirstate, 32-11);
- memcpy(&newdirentry[DIR_NAME], dirinfo.fd_name, 8+3);
-#ifdef CONFIG_FLAT_LCNAMES
- DIR_PUTNTRES(newdirentry, dirinfo.fd_ntflags);
-#else
- DIR_PUTNTRES(newdirentry, 0);
-#endif
+ memcpy(&newdirentry[DIR_ATTRIBUTES], dirstate, DIR_SIZE-DIR_ATTRIBUTES);
fs->fs_dirty = true;
+
+ ret = fat_dirnamewrite(fs, &dirinfo);
+ if (ret < 0)
+ {
+ goto errout_with_semaphore;
+ }
/* Now flush the new directory entry to disk and read the sector
* containing the old directory entry.
@@ -2062,9 +2054,12 @@ int fat_rename(struct inode *mountpt, const char *oldrelpath,
/* Remove the old entry */
- olddirentry[DIR_NAME] = DIR0_EMPTY;
- fs->fs_dirty = true;
-
+ ret = fat_freedirentry(fs, olddirentry);
+ if (ret < 0)
+ {
+ goto errout_with_semaphore;
+ }
+
/* Write the old entry to disk and update FSINFO if necessary */
ret = fat_updatefsinfo(fs);
diff --git a/nuttx/fs/fat/fs_fat32.h b/nuttx/fs/fat/fs_fat32.h
index 76a965215..89c42e2a5 100644
--- a/nuttx/fs/fat/fs_fat32.h
+++ b/nuttx/fs/fat/fs_fat32.h
@@ -773,13 +773,16 @@ EXTERN int32_t fat_extendchain(struct fat_mountpt_s *fs, uint32_t cluster);
#define fat_createchain(fs) fat_extendchain(fs, 0)
-/* Help for traversing directory trees */
+/* Help for traversing directory trees and accessing directory entries */
EXTERN int fat_nextdirentry(struct fat_mountpt_s *fs, struct fs_fatdir_s *dir);
EXTERN int fat_finddirentry(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo,
const char *path);
+EXTERN int fat_dirnamewrite(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo);
+EXTERN int fat_dirwrite(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo,
+ uint8_t attributes, uint32_t fattime);
EXTERN int fat_allocatedirentry(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo);
-
+EXTERN int fat_freedirentry(struct fat_mountpt_s *fs, FAR uint8_t *direntry);
EXTERN int fat_dirname2path(char *path, uint8_t *direntry);
/* File creation and removal helpers */
diff --git a/nuttx/fs/fat/fs_fat32dirent.c b/nuttx/fs/fat/fs_fat32dirent.c
new file mode 100644
index 000000000..58d08e33c
--- /dev/null
+++ b/nuttx/fs/fat/fs_fat32dirent.c
@@ -0,0 +1,964 @@
+/****************************************************************************
+ * fs/fat/fs_fat32dirent.c
+ *
+ * Copyright (C) 2011 Gregory Nutt. All rights reserved.
+ * Author: Gregory Nutt <spudmonkey@racsa.co.cr>
+ *
+ * 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.
+ *
+ ****************************************************************************/
+
+/****************************************************************************
+ * NOTE: If CONFIG_FAT_LFN is defined, then there may be some legal, patent
+ * issues. The following was extracted from the entry "File Allocation Table
+ * from Wikipedia, the free encyclopedia:
+ *
+ * "On December 3, 2003 Microsoft announced it would be offering licenses
+ * for use of its FAT specification and 'associated intellectual property',
+ * at the cost of a US$0.25 royalty per unit sold, with a $250,000 maximum
+ * royalty per license agreement.
+ *
+ * o "U.S. Patent 5,745,902 (http://www.google.com/patents?vid=5745902) -
+ * Method and system for accessing a file using file names having
+ * different file name formats. ...
+ * o "U.S. Patent 5,579,517 (http://www.google.com/patents?vid=5579517) -
+ * Common name space for long and short filenames. ...
+ * o "U.S. Patent 5,758,352 (http://www.google.com/patents?vid=5758352) -
+ * Common name space for long and short filenames. ...
+ * o "U.S. Patent 6,286,013 (http://www.google.com/patents?vid=6286013) -
+ * Method and system for providing a common name space for long and
+ * short file names in an operating system. ...
+ *
+ * "Many technical commentators have concluded that these patents only cover
+ * FAT implementations that include support for long filenames, and that
+ * removable solid state media and consumer devices only using short names
+ * would be unaffected. ..."
+ *
+ * So you have been forewarned: Use the long filename at your own risk!
+ *
+ ****************************************************************************/
+
+/****************************************************************************
+ * Included Files
+ ****************************************************************************/
+
+#include <nuttx/config.h>
+
+#include <sys/types.h>
+
+#include <stdint.h>
+#include <stdbool.h>
+#include <stdlib.h>
+#include <string.h>
+#include <ctype.h>
+#include <assert.h>
+#include <errno.h>
+#include <debug.h>
+
+#include <nuttx/fs.h>
+#include <nuttx/fat.h>
+
+#include "fs_internal.h"
+#include "fs_fat32.h"
+
+/****************************************************************************
+ * Definitions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Private Types
+ ****************************************************************************/
+
+/****************************************************************************
+ * Private Function Prototypes
+ ****************************************************************************/
+
+/****************************************************************************
+ * Private Variables
+ ****************************************************************************/
+
+/****************************************************************************
+ * Public Variables
+ ****************************************************************************/
+
+/****************************************************************************
+ * Private Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Name: fat_path2dirname
+ *
+ * Desciption: Convert a user filename into a properly formatted FAT
+ * (short) filname as it would appear in a directory entry. Here are the
+ * rules for the 11 byte name in the directory:
+ *
+ * The first byte:
+ * - 0xe5 = The directory is free
+ * - 0x00 = This directory and all following directories are free
+ * - 0x05 = Really 0xe5
+ * - 0x20 = May NOT be ' '
+ *
+ * Any bytes
+ * 0x00-0x1f = (except for 0x00 and 0x05 in the first byte)
+ * 0x22 = '"'
+ * 0x2a-0x2c = '*', '+', ','
+ * 0x2e-0x2f = '.', '/'
+ * 0x3a-0x3f = ':', ';', '<', '=', '>', '?'
+ * 0x5b-0x5d = '[', '\\', ;]'
+ * 0x7c = '|'
+ *
+ * Upper case characters are not allowed in directory names (without some
+ * poorly documented operatgions on the NTRes directory byte). Lower case
+ * codes may represent different characters in other character sets ("DOS
+ * code pages". The logic below does not, at present, support any other
+ * character sets.
+ *
+ ****************************************************************************/
+
+static inline int fat_path2dirname(const char **path, struct fat_dirinfo_s *dirinfo,
+ char *terminator)
+{
+#ifdef CONFIG_FAT_LCNAMES
+ unsigned int ntlcenable = FATNTRES_LCNAME | FATNTRES_LCEXT;
+ unsigned int ntlcfound = 0;
+#endif
+ const char *node = *path;
+ int endndx;
+ uint8_t ch;
+ int ndx = 0;
+
+ /* Initialized the name with all spaces */
+
+ memset(dirinfo->fd_name, ' ', 8+3);
+
+ /* Loop until the name is successfully parsed or an error occurs */
+
+ endndx = 8;
+ for (;;)
+ {
+ /* Get the next byte from the path */
+
+ ch = *node++;
+
+ /* Check if this the last byte in this node of the name */
+
+ if ((ch == '\0' || ch == '/') && ndx != 0 )
+ {
+ /* Return the accumulated NT flags and the terminating character */
+#ifdef CONFIG_FAT_LCNAMES
+ dirinfo->fd_ntflags = ntlcfound & ntlcenable;
+#endif
+ *terminator = ch;
+ *path = node;
+ return OK;
+ }
+
+ /* Accept only the printable character set. Note the first byte
+ * of the name could be 0x05 meaning that is it 0xe5, but this is
+ * not a printable character in this character in either case.
+ */
+
+ else if (!isgraph(ch))
+ {
+ goto errout;
+ }
+
+ /* Check for transition from name to extension */
+
+ else if (ch == '.')
+ {
+ /* Starting the extension */
+
+ ndx = 8;
+ endndx = 11;
+ continue;
+ }
+
+ /* Reject printable characters forbidden by FAT */
+
+ else if (ch == '"' || (ch >= '*' && ch <= ',') ||
+ ch == '.' || ch == '/' ||
+ (ch >= ':' && ch <= '?') ||
+ (ch >= '[' && ch <= ']') ||
+ (ch == '|'))
+ {
+ goto errout;
+ }
+
+ /* Check for upper case charaters */
+
+#ifdef CONFIG_FAT_LCNAMES
+ else if (isupper(ch))
+ {
+ /* Some or all of the characters in the name or extension
+ * are upper case. Force all of the characters to be interpreted
+ * as upper case.
+ */
+
+ if ( endndx == 8)
+ {
+ /* Clear lower case name bit in mask*/
+ ntlcenable &= FATNTRES_LCNAME;
+ }
+ else
+ {
+ /* Clear lower case extension in mask */
+ ntlcenable &= FATNTRES_LCNAME;
+ }
+ }
+#endif
+
+ /* Check for lower case characters */
+
+ else if (islower(ch))
+ {
+ /* Convert the character to upper case */
+
+ ch = toupper(ch);
+
+ /* Some or all of the characters in the name or extension
+ * are lower case. They can be interpreted as lower case if
+ * only if all of the characters in the name or extension are
+ * lower case.
+ */
+
+#ifdef CONFIG_FAT_LCNAMES
+ if ( endndx == 8)
+ {
+ /* Set lower case name bit */
+ ntlcfound |= FATNTRES_LCNAME;
+ }
+ else
+ {
+ /* Set lower case extension bit */
+ ntlcfound |= FATNTRES_LCNAME;
+ }
+#endif
+ }
+
+ /* Check if the file name exceeds the size permitted (without
+ * long file name support
+ */
+
+ if (ndx >= endndx)
+ {
+ goto errout;
+ }
+
+ /* Save next character in the accumulated name */
+
+ dirinfo->fd_name[ndx++] = ch;
+ }
+
+ errout:
+ return -EINVAL;
+}
+
+/****************************************************************************
+ * Public Functions
+ ****************************************************************************/
+
+/****************************************************************************
+ * Name: fat_finddirentry
+ *
+ * Desciption: Given a path to something that may or may not be in the file
+ * system, return the directory entry of the item.
+ *
+ ****************************************************************************/
+
+int fat_finddirentry(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo,
+ const char *path)
+{
+ off_t cluster;
+ uint8_t *direntry = NULL;
+ char terminator;
+ int ret;
+
+ /* Initialize to traverse the chain. Set it to the cluster of
+ * the root directory
+ */
+
+ cluster = fs->fs_rootbase;
+ if (fs->fs_type == FSTYPE_FAT32)
+ {
+ /* For FAT32, the root directory is variable sized and is a
+ * cluster chain like any other directory. fs_rootbase holds
+ * the first cluster of the root directory.
+ */
+
+ dirinfo->dir.fd_startcluster = cluster;
+ dirinfo->dir.fd_currcluster = cluster;
+ dirinfo->dir.fd_currsector = fat_cluster2sector(fs, cluster);
+ }
+ else
+ {
+ /* For FAT12/16, the first sector of the root directory is a sector
+ * relative to the first sector of the fat volume.
+ */
+
+ dirinfo->dir.fd_startcluster = 0;
+ dirinfo->dir.fd_currcluster = 0;
+ dirinfo->dir.fd_currsector = cluster;
+ }
+
+ /* fd_index is the index into the current directory table */
+
+ dirinfo->dir.fd_index = 0;
+
+ /* If no path was provided, then the root directory must be exactly
+ * what the caller is looking for.
+ */
+
+ if (*path == '\0')
+ {
+ dirinfo->fd_entry = NULL;
+ return OK;
+ }
+
+ /* Otherwise, loop until the path is found */
+
+ for (;;)
+ {
+ /* Convert the next the path segment name into the kind of
+ * name that we would see in the directory entry.
+ */
+
+ ret = fat_path2dirname(&path, dirinfo, &terminator);
+ if (ret < 0)
+ {
+ /* ERROR: The filename contains invalid characters or is
+ * too long.
+ */
+
+ return ret;
+ }
+
+ /* Now search the current directory entry for an entry with this
+ * matching name.
+ */
+
+ for (;;)
+ {
+ /* Read the next sector into memory */
+
+ ret = fat_fscacheread(fs, dirinfo->dir.fd_currsector);
+ if (ret < 0)
+ {
+ return ret;
+ }
+
+ /* Get a pointer to the directory entry */
+
+ direntry = &fs->fs_buffer[DIRSEC_BYTENDX(fs, dirinfo->dir.fd_index)];
+
+ /* Check if we are at the end of the directory */
+
+ if (direntry[DIR_NAME] == DIR0_ALLEMPTY)
+ {
+ return -ENOENT;
+ }
+
+ /* Check if we have found the directory entry that we are looking for */
+
+ if (direntry[DIR_NAME] != DIR0_EMPTY &&
+ !(DIR_GETATTRIBUTES(direntry) & FATATTR_VOLUMEID) &&
+ !memcmp(&direntry[DIR_NAME], dirinfo->fd_name, 8+3) )
+ {
+ /* Yes.. break out of the loop */
+
+ break;
+ }
+
+ /* No... get the next directory index and try again */
+
+ if (fat_nextdirentry(fs, &dirinfo->dir) != OK)
+ {
+ return -ENOENT;
+ }
+ }
+
+ /* We get here only if we have found a directory entry that matches
+ * the path element that we are looking for.
+ *
+ * If the terminator character in the path was the end of the string
+ * then we have successfully found the directory entry that describes
+ * the path.
+ */
+
+ if (!terminator)
+ {
+ /* Return the pointer to the matching directory entry */
+
+ dirinfo->fd_entry = direntry;
+ return OK;
+ }
+
+ /* No.. then we have found one of the intermediate directories on
+ * the way to the final path target. In this case, make sure
+ * the thing that we found is, indeed, a directory.
+ */
+
+ if (!(DIR_GETATTRIBUTES(direntry) & FATATTR_DIRECTORY))
+ {
+ /* Ooops.. we found something else */
+
+ return -ENOTDIR;
+ }
+
+ /* Get the cluster number of this directory */
+
+ cluster =
+ ((uint32_t)DIR_GETFSTCLUSTHI(direntry) << 16) |
+ DIR_GETFSTCLUSTLO(direntry);
+
+ /* The restart scanning at the new directory */
+
+ dirinfo->dir.fd_currcluster = dirinfo->dir.fd_startcluster = cluster;
+ dirinfo->dir.fd_currsector = fat_cluster2sector(fs, cluster);
+ dirinfo->dir.fd_index = 2;
+ }
+}
+
+/****************************************************************************
+ * Name: fat_allocatedirentry
+ *
+ * Desciption: Find a free directory entry
+ *
+ ****************************************************************************/
+
+int fat_allocatedirentry(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo)
+{
+ int32_t cluster;
+ off_t sector;
+ uint8_t *direntry;
+ uint8_t ch;
+ int ret;
+ int i;
+
+ /* Re-initialize directory object */
+
+ cluster = dirinfo->dir.fd_startcluster;
+ if (cluster)
+ {
+ /* Cluster chain can be extended */
+
+ dirinfo->dir.fd_currcluster = cluster;
+ dirinfo->dir.fd_currsector = fat_cluster2sector(fs, cluster);
+ }
+ else
+ {
+ /* Fixed size FAT12/16 root directory is at fixxed offset/size */
+
+ dirinfo->dir.fd_currsector = fs->fs_rootbase;
+ }
+ dirinfo->dir.fd_index = 0;
+
+ for (;;)
+ {
+ unsigned int dirindex;
+
+ /* Read the directory sector into fs_buffer */
+
+ ret = fat_fscacheread(fs, dirinfo->dir.fd_currsector);
+ if (ret < 0)
+ {
+ return ret;
+ }
+
+ /* Get a pointer to the entry at fd_index */
+
+ dirindex = (dirinfo->dir.fd_index & DIRSEC_NDXMASK(fs)) * DIR_SIZE;
+ direntry = &fs->fs_buffer[dirindex];
+
+ /* Check if this directory entry is empty */
+
+ ch = direntry[DIR_NAME];
+ if (ch == DIR0_ALLEMPTY || ch == DIR0_EMPTY)
+ {
+ /* It is empty -- we have found a directory entry */
+
+ dirinfo->fd_entry = direntry;
+ return OK;
+ }
+
+ ret = fat_nextdirentry(fs, &dirinfo->dir);
+ if (ret < 0)
+ {
+ return ret;
+ }
+ }
+
+ /* If we get here, then we have reached the end of the directory table
+ * in this sector without finding a free directory enty.
+ *
+ * It this is a fixed size dirctory entry, then this is an error.
+ * Otherwise, we can try to extend the directory cluster chain to
+ * make space for the new directory entry.
+ */
+
+ if (!cluster)
+ {
+ /* The size is fixed */
+ return -ENOSPC;
+ }
+
+ /* Try to extend the cluster chain for this directory */
+
+ cluster = fat_extendchain(fs, dirinfo->dir.fd_currcluster);
+ if (cluster < 0)
+ {
+ return cluster;
+ }
+
+ /* Flush out any cached date in fs_buffer.. we are going to use
+ * it to initialize the new directory cluster.
+ */
+
+ ret = fat_fscacheflush(fs);
+ if (ret < 0)
+ {
+ return ret;
+ }
+
+ /* Clear all sectors comprising the new directory cluster */
+
+ fs->fs_currentsector = fat_cluster2sector(fs, cluster);
+ memset(fs->fs_buffer, 0, fs->fs_hwsectorsize);
+
+ sector = sector;
+ for (i = fs->fs_fatsecperclus; i; i--)
+ {
+ ret = fat_hwwrite(fs, fs->fs_buffer, sector, 1);
+ if ( ret < 0)
+ {
+ return ret;
+ }
+ sector++;
+ }
+
+ dirinfo->fd_entry = fs->fs_buffer;
+ return OK;
+}
+
+/****************************************************************************
+ * Name: fat_freedirentry
+ *
+ * Desciption: Free the directory entry.
+ *
+ * Assumptions: (1) the directory enty is in the cache and (2) direntry
+ * points to the directory entry to be deleted. This obvioulsy needs
+ * to be re-designed to support long file names!
+ *
+ ****************************************************************************/
+
+int fat_freedirentry(struct fat_mountpt_s *fs, FAR uint8_t *direntry)
+{
+ direntry[DIR_NAME] = DIR0_EMPTY;
+ fs->fs_dirty = true;
+ return OK;
+}
+
+/****************************************************************************
+ * Name: fat_dirname2path
+ *
+ * Desciption: Convert a filename in a raw directory entry into a user
+ * filename. This is essentially the inverse operation of that performed
+ * by fat_path2dirname. See that function for more details.
+ *
+ ****************************************************************************/
+
+int fat_dirname2path(char *path, uint8_t *direntry)
+{
+#ifdef CONFIG_FAT_LCNAMES
+ uint8_t ntflags;
+#endif
+ int ch;
+ int ndx;
+
+ /* Check if we will be doing upper to lower case conversions */
+
+#ifdef CONFIG_FAT_LCNAMES
+ ntflags = DIR_GETNTRES(direntry);
+#endif
+
+ /* Get the 8-byte filename */
+
+ for (ndx = 0; ndx < 8; ndx++)
+ {
+ /* Get the next filename character from the directory entry */
+
+ ch = direntry[ndx];
+
+ /* Any space (or ndx==8) terminates the filename */
+
+ if (ch == ' ')
+ {
+ break;
+ }
+
+ /* In this version, we never write 0xe5 in the directoryfilenames
+ * (because we do not handle any character sets where 0xe5 is valid
+ * in a filaname), but we could encounted this in a filesystem
+ * written by some other system
+ */
+
+ if (ndx == 0 && ch == DIR0_E5)
+ {
+ ch = 0xe5;
+ }
+
+ /* Check if we should perform upper to lower case conversion
+ * of the (whole) filename.
+ */
+
+#ifdef CONFIG_FAT_LCNAMES
+ if (ntflags & FATNTRES_LCNAME && isupper(ch))
+ {
+ ch = tolower(ch);
+ }
+#endif
+ /* Copy the next character into the filename */
+
+ *path++ = ch;
+ }
+
+ /* Check if there is an extension */
+
+ if (direntry[8] != ' ')
+ {
+ /* Yes, output the dot before the extension */
+
+ *path++ = '.';
+
+ /* Then output the (up to) 3 character extension */
+
+ for (ndx = 8; ndx < 11; ndx++)
+ {
+ /* Get the next extensions character from the directory entry */
+
+ ch = direntry[DIR_NAME + ndx];
+
+ /* Any space (or ndx==11) terminates the extension */
+
+ if (ch == ' ')
+ {
+ break;
+ }
+
+ /* Check if we should perform upper to lower case conversion
+ * of the (whole) filename.
+ */
+
+#ifdef CONFIG_FAT_LCNAMES
+ if (ntflags & FATNTRES_LCEXT && isupper(ch))
+ {
+ ch = tolower(ch);
+ }
+#endif
+ /* Copy the next character into the filename */
+
+ *path++ = ch;
+ }
+ }
+
+ /* Put a null terminator at the end of the filename */
+
+ *path = '\0';
+ return OK;
+}
+
+/****************************************************************************
+ * Name: fat_dirnamewrite
+ *
+ * Desciption: Write the (possibly long) directory entry name.
+ *
+ * Assumption: The directory sector is in the cache.
+ *
+ ****************************************************************************/
+
+int fat_dirnamewrite(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo)
+{
+ uint8_t *direntry = dirinfo->fd_entry;
+
+ memcpy(&direntry[DIR_NAME], dirinfo->fd_name, 8+3);
+#ifdef CONFIG_FLAT_LCNAMES
+ DIR_PUTNTRES(direntry, dirinfo->fd_ntflags);
+#else
+ DIR_PUTNTRES(direntry, 0);
+#endif
+ fs->fs_dirty = true;
+ return OK;
+}
+
+/****************************************************************************
+ * Name: fat_dirwrite
+ *
+ * Desciption: Write a directory entry, possibly with a long file name
+ *
+ * Assumption: The directory sector is in the cache. The caller will write
+ * sector information.
+ *
+ ****************************************************************************/
+
+int fat_dirwrite(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo,
+ uint8_t attributes, uint32_t fattime)
+{
+ uint8_t *direntry;
+
+ /* Initialize the 32-byte directory entry */
+
+ direntry = dirinfo->fd_entry;
+ memset(direntry, 0, DIR_SIZE);
+
+ /* Directory name info */
+
+ (void)fat_dirnamewrite(fs, dirinfo);
+
+ /* Set the attribute attribute, write time, creation time */
+
+ DIR_PUTATTRIBUTES(direntry, attributes);
+
+ /* Set the time information */
+
+ DIR_PUTWRTTIME(direntry, fattime & 0xffff);
+ DIR_PUTCRTIME(direntry, fattime & 0xffff);
+ DIR_PUTWRTDATE(direntry, fattime >> 16);
+ DIR_PUTCRDATE(direntry, fattime >> 16);
+
+ fs->fs_dirty = true;
+ return OK;
+}
+
+/****************************************************************************
+ * Name: fat_dircreate
+ *
+ * Desciption: Create a directory entry for a new file
+ *
+ ****************************************************************************/
+
+int fat_dircreate(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo)
+{
+ uint32_t fattime;
+ int ret;
+
+ /* Allocate a directory entry entry */
+
+ ret = fat_allocatedirentry(fs, dirinfo);
+ if (ret != OK)
+ {
+ /* Failed to set up directory entry */
+
+ return ret;
+ }
+
+ /* Write the directory entry with the current time and the ARCHIVE attribute */
+
+ fattime = fat_systime2fattime();
+ return fat_dirwrite(fs, dirinfo, FATATTR_ARCHIVE, fattime);
+}
+
+/****************************************************************************
+ * Name: fat_remove
+ *
+ * Desciption: Remove a directory or file from the file system. This
+ * implements both rmdir() and unlink().
+ *
+ ****************************************************************************/
+
+int fat_remove(struct fat_mountpt_s *fs, const char *relpath, bool directory)
+{
+ struct fat_dirinfo_s dirinfo;
+ uint32_t dircluster;
+ off_t dirsector;
+ int ret;
+
+ /* Find the directory entry referring to the entry to be deleted */
+
+ ret = fat_finddirentry(fs, &dirinfo, relpath);
+ if (ret != OK)
+ {
+ /* No such path */
+
+ return -ENOENT;
+ }
+
+ /* Check if this is a FAT12/16 root directory */
+
+ if (dirinfo.fd_entry == NULL)
+ {
+ /* The root directory cannot be removed */
+
+ return -EPERM;
+ }
+
+ /* The object has to have write access to be deleted */
+
+ if ((DIR_GETATTRIBUTES(dirinfo.fd_entry) & FATATTR_READONLY) != 0)
+ {
+ /* It is a read-only entry */
+
+ return -EACCES;
+ }
+
+ /* Get the directory sector and cluster containing the
+ * entry to be deleted
+ */
+
+ dirsector = fs->fs_currentsector;
+ dircluster =
+ ((uint32_t)DIR_GETFSTCLUSTHI(dirinfo.fd_entry) << 16) |
+ DIR_GETFSTCLUSTLO(dirinfo.fd_entry);
+
+ /* Is this entry a directory? */
+
+ if (DIR_GETATTRIBUTES(dirinfo.fd_entry) & FATATTR_DIRECTORY)
+ {
+ /* It is a sub-directory. Check if we are be asked to remove
+ * a directory or a file.
+ */
+
+ if (!directory)
+ {
+ /* We are asked to delete a file */
+
+ return -EISDIR;
+ }
+
+ /* We are asked to delete a directory. Check if this
+ * sub-directory is empty
+ */
+
+ dirinfo.dir.fd_currcluster = dircluster;
+ dirinfo.dir.fd_currsector = fat_cluster2sector(fs, dircluster);
+ dirinfo.dir.fd_index = 2;
+
+ /* Loop until either (1) an entry is found in the directory
+ * (error), (2) the directory is found to be empty, or (3) some
+ * error occurs.
+ */
+
+ for (;;)
+ {
+ unsigned int subdirindex;
+ uint8_t *subdirentry;
+
+ /* Make sure that the sector containing the of the
+ * subdirectory sector is in the cache
+ */
+
+ ret = fat_fscacheread(fs, dirinfo.dir.fd_currsector);
+ if (ret < 0)
+ {
+ return ret;
+ }
+
+ /* Get a reference to the next entry in the directory */
+
+ subdirindex = (dirinfo.dir.fd_index & DIRSEC_NDXMASK(fs)) * DIR_SIZE;
+ subdirentry = &fs->fs_buffer[subdirindex];
+
+ /* Is this the last entry in the direcory? */
+
+ if (subdirentry[DIR_NAME] == DIR0_ALLEMPTY)
+ {
+ /* Yes then the directory is empty. Break out of the
+ * loop and delete the directory.
+ */
+
+ break;
+ }
+
+ /* Check if the next entry refers to a file or directory */
+
+ if (subdirentry[DIR_NAME] != DIR0_EMPTY &&
+ !(DIR_GETATTRIBUTES(subdirentry) & FATATTR_VOLUMEID))
+ {
+ /* The directory is not empty */
+
+ return -ENOTEMPTY;
+ }
+
+ /* Get the next directgory entry */
+
+ ret = fat_nextdirentry(fs, &dirinfo.dir);
+ if (ret < 0)
+ {
+ return ret;
+ }
+ }
+ }
+ else
+ {
+ /* It is a file. Check if we are be asked to remove a directory
+ * or a file.
+ */
+
+ if (directory)
+ {
+ /* We are asked to remove a directory */
+
+ return -ENOTDIR;
+ }
+ }
+
+ /* Make sure that the directory containing the entry to be deleted is
+ * in the cache.
+ */
+
+ ret = fat_fscacheread(fs, dirsector);
+ if (ret < 0)
+ {
+ return ret;
+ }
+
+ /* Mark the directory entry 'deleted' */
+
+ ret = fat_freedirentry(fs, dirinfo.fd_entry);
+ if (ret < 0)
+ {
+ return ret;
+ }
+
+ /* And remove the cluster chain making up the subdirectory */
+
+ ret = fat_removechain(fs, dircluster);
+ if (ret < 0)
+ {
+ return ret;
+ }
+
+ /* Update the FSINFO sector (FAT32) */
+
+ ret = fat_updatefsinfo(fs);
+ if (ret < 0)
+ {
+ return ret;
+ }
+
+ return OK;
+}
diff --git a/nuttx/fs/fat/fs_fat32util.c b/nuttx/fs/fat/fs_fat32util.c
index 5bd543163..e54206aff 100644
--- a/nuttx/fs/fat/fs_fat32util.c
+++ b/nuttx/fs/fat/fs_fat32util.c
@@ -50,7 +50,6 @@
#include <stdbool.h>
#include <stdlib.h>
#include <string.h>
-#include <ctype.h>
#include <semaphore.h>
#include <assert.h>
#include <errno.h>
@@ -88,175 +87,6 @@
****************************************************************************/
/****************************************************************************
- * Name: fat_path2dirname
- *
- * Desciption: Convert a user filename into a properly formatted FAT
- * (short) filname as it would appear in a directory entry. Here are the
- * rules for the 11 byte name in the directory:
- *
- * The first byte:
- * - 0xe5 = The directory is free
- * - 0x00 = This directory and all following directories are free
- * - 0x05 = Really 0xe5
- * - 0x20 = May NOT be ' '
- *
- * Any bytes
- * 0x00-0x1f = (except for 0x00 and 0x05 in the first byte)
- * 0x22 = '"'
- * 0x2a-0x2c = '*', '+', ','
- * 0x2e-0x2f = '.', '/'
- * 0x3a-0x3f = ':', ';', '<', '=', '>', '?'
- * 0x5b-0x5d = '[', '\\', ;]'
- * 0x7c = '|'
- *
- * Upper case characters are not allowed in directory names (without some
- * poorly documented operatgions on the NTRes directory byte). Lower case
- * codes may represent different characters in other character sets ("DOS
- * code pages". The logic below does not, at present, support any other
- * character sets.
- *
- ****************************************************************************/
-
-static inline int fat_path2dirname(const char **path, struct fat_dirinfo_s *dirinfo,
- char *terminator)
-{
-#ifdef CONFIG_FAT_LCNAMES
- unsigned int ntlcenable = FATNTRES_LCNAME | FATNTRES_LCEXT;
- unsigned int ntlcfound = 0;
-#endif
- const char *node = *path;
- int endndx;
- uint8_t ch;
- int ndx = 0;
-
- /* Initialized the name with all spaces */
-
- memset(dirinfo->fd_name, ' ', 8+3);
-
- /* Loop until the name is successfully parsed or an error occurs */
-
- endndx = 8;
- for (;;)
- {
- /* Get the next byte from the path */
-
- ch = *node++;
-
- /* Check if this the last byte in this node of the name */
-
- if ((ch == '\0' || ch == '/') && ndx != 0 )
- {
- /* Return the accumulated NT flags and the terminating character */
-#ifdef CONFIG_FAT_LCNAMES
- dirinfo->fd_ntflags = ntlcfound & ntlcenable;
-#endif
- *terminator = ch;
- *path = node;
- return OK;
- }
-
- /* Accept only the printable character set. Note the first byte
- * of the name could be 0x05 meaning that is it 0xe5, but this is
- * not a printable character in this character in either case.
- */
-
- else if (!isgraph(ch))
- {
- goto errout;
- }
-
- /* Check for transition from name to extension */
-
- else if (ch == '.')
- {
- /* Starting the extension */
-
- ndx = 8;
- endndx = 11;
- continue;
- }
-
- /* Reject printable characters forbidden by FAT */
-
- else if (ch == '"' || (ch >= '*' && ch <= ',') ||
- ch == '.' || ch == '/' ||
- (ch >= ':' && ch <= '?') ||
- (ch >= '[' && ch <= ']') ||
- (ch == '|'))
- {
- goto errout;
- }
-
- /* Check for upper case charaters */
-
-#ifdef CONFIG_FAT_LCNAMES
- else if (isupper(ch))
- {
- /* Some or all of the characters in the name or extension
- * are upper case. Force all of the characters to be interpreted
- * as upper case.
- */
-
- if ( endndx == 8)
- {
- /* Clear lower case name bit in mask*/
- ntlcenable &= FATNTRES_LCNAME;
- }
- else
- {
- /* Clear lower case extension in mask */
- ntlcenable &= FATNTRES_LCNAME;
- }
- }
-#endif
-
- /* Check for lower case characters */
-
- else if (islower(ch))
- {
- /* Convert the character to upper case */
-
- ch = toupper(ch);
-
- /* Some or all of the characters in the name or extension
- * are lower case. They can be interpreted as lower case if
- * only if all of the characters in the name or extension are
- * lower case.
- */
-
-#ifdef CONFIG_FAT_LCNAMES
- if ( endndx == 8)
- {
- /* Set lower case name bit */
- ntlcfound |= FATNTRES_LCNAME;
- }
- else
- {
- /* Set lower case extension bit */
- ntlcfound |= FATNTRES_LCNAME;
- }
-#endif
- }
-
- /* Check if the file name exceeds the size permitted (without
- * long file name support
- */
-
- if (ndx >= endndx)
- {
- goto errout;
- }
-
- /* Save next character in the accumulated name */
-
- dirinfo->fd_name[ndx++] = ch;
- }
-
- errout:
- return -EINVAL;
-}
-
-/****************************************************************************
* Name: fat_checkfsinfo
*
* Desciption: Read the FAT32 FSINFO sector
@@ -1438,394 +1268,6 @@ int fat_nextdirentry(struct fat_mountpt_s *fs, struct fs_fatdir_s *dir)
}
/****************************************************************************
- * Name: fat_finddirentry
- *
- * Desciption: Given a path to something that may or may not be in the file
- * system, return the directory entry of the item.
- *
- ****************************************************************************/
-
-int fat_finddirentry(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo,
- const char *path)
-{
- off_t cluster;
- uint8_t *direntry = NULL;
- char terminator;
- int ret;
-
- /* Initialize to traverse the chain. Set it to the cluster of
- * the root directory
- */
-
- cluster = fs->fs_rootbase;
- if (fs->fs_type == FSTYPE_FAT32)
- {
- /* For FAT32, the root directory is variable sized and is a
- * cluster chain like any other directory. fs_rootbase holds
- * the first cluster of the root directory.
- */
-
- dirinfo->dir.fd_startcluster = cluster;
- dirinfo->dir.fd_currcluster = cluster;
- dirinfo->dir.fd_currsector = fat_cluster2sector(fs, cluster);
- }
- else
- {
- /* For FAT12/16, the first sector of the root directory is a sector
- * relative to the first sector of the fat volume.
- */
-
- dirinfo->dir.fd_startcluster = 0;
- dirinfo->dir.fd_currcluster = 0;
- dirinfo->dir.fd_currsector = cluster;
- }
-
- /* fd_index is the index into the current directory table */
-
- dirinfo->dir.fd_index = 0;
-
- /* If no path was provided, then the root directory must be exactly
- * what the caller is looking for.
- */
-
- if (*path == '\0')
- {
- dirinfo->fd_entry = NULL;
- return OK;
- }
-
- /* Otherwise, loop until the path is found */
-
- for (;;)
- {
- /* Convert the next the path segment name into the kind of
- * name that we would see in the directory entry.
- */
-
- ret = fat_path2dirname(&path, dirinfo, &terminator);
- if (ret < 0)
- {
- /* ERROR: The filename contains invalid characters or is
- * too long.
- */
-
- return ret;
- }
-
- /* Now search the current directory entry for an entry with this
- * matching name.
- */
-
- for (;;)
- {
- /* Read the next sector into memory */
-
- ret = fat_fscacheread(fs, dirinfo->dir.fd_currsector);
- if (ret < 0)
- {
- return ret;
- }
-
- /* Get a pointer to the directory entry */
-
- direntry = &fs->fs_buffer[DIRSEC_BYTENDX(fs, dirinfo->dir.fd_index)];
-
- /* Check if we are at the end of the directory */
-
- if (direntry[DIR_NAME] == DIR0_ALLEMPTY)
- {
- return -ENOENT;
- }
-
- /* Check if we have found the directory entry that we are looking for */
-
- if (direntry[DIR_NAME] != DIR0_EMPTY &&
- !(DIR_GETATTRIBUTES(direntry) & FATATTR_VOLUMEID) &&
- !memcmp(&direntry[DIR_NAME], dirinfo->fd_name, 8+3) )
- {
- /* Yes.. break out of the loop */
- break;
- }
-
- /* No... get the next directory index and try again */
-
- if (fat_nextdirentry(fs, &dirinfo->dir) != OK)
- {
- return -ENOENT;
- }
- }
-
- /* We get here only if we have found a directory entry that matches
- * the path element that we are looking for.
- *
- * If the terminator character in the path was the end of the string
- * then we have successfully found the directory entry that describes
- * the path.
- */
-
- if (!terminator)
- {
- /* Return the pointer to the matching directory entry */
- dirinfo->fd_entry = direntry;
- return OK;
- }
-
- /* No.. then we have found one of the intermediate directories on
- * the way to the final path target. In this case, make sure
- * the thing that we found is, indeed, a directory.
- */
-
- if (!(DIR_GETATTRIBUTES(direntry) & FATATTR_DIRECTORY))
- {
- /* Ooops.. we found something else */
- return -ENOTDIR;
- }
-
- /* Get the cluster number of this directory */
-
- cluster =
- ((uint32_t)DIR_GETFSTCLUSTHI(direntry) << 16) |
- DIR_GETFSTCLUSTLO(direntry);
-
- /* The restart scanning at the new directory */
-
- dirinfo->dir.fd_currcluster = dirinfo->dir.fd_startcluster = cluster;
- dirinfo->dir.fd_currsector = fat_cluster2sector(fs, cluster);
- dirinfo->dir.fd_index = 2;
- }
-}
-
-/****************************************************************************
- * Name: fat_allocatedirentry
- *
- * Desciption: Find a free directory entry
- *
- ****************************************************************************/
-
-int fat_allocatedirentry(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo)
-{
- int32_t cluster;
- off_t sector;
- uint8_t *direntry;
- uint8_t ch;
- int ret;
- int i;
-
- /* Re-initialize directory object */
-
- cluster = dirinfo->dir.fd_startcluster;
- if (cluster)
- {
- /* Cluster chain can be extended */
-
- dirinfo->dir.fd_currcluster = cluster;
- dirinfo->dir.fd_currsector = fat_cluster2sector(fs, cluster);
- }
- else
- {
- /* Fixed size FAT12/16 root directory is at fixxed offset/size */
-
- dirinfo->dir.fd_currsector = fs->fs_rootbase;
- }
- dirinfo->dir.fd_index = 0;
-
- for (;;)
- {
- unsigned int dirindex;
-
- /* Read the directory sector into fs_buffer */
-
- ret = fat_fscacheread(fs, dirinfo->dir.fd_currsector);
- if (ret < 0)
- {
- return ret;
- }
-
- /* Get a pointer to the entry at fd_index */
-
- dirindex = (dirinfo->dir.fd_index & DIRSEC_NDXMASK(fs)) * 32;
- direntry = &fs->fs_buffer[dirindex];
-
- /* Check if this directory entry is empty */
-
- ch = direntry[DIR_NAME];
- if (ch == DIR0_ALLEMPTY || ch == DIR0_EMPTY)
- {
- /* It is empty -- we have found a directory entry */
-
- dirinfo->fd_entry = direntry;
- return OK;
- }
-
- ret = fat_nextdirentry(fs, &dirinfo->dir);
- if (ret < 0)
- {
- return ret;
- }
- }
-
- /* If we get here, then we have reached the end of the directory table
- * in this sector without finding a free directory enty.
- *
- * It this is a fixed size dirctory entry, then this is an error.
- * Otherwise, we can try to extend the directory cluster chain to
- * make space for the new directory entry.
- */
-
- if (!cluster)
- {
- /* The size is fixed */
- return -ENOSPC;
- }
-
- /* Try to extend the cluster chain for this directory */
-
- cluster = fat_extendchain(fs, dirinfo->dir.fd_currcluster);
- if (cluster < 0)
- {
- return cluster;
- }
-
- /* Flush out any cached date in fs_buffer.. we are going to use
- * it to initialize the new directory cluster.
- */
-
- ret = fat_fscacheflush(fs);
- if (ret < 0)
- {
- return ret;
- }
-
- /* Clear all sectors comprising the new directory cluster */
-
- fs->fs_currentsector = fat_cluster2sector(fs, cluster);
- memset(fs->fs_buffer, 0, fs->fs_hwsectorsize);
-
- sector = sector;
- for (i = fs->fs_fatsecperclus; i; i--)
- {
- ret = fat_hwwrite(fs, fs->fs_buffer, sector, 1);
- if ( ret < 0)
- {
- return ret;
- }
- sector++;
- }
-
- dirinfo->fd_entry = fs->fs_buffer;
- return OK;
-}
-
-/****************************************************************************
- * Name: fat_dirname2path
- *
- * Desciption: Convert a filename in a raw directory entry into a user
- * filename. This is essentially the inverse operation of that performed
- * by fat_path2dirname. See that function for more details.
- *
- ****************************************************************************/
-
-int fat_dirname2path(char *path, uint8_t *direntry)
-{
-#ifdef CONFIG_FAT_LCNAMES
- uint8_t ntflags;
-#endif
- int ch;
- int ndx;
-
- /* Check if we will be doing upper to lower case conversions */
-
-#ifdef CONFIG_FAT_LCNAMES
- ntflags = DIR_GETNTRES(direntry);
-#endif
-
- /* Get the 8-byte filename */
-
- for (ndx = 0; ndx < 8; ndx++)
- {
- /* Get the next filename character from the directory entry */
-
- ch = direntry[ndx];
-
- /* Any space (or ndx==8) terminates the filename */
-
- if (ch == ' ')
- {
- break;
- }
-
- /* In this version, we never write 0xe5 in the directoryfilenames
- * (because we do not handle any character sets where 0xe5 is valid
- * in a filaname), but we could encounted this in a filesystem
- * written by some other system
- */
-
- if (ndx == 0 && ch == DIR0_E5)
- {
- ch = 0xe5;
- }
-
- /* Check if we should perform upper to lower case conversion
- * of the (whole) filename.
- */
-
-#ifdef CONFIG_FAT_LCNAMES
- if (ntflags & FATNTRES_LCNAME && isupper(ch))
- {
- ch = tolower(ch);
- }
-#endif
- /* Copy the next character into the filename */
-
- *path++ = ch;
- }
-
- /* Check if there is an extension */
-
- if (direntry[8] != ' ')
- {
- /* Yes, output the dot before the extension */
-
- *path++ = '.';
-
- /* Then output the (up to) 3 character extension */
-
- for (ndx = 8; ndx < 11; ndx++)
- {
- /* Get the next extensions character from the directory entry */
-
- ch = direntry[DIR_NAME + ndx];
-
- /* Any space (or ndx==11) terminates the extension */
-
- if (ch == ' ')
- {
- break;
- }
-
- /* Check if we should perform upper to lower case conversion
- * of the (whole) filename.
- */
-
-#ifdef CONFIG_FAT_LCNAMES
- if (ntflags & FATNTRES_LCEXT && isupper(ch))
- {
- ch = tolower(ch);
- }
-#endif
- /* Copy the next character into the filename */
-
- *path++ = ch;
- }
- }
-
- /* Put a null terminator at the end of the filename */
-
- *path = '\0';
- return OK;
-}
-
-/****************************************************************************
* Name: fat_dirtruncate
*
* Desciption: Truncate an existing file to zero length
@@ -1888,233 +1330,6 @@ int fat_dirtruncate(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo)
}
/****************************************************************************
- * Name: fat_dircreate
- *
- * Desciption: Create a directory entry for a new file
- *
- ****************************************************************************/
-
-int fat_dircreate(struct fat_mountpt_s *fs, struct fat_dirinfo_s *dirinfo)
-{
- uint8_t *direntry;
- uint32_t fattime;
- int ret;
-
- /* Set up the directory entry */
-
- ret = fat_allocatedirentry(fs, dirinfo);
- if (ret != OK)
- {
- /* Failed to set up directory entry */
- return ret;
- }
-
- /* Initialize the 32-byte directory entry */
-
- direntry = dirinfo->fd_entry;
- memset(direntry, 0, 32);
-
- /* Directory name info */
-
- memcpy(&direntry[DIR_NAME], dirinfo->fd_name, 8+3);
-#ifdef CONFIG_FLAT_LCNAMES
- DIR_PUTNTRES(dirinfo->fd_entry, dirinfo->fd_ntflags);
-#else
- DIR_PUTNTRES(dirinfo->fd_entry, 0);
-#endif
-
- /* ARCHIVE attribute, write time, creation time */
- DIR_PUTATTRIBUTES(dirinfo->fd_entry, FATATTR_ARCHIVE);
-
- fattime = fat_systime2fattime();
- DIR_PUTWRTTIME(dirinfo->fd_entry, fattime & 0xffff);
- DIR_PUTCRTIME(dirinfo->fd_entry, fattime & 0xffff);
- DIR_PUTWRTDATE(dirinfo->fd_entry, fattime >> 16);
- DIR_PUTCRDATE(dirinfo->fd_entry, fattime >> 16);
-
- fs->fs_dirty = true;
- return OK;
-}
-
-/****************************************************************************
- * Name: fat_remove
- *
- * Desciption: Remove a directory or file from the file system. This
- * implements both rmdir() and unlink().
- *
- ****************************************************************************/
-
-int fat_remove(struct fat_mountpt_s *fs, const char *relpath, bool directory)
-{
- struct fat_dirinfo_s dirinfo;
- uint32_t dircluster;
- off_t dirsector;
- int ret;
-
- /* Find the directory entry referring to the entry to be deleted */
-
- ret = fat_finddirentry(fs, &dirinfo, relpath);
- if (ret != OK)
- {
- /* No such path */
-
- return -ENOENT;
- }
-
- /* Check if this is a FAT12/16 root directory */
-
- if (dirinfo.fd_entry == NULL)
- {
- /* The root directory cannot be removed */
-
- return -EPERM;
- }
-
- /* The object has to have write access to be deleted */
-
- if ((DIR_GETATTRIBUTES(dirinfo.fd_entry) & FATATTR_READONLY) != 0)
- {
- /* It is a read-only entry */
-
- return -EACCES;
- }
-
- /* Get the directory sector and cluster containing the
- * entry to be deleted
- */
-
- dirsector = fs->fs_currentsector;
- dircluster =
- ((uint32_t)DIR_GETFSTCLUSTHI(dirinfo.fd_entry) << 16) |
- DIR_GETFSTCLUSTLO(dirinfo.fd_entry);
-
- /* Is this entry a directory? */
-
- if (DIR_GETATTRIBUTES(dirinfo.fd_entry) & FATATTR_DIRECTORY)
- {
- /* It is a sub-directory. Check if we are be asked to remove
- * a directory or a file.
- */
-
- if (!directory)
- {
- /* We are asked to delete a file */
-
- return -EISDIR;
- }
-
- /* We are asked to delete a directory. Check if this
- * sub-directory is empty
- */
-
- dirinfo.dir.fd_currcluster = dircluster;
- dirinfo.dir.fd_currsector = fat_cluster2sector(fs, dircluster);
- dirinfo.dir.fd_index = 2;
-
- /* Loop until either (1) an entry is found in the directory
- * (error), (2) the directory is found to be empty, or (3) some
- * error occurs.
- */
-
- for (;;)
- {
- unsigned int subdirindex;
- uint8_t *subdirentry;
-
- /* Make sure that the sector containing the of the
- * subdirectory sector is in the cache
- */
-
- ret = fat_fscacheread(fs, dirinfo.dir.fd_currsector);
- if (ret < 0)
- {
- return ret;
- }
-
- /* Get a reference to the next entry in the directory */
-
- subdirindex = (dirinfo.dir.fd_index & DIRSEC_NDXMASK(fs)) * 32;
- subdirentry = &fs->fs_buffer[subdirindex];
-
- /* Is this the last entry in the direcory? */
-
- if (subdirentry[DIR_NAME] == DIR0_ALLEMPTY)
- {
- /* Yes then the directory is empty. Break out of the
- * loop and delete the directory.
- */
-
- break;
- }
-
- /* Check if the next entry refers to a file or directory */
-
- if (subdirentry[DIR_NAME] != DIR0_EMPTY &&
- !(DIR_GETATTRIBUTES(subdirentry) & FATATTR_VOLUMEID))
- {
- /* The directory is not empty */
-
- return -ENOTEMPTY;
- }
-
- /* Get the next directgory entry */
-
- ret = fat_nextdirentry(fs, &dirinfo.dir);
- if (ret < 0)
- {
- return ret;
- }
- }
- }
- else
- {
- /* It is a file. Check if we are be asked to remove a directory
- * or a file.
- */
-
- if (directory)
- {
- /* We are asked to remove a directory */
-
- return -ENOTDIR;
- }
- }
-
- /* Make sure that the directory containing the entry to be deleted is
- * in the cache.
- */
-
- ret = fat_fscacheread(fs, dirsector);
- if (ret < 0)
- {
- return ret;
- }
-
- /* Mark the directory entry 'deleted' */
-
- dirinfo.fd_entry[DIR_NAME] = DIR0_EMPTY;
- fs->fs_dirty = true;
-
- /* And remove the cluster chain making up the subdirectory */
-
- ret = fat_removechain(fs, dircluster);
- if (ret < 0)
- {
- return ret;
- }
-
- /* Update the FSINFO sector (FAT32) */
-
- ret = fat_updatefsinfo(fs);
- if (ret < 0)
- {
- return ret;
- }
-
- return OK;
-}
-
-/****************************************************************************
* Name: fat_fscacheflush
*
* Desciption: Flush any dirty sector if fs_buffer as necessary