Re: [PATCH] fs: add jfsv3 (AIX powerpc native JFS file system) support

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

 



Il 17/05/2013 01:46, Philippe De Muyter ha scritto:
This is a file system driver for the file system called JFS on AIX, but
different from what's called jfs on linux.  In AIX header files this
file system seems to be called "Version 3" or "Version 3p", hence its
name here.  This driver supports only read-only access to such file systems,
and has been tested successfully on AIX 3.5, AIX 4.1 and AIX 4.2 filesystems..

Signed-off-by: Philippe De Muyter <phdm@xxxxxxxxx>
Tested-by: Jori Mantysalo <Jori.Mantysalo@xxxxxx>
---
  fs/Kconfig        |    1 +
  fs/Makefile       |    1 +
  fs/jfsv3/Kconfig  |   10 +
  fs/jfsv3/Makefile |    7 +
  fs/jfsv3/inode.c  |  721 +++++++++++++++++++++++++++++++++++++++++++++++++++++
  5 files changed, 740 insertions(+), 0 deletions(-)
  create mode 100644 fs/jfsv3/Kconfig
  create mode 100644 fs/jfsv3/Makefile
  create mode 100644 fs/jfsv3/inode.c

diff --git a/fs/Kconfig b/fs/Kconfig
index c229f82..807823a 100644
--- a/fs/Kconfig
+++ b/fs/Kconfig
@@ -212,6 +212,7 @@ source "fs/ufs/Kconfig"
  source "fs/exofs/Kconfig"
  source "fs/f2fs/Kconfig"
  source "fs/efivarfs/Kconfig"
+source "fs/jfsv3/Kconfig"

  endif # MISC_FILESYSTEMS

diff --git a/fs/Makefile b/fs/Makefile
index 4fe6df3..99cd8e6 100644
--- a/fs/Makefile
+++ b/fs/Makefile
@@ -122,6 +122,7 @@ obj-$(CONFIG_OCFS2_FS)		+= ocfs2/
  obj-$(CONFIG_BTRFS_FS)		+= btrfs/
  obj-$(CONFIG_GFS2_FS)           += gfs2/
  obj-$(CONFIG_F2FS_FS)		+= f2fs/
+obj-$(CONFIG_JFSV3_FS)		+= jfsv3/
  obj-y				+= exofs/ # Multiple modules
  obj-$(CONFIG_CEPH_FS)		+= ceph/
  obj-$(CONFIG_PSTORE)		+= pstore/
