OpenCores
URL https://opencores.org/ocsvn/or1k/or1k/trunk

Subversion Repositories or1k

[/] [or1k/] [trunk/] [linux/] [linux-2.4/] [fs/] [reiserfs/] [ioctl.c] - Diff between revs 1275 and 1765

Only display areas with differences | Details | Blame | View Log

Rev 1275 Rev 1765
/*
/*
 * Copyright 2000-2002 by Hans Reiser, licensing governed by reiserfs/README
 * Copyright 2000-2002 by Hans Reiser, licensing governed by reiserfs/README
 */
 */
 
 
#include <linux/fs.h>
#include <linux/fs.h>
#include <linux/reiserfs_fs.h>
#include <linux/reiserfs_fs.h>
#include <linux/sched.h>
#include <linux/sched.h>
#include <asm/uaccess.h>
#include <asm/uaccess.h>
#include <linux/smp_lock.h>
#include <linux/smp_lock.h>
#include <linux/locks.h>
#include <linux/locks.h>
 
 
/*
/*
** reiserfs_ioctl - handler for ioctl for inode
** reiserfs_ioctl - handler for ioctl for inode
** supported commands:
** supported commands:
**  1) REISERFS_IOC_UNPACK - try to unpack tail from direct item into indirect
**  1) REISERFS_IOC_UNPACK - try to unpack tail from direct item into indirect
**                           and prevent packing file (argument arg has to be non-zero)
**                           and prevent packing file (argument arg has to be non-zero)
**  2) REISERFS_IOC_[GS]ETFLAGS, REISERFS_IOC_[GS]ETVERSION
**  2) REISERFS_IOC_[GS]ETFLAGS, REISERFS_IOC_[GS]ETVERSION
**  3) That's all for a while ...
**  3) That's all for a while ...
*/
*/
int reiserfs_ioctl (struct inode * inode, struct file * filp, unsigned int cmd,
int reiserfs_ioctl (struct inode * inode, struct file * filp, unsigned int cmd,
                unsigned long arg)
                unsigned long arg)
{
{
        unsigned int flags;
        unsigned int flags;
 
 
        switch (cmd) {
        switch (cmd) {
            case REISERFS_IOC_UNPACK:
            case REISERFS_IOC_UNPACK:
                if( S_ISREG( inode -> i_mode ) ) {
                if( S_ISREG( inode -> i_mode ) ) {
                if (arg)
                if (arg)
                    return reiserfs_unpack (inode, filp);
                    return reiserfs_unpack (inode, filp);
                        else
                        else
                                return 0;
                                return 0;
                } else
                } else
                        return -ENOTTY;
                        return -ENOTTY;
        /*
        /*
         * Following {G,S}ETFLAGS, and {G,S}ETVERSION are providing ext2
         * Following {G,S}ETFLAGS, and {G,S}ETVERSION are providing ext2
         * binary compatible interface (used by lsattr(1), and chattr(1)) and
         * binary compatible interface (used by lsattr(1), and chattr(1)) and
         * are * thus conceptually similar to appropriate pieces of
         * are * thus conceptually similar to appropriate pieces of
         * fs/ext2/ioctl.c
         * fs/ext2/ioctl.c
         */
         */
        case REISERFS_IOC_GETFLAGS:
        case REISERFS_IOC_GETFLAGS:
                flags = inode -> u.reiserfs_i.i_attrs;
                flags = inode -> u.reiserfs_i.i_attrs;
                i_attrs_to_sd_attrs( inode, ( __u16 * ) &flags );
                i_attrs_to_sd_attrs( inode, ( __u16 * ) &flags );
                return put_user(flags, (int *) arg);
                return put_user(flags, (int *) arg);
        case REISERFS_IOC_SETFLAGS: {
        case REISERFS_IOC_SETFLAGS: {
                if (IS_RDONLY(inode))
                if (IS_RDONLY(inode))
                        return -EROFS;
                        return -EROFS;
 
 
                if ((current->fsuid != inode->i_uid) && !capable(CAP_FOWNER))
                if ((current->fsuid != inode->i_uid) && !capable(CAP_FOWNER))
                        return -EPERM;
                        return -EPERM;
 
 
                if (get_user(flags, (int *) arg))
                if (get_user(flags, (int *) arg))
                        return -EFAULT;
                        return -EFAULT;
 
 
                if ( ( ( flags ^ inode->u.reiserfs_i.i_attrs) & ( REISERFS_IMMUTABLE_FL | REISERFS_APPEND_FL)) &&
                if ( ( ( flags ^ inode->u.reiserfs_i.i_attrs) & ( REISERFS_IMMUTABLE_FL | REISERFS_APPEND_FL)) &&
                     !capable( CAP_LINUX_IMMUTABLE ) )
                     !capable( CAP_LINUX_IMMUTABLE ) )
                        return -EPERM;
                        return -EPERM;
 
 
                if( ( flags & REISERFS_NOTAIL_FL ) &&
                if( ( flags & REISERFS_NOTAIL_FL ) &&
                    S_ISREG( inode -> i_mode ) ) {
                    S_ISREG( inode -> i_mode ) ) {
                                int result;
                                int result;
 
 
                                result = reiserfs_unpack( inode, filp );
                                result = reiserfs_unpack( inode, filp );
                                if( result )
                                if( result )
                                        return result;
                                        return result;
                }
                }
                sd_attrs_to_i_attrs( flags, inode );
                sd_attrs_to_i_attrs( flags, inode );
                inode -> u.reiserfs_i.i_attrs = flags;
                inode -> u.reiserfs_i.i_attrs = flags;
                inode->i_ctime = CURRENT_TIME;
                inode->i_ctime = CURRENT_TIME;
                mark_inode_dirty(inode);
                mark_inode_dirty(inode);
                return 0;
                return 0;
        }
        }
        case REISERFS_IOC_GETVERSION:
        case REISERFS_IOC_GETVERSION:
                return put_user(inode->i_generation, (int *) arg);
                return put_user(inode->i_generation, (int *) arg);
        case REISERFS_IOC_SETVERSION:
        case REISERFS_IOC_SETVERSION:
                if ((current->fsuid != inode->i_uid) && !capable(CAP_FOWNER))
                if ((current->fsuid != inode->i_uid) && !capable(CAP_FOWNER))
                        return -EPERM;
                        return -EPERM;
                if (IS_RDONLY(inode))
                if (IS_RDONLY(inode))
                        return -EROFS;
                        return -EROFS;
                if (get_user(inode->i_generation, (int *) arg))
                if (get_user(inode->i_generation, (int *) arg))
                        return -EFAULT;
                        return -EFAULT;
                inode->i_ctime = CURRENT_TIME;
                inode->i_ctime = CURRENT_TIME;
                mark_inode_dirty(inode);
                mark_inode_dirty(inode);
                return 0;
                return 0;
        default:
        default:
                return -ENOTTY;
                return -ENOTTY;
        }
        }
        return 0;
        return 0;
}
}
 
 
/*
/*
** reiserfs_unpack
** reiserfs_unpack
** Function try to convert tail from direct item into indirect.
** Function try to convert tail from direct item into indirect.
** It set up nopack attribute in the inode.u.reiserfs_i.nopack
** It set up nopack attribute in the inode.u.reiserfs_i.nopack
*/
*/
int reiserfs_unpack (struct inode * inode, struct file * filp)
int reiserfs_unpack (struct inode * inode, struct file * filp)
{
{
    int retval = 0;
    int retval = 0;
    int index ;
    int index ;
    struct page *page ;
    struct page *page ;
    unsigned long write_from ;
    unsigned long write_from ;
    unsigned long blocksize = inode->i_sb->s_blocksize ;
    unsigned long blocksize = inode->i_sb->s_blocksize ;
 
 
    if (inode->i_size == 0) {
    if (inode->i_size == 0) {
        inode->u.reiserfs_i.i_flags |= i_nopack_mask;
        inode->u.reiserfs_i.i_flags |= i_nopack_mask;
        return 0 ;
        return 0 ;
    }
    }
    /* ioctl already done */
    /* ioctl already done */
    if (inode->u.reiserfs_i.i_flags & i_nopack_mask) {
    if (inode->u.reiserfs_i.i_flags & i_nopack_mask) {
        return 0 ;
        return 0 ;
    }
    }
    lock_kernel();
    lock_kernel();
 
 
    /* we need to make sure nobody is changing the file size beneath
    /* we need to make sure nobody is changing the file size beneath
    ** us
    ** us
    */
    */
    down(&inode->i_sem) ;
    down(&inode->i_sem) ;
 
 
    write_from = inode->i_size & (blocksize - 1) ;
    write_from = inode->i_size & (blocksize - 1) ;
    /* if we are on a block boundary, we are already unpacked.  */
    /* if we are on a block boundary, we are already unpacked.  */
    if ( write_from == 0) {
    if ( write_from == 0) {
        inode->u.reiserfs_i.i_flags |= i_nopack_mask;
        inode->u.reiserfs_i.i_flags |= i_nopack_mask;
        goto out ;
        goto out ;
    }
    }
 
 
    /* we unpack by finding the page with the tail, and calling
    /* we unpack by finding the page with the tail, and calling
    ** reiserfs_prepare_write on that page.  This will force a
    ** reiserfs_prepare_write on that page.  This will force a
    ** reiserfs_get_block to unpack the tail for us.
    ** reiserfs_get_block to unpack the tail for us.
    */
    */
    index = inode->i_size >> PAGE_CACHE_SHIFT ;
    index = inode->i_size >> PAGE_CACHE_SHIFT ;
    page = grab_cache_page(inode->i_mapping, index) ;
    page = grab_cache_page(inode->i_mapping, index) ;
    retval = -ENOMEM;
    retval = -ENOMEM;
    if (!page) {
    if (!page) {
        goto out ;
        goto out ;
    }
    }
    retval = reiserfs_prepare_write(NULL, page, write_from, blocksize) ;
    retval = reiserfs_prepare_write(NULL, page, write_from, blocksize) ;
    if (retval)
    if (retval)
        goto out_unlock ;
        goto out_unlock ;
 
 
    /* conversion can change page contents, must flush */
    /* conversion can change page contents, must flush */
    flush_dcache_page(page) ;
    flush_dcache_page(page) ;
    inode->u.reiserfs_i.i_flags |= i_nopack_mask;
    inode->u.reiserfs_i.i_flags |= i_nopack_mask;
    kunmap(page) ; /* mapped by prepare_write */
    kunmap(page) ; /* mapped by prepare_write */
 
 
out_unlock:
out_unlock:
    UnlockPage(page) ;
    UnlockPage(page) ;
    page_cache_release(page) ;
    page_cache_release(page) ;
 
 
out:
out:
    up(&inode->i_sem) ;
    up(&inode->i_sem) ;
    unlock_kernel();
    unlock_kernel();
    return retval;
    return retval;
}
}
 
 

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.