Re: [RFC][patch] VMUFAT filesystem - v2

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Hi Adrian.

Some nitpicks. I do not know filesystems so nothing value on that front.

> 
> I will be writing appropriate userland tools and more documentation in
> due course.
> 
>  fs/Kconfig         |    1 +
>  fs/Makefile        |    1 +
>  fs/vmufat/Kconfig  |   14 +
>  fs/vmufat/Makefile |    7 +
>  fs/vmufat/inode.c  | 1400 ++++++++++++++++++++++++++++++++++++++++++++++++++++
>  5 files changed, 1423 insertions(+), 0 deletions(-)

That was a lot of code in one file.

> 
> Signed-off by: Adrian McMenamin <adrian@xxxxxxxxxxxxxxxxx>

It is:
Signed-off-by: ...


> +++ b/fs/vmufat/Makefile
> @@ -0,0 +1,7 @@
> +#
> +# Makefile for VMUFAT filesystem
> +#
> +
> +obj-$(CONFIG_VMUFAT_FS) += vmufat.o
> +
> +vmufat-objs := inode.o

Please use:
vmufat-y := inode.o

This is preferred syntax these days.

> diff --git a/fs/vmufat/inode.c b/fs/vmufat/inode.c
> new file mode 100644
> index 0000000..8ac0cba
> --- /dev/null
> +++ b/fs/vmufat/inode.c
> @@ -0,0 +1,1400 @@
> +/*
> + * inode operations for the VMU file system
> + *
> + * Copyright (C) 2002 - 2009	Adrian McMenamin
> + * Copyright (C) 2002		Paul Mundt
> + *
> + * Released under the terms of the GNU GPL.
> + */
> +#include <linux/module.h>
> +#include <linux/init.h>
> +#include <linux/fs.h>
> +#include <linux/slab.h>
> +#include <linux/blkdev.h>
> +#include <linux/buffer_head.h>
> +#include <linux/statfs.h>
> +#include <linux/mpage.h>
> +#include <linux/bcd.h>

The preference is to either sort includes by length (longest first)
or in alphabetic order.


> +
> +#define VMUFAT_NAMELEN 12
> +
> +/* GNU utils won't list files with inode num 0 */
> +#define VMUFAT_ZEROBLOCK 32768
> +#define VMU_BLK_SZ 512
> +#define VMUFAT_MAGIC 0x55555555
No so magic?
And it belongs in <linux/magic.h>

> +
> +static struct kmem_cache *vmufat_inode_cachep;
> +static struct kmem_cache *vmufat_blist_cachep;
> +static const struct inode_operations vmufat_inode_operations;
> +static const struct file_operations vmufat_file_operations;
> +static const struct address_space_operations vmufat_address_space_operations;
> +static const struct file_operations vmufat_file_dir_operations;
> +static const struct super_operations vmufat_super_operations;
> +
> +static struct inode *vmufat_get_inode(struct super_block *sb, long ino);
> +static int vmufat_list_blocks(struct inode *in);
Can you rearrange things so we do not need these forward decalarations?

> +struct memcard {
> +	long sb_bnum;
> +	long fat_bnum;
> +	long fat_len;
> +	long dir_bnum;
> +	long dir_len;
> +	long numblocks;
> +};
> +
> +struct vmufat_block_list {
> +	struct list_head b_list;
> +	int bno;
> +};
> +
> +struct vmufat_inode {
> +	struct vmufat_block_list blocks;
> +	int nblcks;
> +	struct inode vfs_inode;
> +};
> +
> +static struct vmufat_inode *VMUFAT_I(struct inode *in)
> +{
> +	return container_of(in, struct vmufat_inode, vfs_inode);
> +}
> +
> +struct vmufat_file_info {
> +	__u8 ftype;
> +	__u8 copy_pro;
> +	__u16 fblk;
> +	char fname[12];
> +};
VMUFAT_NAMELEN?

> +/* VMU hardware is flaky, so let's compensate for that
> + * without losing hardare independence -
> + * as it is likely to be where this filesystem is used
> + */
> +static inline struct buffer_head *vmufat_sb_bread(struct super_block *sb,
> +	sector_t block)
> +{
> +	struct buffer_head *bh;
> +	bh = sb_bread(sb, block);
> +	if (bh)
> +		return bh;
> +	return sb_bread(sb, block);
> +}
Looks wrong that we just try the same thing twice.