diff --git a/fs/jfsv3/Kconfig b/fs/jfsv3/Kconfig
new file mode 100644
index 0000000..4ba73c5
--- /dev/null
+++ b/fs/jfsv3/Kconfig
@@ -0,0 +1,10 @@
+config JFSV3_FS
+	tristate "AIX jfsv3 file system support"
+	---help---
+	  Read-only support for AIX jfs file systems (not to be confused
+	  with linux jfs).  You should normally also select support for
+	  AIX LVM partitions, but if you manage to get a AIX file system
+	  image by another way (dd, e.g.), selecting this is enough.  You'll
+	  be able to mount your disk image using the loop driver.
+	  To compile this file system support as a module, choose M here: the
+	  module will be called jfsv3.
diff --git a/fs/jfsv3/Makefile b/fs/jfsv3/Makefile
new file mode 100644
index 0000000..d6ecd66
--- /dev/null
+++ b/fs/jfsv3/Makefile
@@ -0,0 +1,7 @@
+#
+# Makefile for the AIX jfsv3 filesystem routines.
+#
+
+obj-$(CONFIG_JFSV3_FS) += jfsv3.o
+
+jfsv3-objs := inode.o
diff --git a/fs/jfsv3/inode.c b/fs/jfsv3/inode.c
new file mode 100644
index 0000000..3160f08
--- /dev/null
+++ b/fs/jfsv3/inode.c
@@ -0,0 +1,721 @@
+/*
+ * AIX JFS Version 3/3p file system, Linux read-only implementation
+ *
+ * Copyright (C) 2012-2013  Philippe De Muyter <phdm@xxxxxxxxx>
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version
+ * 2 of the License, or (at your option) any later version.
+ */
+
+#include <linux/module.h>
+#include <linux/namei.h>
+#include <linux/statfs.h>
+#include <linux/buffer_head.h>
+#include <linux/mpage.h>
+#include <linux/fs.h>
+#include <linux/slab.h>
+
+struct  jfsv3_superblock {
+	unsigned char   s_magic[4];     /* magic number */

Is it needed to use char[4] instead of int for example? It'd be better to use a number with a fix endianess.

+	char   s_flag[4];      /* flag word (see below) */

Where? I can't see s_flag used.

+	__be32 s_agsize;       /* fragments per allocation group */
+	__be32 s_logserial;    /* serial number of log when fs mounted */
+	__be32 s_fsize;        /* size (in 512 bytes) of entire fs */
+	__be16 s_bsize;        /* block size in bytes */
+	__be16 s_spare;        /* unused. */
+	char   s_fname[6];     /* name of this file system */
+	char   s_fpack[6];     /* name of this volume */
+	__be32 s_logdev;       /* device address of log */
+
+	/* current file system state information, values change over time */
+	char   s_fmod;         /* flag: set when file system is mounted */
+	char   s_ronly;        /* flag: file system is read only */
+	__be32 s_time;         /* time of last superblock update */
+
+	/* more persistent information */
+	__be32 s_version;      /* version number */
+	__be32 s_fragsize;     /* fragment size in bytes (fsv3p only) */
+	__be32 s_iagsize;      /* disk inode per alloc grp (fsv3p only) */
+	__be32 s_compress;     /* > 0 if data compression */
+};
+
+#define fsv3magic  "\103\041\207\145"   /* Version 3 fs magic number */
+#define fsv3pmagic "\145\207\041\103"   /* Version 3p fs magic number */
+

In magic.h please.

+#define fsv3vers   0                    /* Version 3 fs version number */
+#define fsv3pvers  1                    /* Version 3p fs version number */
+
+#define D_PRIVATE 48			/* max len of in-inode symlink */
+
+struct jfsv3_dinode {
+	__be32 di_gen;
+	__be32 di_mode;
+	__be16 di_nlink;
+	__be16 di_acct;
+	__be32 di_uid;
+	__be32 di_gid;
+	__be32 di_size;
+	__be32 di_nblocks;
+	__be32 di_mtime;
+	char   res32[4];
+	__be32 di_atime;
+	char   res40[4];
+	__be32 di_ctime;
+	char   res48[4];
+	char   res52[28];
+	__be32 di_rdaddr[8];
+	char   res112[4];
+	__be32 di_rindirect;
+	char   res120[8];
+};
+
+struct jfsv3_direct {
+        __be32  d_ino;
+        __be16  d_reclen;
+        __be16  d_namlen;
+        char    d_name[0];    /* NULL terminated      */
+};
+
+struct jfsv3_fs_info {
+	unsigned long  s_iagsize;	/* disk inodes per alloc grp */
+	unsigned long  s_fsize;		/* size of fs in 512 bytes-blocks */
+	unsigned long  s_fragsize;	/* basic block size */
+	unsigned long  s_agsize;	/* blocks per allocation group */
+};
+

I would use a separate .h for these declarations.

