!C99Shell v. 1.0 pre-release build #13!

Software: Apache/2.0.54 (Unix) mod_perl/1.99_09 Perl/v5.8.0 mod_ssl/2.0.54 OpenSSL/0.9.7l DAV/2 FrontPage/5.0.2.2635 PHP/4.4.0 mod_gzip/2.0.26.1a 

uname -a: Linux snow.he.net 4.4.276-v2-mono-1 #1 SMP Wed Jul 21 11:21:17 PDT 2021 i686 

uid=99(nobody) gid=98(nobody) groups=98(nobody) 

Safe-mode: OFF (not secure)

/usr/src/linux-2.4.18-xfs-1.1/fs/xfs/linux/   drwxr-xr-x
Free 318.32 GB of 458.09 GB (69.49%)
Home    Back    Forward    UPDIR    Refresh    Search    Buffer    Encoder    Tools    Proc.    FTP brute    Sec.    SQL    PHP-code    Update    Feedback    Self remove    Logout    


Viewing file:     xfs_super.c (23.68 KB)      -rw-r--r--
Select action/file-type:
(+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
/*
 * Copyright (c) 2000 Silicon Graphics, Inc.  All Rights Reserved.
 * 
 * This program is free software; you can redistribute it and/or modify it
 * under the terms of version 2 of the GNU General Public License as
 * published by the Free Software Foundation.
 * 
 * This program is distributed in the hope that it would be useful, but
 * WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
 * 
 * Further, this software is distributed without any warranty that it is
 * free of the rightful claim of any third person regarding infringement
 * or the like.  Any license provided herein, whether implied or
 * otherwise, applies only to this software file.  Patent licenses, if
 * any, provided herein do not apply to combinations of this program with
 * other software, or any other product whatsoever.
 * 
 * You should have received a copy of the GNU General Public License along
 * with this program; if not, write the Free Software Foundation, Inc., 59
 * Temple Place - Suite 330, Boston MA 02111-1307, USA.
 * 
 * Contact information: Silicon Graphics, Inc., 1600 Amphitheatre Pkwy,
 * Mountain View, CA  94043, or:
 * 
 * http://www.sgi.com 
 * 
 * For further information regarding this notice, see: 
 * 
 * http://oss.sgi.com/projects/GenInfo/SGIGPLNoticeExplan/
 */

#include <xfs.h>
#include <linux/bitops.h>
#include <linux/locks.h>
#include <linux/smp_lock.h>
#include <linux/xfs_iops.h>
#include <linux/blkdev.h>
#include <linux/init.h>

/* xfs_vfs[ops].c */
extern void vfsinit(void);
extern int  xfs_init(int fstype);
extern void xfs_cleanup(void);

#ifdef CONFIG_XFS_GRIO
extern void xfs_grio_init(void);
extern void xfs_grio_uninit(void);
#else
# define xfs_grio_init()    do { } while (0)
# define xfs_grio_uninit()    do { } while (0)
#endif

#ifdef CELL_CAPABLE
extern int cxfs_parseargs(char *, int, struct xfs_args *);
#else
# define cxfs_parseargs(opt,flag,xargs)    (0) /* success */
#endif

#ifdef CONFIG_FS_POSIX_ACL
# define set_posix_acl(sb)    ((sb)->s_xattr_flags|= XATTR_MNT_FLAG_POSIX_ACL)
#else
# define set_posix_acl(sb)    do { } while (0)
#endif

/* For kernels which have the s_maxbytes field - set it */
#ifdef MAX_NON_LFS
# define set_max_bytes(sb)    ((sb)->s_maxbytes = XFS_MAX_FILE_OFFSET)
#else
# define set_max_bytes(sb)    do { } while (0)
#endif

#ifdef CONFIG_XFS_QUOTA
static struct quota_operations linvfs_qops = {
    getxstate:        linvfs_getxstate,
    setxstate:        linvfs_setxstate,
    getxquota:        linvfs_getxquota,
    setxquota:        linvfs_setxquota,
};
# define set_quota_ops(sb)    ((sb)->s_qop = &linvfs_qops)
#else
# define set_quota_ops(sb)    do { } while (0)
#endif

static struct super_operations linvfs_sops;

#define MNTOPT_LOGBUFS  "logbufs"       /* number of XFS log buffers */
#define MNTOPT_LOGBSIZE "logbsize"      /* size of XFS log buffers */
#define MNTOPT_LOGDEV    "logdev"    /* log device */
#define MNTOPT_RTDEV    "rtdev"        /* realtime I/O device */
#define MNTOPT_DMAPI    "dmapi"         /* DMI enabled (DMAPI / XDSM) */
#define MNTOPT_XDSM     "xdsm"          /* DMI enabled (DMAPI / XDSM) */
#define MNTOPT_BIOSIZE  "biosize"       /* log2 of preferred buffered io size */
#define MNTOPT_WSYNC    "wsync"         /* safe-mode nfs compatible mount */
#define MNTOPT_NOATIME  "noatime"       /* don't modify access times on reads */
#define MNTOPT_INO64    "ino64"         /* force inodes into 64-bit range */
#define MNTOPT_NOALIGN  "noalign"       /* turn off stripe alignment */
#define MNTOPT_SUNIT    "sunit"         /* data volume stripe unit */
#define MNTOPT_SWIDTH   "swidth"        /* data volume stripe width */
#define MNTOPT_NORECOVERY "norecovery"  /* don't run XFS recovery */
#define MNTOPT_OSYNCISDSYNC "osyncisdsync" /* o_sync == o_dsync on this fs */
                       /* (this is now the default!) */
#define MNTOPT_OSYNCISOSYNC "osyncisosync" /* o_sync is REALLY o_sync */
#define MNTOPT_QUOTA    "quota"         /* disk quotas */
#define MNTOPT_MRQUOTA  "mrquota"       /* don't turnoff if SB has quotas on */
#define MNTOPT_NOSUID   "nosuid"        /* disallow setuid program execution */
#define MNTOPT_NOQUOTA  "noquota"       /* no quotas */
#define MNTOPT_UQUOTA   "usrquota"      /* user quota enabled */
#define MNTOPT_GQUOTA   "grpquota"      /* group quota enabled */
#define MNTOPT_UQUOTANOENF "uqnoenforce"/* user quota limit enforcement */
#define MNTOPT_GQUOTANOENF "gqnoenforce"/* group quota limit enforcement */
#define MNTOPT_QUOTANOENF  "qnoenforce" /* same as uqnoenforce */
#define MNTOPT_RO       "ro"            /* read only */
#define MNTOPT_RW       "rw"            /* read/write */
#define MNTOPT_NOUUID   "nouuid"    /* Ignore FS uuid */

STATIC int
xfs_parseargs(
    char        *options,
    int        flags,
    struct xfs_args    *args)
{
    char        *this_char, *value, *eov;
    int        logbufs = -1;
    int        logbufsize = -1;
    int        dsunit, dswidth, vol_dsunit, vol_dswidth;
    int        iosize;
    int        rval = 1;    /* failure is default */

    iosize = dsunit = dswidth = vol_dsunit = vol_dswidth = 0;
    memset(args, 0, sizeof(struct xfs_args));
    args->slcount = args->stimeout = args->ctimeout = -1;

    /* Copy the already-parsed mount(2) flags we're interested in */
    args->flags = flags & MS_RDONLY;
    if (flags & MS_NOATIME)
        args->flags |= XFSMNT_NOATIME;

    args->flags |= XFSMNT_32BITINODES;

    if (!options) {
        args->logbufs = logbufs;
        args->logbufsize = logbufsize;
        return 0;
    }
    
    for (this_char = strtok (options, ",");
         this_char != NULL;
         this_char = strtok (NULL, ",")) {

        if ((value = strchr (this_char, '=')) != NULL)
            *value++ = 0;

        if (!strcmp(this_char, MNTOPT_LOGBUFS)) {
            if (!strcmp(value, "none")) {
                logbufs = 0;
                printk(
                "XFS: this FS is trash after writing to it\n");
            } else {
                logbufs = simple_strtoul(value, &eov, 10);
                if (logbufs < XLOG_NUM_ICLOGS ||
                    logbufs > XLOG_MAX_ICLOGS) {
                    printk(
                    "XFS: Illegal logbufs: %d [not %d-%d]\n",
                        logbufs, XLOG_NUM_ICLOGS,
                        XLOG_MAX_ICLOGS);
                    return rval;
                }
            }
        } else if (!strcmp(this_char, MNTOPT_LOGBSIZE)) {
            logbufsize = simple_strtoul(value, &eov, 10);
            if (logbufsize != 16*1024 && logbufsize != 32*1024) {
                printk(
            "XFS: Illegal logbufsize: %d [not 16k or 32k]\n",
                    logbufsize);
                return rval;
            }
        } else if (!strcmp(this_char, MNTOPT_LOGDEV)) {
            strncpy(args->logname, value, MAXNAMELEN);
        } else if (!strcmp(this_char, MNTOPT_DMAPI)) {
            args->flags |= XFSMNT_DMAPI;
        } else if (!strcmp(this_char, MNTOPT_XDSM)) {
            args->flags |= XFSMNT_DMAPI;
                } else if (!strcmp(this_char, MNTOPT_RTDEV)) {
            strncpy(args->rtname, value, MAXNAMELEN);
        } else if (!strcmp(this_char, MNTOPT_BIOSIZE)) {
            iosize = simple_strtoul(value, &eov, 10);
            if (iosize > 255 || iosize <= 0) {
                printk(
            "XFS: biosize value %d is out of bounds [0-255]\n",
                    iosize);
                return rval;
            }
            args->flags |= XFSMNT_IOSIZE;
            args->iosizelog = (uint8_t) iosize;
        } else if (!strcmp(this_char, MNTOPT_WSYNC)) {
            args->flags |= XFSMNT_WSYNC;
        } else if (!strcmp(this_char, MNTOPT_NOATIME)) {
            args->flags |= XFSMNT_NOATIME;
        } else if (!strcmp(this_char, MNTOPT_OSYNCISDSYNC)) {
            /* no-op, this is now the default */
        } else if (!strcmp(this_char, MNTOPT_OSYNCISOSYNC)) {
            args->flags |= XFSMNT_OSYNCISOSYNC;
        } else if (!strcmp(this_char, MNTOPT_NORECOVERY)) {
            args->flags |= XFSMNT_NORECOVERY;
        } else if (!strcmp(this_char, MNTOPT_INO64)) {
#ifdef XFS_BIG_FILESYSTEMS
            args->flags |= XFSMNT_INO64;
#else
            printk("XFS: ino64 option not allowed on this system\n");
            return rval;
#endif
        } else if (!strcmp(this_char, MNTOPT_UQUOTA)) {
            args->flags |= XFSMNT_UQUOTA | XFSMNT_UQUOTAENF;
        } else if (!strcmp(this_char, MNTOPT_QUOTA)) {
            args->flags |= XFSMNT_UQUOTA | XFSMNT_UQUOTAENF;
        } else if (!strcmp(this_char, MNTOPT_UQUOTANOENF)) {
            args->flags |= XFSMNT_UQUOTA;
            args->flags &= ~XFSMNT_UQUOTAENF;
        } else if (!strcmp(this_char, MNTOPT_QUOTANOENF)) {
            args->flags |= XFSMNT_UQUOTA;
            args->flags &= ~XFSMNT_UQUOTAENF;
        } else if (!strcmp(this_char, MNTOPT_MRQUOTA)) {
            args->flags |= XFSMNT_QUOTAMAYBE;
        } else if (!strcmp(this_char, MNTOPT_GQUOTA)) {
            args->flags |= XFSMNT_GQUOTA | XFSMNT_GQUOTAENF;
        } else if (!strcmp(this_char, MNTOPT_GQUOTANOENF)) {
            args->flags |= XFSMNT_GQUOTA;
            args->flags &= ~XFSMNT_GQUOTAENF;
        } else if (!strcmp(this_char, MNTOPT_NOALIGN)) {
            args->flags |= XFSMNT_NOALIGN;
        } else if (!strcmp(this_char, MNTOPT_SUNIT)) {
            dsunit = simple_strtoul(value, &eov, 10);
        } else if (!strcmp(this_char, MNTOPT_SWIDTH)) {
            dswidth = simple_strtoul(value, &eov, 10);
        } else if (!strcmp(this_char, MNTOPT_RO)) {
            args->flags |= MS_RDONLY;
        } else if (!strcmp(this_char, MNTOPT_NOSUID)) {
            args->flags |= MS_NOSUID;
        } else if (!strcmp(this_char, MNTOPT_NOUUID)) {
            args->flags |= XFSMNT_NOUUID; 
        } else {
            printk("XFS: unknown mount option [%s].\n", this_char);
            return rval;
                }
    }

    if (args->flags & XFSMNT_NORECOVERY) {
        if ((args->flags & MS_RDONLY) == 0) {
            printk("XFS: no-recovery mounts must be read-only.\n");
            return rval;
        }
    }

    if ((args->flags & XFSMNT_NOALIGN) && (dsunit || dswidth)) {
        printk(
    "XFS: sunit and swidth options incompatible with the noalign option\n");
        return rval;
    }

    if ((dsunit && !dswidth) || (!dsunit && dswidth)) {
        printk("XFS: sunit and swidth must be specified together\n");
        return rval;
    }

    if (dsunit && (dswidth % dsunit != 0)) {
        printk(
    "XFS: stripe width (%d) must be a multiple of the stripe unit (%d)\n",
            dswidth, dsunit);
        return rval;
    }

    if ((args->flags & XFSMNT_NOALIGN) != XFSMNT_NOALIGN) {
        if (dsunit) { 
            args->sunit = dsunit;
            args->flags |= XFSMNT_RETERR;
        } else
            args->sunit = vol_dsunit;    
        dswidth ? (args->swidth = dswidth) : 
              (args->swidth = vol_dswidth);
    } else 
        args->sunit = args->swidth = 0;

    args->logbufs = logbufs;
    args->logbufsize = logbufsize;

    rval = cxfs_parseargs(options, flags, args);
    return rval;
}

/*
 * Convert one device special file to a dev_t.
 * Helper routine, used only by spectodevs below.
 */
STATIC int
spectodev(
    const char    *name,
    const char    *id,
    dev_t        *dev)
{
    struct nameidata nd;
    int        rval = 0;

    if (path_init(name, LOOKUP_FOLLOW, &nd))
        rval = path_walk(name, &nd);
    /* Watch out for negative dentries */
    if (!nd.dentry->d_inode)
        rval = -ENOENT;
    if (rval)
        printk("XFS: Invalid %s device [%s], err=%d\n", id, name, rval);
    else
        *dev = nd.dentry->d_inode->i_rdev;
    path_release(&nd);
    return rval;
}

/*
 * Convert device special files to dev_t for data, log, realtime.
 */
int
spectodevs(
    struct super_block *sb,
    struct xfs_args    *args,
    dev_t        *ddevp,
    dev_t        *logdevp,
    dev_t        *rtdevp)
{
    int        rval = 0;

    *ddevp = *logdevp = sb->s_dev;
    if (args->logname[0])
        rval = spectodev(args->logname, "log", logdevp);
    if (args->rtname[0] && !rval)
        rval = spectodev(args->rtname, "realtime", rtdevp);
    else
        *rtdevp = 0;
    return rval;
}

/*
 * Initialise a device forming part of a filessystem.
 * The "data" flag indicates if this is part of the data
 * volume - if so, we don't need to initialize a block
 * device reference as the VFS has already done this.
 *
 * In the other cases (external log, rt volume) we must
 * initialise the device because:
 *  - its device driver may not yet have been loaded
 *  - without opening the block device, the kernel has no
 *    knowledge that we are actually using the device, so
 *    the driver may choose to unload at any time, etc.
 */
int
linvfs_fill_buftarg(
    struct buftarg        *btp,
    dev_t            dev,
    struct super_block    *sb,
    int            data)
{
    int            rval;
    struct block_device    *bdev = NULL;

    if (!data) {
        bdev = bdget(dev);
        if (!bdev)
            return -ENOMEM;
        rval = blkdev_get(bdev, FMODE_READ|FMODE_WRITE, 0, BDEV_FS);
        if (rval) {
            printk("XFS: blkdev_get failed on device %d\n", dev);
            bdput(bdev);
            return rval;
        }
    }

    btp->bd_targ = bdev;
    btp->pb_targ = pagebuf_lock_enable(dev, sb);
    btp->dev = dev;
    return 0;
}

void
linvfs_bsize_buftarg(
    struct buftarg        *btp,
    unsigned int        blocksize)
{
    pagebuf_target_blocksize(btp->pb_targ, blocksize);
}

void
linvfs_release_buftarg(
    struct buftarg        *btp)
{
    struct pb_target    *target = btp->pb_targ;
    struct block_device    *bdev = btp->bd_targ;

    if (target) {
        pagebuf_delwri_flush(target, PBDF_WAIT, NULL);
        pagebuf_lock_disable(target);
    }
    if (bdev) {
        blkdev_put(bdev, BDEV_FS);
    }
}

struct super_block *
linvfs_read_super(
    struct super_block *sb,
    void        *data,
    int        silent)
{
    vfs_t        *vfsp;
    vfsops_t    *vfsops;
    vnode_t        *cvp, *rootvp;
    struct inode    *ip, *cip;
    struct mounta    ap;
    struct xfs_args    args;
    statvfs_t    statvfs;
    int        error;

    if (xfs_parseargs((char *)data, sb->s_flags, &args))
        return NULL;
    strncpy(args.fsname, bdevname(sb->s_dev), MAXNAMELEN);
    /* args.rtdev and args.logdev done in xfs_parseargs */

    /*  Setup the generic "mounta" structure  */
    memset(&ap, 0, sizeof(struct mounta));
    ap.dataptr = (char *)&args;
    ap.datalen = sizeof(struct xfs_args);
    ap.spec = args.fsname;

    /*  Kludge in XFS until we have other VFS/VNODE FSs  */
    vfsops = &xfs_vfsops;

    /*  Set up the vfs_t structure  */
    vfsp = vfs_allocate();
    if (!vfsp) 
        return NULL; 

    if (sb->s_flags & MS_RDONLY)
        vfsp->vfs_flag |= VFS_RDONLY;

    /*  Setup up the cvp structure  */

    cip = (struct inode *)kmem_alloc(sizeof(struct inode),0);
    if (!cip) { 
        vfs_deallocate(vfsp);
        return NULL;
    } 

    bzero(cip, sizeof(*cip));
    atomic_set(&cip->i_count, 1);

    cvp = LINVFS_GET_VN_ADDRESS(cip);
    cvp->v_type   = VDIR;
    cvp->v_number = 1;        /* Place holder */
    cvp->v_inode  = cip;

#ifdef CONFIG_XFS_VNODE_TRACING
    cvp->v_trace = ktrace_alloc(VNODE_TRACE_SIZE, KM_SLEEP);
    vn_trace_entry(cvp, "linvfs_read_super", (inst_t *)__return_address);
#endif /* CONFIG_XFS_VNODE_TRACING */

        cvp->v_flag |= VMOUNTING;
    vn_bhv_head_init(VN_BHV_HEAD(cvp), "vnode");    /* for DMAPI */
    LINVFS_SET_CVP(sb, cvp);
    vfsp->vfs_super = sb;

    sb->s_blocksize = BBSIZE;
    sb->s_blocksize_bits = BBSHIFT;
    set_blocksize(sb->s_dev, BBSIZE);
    set_posix_acl(sb);
    sb->s_xattr_flags |= XATTR_MNT_FLAG_USER;
    set_max_bytes(sb);
    set_quota_ops(sb);
    sb->s_op = &linvfs_sops;

    LINVFS_SET_VFS(sb, vfsp);

    VFSOPS_MOUNT(vfsops, vfsp, cvp, &ap, NULL, sys_cred, error);
    if (error)
        goto fail_vfsop;

    VFS_STATVFS(vfsp, &statvfs, NULL, error);
    if (error)
        goto fail_unmount;

    sb->s_magic = XFS_SB_MAGIC;
    sb->s_dirt = 1;

        VFS_ROOT(vfsp, &rootvp, error);
        if (error)
                goto fail_unmount;

    ip = LINVFS_GET_IP(rootvp);
    linvfs_set_inode_ops(ip);
    linvfs_revalidate_core(ip, ATTR_COMM);

    sb->s_root = d_alloc_root(ip);
    if (!sb->s_root)
        goto fail_vnrele;
    if (is_bad_inode((struct inode *)sb->s_root))
        goto fail_vnrele;

    /* Don't set the VFS_DMI flag until here because we don't want
     * to send events while replaying the log.
     */
    if (args.flags & XFSMNT_DMAPI)
        vfsp->vfs_flag |= VFS_DMI;

    vn_trace_exit(rootvp, "linvfs_read_super", (inst_t *)__return_address);

    return(sb);

fail_vnrele:
    if (sb->s_root) {
        dput(sb->s_root);
        sb->s_root = NULL;
    } else {
        VN_RELE(rootvp);
    }

fail_unmount:
    VFS_UNMOUNT(vfsp, 0, sys_cred, error);
    /*  We need to do something here to shut down the VNODE/VFS layer.  */

fail_vfsop:
    vfs_deallocate(vfsp);

#ifdef  CONFIG_XFS_VNODE_TRACING
    ktrace_free(cvp->v_trace);
    cvp->v_trace = NULL;
#endif  /* CONFIG_XFS_VNODE_TRACING */

    kfree(cvp->v_inode);
    return(NULL);
}

void
linvfs_set_inode_ops(
    struct inode    *inode)
{
    vnode_t        *vp;

    vp = LINVFS_GET_VN_ADDRESS(inode);

    inode->i_mode = VTTOIF(vp->v_type);

    if (vp->v_type == VNON) {
        make_bad_inode(inode);
    } else if (S_ISREG(inode->i_mode)) {
        inode->i_op = &linvfs_file_inode_operations;
        inode->i_fop = &linvfs_file_operations;
        inode->i_mapping->a_ops = &linvfs_aops;
    } else if (S_ISDIR(inode->i_mode)) {
        inode->i_op = &linvfs_dir_inode_operations;
        inode->i_fop = &linvfs_dir_operations;
    } else if (S_ISLNK(inode->i_mode)) {
        inode->i_op = &linvfs_symlink_inode_operations;
        if (inode->i_blocks)
            inode->i_mapping->a_ops = &linvfs_aops;
    } else {
        inode->i_op = &linvfs_file_inode_operations;
        init_special_inode(inode, inode->i_mode, inode->i_rdev);
    }
}

void
linvfs_read_inode(
    struct inode    *inode)
{
    vfs_t        *vfsp = LINVFS_GET_VFS(inode->i_sb);

    if (vfsp) {
        vn_initialize(vfsp, inode, 1);
    } else {
        make_bad_inode(inode);
        return;
    }

    inode->i_version = ++event;
}

/*
 * We do not actually write the inode here, just mark the
 * super block dirty so that sync_supers calls us and
 * forces the flush.
 */
void
linvfs_write_inode(
    struct inode    *inode,
        int             sync)
{
    vnode_t    *vp = LINVFS_GET_VP(inode);

    if (vp) {
        vn_trace_entry(vp, "linvfs_write_inode",
                    (inst_t *)__return_address);
    }
    inode->i_sb->s_dirt = 1;
}

void
linvfs_delete_inode(
    struct inode    *inode)
{
    vnode_t    *vp = LINVFS_GET_VP(inode);

    if (vp) {
        vn_trace_entry(vp, "linvfs_delete_inode",
                    (inst_t *)__return_address);
        /*
         * Remove the vnode, the nlink count
         * is zero & the unlink will complete.
         */
        vp->v_flag |= VPURGE;
        vn_remove(vp);
    }

    clear_inode(inode);
}

void
linvfs_clear_inode(
    struct inode    *inode)
{
    vnode_t        *vp = LINVFS_GET_VP(inode);

    if (vp) {
        vn_trace_entry(vp, "linvfs_clear_inode",
                    (inst_t *)__return_address);
        /*
         * Do all our cleanup, and remove this vnode.
         */
        vp->v_flag |= VPURGE;
        vn_remove(vp);
    }
}

void 
linvfs_put_inode(
    struct inode    *inode)
{
    vnode_t        *vp = LINVFS_GET_VP(inode);

        if (vp) vn_put(vp);
}

void
linvfs_put_super(
    struct super_block *sb)
{
    int        error;
    int        sector_size = BBSIZE;
    kdev_t        dev = sb->s_dev;
    vfs_t         *vfsp = LINVFS_GET_VFS(sb);
    vnode_t        *cvp;

    VFS_DOUNMOUNT(vfsp, 0, NULL, sys_cred, error); 
    if (error) {
        printk("XFS unmount got error %d\n", error);
        printk("linvfs_put_super: vfsp/0x%p left dangling!\n", vfsp);
        return;
    }

    vfs_deallocate(vfsp);

    cvp = LINVFS_GET_CVP(sb);
#ifdef  CONFIG_XFS_VNODE_TRACING
    ktrace_free(cvp->v_trace);
    cvp->v_trace = NULL;
#endif  /* CONFIG_XFS_VNODE_TRACING */
    kfree(cvp->v_inode);

    /*  Do something to get rid of the VNODE/VFS layer here  */

    /* Reset device block size */
    if (hardsect_size[MAJOR(dev)])
        sector_size = hardsect_size[MAJOR(dev)][MINOR(dev)];
    set_blocksize(dev, sector_size);
}

void
linvfs_write_super(
    struct super_block *sb)
{
     vfs_t        *vfsp = LINVFS_GET_VFS(sb); 
     int        error; 

    sb->s_dirt = 0;
    if (sb->s_flags & MS_RDONLY)
        return;
    VFS_SYNC(vfsp, SYNC_FSDATA|SYNC_BDFLUSH|SYNC_NOWAIT|SYNC_ATTR,
        sys_cred, error);
}

int
linvfs_statfs(
    struct super_block *sb,
    struct statfs    *buf)
{
    vfs_t        *vfsp = LINVFS_GET_VFS(sb);
    statvfs_t    stat;
    int        error;

    VFS_STATVFS(vfsp, &stat, NULL, error);
    if (error)
        return(-error);

    buf->f_type = XFS_SB_MAGIC;
    buf->f_bsize = stat.f_bsize;
    buf->f_blocks = stat.f_blocks;
    buf->f_bfree = stat.f_bfree;
    buf->f_bavail = stat.f_bavail;
    buf->f_files = stat.f_files;
    buf->f_ffree = stat.f_ffree;
    buf->f_fsid.val[0] = stat.f_fsid;
    buf->f_fsid.val[1] = 0;
    buf->f_namelen = stat.f_namemax;

    return 0;
}

int
linvfs_remount(
    struct super_block *sb,
    int        *flags,
    char        *options)
{
    struct xfs_args    args;
    vfs_t        *vfsp;
    vnode_t        *cvp;

    vfsp = LINVFS_GET_VFS(sb);
    cvp = LINVFS_GET_CVP(sb);

    set_posix_acl(sb);
    sb->s_xattr_flags |= XATTR_MNT_FLAG_USER;

    if ((*flags & MS_RDONLY) == (sb->s_flags & MS_RDONLY))
        return 0;

    if (xfs_parseargs(options, *flags, &args))
                return -EINVAL;

    if (*flags & MS_RDONLY || args.flags & MS_RDONLY) {
        sb->s_flags |= MS_RDONLY;
        XFS_log_write_unmount_ro(vfsp->vfs_fbhv);
        vfsp->vfs_flag |= VFS_RDONLY;
    } else {
        vfsp->vfs_flag &= ~VFS_RDONLY;
        sb->s_flags &= ~MS_RDONLY;
    }

    return 0;
}

void
linvfs_freeze_fs(
    struct super_block *sb)
{
    vfs_t        *vfsp;
    vnode_t        *vp;
    int        error;

    vfsp = LINVFS_GET_VFS(sb);
        if (sb->s_flags & MS_RDONLY)
                return;
    VFS_ROOT(vfsp, &vp, error);
    VOP_IOCTL(vp, LINVFS_GET_IP(vp), NULL, XFS_IOC_FREEZE, 0, error);
    VN_RELE(vp);
}

void
linvfs_unfreeze_fs(
    struct super_block *sb)
{
    vfs_t        *vfsp;
    vnode_t        *vp;
    int        error;

    vfsp = LINVFS_GET_VFS(sb);
    VFS_ROOT(vfsp, &vp, error);
    VOP_IOCTL(vp, LINVFS_GET_IP(vp), NULL, XFS_IOC_THAW, 0, error);
    VN_RELE(vp);
}

int
linvfs_dmapi_mount(
    struct super_block *sb,
    char        *dir_name)
{
    vfsops_t    *vfsops;
    char        fsname[MAXNAMELEN];
    vnode_t        *cvp;    /* covered vnode */
    vfs_t        *vfsp; /* mounted vfs */
    int        error;

    vfsp = LINVFS_GET_VFS(sb);
    if ( ! (vfsp->vfs_flag & VFS_DMI) )
        return 0;
    cvp = LINVFS_GET_CVP(sb);
    strncpy(fsname, bdevname(sb->s_dev), MAXNAMELEN);

    /*  Kludge in XFS until we have other VFS/VNODE FSs  */
    vfsops = &xfs_vfsops;

    VFSOPS_DMAPI_MOUNT(vfsops, vfsp, cvp, dir_name, fsname, error);
    if (error) {
        if (atomic_read(&sb->s_active) == 1)
            vfsp->vfs_flag &= ~VFS_DMI;
        return -error;
    }
    return 0;
}

#if 0
STATIC int linvfs_dentry_to_fh(
    struct dentry *dentry,
    __u32 *data,
    int *lenp,
    int need_parent)
{
    struct inode *inode = dentry->d_inode ;
    vnode_t *vp = LINVFS_GET_VN_ADDRESS(inode);
    int maxlen = *lenp;
    xfs_fid2_t fid;
    int error;

    if (maxlen < 3)
        return 255 ;

    VOP_FID2(vp, (struct fid *)&fid, error);
    memcpy(&data[0], &fid.fid_ino, sizeof(__u64));
    data[2] = fid.fid_gen;

    *lenp = 3 ;
    if (maxlen < 5 || ! need_parent)
        return 3 ;

    inode = dentry->d_parent->d_inode ;
    vp = LINVFS_GET_VN_ADDRESS(inode);

    VOP_FID2(vp, (struct fid *)&fid, error);
    memcpy(&data[3], &fid.fid_ino, sizeof(__u64));
    *lenp = 5 ;
    if (maxlen < 6)
        return 5 ;
    data[5] = fid.fid_gen;
    *lenp = 6 ;
    return 6 ;
}

STATIC struct dentry *linvfs_fh_to_dentry(
    struct super_block *sb,
    __u32 *data,
    int len,
    int fhtype,
    int parent)
{
    vnode_t *vp;
    struct inode *inode = NULL;
    struct list_head *lp;
    struct dentry *result;
    xfs_fid2_t xfid;
    vfs_t *vfsp = LINVFS_GET_VFS(sb);
    int error;

    xfid.fid_len = sizeof(xfs_fid2_t) - sizeof(xfid.fid_len);
    xfid.fid_pad = 0;

    if (!parent) {
        xfid.fid_gen = data[2];
        memcpy(&xfid.fid_ino, &data[0], sizeof(__u64));
    } else {
        if (fhtype == 6)    
            xfid.fid_gen = data[5];
        else
            xfid.fid_gen = 0;
        memcpy(&xfid.fid_ino, &data[3], sizeof(__u64));
    }

    VFS_VGET(vfsp, &vp, (fid_t *)&xfid, error);
    if (error || vp == NULL)
        return ERR_PTR(-ESTALE) ;

    inode = LINVFS_GET_IP(vp);
    spin_lock(&dcache_lock);
    for (lp = inode->i_dentry.next; lp != &inode->i_dentry ; lp=lp->next) {
        result = list_entry(lp,struct dentry, d_alias);
        if (! (result->d_flags & DCACHE_NFSD_DISCONNECTED)) {
            dget_locked(result);
            result->d_vfs_flags |= DCACHE_REFERENCED;
            spin_unlock(&dcache_lock);
            iput(inode);
            return result;
        }
    }
    spin_unlock(&dcache_lock);
    result = d_alloc_root(inode);
    if (result == NULL) {
        iput(inode);
        return ERR_PTR(-ENOMEM);
    }
    result->d_flags |= DCACHE_NFSD_DISCONNECTED;
    return result;
}
#endif

static struct super_operations linvfs_sops = {
    read_inode:        linvfs_read_inode,
    write_inode:        linvfs_write_inode,
#ifdef CONFIG_HAVE_XFS_DMAPI
    dmapi_mount_event:    linvfs_dmapi_mount,
#endif
    put_inode:        linvfs_put_inode,
    delete_inode:        linvfs_delete_inode,
    clear_inode:        linvfs_clear_inode,
    put_super:        linvfs_put_super,
    write_super:        linvfs_write_super,
    write_super_lockfs:    linvfs_freeze_fs,
    unlockfs:        linvfs_unfreeze_fs,
    statfs:            linvfs_statfs,
    remount_fs:        linvfs_remount,

#if 0
    fh_to_dentry:        linvfs_fh_to_dentry,
    dentry_to_fh:        linvfs_dentry_to_fh,
#endif
};

DECLARE_FSTYPE_DEV(xfs_fs_type, XFS_NAME, linvfs_read_super);

static int __init init_xfs_fs(void)
{
    int error;
    struct sysinfo    si;
    static char message[] __initdata =
        KERN_INFO "SGI XFS with " XFS_BUILD_OPTIONS " enabled\n";

    error = pagebuf_init();
    if (error < 0)
        return error;
    si_meminfo(&si);
    xfs_physmem = si.totalram;

    printk(message);

    cred_init();
    vfsinit();
    xfs_init(0);
    xfs_grio_init();

    return register_filesystem(&xfs_fs_type);
}


static void __exit exit_xfs_fs(void)
{
        xfs_grio_uninit();
    xfs_cleanup();
        unregister_filesystem(&xfs_fs_type);
    pagebuf_terminate();
}

EXPORT_NO_SYMBOLS;

module_init(init_xfs_fs);
module_exit(exit_xfs_fs);

MODULE_AUTHOR("SGI <sgi.com>");
MODULE_DESCRIPTION("SGI XFS with " XFS_BUILD_OPTIONS " enabled");
MODULE_LICENSE("GPL");

:: Command execute ::

Enter:
 
Select:
 

:: Search ::
  - regexp 

:: Upload ::
 
[ Read-Only ]

:: Make Dir ::
 
[ Read-Only ]
:: Make File ::
 
[ Read-Only ]

:: Go Dir ::
 
:: Go File ::
 

--[ c99shell v. 1.0 pre-release build #13 powered by Captain Crunch Security Team | http://ccteam.ru | Generation time: 0.0059 ]--