> +
> +/* Linear day numbers of the respective 1sts in non-leap years. */
> +static int day_n[] =
> +    {0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334};
> +
> +static struct dentry *vmufat_inode_lookup(struct inode *in, struct dentry *dent,
> +	struct nameidata *nd)
> +{
> +	struct super_block *sb;
> +	struct memcard *vmudetails;
> +	struct buffer_head *bh;
> +	struct inode *ino;
> +	char name[VMUFAT_NAMELEN];
> +	long blck_read;
> +	int error = 0, fno = 0;
> +
> +	if (dent->d_name.len > VMUFAT_NAMELEN) {
> +		error = -ENAMETOOLONG;
> +		goto out;
> +	}
> +
> +	sb = in->i_sb;
> +	vmudetails = sb->s_fs_info;
> +	blck_read = vmudetails->dir_bnum;
> +
> +	bh = vmufat_sb_bread(sb, blck_read);
> +	if (!bh) {
> +		error = -EIO;
> +		goto out;
> +	}
> +
> +	do {
> +		/* Have we got a file? */
> +		if (bh->b_data[vmufat_index(fno)] == 0)
> +			goto next;
> +
> +		/* get file name */
> +		memcpy(name,
> +		       bh->b_data + 4 + vmufat_index(fno), VMUFAT_NAMELEN);
> +		/* do names match ?*/
> +		if (memcmp(dent->d_name.name, name, dent->d_name.len) == 0) {
> +			/* read the inode number from the directory */
> +			ino = vmufat_get_inode(sb,
> +				le16_to_cpu(((u16 *) bh->b_data)
> +					[1 + vmufat_index_16(fno)]));
> +			if (!ino) {
> +				error = -EACCES;
> +				goto release_bh;
> +			}
> +			if (IS_ERR(ino)) {
> +				error = PTR_ERR(ino);
> +				goto release_bh;
> +			}
> +			/* return the entry */
> +			d_add(dent, ino);
> +			goto release_bh;
> +		}
> +next:
> +		/* did not match, so try the next file */
> +		fno++;
> +		/* do we need to move to the next block in the directory? */
> +		if (fno >= 0x10) {

Can we have a descriptive constant here..


> +			fno = 0;
> +			blck_read--;
> +			if (blck_read <=
> +				vmudetails->dir_bnum - vmudetails->dir_len) {
> +				d_add(dent, NULL);
> +				break;
> +			}
> +			brelse(bh);
> +			bh = vmufat_sb_bread(sb, blck_read);
> +			if (!bh) {
> +				error = -EIO;
> +				goto out;
> +			}
> +		}
> +	} while (1);
> +
> +release_bh:
> +	brelse(bh);
> +out:
> +	return ERR_PTR(error);
> +}
> +

> +/*
> + * Find a free block in the FAT and mark it
> + * as the end of a file
> + */
> +static int vmufat_find_free(struct super_block *sb)
> +{
> +	struct memcard *vmudetails = sb->s_fs_info;
> +	int nextblock, x;
> +	u16 fatdata;
> +	struct buffer_head *bh_fat;
> +	int error = 0;
> +
> +	nextblock = vmudetails->fat_bnum + vmudetails->fat_len - 1;
> +	x = VMU_BLK_SZ;
> +	bh_fat =
> +	    vmufat_sb_bread(sb, nextblock);
> +	if (!bh_fat) {
> +		error = -EIO;
> +		goto fail;
> +	}
> +
> +	do {
> +		fatdata = ((u16 *) bh_fat->b_data)[x];
> +		if (fatdata == 0xfffc)
> +			break;	/*empty block */
Is this endian safe - you use le16_to_cpu before.
Hmmm, you are reading a char * and casting it to a u16 *.
So -4 equals 0xfffc. That should be ok.
But it would be nicer with an accessor function to read this.

Use a constant for 0xfffc (you use it multiple times)

> +		if (--x < 0) {
> +			put_bh(bh_fat);
> +			if (--nextblock >= vmudetails->fat_bnum) {
> +				x = VMU_BLK_SZ;
> +				bh_fat = vmufat_sb_bread(sb, nextblock);
> +				if (!bh_fat) {
> +					error = -EIO;
> +					goto fail;
> +				}
> +			} else
> +				break;
> +		}
> +	} while (1);
> +
> +	if (nextblock < vmudetails->fat_bnum) {
> +		printk(KERN_ERR "VMUFAT: device is full\n");
> +		error = -ENOSPC;
> +		put_bh(bh_fat);
> +		goto fail;
> +	}
> +	put_bh(bh_fat);
> +	return  x + (nextblock - vmudetails->fat_bnum) * VMU_BLK_SZ;
> +
> +fail:
> +	return error;
> +}