+static loff_t dinode_offset(struct super_block *sb, unsigned long ino)
+{
+	struct jfsv3_fs_info *fsi = sb->s_fs_info;
+	unsigned long iagsize = fsi->s_iagsize;
+
+	/* The first i-nodes are at block 256 (32 x 4096 bytes) */
+	/* Next i-nodes are at block fsi->s_agsize * fsi->s_fragsize * n */
+	if (ino < iagsize)
+		return (256 * 512) + ino * 128;
+	else
+		return (loff_t)((ino / iagsize) * fsi->s_agsize * (fsi->s_fragsize / 128) + (ino % iagsize)) * 128;
+}
+
+struct jfsv3_dirpage {
+	char *p_data;
+	struct buffer_head *p_bh;
+};
+
+void jfsv3_readdirpage(struct jfsv3_dirpage *p, struct super_block *sb, u32 addr)
+{
+	struct buffer_head *bh;
+
+	if (sb->s_blocksize == 4096) {
+		bh = __bread(sb->s_bdev, addr, 4096);
+		p->p_bh = bh;
+		if (bh)
+			p->p_data = bh->b_data;
+		else
+			p->p_data = NULL;
+	} else {
+		u32 nfrags;
+		char *data;
+
+		nfrags = addr >> 28;
+		nfrags = 8 - nfrags;
+		if (nfrags * 512 == sb->s_blocksize) {
+			addr &= 0xfffffff;
+			bh = __bread(sb->s_bdev, addr, sb->s_blocksize);
+			p->p_bh = bh;
+			if (bh)
+				p->p_data = bh->b_data;
+			else
+				p->p_data = NULL;
+			return;
+		}
+		data = kmalloc(nfrags * 512, GFP_KERNEL);
+		p->p_data = data;
+		p->p_bh = NULL;
+		if (!data)
+			return;
+		addr &= 0xfffffff;
+#if 0
+		printk("addr = %u, nfrags = %u\n", addr, nfrags);
+#endif

Remove these "#if 0" and use pr_* functions. In addition you have to use checkpatch script before to send the patch.

+		do {
+			struct buffer_head *bh;
+
+			bh = __bread(sb->s_bdev, addr, 512);
+			if (bh) {
+#if 0
+print_hex_dump(KERN_DEBUG, "", DUMP_PREFIX_OFFSET,
+	16, 1, bh->b_data, 512, 1);
+#endif
+				memcpy(data, bh->b_data, 512);
+				brelse(bh);
+			}
+			addr += 1;
+			data += 512;
+		} while (--nfrags);
+	}
+}
+
+void jfsv3_freedirpage(struct jfsv3_dirpage *p)
+{
+	if (p->p_data) {
+		if (p->p_bh)
+			brelse(p->p_bh);
+		else
+			kfree(p->p_data);
+	}
+}
+
+static int
+jfsv3_readdir(struct file *filp, void *dirent, filldir_t filldir)
+{
+	int stored = 0;
+	struct inode *i = file_inode(filp);
+#if 0
+printk("jfsv3_readdir(i_ino = %lu, f_pos = %llu)\n", i->i_ino, filp->f_pos);
+#endif
+	if (filp->f_pos < 8 * 4096) {
+	    if (i->i_ino) {
+		struct super_block *sb = i->i_sb;
+		struct buffer_head *ibh;
+		loff_t fs_offset;
+		unsigned long blk_offset;
+
+		fs_offset = dinode_offset(sb, i->i_ino);
+		blk_offset = fs_offset & (sb->s_blocksize - 1);
+		ibh = __bread(sb->s_bdev, fs_offset >> sb->s_blocksize_bits, sb->s_blocksize);
+		if (ibh) {
+			struct jfsv3_dinode *aji;
+			u32 curpage = filp->f_pos / 4096;
+			u32 addr;
+			struct jfsv3_dirpage page;
+
+			aji = (struct jfsv3_dinode *)(ibh->b_data + blk_offset);
+#if 0
+			printk("inode %lu : nlink = %u\n", i->i_ino, be32_to_cpu(aji->di_nlink));
+			printk("mode = 0%o, size = %u, nblocks = %u, rdaddr[0] = %u\n",
+				be32_to_cpu(aji->di_mode), be32_to_cpu(aji->di_size),
+				be32_to_cpu(aji->di_nblocks), be32_to_cpu(aji->di_rdaddr[0]));
+#endif
+			addr = be32_to_cpu(aji->di_rdaddr[curpage]);
+			jfsv3_readdirpage(&page, sb, addr);
+			if (page.p_data) {
+				struct jfsv3_direct *ajd;
+
+				ajd = (struct jfsv3_direct *)(page.p_data + filp->f_pos % 4096);
+				while (filp->f_pos < be32_to_cpu(aji->di_size)
+				&& filp->f_pos / 4096 == curpage) {
+#if 0
+					printk("%u %s\n", be32_to_cpu(ajd->d_ino), ajd->d_name);
+#endif
+					if (filldir(dirent, ajd->d_name, strlen(ajd->d_name), filp->f_pos, be32_to_cpu(ajd->d_ino),
+			    DT_UNKNOWN) < 0) {
+#if 0
+				printk("jfsv3_readdir: filldir(%s, %u) failed\n", ajd->d_name, be32_to_cpu(ajd->d_ino));
+#endif
+						break;
+					}
+					stored++;
+					ajd = (struct jfsv3_direct *)((void*)ajd + be16_to_cpu(ajd->d_reclen));
+					filp->f_pos = (char*)ajd - page.p_data + (filp->f_pos & -4096);
+				}
+				jfsv3_freedirpage(&page);
+			} else {
+				u32 addr = be32_to_cpu(aji->di_rdaddr[curpage]);
+				u32 freefrags;
+
+				printk("jfsv3_readdir: __bread(%08x) failed\n", addr);
+				freefrags = addr >> 28;
+			}
+			brelse(ibh);
+		}
+	} else {
+		/* FIXME : put jfsv3_get_block's indirect block handling
+		 * in a separate function and use it here also.
+		 */
+		printk("jfsv3_readdir: huge dir support not yet implemented\n");
+	    }
+	}
+#if 0
+printk("exiting with f_pos = %llu, stored = %u\n", filp->f_pos, stored);
+#endif
+	return stored;
+}
+
+static const struct file_operations jfsv3_dir_operations = {
+	.read		= generic_read_dir,
+	.readdir	= jfsv3_readdir,
+};
+
+void jfsv3_map_bh(struct buffer_head *bh_result, struct super_block *sb, u32 addr, sector_t lblock)
+{
+#if 0
+	printk("jfsv3_map_bh(%08x, %llu)\n", addr, lblock);
+#endif
+	if (sb->s_blocksize != 4096) {
+		addr &= 0xfffffff;
+		addr += lblock & (0xfff >> sb->s_blocksize_bits);
+	}
+#if 0
+	printk("map_bh(%08x)\n", addr);
+#endif
+	map_bh(bh_result, sb, addr);
+}
+
+int jfsv3_get_block(struct inode *i, sector_t lblock,
+		  struct buffer_head *bh_result, int create)
+{
+	int ret = -EIO;
+
+#if 0
+printk("jfsv3_get_block(block %08llx)\n", (unsigned long long)lblock);
+#endif
+	if (i->i_ino) {
+		struct super_block *sb = i->i_sb;
+		struct buffer_head *ibh;
+		loff_t fs_offset;
+		unsigned long blk_offset;
+
+		fs_offset = dinode_offset(sb, i->i_ino);
+		blk_offset = fs_offset & (sb->s_blocksize - 1);
+		ibh = __bread(sb->s_bdev, fs_offset >> sb->s_blocksize_bits, sb->s_blocksize);
+		if (ibh) {
+			struct jfsv3_dinode *aji;
+			u32 curpage = lblock >> (12 - sb->s_blocksize_bits);
+
+			aji = (struct jfsv3_dinode *)(ibh->b_data + blk_offset);
+#if 0
+			printk("inode %lu : nlink = %u\n", i->i_ino, be32_to_cpu(aji->di_nlink));
+			printk("mode = 0%o, size = %u, nblocks = %u, rdaddr[0] = %u\n",
+				be32_to_cpu(aji->di_mode), be32_to_cpu(aji->di_size),
+				be32_to_cpu(aji->di_nblocks), be32_to_cpu(aji->di_rdaddr[0]));
+#endif
+			if (curpage < 8) {
+				jfsv3_map_bh(bh_result, sb, be32_to_cpu(aji->di_rdaddr[curpage]), lblock);
+				ret = 0;
+			} else {
+				u32 addr = be32_to_cpu(aji->di_rindirect);
+				u32 frags_per_page = 4096 / sb->s_blocksize;
+				u32 addrs_per_block = 1024 / frags_per_page;
+#if 0
+				printk("r_indirect = %08x, curpage = %u\n", addr, curpage);
+#endif
+				if (be32_to_cpu(aji->di_size) <= 1024 * 4096) {
+					struct buffer_head *sibh;
+
+					addr += curpage / addrs_per_block;
+					sibh = __bread(sb->s_bdev, addr, sb->s_blocksize);
+					if (sibh) {
+						unsigned long *b = (unsigned long *)sibh->b_data;
+
+#if 0
+printk("mapping block %08x\n", be32_to_cpu(b[curpage % addrs_per_block]));
+#endif
+						jfsv3_map_bh(bh_result, sb, be32_to_cpu(b[curpage % addrs_per_block]), lblock);
+						ret = 0;
+						brelse(sibh);
+					}
+				} else {
+					/*
+					 * 4k double-indirect block contains
+					 * 512 addresses of indirect blocks,
+					 * each containing 1024 block addresses.
+					 */
+					struct buffer_head *sibh;
+					unsigned long diblock = curpage / 1024;
+					unsigned long iblock = curpage % 1024;
+
+					if (sb->s_blocksize == 512) {
+						/* 64 = 512 / 8 */
+						addr += diblock / 64;
+						diblock &= 63;
+					}
+					sibh = __bread(sb->s_bdev, addr, sb->s_blocksize);
+					if (sibh) {
+						unsigned long *ib = (unsigned long *)sibh->b_data;
+						struct buffer_head *dibh;
+
+						addr = be32_to_cpu(ib[diblock * 2 + 1]);
+#if 0
+printk("reading addresses block %08x\n", addr);
+#endif
+						addr += iblock / addrs_per_block;
+						dibh = __bread(sb->s_bdev, addr, sb->s_blocksize);
+						if (dibh) {
+							unsigned long *b = (unsigned long *)dibh->b_data;
+#if 0
+printk("mapping block %08x\n", be32_to_cpu(b[iblock % addrs_per_block]));
+#endif
+							jfsv3_map_bh(bh_result, sb, be32_to_cpu(b[iblock % addrs_per_block]), lblock);
+							ret = 0;
+							brelse(dibh);
+						}
+						brelse(sibh);
+					}
+				}
+			}
+			brelse(ibh);
+		}
+	}
+	return ret;
+}
+
+static int jfsv3_readpage(struct file *file, struct page *page)
+{
+	return mpage_readpage(page, jfsv3_get_block);
+}
+
+static const struct address_space_operations jfsv3_aops = {
+	.readpage = jfsv3_readpage
+};
+
+static struct inode *jfsv3_iget(struct super_block *sb, ino_t ino);
+
+static struct dentry *
+jfsv3_lookup(struct inode *i, struct dentry *dentry, unsigned int flags)
+{
+	int res = -EACCES;
+	const char *name = dentry->d_name.name;
+	int len = dentry->d_name.len;
+	struct inode *inode = NULL;
+
+#if 0
+printk("jfsv3_lookup(i_ino = %lu, %.*s)\n", i->i_ino, len, name);
+#endif
+	if (i->i_ino) {
+		struct super_block *sb = i->i_sb;
+		struct buffer_head *ibh;
+		loff_t fs_offset;
+		unsigned long blk_offset;
+
+		fs_offset = dinode_offset(sb, i->i_ino);
+		blk_offset = fs_offset & (sb->s_blocksize - 1);
+		ibh = __bread(sb->s_bdev, fs_offset >> sb->s_blocksize_bits, sb->s_blocksize);
+		if (ibh) {
+			struct jfsv3_dinode *aji;
+			int curpage = 0;
+			u32 addr;
+			struct jfsv3_dirpage page;
+
+			aji = (struct jfsv3_dinode *)(ibh->b_data + blk_offset);
+#if 0
+			printk("inode %lu : nlink = %u\n", i->i_ino, be32_to_cpu(aji->di_nlink));
+			printk("mode = 0%o, size = %u, nblocks = %u, rdaddr[0] = %u\n",
+				be32_to_cpu(aji->di_mode), be32_to_cpu(aji->di_size),
+				be32_to_cpu(aji->di_nblocks), be32_to_cpu(aji->di_rdaddr[0]));
+#endif
+nextpage:
+			addr = be32_to_cpu(aji->di_rdaddr[curpage]);
+			jfsv3_readdirpage(&page, sb, addr);
+			if (page.p_data) {
+				struct jfsv3_direct *ajd;
+
+				ajd = (struct jfsv3_direct *)page.p_data;
+				while ((char*)ajd - page.p_data < 4096
+				&& (char*)ajd - page.p_data + curpage * 4096 < be32_to_cpu(aji->di_size)) {
+#if 0
+					printk("%u %s\n", be32_to_cpu(ajd->d_ino), ajd->d_name);
+#endif
+					if (len == strlen(ajd->d_name)
+					&& memcmp(ajd->d_name, name, len) == 0) {
+						inode = jfsv3_iget(i->i_sb, be32_to_cpu(ajd->d_ino));
+						res = 0;
+						break;
+					}
+					ajd = (struct jfsv3_direct *)((void*)ajd + be16_to_cpu(ajd->d_reclen));
+				}
+				jfsv3_freedirpage(&page);
+			}
+			curpage += 1;
+			if (res && curpage * 4096 < be32_to_cpu(aji->di_size) && curpage < 8)
+				goto nextpage;
+			brelse(ibh);
+		}
+	}
+	d_add(dentry, inode);
+if (res)
+	printk("jfsv3_lookup(i_ino = %lu, %.*s) failed\n", i->i_ino, len, name);
+	return ERR_PTR(res);
+}
+
+static const struct inode_operations jfsv3_dir_inode_operations = {
+	.lookup		= jfsv3_lookup,
+};
+
+static void *jfsv3_follow_link(struct dentry *dentry, struct nameidata *nd)
+{
+	struct inode *i = dentry->d_inode;
+	char *target_path = NULL;
+
+	if (i->i_ino) {
+		struct super_block *sb = i->i_sb;
+		struct buffer_head *ibh;
+		loff_t fs_offset;
+		unsigned long blk_offset;
+
+		fs_offset = dinode_offset(sb, i->i_ino);
+		blk_offset = fs_offset & (sb->s_blocksize - 1);
+		ibh = __bread(sb->s_bdev, fs_offset >> sb->s_blocksize_bits, sb->s_blocksize);
+		if (ibh) {
+			struct jfsv3_dinode *aji;
+
+			aji = (struct jfsv3_dinode *)(ibh->b_data + blk_offset);
+			target_path = kmalloc(D_PRIVATE + 1, GFP_KERNEL);
+			if (target_path) {
+				memcpy(target_path, (char *)aji->di_rdaddr, D_PRIVATE);
+				target_path[D_PRIVATE] = '\0';
+			}
+			nd_set_link(nd, target_path);
+			brelse(ibh);
+		}
+	}
+	return target_path;
+}
+
+static void jfsv3_put_link(struct dentry *dentry, struct nameidata *nd, void *target_path)
+{
+	if (target_path)
+		kfree(target_path);
+}
+
+const struct inode_operations jfsv3_symlink_inode_operations = {
+	.readlink	= generic_readlink,
+	.follow_link	= jfsv3_follow_link,
+	.put_link	= jfsv3_put_link,
+};
+
+static void
+jfsv3_read_inode(struct inode *i)
+{
+#if 0
+	printk("jfsv3_read_inode(%lu)\n", i->i_ino);
+#endif
+	if (i->i_ino) {
+		struct super_block *sb = i->i_sb;
+		struct buffer_head *ibh;
+		loff_t fs_offset;
+		unsigned long blk_offset;
+
+		fs_offset = dinode_offset(sb, i->i_ino);
+		blk_offset = fs_offset & (sb->s_blocksize - 1);
+		ibh = __bread(sb->s_bdev, fs_offset >> sb->s_blocksize_bits, sb->s_blocksize);
+		if (!ibh) {
+printk("__bread(%llu, %lu) = NULL\n", fs_offset >> sb->s_blocksize_bits, sb->s_blocksize);

no error management here? only a print?

+		} else {
+			struct jfsv3_dinode *aji;
+
+			aji = (struct jfsv3_dinode *)(ibh->b_data + blk_offset);
+#if 0
+			printk("inode %lu : nlink = %u\n", i->i_ino, be32_to_cpu(aji->di_nlink));
+			printk("mode = 0%o, size = %u, nblocks = %u, rdaddr[0] = %u\n",
+				be32_to_cpu(aji->di_mode), be32_to_cpu(aji->di_size),
+				be32_to_cpu(aji->di_nblocks), be32_to_cpu(aji->di_rdaddr[0]));
+#endif
+			set_nlink(i, be32_to_cpu(aji->di_nlink));
+			i->i_mode = be32_to_cpu(aji->di_mode);
+			i->i_size = be32_to_cpu(aji->di_size);
+			i->i_blocks = ((i->i_size + sb->s_blocksize - 1) >> sb->s_blocksize_bits) << (sb->s_blocksize_bits - 9);
+			i->i_uid = be32_to_cpu(aji->di_uid);
+			i->i_gid = be32_to_cpu(aji->di_gid);
+			i->i_mtime.tv_sec = be32_to_cpu(aji->di_mtime);
+			i->i_atime.tv_sec = be32_to_cpu(aji->di_atime);
+			i->i_ctime.tv_sec = be32_to_cpu(aji->di_ctime);
+			i->i_mtime.tv_nsec = i->i_atime.tv_nsec = i->i_ctime.tv_nsec = 0;
+			if (S_ISREG(i->i_mode)) {
+				i->i_fop = &generic_ro_fops;
+				i->i_data.a_ops = &jfsv3_aops;
+			} else if (S_ISDIR(i->i_mode)) {
+				i->i_op = &jfsv3_dir_inode_operations;
+				i->i_fop = &jfsv3_dir_operations;
+			} else if (S_ISLNK(i->i_mode)) {
+				if (i->i_size > D_PRIVATE) {
+					i->i_op = &page_symlink_inode_operations;
+					i->i_data.a_ops = &jfsv3_aops;
+				} else
+					i->i_op = &jfsv3_symlink_inode_operations;
+			} else {
+				if (!S_ISCHR(i->i_mode)
+				&& !S_ISBLK(i->i_mode)
+				&& !S_ISFIFO(i->i_mode)
+				&& !S_ISSOCK(i->i_mode)) {
+					printk("inode %lu : fs_offset = %lld, blk_offset = %lu\n",
+						i->i_ino, fs_offset, blk_offset);
+					printk("nlink = %u, mode = 0%o, size = %u, nblocks = %u, rdaddr[0] = %u\n",
+						be32_to_cpu(aji->di_nlink),
+						be32_to_cpu(aji->di_mode),
+						be32_to_cpu(aji->di_size),
+						be32_to_cpu(aji->di_nblocks),
+						be32_to_cpu(aji->di_rdaddr[0]));
+				}
+				i->i_rdev = be32_to_cpu(aji->di_rdaddr[0]);
+				init_special_inode(i, i->i_mode, i->i_rdev);
+			}
+			brelse(ibh);
+		}
+	}
+}
+
+static struct inode *jfsv3_iget(struct super_block *sb, ino_t ino)
+{
+	struct inode *inode;
+
+	inode = iget_locked(sb, ino);
+	if (!inode)
+		return ERR_PTR(-ENOMEM);
+	if (inode->i_state & I_NEW) {
+		jfsv3_read_inode(inode);
+		unlock_new_inode(inode);
+	}
+	return inode;
+}
+
+static void jfsv3_put_super(struct super_block *sb)
+{
+	kfree(sb->s_fs_info);
+	sb->s_fs_info = NULL;
+}
+
+static int
+jfsv3_statfs(struct dentry *dentry, struct kstatfs *buf)
+{
+	struct jfsv3_fs_info *fsi = dentry->d_sb->s_fs_info;
+
+	buf->f_type = 0x65872143;
+	buf->f_bsize = 4096;
+	buf->f_bfree = buf->f_bavail = buf->f_ffree;
+	buf->f_blocks = fsi->s_fsize;
+	buf->f_namelen = 256;

No *magic* numbers here, you can use a macro to code them.

+	return 0;
+}
+
+static int jfsv3_remount(struct super_block *sb, int *flags, char *data)
+{
+	*flags |= MS_RDONLY;
+	return 0;
+}
+
+static const struct super_operations jfsv3_ops = {
+	.put_super	= jfsv3_put_super,
+	.statfs		= jfsv3_statfs,
+	.remount_fs	= jfsv3_remount,
+};
+
+static int jfsv3_fill_super(struct super_block *sb, void *data, int silent)
+{
+	struct buffer_head *bh;
+	struct jfsv3_superblock *aj_sb;
+	struct inode *root;
+	enum jfsv3type { NOTJFSV3, JFSV3, JFSV3P1, JFSV3P2 };
+	enum jfsv3type fstype = NOTJFSV3;
+
+	/* I would parse the options here, but there are none.. :) */
+
+	sb_set_blocksize(sb, 4096);
+
+	bh = __bread(sb->s_bdev, 0x1000 >> sb->s_blocksize_bits, sb->s_blocksize);
+	if (!bh)
+		return -EINVAL;
+
+	aj_sb = (struct jfsv3_superblock *)bh->b_data;
+	if (strncmp(aj_sb->s_magic, "\x43\x21\x87\x65", 4) == 0)
+		fstype = JFSV3;
+	else if (strncmp(aj_sb->s_magic, "\x65\x87\x21\x43", 4) == 0) {
+		if (aj_sb->s_version == cpu_to_be32(1))
+			fstype = JFSV3P1;
+		else if (aj_sb->s_version == cpu_to_be32(2))
+			fstype = JFSV3P2;
+	}
+	if (fstype != NOTJFSV3) {
+		struct jfsv3_fs_info *fsi;
+
+		sb->s_magic = 0x65872143;
+		fsi = kmalloc(sizeof(struct jfsv3_fs_info), GFP_KERNEL);
+		fsi->s_iagsize = fstype == JFSV3 ? 2048 : be32_to_cpu(aj_sb->s_iagsize);
+		fsi->s_fsize = be32_to_cpu(aj_sb->s_fsize);
+		sb->s_fs_info = fsi;
+		printk("AIX jfs file system, lv %.6s, name %.6s, %s version %u\n",
+			aj_sb->s_fpack, aj_sb->s_fname,
+			fstype == JFSV3 ? "jfsv3" : "jfsv3p",
+			be32_to_cpu(aj_sb->s_version));
+		printk("fs size : %lu 512 bytes blocks, block size : %u bytes\n",
+			fsi->s_fsize, be16_to_cpu(aj_sb->s_bsize));
+		printk("state : %u\n", aj_sb->s_fmod);
+		fsi->s_agsize = be32_to_cpu(aj_sb->s_agsize);
+		printk("fragments per allocation group : %lu\n", fsi->s_agsize);
+		printk("disk i-nodes per allocation group : %lu\n", fsi->s_iagsize);
+		fsi->s_fragsize = be32_to_cpu(aj_sb->s_fragsize);
+		brelse(bh);
+		printk("fragments size (default 4096) : %lu bytes\n", fsi->s_fragsize);
+		if (!fsi->s_fragsize)
+			fsi->s_fragsize = 4096;
+		else if (sb_set_blocksize(sb, fsi->s_fragsize) != fsi->s_fragsize)
+			goto out;
+	} else {
+		printk("found s_magic : %x %x %x %x\n",
+			aj_sb->s_magic[0], aj_sb->s_magic[1],
+			aj_sb->s_magic[2], aj_sb->s_magic[3]);
+		brelse(bh);
+		goto out;
+	}
+
+	sb->s_maxbytes = 0xFFFFFFFF;

Maybe MAX_LFS_FILESIZE? It'd be better to set even s_time_gran and s_max_links.

Regards,

Marco
--
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