summaryrefslogtreecommitdiffstats
path: root/fs/coda/inode.c
diff options
context:
space:
mode:
Diffstat (limited to 'fs/coda/inode.c')
-rw-r--r--fs/coda/inode.c369
1 files changed, 369 insertions, 0 deletions
diff --git a/fs/coda/inode.c b/fs/coda/inode.c
new file mode 100644
index 000000000..96d07e265
--- /dev/null
+++ b/fs/coda/inode.c
@@ -0,0 +1,369 @@
+/*
+ * Super block/filesystem wide operations
+ *
+ * Copryright (C) 1996 Peter J. Braam <braam@maths.ox.ac.uk> and
+ * Michael Callahan <callahan@maths.ox.ac.uk>
+ *
+ * Rewritten for Linux 2.1. Peter Braam <braam@cs.cmu.edu>
+ * Copyright (C) Carnegie Mellon University
+ */
+
+#define __NO_VERSION__
+#include <linux/module.h>
+#include <linux/kernel.h>
+#include <linux/mm.h>
+#include <linux/string.h>
+#include <linux/stat.h>
+#include <linux/errno.h>
+#include <linux/locks.h>
+#include <linux/unistd.h>
+
+#include <asm/system.h>
+#include <asm/uaccess.h>
+
+#include <linux/fs.h>
+#include <linux/stat.h>
+#include <linux/errno.h>
+#include <linux/locks.h>
+#include <linux/string.h>
+#include <asm/uaccess.h>
+#include <linux/vmalloc.h>
+#include <asm/segment.h>
+
+#include <linux/coda.h>
+#include <linux/coda_linux.h>
+#include <linux/coda_psdev.h>
+#include <linux/coda_fs_i.h>
+#include <linux/coda_cache.h>
+
+
+/* VFS super_block ops */
+static struct super_block *coda_read_super(struct super_block *, void *, int);
+static void coda_read_inode(struct inode *);
+static int coda_notify_change(struct dentry *dentry, struct iattr *attr);
+static void coda_put_inode(struct inode *);
+static void coda_delete_inode(struct inode *);
+static void coda_put_super(struct super_block *);
+static int coda_statfs(struct super_block *sb, struct statfs *buf,
+ int bufsiz);
+
+/* helper functions */
+static inline struct vcomm *coda_psinode2vcomm(struct inode *inode);
+static int coda_get_psdev(void *, struct inode **);
+static struct coda_sb_info *coda_psinode2sbi(struct inode *inode);
+
+/* exported operations */
+struct super_operations coda_super_operations =
+{
+ coda_read_inode, /* read_inode */
+ NULL, /* write_inode */
+ coda_put_inode, /* put_inode */
+ coda_delete_inode, /* delete_inode */
+ coda_notify_change, /* notify_change */
+ coda_put_super, /* put_super */
+ NULL, /* write_super */
+ coda_statfs, /* statfs */
+ NULL /* remount_fs */
+};
+
+/*
+ * globals
+ */
+struct coda_sb_info coda_super_info[MAX_CODADEVS];
+
+
+static struct super_block * coda_read_super(struct super_block *sb,
+ void *data, int silent)
+{
+ struct inode *psdev = 0, *root = 0;
+ struct coda_sb_info *sbi = NULL;
+ struct vcomm *vc = NULL;
+ ViceFid fid;
+ kdev_t dev = sb->s_dev;
+ int error;
+
+ ENTRY;
+ MOD_INC_USE_COUNT;
+ if (coda_get_psdev(data, &psdev))
+ goto error;
+
+ vc = coda_psinode2vcomm(psdev);
+ if ( !vc )
+ goto error;
+ vc->vc_sb = sb;
+ vc->vc_inuse = 1;
+
+ sbi = coda_psinode2sbi(psdev);
+ if ( !sbi )
+ goto error;
+ sbi->sbi_psdev = psdev;
+ sbi->sbi_vcomm = vc;
+ INIT_LIST_HEAD(&(sbi->sbi_cchead));
+ INIT_LIST_HEAD(&(sbi->sbi_volroothead));
+
+ lock_super(sb);
+ sb->u.generic_sbp = sbi;
+ sb->s_blocksize = 1024; /* XXXXX what do we put here?? */
+ sb->s_blocksize_bits = 10;
+ sb->s_magic = CODA_SUPER_MAGIC;
+ sb->s_dev = dev;
+ sb->s_op = &coda_super_operations;
+
+ /* get root fid from Venus: this needs the root inode */
+ error = venus_rootfid(sb, &fid);
+ if ( error ) {
+ printk("coda_read_super: coda_get_rootfid failed with %d\n",
+ error);
+ sb->s_dev = 0;
+ unlock_super(sb);
+ goto error;
+ }
+ printk("coda_read_super: rootfid is %s\n", coda_f2s(&fid));
+
+ /* make root inode */
+ error = coda_cnode_make(&root, &fid, sb);
+ if ( error || !root ) {
+ printk("Failure of coda_cnode_make for root: error %d\n", error);
+ sb->s_dev = 0;
+ unlock_super(sb);
+ goto error;
+ }
+
+ printk("coda_read_super: rootinode is %ld dev %d\n",
+ root->i_ino, root->i_dev);
+ sbi->sbi_root = root;
+ sb->s_root = d_alloc_root(root, NULL);
+ unlock_super(sb);
+ EXIT;
+ return sb;
+
+error:
+EXIT;
+ MOD_DEC_USE_COUNT;
+ if (sbi) {
+ sbi->sbi_vcomm = NULL;
+ sbi->sbi_root = NULL;
+ }
+ if ( vc ) {
+ vc->vc_sb = NULL;
+ vc->vc_inuse = 0;
+ }
+ if (root) {
+ iput(root);
+ }
+ sb->s_dev = 0;
+ return NULL;
+}
+
+static void coda_put_super(struct super_block *sb)
+{
+ struct coda_sb_info *sb_info;
+
+ ENTRY;
+
+ lock_super(sb);
+
+ sb->s_dev = 0;
+ coda_cache_clear_all(sb);
+ sb_info = coda_sbp(sb);
+ sb_info->sbi_vcomm->vc_inuse = 0;
+ sb_info->sbi_vcomm->vc_sb = NULL;
+ printk("Coda: Bye bye.\n");
+ memset(sb_info, 0, sizeof(* sb_info));
+
+ unlock_super(sb);
+ MOD_DEC_USE_COUNT;
+ EXIT;
+}
+
+/* all filling in of inodes postponed until lookup */
+static void coda_read_inode(struct inode *inode)
+{
+ struct coda_inode_info *cii;
+ ENTRY;
+ cii = ITOC(inode);
+ cii->c_magic = 0;
+ return;
+}
+
+static void coda_put_inode(struct inode *in)
+{
+ ENTRY;
+
+ CDEBUG(D_INODE,"ino: %ld, count %d\n", in->i_ino, in->i_count);
+
+ if ( in->i_count == 1 )
+ in->i_nlink = 0;
+
+}
+
+static void coda_delete_inode(struct inode *inode)
+{
+ struct coda_inode_info *cii;
+ struct inode *open_inode;
+
+ ENTRY;
+ CDEBUG(D_SUPER, " inode->ino: %ld, count: %d\n",
+ inode->i_ino, inode->i_count);
+
+ cii = ITOC(inode);
+ if ( inode->i_ino == CTL_INO || cii->c_magic != CODA_CNODE_MAGIC ) {
+ clear_inode(inode);
+ return;
+ }
+
+
+ if ( coda_fid_is_volroot(&cii->c_fid) )
+ list_del(&cii->c_volrootlist);
+
+ open_inode = cii->c_ovp;
+ if ( open_inode ) {
+ CDEBUG(D_SUPER, "DELINO cached file: ino %ld count %d.\n",
+ open_inode->i_ino, open_inode->i_count);
+ cii->c_ovp = NULL;
+ iput(open_inode);
+ }
+
+ coda_cache_clear_inode(inode);
+
+ inode->u.generic_ip = NULL;
+ clear_inode(inode);
+ EXIT;
+}
+
+static int coda_notify_change(struct dentry *de, struct iattr *iattr)
+{
+ struct inode *inode = de->d_inode;
+ struct coda_inode_info *cii;
+ struct coda_vattr vattr;
+ int error;
+
+ ENTRY;
+ memset(&vattr, 0, sizeof(vattr));
+ cii = ITOC(inode);
+ CHECK_CNODE(cii);
+
+ coda_iattr_to_vattr(iattr, &vattr);
+ vattr.va_type = C_VNON; /* cannot set type */
+ CDEBUG(D_SUPER, "vattr.va_mode %o\n", vattr.va_mode);
+
+ error = venus_setattr(inode->i_sb, &cii->c_fid, &vattr);
+
+ if ( !error ) {
+ coda_vattr_to_iattr(inode, &vattr);
+ coda_cache_clear_inode(inode);
+ }
+ CDEBUG(D_SUPER, "inode.i_mode %o, error %d\n",
+ inode->i_mode, error);
+
+ EXIT;
+ return error;
+}
+
+/* we need _something_ for this routine. Let's mimic AFS */
+static int coda_statfs(struct super_block *sb, struct statfs *buf,
+ int bufsiz)
+{
+ struct statfs tmp;
+
+ tmp.f_type = CODA_SUPER_MAGIC;
+ tmp.f_bsize = 1024;
+ tmp.f_blocks = 9000000;
+ tmp.f_bfree = 9000000;
+ tmp.f_bavail = 9000000 ;
+ tmp.f_files = 9000000;
+ tmp.f_ffree = 9000000;
+ tmp.f_namelen = 0;
+ copy_to_user(buf, &tmp, bufsiz);
+ return 0;
+}
+
+
+/* init_coda: used by filesystems.c to register coda */
+
+struct file_system_type coda_fs_type = {
+ "coda", 0, coda_read_super, NULL
+};
+
+int init_coda_fs(void)
+{
+ return register_filesystem(&coda_fs_type);
+}
+
+/* MODULE stuff is in psdev.c */
+
+/* helpers */
+static inline struct vcomm *coda_psinode2vcomm(struct inode *inode)
+{
+
+ unsigned int minor = MINOR(inode->i_rdev);
+ CDEBUG(D_PSDEV,"minor %d\n", minor);
+ if ( minor < MAX_CODADEVS )
+ return &(psdev_vcomm[minor]);
+ else
+ return NULL;
+}
+
+static struct coda_sb_info *coda_psinode2sbi(struct inode *inode)
+{
+ unsigned int minor = MINOR(inode->i_rdev);
+
+ CDEBUG(D_PSDEV,"minor %d\n", minor);
+ if ( (minor >= 0) && (minor < MAX_CODADEVS))
+ return &(coda_super_info[minor]);
+ else
+ return NULL;
+}
+
+/* name lookup for psdev passed in by mount */
+static int coda_get_psdev(void *data, struct inode **res_dev)
+{
+ char **psdev_path;
+ struct inode *psdev = 0;
+ struct dentry *ent=NULL;
+
+
+ if ( ! data ) {
+ printk("coda_get_psdev: no data!\n");
+ return 1;
+ }
+
+ psdev_path = data;
+ ent = namei((char *) *psdev_path);
+ if (IS_ERR(ent)) {
+ printk("namei error %ld for %d\n", PTR_ERR(ent),
+ (int) psdev_path);
+ return 1;
+ }
+ psdev = ent->d_inode;
+
+ if (!S_ISCHR(psdev->i_mode)) {
+ printk("not a character device\n");
+ return 1;
+ }
+ CDEBUG(D_PSDEV,"major %d, minor %d, count %d\n",
+ MAJOR(psdev->i_rdev),
+ MINOR(psdev->i_rdev), psdev->i_count);
+
+ if (MAJOR(psdev->i_rdev) != CODA_PSDEV_MAJOR) {
+ printk("device %d not a Coda PSDEV device\n",
+ MAJOR(psdev->i_rdev));
+ return 1;
+ }
+
+ if (MINOR(psdev->i_rdev) >= MAX_CODADEVS) {
+ printk("minor %d not an allocated Coda PSDEV\n",
+ psdev->i_rdev);
+ return 1;
+ }
+
+ if (psdev->i_count < 1) {
+ printk("coda device minor %d not open (i_count = %d)\n",
+ MINOR(psdev->i_rdev), psdev->i_count);
+ return 1;
+ }
+
+ *res_dev = psdev;
+ EXIT;
+ return 0;
+}