> +
> +/* read the FAT for a given block */
> +static u16 vmufat_get_fat(struct super_block *sb, long block)
> +{
> +	struct memcard *vmudetails = sb->s_fs_info;
> +	struct buffer_head *bh;
> +	int offset;
> +	u16 block_content;
> +
> +	offset = block/(VMU_BLK_SZ/2);
Spaces around operators..

> +	if (offset >= vmudetails->fat_len)
> +		return 0xFFFE;

Constant for 0xFFFE

> +
> +	bh = vmufat_sb_bread(sb, offset + 1 +
> +		vmudetails->fat_bnum - vmudetails->fat_len);
> +	if (!bh)
> +		return 0xFFFF;

Constant for 0xFFFF

> +
> +
> +	block_content = ((u16 *)bh->b_data)[block % (VMU_BLK_SZ / 2)];

What is the difference between block % (VMU_BLK_SZ / 2) and block / (VMU_BLK_SZ / 2)?


> +	put_bh(bh);
> +	return block_content;
> +}

> +
> +/* set the FAT for a given block */
> +static int vmufat_set_fat(struct super_block *sb, long block, u16 set)
> +{
> +	struct memcard *vmudetails = sb->s_fs_info;
> +	struct buffer_head *bh;
> +	int offset;
> +
> +	offset = block/(VMU_BLK_SZ/2);
Spaces. And same calculation as before.

> +	if (offset >= vmudetails->fat_len)
> +		return -EINVAL;
> +
> +	bh = vmufat_sb_bread(sb, offset + 1 +
> +		vmudetails->fat_bnum - vmudetails->fat_len);
> +	if (!bh)
> +		return -EIO;
> +
> +	((u16 *) bh->b_data)[block % (VMU_BLK_SZ / 2)] = set;
> +	mark_buffer_dirty(bh);
> +	put_bh(bh);
> +	return 0;
> +}
> +
> +static int vmufat_inode_create(struct inode *dir, struct dentry *de,
> +		int imode, struct nameidata *nd)
> +{
> +	/* Create an inode */
> +	int x = 0, y, z, error = 0, q;
> +	long blck_read;
> +	struct inode *inode;
> +	struct super_block *sb;
> +	struct memcard *vmudetails;
> +	struct buffer_head *bh_fat = NULL, *bh;
> +	unsigned long unix_date;
> +	int year, day, nl_day, month;	/*inspired by FAT driver */
> +	u8 century, u8year;
> +
> +	if (de->d_name.len > VMUFAT_NAMELEN)
> +		return -ENAMETOOLONG;
> +
> +	sb = dir->i_sb;
> +	vmudetails = sb->s_fs_info;
> +
> +	inode = new_inode(sb);
> +	if (!inode) {
> +		error = -ENOSPC;
> +		goto out;
> +	}
> +
> +	/* Walk through blocks looking for place to write
> +	* Is this an executible file? */
> +	if (imode & 73) {	/*Octal 111 */

Then write an octal value?

> +		inode->i_ino = VMUFAT_ZEROBLOCK;
> +		/* But this already allocated? */
> +		if (vmufat_get_fat(sb, 0) != 0xFFFC) {
> +			printk(KERN_ERR
> +				"VMUFAT: cannot write executible file to"
> +				" filesystem - block 0 already allocated.\n");
> +			error = -ENOSPC;
> +			goto clean_inode;
> +		}
> +		q = 0;
> +	} else {
> +		q = vmufat_find_free(sb);
> +		if (q < 0) {
> +			error = q;
> +			goto clean_inode;
> +		}
> +		inode->i_ino = q;
> +	}
> +
> +	error = vmufat_set_fat(sb, q, 0xFFFA);
> +	if (error)
> +		goto clean_inode;
> +
> +	inode->i_uid = 0;
> +	inode->i_gid = 0;
> +	inode->i_mtime = inode->i_atime = inode->i_ctime = CURRENT_TIME;
> +	inode->i_mode = imode;
> +	inode->i_blocks = 1;
> +	inode->i_sb = sb;
> +	insert_inode_hash(inode);
> +	inode->i_op = &vmufat_inode_operations;
> +	inode->i_fop = &vmufat_file_operations;
> +	inode->i_mapping->a_ops = &vmufat_address_space_operations;
> +
> +	/* Write to the directory
> +	* Now search for space for the directory entry */
> +	blck_read = vmudetails->dir_bnum;
> +	bh = vmufat_sb_bread(sb, blck_read);
> +	if (!bh) {
> +		error = -EIO;
> +		goto clean_inode;
> +	}
> +
> +	for (y = 0; y < (vmudetails->dir_len * 0x10); y++) {
> +		if ((y / 0x10) > (vmudetails->dir_bnum - blck_read)) {
> +			brelse(bh);
> +			blck_read--;
> +			bh = vmufat_sb_bread(sb, blck_read);
> +			if (!bh) {
> +				error = -EIO;
> +				goto clean_fat;
> +			}
> +		}
> +		if (((bh->b_data)[vmufat_index(y)]) == 0)
> +			break;
> +	}
> +	/* Have the directory entry
> +	 * so now update it */
> +	z = vmufat_index(y);
> +	if (inode->i_ino != VMUFAT_ZEROBLOCK)
> +		bh->b_data[z] = 0x33;	/* data file */
> +	else
> +		bh->b_data[z] = 0xcc;
> +
> +	if ((bh->b_data[z + 1] != (char) 0x00) &&
> +		(bh->b_data[z + 1] != (char) 0xff))
> +		bh->b_data[z + 1] = (char) 0x00;
> +
> +	if (inode->i_ino != VMUFAT_ZEROBLOCK) {
> +		((u16 *) bh->b_data)[z / 2 + 1] =
> +		    cpu_to_le16(inode->i_ino);
> +		((u16 *) bh->b_data)[z / 2 + 0x0D] = 0;
> +	} else {
> +		((u16 *) bh->b_data)[z / 2 + 1] = 0;
> +		((u16 *) bh->b_data)[z / 2 + 0x0D] = 1;
> +	}
> +
> +	/* Name */
> +	memset((char *) (bh->b_data + z + 0x04), '\0', 0x0C);
> +	memcpy((char *) (bh->b_data + z + 0x04), ((de->d_name).name),
> +		de->d_name.len);
> +
Lots of hardcoded constants above that I think describe the filesystem layout..

> +	/* BCD timestamp it */
> +	unix_date = CURRENT_TIME.tv_sec;
> +	day = unix_date / 86400 - 3652;
> +	year = day / 365;
> +
> +	if ((year + 3) / 4 + 365 * year > day)
> +		year--;
> +
> +	day -= (year + 3) / 4 + 365 * year;
> +	if (day == 59 && !(year & 3)) {
> +		nl_day = day;
> +		month = 2;
> +	} else {
> +		nl_day = (year & 3) || day <= 59 ? day : day - 1;
> +		for (month = 0; month < 12; month++)
> +			if (day_n[month] > nl_day)
> +				break;
> +	}
> +
> +	century = 19;
> +	if (year > 19)
> +		century = 20;
> +
> +	bh->b_data[z + 0x10] = bin2bcd(century);
> +	u8year = year + 80;
> +	if (u8year > 99)
> +		u8year = u8year - 100;
> +
> +	bh->b_data[z + 0x11] = bin2bcd(u8year);
> +	bh->b_data[z + 0x12] = bin2bcd(month);
> +	bh->b_data[z + 0x13] =
> +	    bin2bcd(day - day_n[month - 1] + 1);
> +	bh->b_data[z + 0x14] =
> +	    bin2bcd((unix_date / 3600) % 24);
> +	bh->b_data[z + 0x15] = bin2bcd((unix_date / 60) % 60);
> +	bh->b_data[z + 0x16] = bin2bcd(unix_date % 60);
The above should be in a separate function.

> +
> +	((u16 *) bh->b_data)[z / 2 + 0x0C] =
> +	    cpu_to_le16(inode->i_blocks);
> +
> +	inode->i_mtime.tv_sec = unix_date;
> +	mark_buffer_dirty(bh);
> +	brelse(bh);
> +
> +	error = vmufat_list_blocks(inode);
> +	if (error)
> +		goto clean_fat;
> +
> +	d_instantiate(de, inode); printk("created inode 0x%lX\n", inode->i_ino);
> +	brelse(bh_fat);
> +	return error;
> +
> +clean_fat:
> +	((u16 *)bh_fat->b_data)[x] = 0xfffc;
> +	mark_buffer_dirty(bh_fat);
> +	brelse(bh_fat);
> +clean_inode:
> +	iput(inode);
> +out:
> +	return error;
> +}
> +
> +static int vmufat_inode_rename(struct inode *in_source,
> +			      struct dentry *de_source,
> +			      struct inode *in_target,
> +			      struct dentry *de_target)
> +{
> +	return -EPERM;
> +}
> +
> +static int vmufat_readdir(struct file *filp, void *dirent, filldir_t filldir)
> +{
> +	int filenamelen, i, error = 0;
> +	struct vmufat_file_info *saved_file = NULL;
> +	struct dentry *dentry = filp->f_dentry;
> +	struct inode *inode = dentry->d_inode;
> +	struct super_block *sb = inode->i_sb;
> +	struct memcard *vmudetails = sb->s_fs_info;
> +	struct buffer_head *bh;
> +
> +	int blck_read = vmudetails->dir_bnum;
> +	bh = vmufat_sb_bread(sb, blck_read);
> +	if (!bh) {
> +		error = -EIO;
> +		goto out;
> +	}
> +
> +	i = filp->f_pos;
> +
> +	/* handle . for this directory and .. for parent */
> +	switch ((unsigned int) filp->f_pos) {
> +	case 0:
> +		if (filldir(dirent, ".", 1, i++, inode->i_ino, DT_DIR) < 0)
> +			goto finish;
> +
> +		filp->f_pos++;
> +	case 1:
> +		if (filldir(dirent, "..", 2, i++,
> +			    dentry->d_parent->d_inode->i_ino, DT_DIR) < 0)
> +			goto finish;
> +
> +		filp->f_pos++;
> +	default:
> +		break;
> +	}
> +
> +	/* trap reading beyond the end of the directory */
> +	if ((i - 2) > (vmudetails->dir_len * 0x10)) {
> +		error = -EINVAL;
> +		goto release_bh;
> +	}
> +
> +	saved_file =
> +	    kmalloc(sizeof(struct vmufat_file_info), GFP_KERNEL);
> +	if (!saved_file) {
> +		error = -ENOMEM;
> +		goto release_bh;
> +	}
> +
> +	do {
> +		if ((i - 2) / 0x10 > (vmudetails->dir_bnum - blck_read)) {
> +			/* move to next block in directory */
> +			blck_read--;
> +			if (vmudetails->dir_bnum - vmudetails->dir_len <=
> +				blck_read)
> +				break;
> +			brelse(bh);
> +			bh = vmufat_sb_bread(sb, blck_read);
> +			if (!bh) {
> +				kfree(saved_file);
> +				error = -EIO;
> +				goto out;
> +			}
It is simpler to just do:
			if (!bh) {
				error = -EIO;
				goto finish;
			}
kfree() is a noop for NULL pointers, and likewise for brelse().
So simplify your error paths with this in mind.
This is true at several places in the file.


> +		}
> +
> +		saved_file->ftype = bh->b_data[vmufat_index(i - 2)];
> +
> +		if (saved_file->ftype == 0)
> +			break;
> +
> +		saved_file->fblk =
> +		    le16_to_cpu(((u16 *) bh->b_data)[1 +
> +			vmufat_index_16(i - 2)]);
This seems to be a common operation - use a helper function.

> +		if (saved_file->fblk == 0)
> +			saved_file->fblk = VMUFAT_ZEROBLOCK;
> +
> +		memcpy(saved_file->fname,
> +		       bh->b_data + 4 + vmufat_index(i - 2), VMUFAT_NAMELEN);
> +		filenamelen = strlen(saved_file->fname);
> +		if (filenamelen > VMUFAT_NAMELEN)
> +			filenamelen = VMUFAT_NAMELEN;
> +		if (filldir
> +		    (dirent, saved_file->fname, filenamelen, i++,
> +		     saved_file->fblk, DT_REG) < 0) {
This is an ugly way to avoid the 80 char limit - please redo.

> +			goto finish;
> +		}
> +
> +		filp->f_pos++;
> +	} while (1);
> +
> +finish:
> +	kfree(saved_file);
> +release_bh:
> +	brelse(bh);
> +out:
> +	return error;
> +}

I did not read furter from here.
But please try to see which comments apply to the rest of the file.

	Sam
--
To unsubscribe from this list: send the line "unsubscribe linux-fsdevel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Linux Ext4 Filesystem]     [Union Filesystem]     [Filesystem Testing]     [Ceph Users]     [Ecryptfs]     [AutoFS]     [Kernel Newbies]     [Share Photos]     [Security]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux Cachefs]     [Reiser Filesystem]     [Linux RAID]     [Samba]     [Device Mapper]     [CEPH Development]
  Powered by Linux