[PATCH v2 4/4] fs: fat: Support basic file rename

Daniel Venzin Daniel.Venzin at mt.com
Mon Feb 3 11:24:51 CET 2025


This implementation currently does not support
moving files between directories.

Signed-off-by: Daniel Venzin <Daniel.Venzin at mt.com>

---

Changes in v2:
- Separate the rename implementation from changes in the filesystem layer
- Eliminate code duplication with delete_dentry_long(..)
- Ensure that the source file exists, but not the destination file
- Free the fatbuf if it has been reinitialized with fat_itr_root
- Delete orphaned FAT entry after copying file metadata

 fs/fat/fat_write.c | 110 +++++++++++++++++++++++++++++++++++++++++++++
 fs/fs.c            |   2 +-
 include/fat.h      |   1 +
 3 files changed, 112 insertions(+), 1 deletion(-)

diff --git a/fs/fat/fat_write.c b/fs/fat/fat_write.c
index c78b80e0b7a..2273bf16053 100644
--- a/fs/fat/fat_write.c
+++ b/fs/fat/fat_write.c
@@ -1821,3 +1821,113 @@ exit:
 	free(dotdent);
 	return ret;
 }
+
+int file_fat_rename(const char *path, const char *new_filename)
+{
+	fsdata datablock = { .fatbuf = NULL, };
+	fat_itr *itr = NULL;
+
+	dir_entry *dentry;
+
+	char new_path[VFAT_MAXLEN_BYTES];
+	char *path_copy, *dirname, *basename;
+
+	loff_t actwrite;
+
+	__u16 starthi, start;
+	__u32 size, entry;
+
+	int ret = -1;
+
+	path_copy = strdup(path);
+	if (!path_copy)
+		goto exit;
+
+	split_filename(path_copy, &dirname, &basename);
+	snprintf(new_path, sizeof(new_path), "%s/%s", dirname, new_filename);
+
+	if (!fat_exists(path)) {
+		ret = -ENOENT;
+		goto exit;
+	}
+
+	if (fat_exists(new_path)) {
+		ret = -EEXIST;
+		goto exit;
+	}
+
+	ret = file_fat_write(new_path, &actwrite, 0, 0, &actwrite);
+	if (ret)
+		goto exit;
+
+	itr = malloc_cache_aligned(sizeof(*itr));
+	if (!itr) {
+		ret = -ENOMEM;
+		goto exit;
+	}
+
+	ret = fat_itr_root(itr, &datablock);
+	if (ret)
+		goto exit;
+
+	ret = fat_itr_resolve(itr, path, TYPE_FILE);
+	if (ret)
+		goto exit;
+
+	dentry = itr->dent;
+
+	starthi = dentry->starthi;
+	start = dentry->start;
+	size = dentry->size;
+
+	free(datablock.fatbuf);
+	datablock.fatbuf = NULL;
+
+	ret = fat_itr_root(itr, &datablock);
+	if (ret)
+		goto exit;
+
+	ret = fat_itr_resolve(itr, new_path, TYPE_FILE);
+	if (ret)
+		goto exit;
+
+	dentry = itr->dent;
+
+	fsdata *mydata = itr->fsdata;
+
+	entry = START(dentry);
+
+	dentry->starthi = starthi;
+	dentry->start = start;
+	dentry->size = size;
+
+	ret = flush_dir(itr);
+	if (ret)
+		goto exit;
+
+	clear_fatent(mydata, entry);
+	if (flush_dirty_fat_buffer(mydata) < 0) {
+		printf("Error: flush fat buffer\n");
+		ret = -EIO;
+		goto exit;
+	}
+
+	free(datablock.fatbuf);
+	datablock.fatbuf = NULL;
+
+	ret = fat_itr_root(itr, &datablock);
+	if (ret)
+		goto exit;
+
+	ret = fat_itr_resolve(itr, path, TYPE_FILE);
+	if (ret)
+		goto exit;
+
+	ret = delete_dentry_long(itr, FATENT_CLEAR_SKIP);
+
+exit:
+	free(path_copy);
+	free(datablock.fatbuf);
+	free(itr);
+	return ret;
+}
diff --git a/fs/fs.c b/fs/fs.c
index b746d05ebcd..3576d5c6644 100644
--- a/fs/fs.c
+++ b/fs/fs.c
@@ -208,7 +208,7 @@ static struct fstype_info fstypes[] = {
 		.write = file_fat_write,
 		.unlink = fat_unlink,
 		.mkdir = fat_mkdir,
-		.rename = fs_rename_unsupported,
+		.rename = file_fat_rename,
 #else
 		.write = fs_write_unsupported,
 		.unlink = fs_unlink_unsupported,
diff --git a/include/fat.h b/include/fat.h
index 3dce99a23cf..415ebd7b7c8 100644
--- a/include/fat.h
+++ b/include/fat.h
@@ -207,6 +207,7 @@ int fat_readdir(struct fs_dir_stream *dirs, struct fs_dirent **dentp);
 void fat_closedir(struct fs_dir_stream *dirs);
 int fat_unlink(const char *filename);
 int fat_mkdir(const char *dirname);
+int file_fat_rename(const char *path, const char *new_filename);
 void fat_close(void);
 void *fat_next_cluster(fat_itr *itr, unsigned int *nbytes);
 
-- 
2.47.2



More information about the U-Boot mailing list