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

Subversion Repositories or1k_old

[/] [or1k_old/] [trunk/] [rc203soc/] [sw/] [uClinux/] [fs/] [open.c] - Diff between revs 1627 and 1765

Go to most recent revision | Only display areas with differences | Details | Blame | View Log

Rev 1627 Rev 1765
/*
/*
 *  linux/fs/open.c
 *  linux/fs/open.c
 *
 *
 *  Copyright (C) 1991, 1992  Linus Torvalds
 *  Copyright (C) 1991, 1992  Linus Torvalds
 */
 */
 
 
/*
/*
 * uClinux revisions for NO_MM
 * uClinux revisions for NO_MM
 * Copyright (C) 1998  Kenneth Albanowski <kjahds@kjahds.com>,
 * Copyright (C) 1998  Kenneth Albanowski <kjahds@kjahds.com>,
 *                     The Silver Hammer Group, Ltd.
 *                     The Silver Hammer Group, Ltd.
 */
 */
 
 
#include <linux/vfs.h>
#include <linux/vfs.h>
#include <linux/types.h>
#include <linux/types.h>
#include <linux/utime.h>
#include <linux/utime.h>
#include <linux/errno.h>
#include <linux/errno.h>
#include <linux/fcntl.h>
#include <linux/fcntl.h>
#include <linux/stat.h>
#include <linux/stat.h>
#include <linux/string.h>
#include <linux/string.h>
#include <linux/sched.h>
#include <linux/sched.h>
#include <linux/kernel.h>
#include <linux/kernel.h>
#include <linux/signal.h>
#include <linux/signal.h>
#include <linux/tty.h>
#include <linux/tty.h>
#include <linux/time.h>
#include <linux/time.h>
#include <linux/mm.h>
#include <linux/mm.h>
#include <linux/file.h>
#include <linux/file.h>
 
 
#include <asm/segment.h>
#include <asm/segment.h>
#include <asm/bitops.h>
#include <asm/bitops.h>
 
 
asmlinkage int sys_statfs(const char * path, struct statfs * buf)
asmlinkage int sys_statfs(const char * path, struct statfs * buf)
{
{
        struct inode * inode;
        struct inode * inode;
        int error;
        int error;
 
 
        error = verify_area(VERIFY_WRITE, buf, sizeof(struct statfs));
        error = verify_area(VERIFY_WRITE, buf, sizeof(struct statfs));
        if (error)
        if (error)
                return error;
                return error;
        error = namei(path,&inode);
        error = namei(path,&inode);
        if (error)
        if (error)
                return error;
                return error;
        if (!inode->i_sb->s_op->statfs) {
        if (!inode->i_sb->s_op->statfs) {
                iput(inode);
                iput(inode);
                return -ENOSYS;
                return -ENOSYS;
        }
        }
        inode->i_sb->s_op->statfs(inode->i_sb, buf, sizeof(struct statfs));
        inode->i_sb->s_op->statfs(inode->i_sb, buf, sizeof(struct statfs));
        iput(inode);
        iput(inode);
        return 0;
        return 0;
}
}
 
 
asmlinkage int sys_fstatfs(unsigned int fd, struct statfs * buf)
asmlinkage int sys_fstatfs(unsigned int fd, struct statfs * buf)
{
{
        struct inode * inode;
        struct inode * inode;
        struct file * file;
        struct file * file;
        int error;
        int error;
 
 
        error = verify_area(VERIFY_WRITE, buf, sizeof(struct statfs));
        error = verify_area(VERIFY_WRITE, buf, sizeof(struct statfs));
        if (error)
        if (error)
                return error;
                return error;
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
                return -EBADF;
                return -EBADF;
        if (!(inode = file->f_inode))
        if (!(inode = file->f_inode))
                return -ENOENT;
                return -ENOENT;
        if (!inode->i_sb)
        if (!inode->i_sb)
                return -ENODEV;
                return -ENODEV;
        if (!inode->i_sb->s_op->statfs)
        if (!inode->i_sb->s_op->statfs)
                return -ENOSYS;
                return -ENOSYS;
        inode->i_sb->s_op->statfs(inode->i_sb, buf, sizeof(struct statfs));
        inode->i_sb->s_op->statfs(inode->i_sb, buf, sizeof(struct statfs));
        return 0;
        return 0;
}
}
 
 
int do_truncate(struct inode *inode, unsigned long length)
int do_truncate(struct inode *inode, unsigned long length)
{
{
        int error;
        int error;
        struct iattr newattrs;
        struct iattr newattrs;
 
 
        /* Not pretty: "inode->i_size" shouldn't really be "off_t". But it is. */
        /* Not pretty: "inode->i_size" shouldn't really be "off_t". But it is. */
        if ((off_t) length < 0)
        if ((off_t) length < 0)
                return -EINVAL;
                return -EINVAL;
 
 
        down(&inode->i_sem);
        down(&inode->i_sem);
        newattrs.ia_size = length;
        newattrs.ia_size = length;
        newattrs.ia_valid = ATTR_SIZE | ATTR_CTIME;
        newattrs.ia_valid = ATTR_SIZE | ATTR_CTIME;
        error = notify_change(inode, &newattrs);
        error = notify_change(inode, &newattrs);
        if (!error) {
        if (!error) {
                /* truncate virtual mappings of this file */
                /* truncate virtual mappings of this file */
#ifndef NO_MM
#ifndef NO_MM
                vmtruncate(inode, length);
                vmtruncate(inode, length);
#endif /* !NO_MM */
#endif /* !NO_MM */
                if (inode->i_op && inode->i_op->truncate)
                if (inode->i_op && inode->i_op->truncate)
                        inode->i_op->truncate(inode);
                        inode->i_op->truncate(inode);
        }
        }
        up(&inode->i_sem);
        up(&inode->i_sem);
        return error;
        return error;
}
}
 
 
asmlinkage int sys_truncate(const char * path, unsigned long length)
asmlinkage int sys_truncate(const char * path, unsigned long length)
{
{
        struct inode * inode;
        struct inode * inode;
        int error;
        int error;
 
 
        error = namei(path,&inode);
        error = namei(path,&inode);
        if (error)
        if (error)
                return error;
                return error;
 
 
        error = -EACCES;
        error = -EACCES;
        if (S_ISDIR(inode->i_mode))
        if (S_ISDIR(inode->i_mode))
                goto out;
                goto out;
 
 
        error = permission(inode,MAY_WRITE);
        error = permission(inode,MAY_WRITE);
        if (error)
        if (error)
                goto out;
                goto out;
 
 
        error = -EROFS;
        error = -EROFS;
        if (IS_RDONLY(inode))
        if (IS_RDONLY(inode))
                goto out;
                goto out;
 
 
        error = -EPERM;
        error = -EPERM;
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
                goto out;
                goto out;
 
 
        error = get_write_access(inode);
        error = get_write_access(inode);
        if (error)
        if (error)
                goto out;
                goto out;
 
 
        error = locks_verify_area(FLOCK_VERIFY_WRITE, inode, NULL,
        error = locks_verify_area(FLOCK_VERIFY_WRITE, inode, NULL,
                                  length < inode->i_size ? length : inode->i_size,
                                  length < inode->i_size ? length : inode->i_size,
                                  abs(inode->i_size - length));
                                  abs(inode->i_size - length));
        if (!error) {
        if (!error) {
                if (inode->i_sb && inode->i_sb->dq_op)
                if (inode->i_sb && inode->i_sb->dq_op)
                        inode->i_sb->dq_op->initialize(inode, -1);
                        inode->i_sb->dq_op->initialize(inode, -1);
                error = do_truncate(inode, length);
                error = do_truncate(inode, length);
        }
        }
        put_write_access(inode);
        put_write_access(inode);
out:
out:
        iput(inode);
        iput(inode);
        return error;
        return error;
}
}
 
 
asmlinkage int sys_ftruncate(unsigned int fd, unsigned long length)
asmlinkage int sys_ftruncate(unsigned int fd, unsigned long length)
{
{
        struct inode * inode;
        struct inode * inode;
        struct file * file;
        struct file * file;
        int error;
        int error;
 
 
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
                return -EBADF;
                return -EBADF;
        if (!(inode = file->f_inode))
        if (!(inode = file->f_inode))
                return -ENOENT;
                return -ENOENT;
        if (S_ISDIR(inode->i_mode) || !(file->f_mode & FMODE_WRITE))
        if (S_ISDIR(inode->i_mode) || !(file->f_mode & FMODE_WRITE))
                return -EACCES;
                return -EACCES;
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
                return -EPERM;
                return -EPERM;
        error = locks_verify_area(FLOCK_VERIFY_WRITE, inode, file,
        error = locks_verify_area(FLOCK_VERIFY_WRITE, inode, file,
                                  length < inode->i_size ? length : inode->i_size,
                                  length < inode->i_size ? length : inode->i_size,
                                  abs(inode->i_size - length));
                                  abs(inode->i_size - length));
        if (!error)
        if (!error)
                error = do_truncate(inode, length);
                error = do_truncate(inode, length);
        return error;
        return error;
}
}
 
 
#ifndef __alpha__
#ifndef __alpha__
 
 
/*
/*
 * sys_utime() can be implemented in user-level using sys_utimes().
 * sys_utime() can be implemented in user-level using sys_utimes().
 * Is this for backwards compatibility?  If so, why not move it
 * Is this for backwards compatibility?  If so, why not move it
 * into the appropriate arch directory (for those architectures that
 * into the appropriate arch directory (for those architectures that
 * need it).
 * need it).
 */
 */
 
 
/* If times==NULL, set access and modification to current time,
/* If times==NULL, set access and modification to current time,
 * must be owner or have write permission.
 * must be owner or have write permission.
 * Else, update from *times, must be owner or super user.
 * Else, update from *times, must be owner or super user.
 */
 */
asmlinkage int sys_utime(char * filename, struct utimbuf * times)
asmlinkage int sys_utime(char * filename, struct utimbuf * times)
{
{
        int error;
        int error;
        struct inode * inode;
        struct inode * inode;
        struct iattr newattrs;
        struct iattr newattrs;
 
 
        error = namei(filename,&inode);
        error = namei(filename,&inode);
        if (error)
        if (error)
                return error;
                return error;
        if (IS_RDONLY(inode)) {
        if (IS_RDONLY(inode)) {
                iput(inode);
                iput(inode);
                return -EROFS;
                return -EROFS;
        }
        }
        /* Don't worry, the checks are done in inode_change_ok() */
        /* Don't worry, the checks are done in inode_change_ok() */
        newattrs.ia_valid = ATTR_CTIME | ATTR_MTIME | ATTR_ATIME;
        newattrs.ia_valid = ATTR_CTIME | ATTR_MTIME | ATTR_ATIME;
        if (times) {
        if (times) {
                error = verify_area(VERIFY_READ, times, sizeof(*times));
                error = verify_area(VERIFY_READ, times, sizeof(*times));
                if (error) {
                if (error) {
                        iput(inode);
                        iput(inode);
                        return error;
                        return error;
                }
                }
                newattrs.ia_atime = get_user(&times->actime);
                newattrs.ia_atime = get_user(&times->actime);
                newattrs.ia_mtime = get_user(&times->modtime);
                newattrs.ia_mtime = get_user(&times->modtime);
                newattrs.ia_valid |= ATTR_ATIME_SET | ATTR_MTIME_SET;
                newattrs.ia_valid |= ATTR_ATIME_SET | ATTR_MTIME_SET;
        } else {
        } else {
                if (current->fsuid != inode->i_uid &&
                if (current->fsuid != inode->i_uid &&
                    (error = permission(inode,MAY_WRITE)) != 0) {
                    (error = permission(inode,MAY_WRITE)) != 0) {
                        iput(inode);
                        iput(inode);
                        return error;
                        return error;
                }
                }
        }
        }
        error = notify_change(inode, &newattrs);
        error = notify_change(inode, &newattrs);
        iput(inode);
        iput(inode);
        return error;
        return error;
}
}
 
 
#endif
#endif
 
 
/* If times==NULL, set access and modification to current time,
/* If times==NULL, set access and modification to current time,
 * must be owner or have write permission.
 * must be owner or have write permission.
 * Else, update from *times, must be owner or super user.
 * Else, update from *times, must be owner or super user.
 */
 */
asmlinkage int sys_utimes(char * filename, struct timeval * utimes)
asmlinkage int sys_utimes(char * filename, struct timeval * utimes)
{
{
        int error;
        int error;
        struct inode * inode;
        struct inode * inode;
        struct iattr newattrs;
        struct iattr newattrs;
 
 
        error = namei(filename,&inode);
        error = namei(filename,&inode);
        if (error)
        if (error)
                return error;
                return error;
        if (IS_RDONLY(inode)) {
        if (IS_RDONLY(inode)) {
                iput(inode);
                iput(inode);
                return -EROFS;
                return -EROFS;
        }
        }
        /* Don't worry, the checks are done in inode_change_ok() */
        /* Don't worry, the checks are done in inode_change_ok() */
        newattrs.ia_valid = ATTR_CTIME | ATTR_MTIME | ATTR_ATIME;
        newattrs.ia_valid = ATTR_CTIME | ATTR_MTIME | ATTR_ATIME;
        if (utimes) {
        if (utimes) {
                struct timeval times[2];
                struct timeval times[2];
                error = verify_area(VERIFY_READ, utimes, sizeof(times));
                error = verify_area(VERIFY_READ, utimes, sizeof(times));
                if (error) {
                if (error) {
                        iput(inode);
                        iput(inode);
                        return error;
                        return error;
                }
                }
                memcpy_fromfs(&times, utimes, sizeof(times));
                memcpy_fromfs(&times, utimes, sizeof(times));
                newattrs.ia_atime = times[0].tv_sec;
                newattrs.ia_atime = times[0].tv_sec;
                newattrs.ia_mtime = times[1].tv_sec;
                newattrs.ia_mtime = times[1].tv_sec;
                newattrs.ia_valid |= ATTR_ATIME_SET | ATTR_MTIME_SET;
                newattrs.ia_valid |= ATTR_ATIME_SET | ATTR_MTIME_SET;
        } else {
        } else {
                if ((error = permission(inode,MAY_WRITE)) != 0) {
                if ((error = permission(inode,MAY_WRITE)) != 0) {
                        iput(inode);
                        iput(inode);
                        return error;
                        return error;
                }
                }
        }
        }
        error = notify_change(inode, &newattrs);
        error = notify_change(inode, &newattrs);
        iput(inode);
        iput(inode);
        return error;
        return error;
}
}
 
 
/*
/*
 * access() needs to use the real uid/gid, not the effective uid/gid.
 * access() needs to use the real uid/gid, not the effective uid/gid.
 * We do this by temporarily setting fsuid/fsgid to the wanted values
 * We do this by temporarily setting fsuid/fsgid to the wanted values
 */
 */
asmlinkage int sys_access(const char * filename, int mode)
asmlinkage int sys_access(const char * filename, int mode)
{
{
        struct inode * inode;
        struct inode * inode;
        int old_fsuid, old_fsgid;
        int old_fsuid, old_fsgid;
        int res;
        int res;
 
 
        if (mode != (mode & S_IRWXO))   /* where's F_OK, X_OK, W_OK, R_OK? */
        if (mode != (mode & S_IRWXO))   /* where's F_OK, X_OK, W_OK, R_OK? */
                return -EINVAL;
                return -EINVAL;
        old_fsuid = current->fsuid;
        old_fsuid = current->fsuid;
        old_fsgid = current->fsgid;
        old_fsgid = current->fsgid;
        current->fsuid = current->uid;
        current->fsuid = current->uid;
        current->fsgid = current->gid;
        current->fsgid = current->gid;
        res = namei(filename,&inode);
        res = namei(filename,&inode);
        if (!res) {
        if (!res) {
                res = permission(inode, mode);
                res = permission(inode, mode);
                iput(inode);
                iput(inode);
        }
        }
        current->fsuid = old_fsuid;
        current->fsuid = old_fsuid;
        current->fsgid = old_fsgid;
        current->fsgid = old_fsgid;
        return res;
        return res;
}
}
 
 
asmlinkage int sys_chdir(const char * filename)
asmlinkage int sys_chdir(const char * filename)
{
{
        struct inode * inode;
        struct inode * inode;
        int error;
        int error;
 
 
        error = namei(filename,&inode);
        error = namei(filename,&inode);
        if (error)
        if (error)
                return error;
                return error;
        if (!S_ISDIR(inode->i_mode)) {
        if (!S_ISDIR(inode->i_mode)) {
                iput(inode);
                iput(inode);
                return -ENOTDIR;
                return -ENOTDIR;
        }
        }
        if ((error = permission(inode,MAY_EXEC)) != 0) {
        if ((error = permission(inode,MAY_EXEC)) != 0) {
                iput(inode);
                iput(inode);
                return error;
                return error;
        }
        }
        iput(current->fs->pwd);
        iput(current->fs->pwd);
        current->fs->pwd = inode;
        current->fs->pwd = inode;
        return (0);
        return (0);
}
}
 
 
asmlinkage int sys_fchdir(unsigned int fd)
asmlinkage int sys_fchdir(unsigned int fd)
{
{
        struct inode * inode;
        struct inode * inode;
        struct file * file;
        struct file * file;
        int error;
        int error;
 
 
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
                return -EBADF;
                return -EBADF;
        if (!(inode = file->f_inode))
        if (!(inode = file->f_inode))
                return -ENOENT;
                return -ENOENT;
        if (!S_ISDIR(inode->i_mode))
        if (!S_ISDIR(inode->i_mode))
                return -ENOTDIR;
                return -ENOTDIR;
        if ((error = permission(inode,MAY_EXEC)) != 0)
        if ((error = permission(inode,MAY_EXEC)) != 0)
                return error;
                return error;
        iput(current->fs->pwd);
        iput(current->fs->pwd);
        current->fs->pwd = inode;
        current->fs->pwd = inode;
        inode->i_count++;
        inode->i_count++;
        return (0);
        return (0);
}
}
 
 
asmlinkage int sys_chroot(const char * filename)
asmlinkage int sys_chroot(const char * filename)
{
{
        struct inode * inode;
        struct inode * inode;
        int error;
        int error;
 
 
        error = namei(filename,&inode);
        error = namei(filename,&inode);
        if (error)
        if (error)
                return error;
                return error;
        if (!S_ISDIR(inode->i_mode)) {
        if (!S_ISDIR(inode->i_mode)) {
                iput(inode);
                iput(inode);
                return -ENOTDIR;
                return -ENOTDIR;
        }
        }
        if (!fsuser()) {
        if (!fsuser()) {
                iput(inode);
                iput(inode);
                return -EPERM;
                return -EPERM;
        }
        }
        iput(current->fs->root);
        iput(current->fs->root);
        current->fs->root = inode;
        current->fs->root = inode;
        return (0);
        return (0);
}
}
 
 
asmlinkage int sys_fchmod(unsigned int fd, mode_t mode)
asmlinkage int sys_fchmod(unsigned int fd, mode_t mode)
{
{
        struct inode * inode;
        struct inode * inode;
        struct file * file;
        struct file * file;
        struct iattr newattrs;
        struct iattr newattrs;
 
 
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
                return -EBADF;
                return -EBADF;
        if (!(inode = file->f_inode))
        if (!(inode = file->f_inode))
                return -ENOENT;
                return -ENOENT;
        if (IS_RDONLY(inode))
        if (IS_RDONLY(inode))
                return -EROFS;
                return -EROFS;
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
                return -EPERM;
                return -EPERM;
        if (mode == (mode_t) -1)
        if (mode == (mode_t) -1)
                mode = inode->i_mode;
                mode = inode->i_mode;
        newattrs.ia_mode = (mode & S_IALLUGO) | (inode->i_mode & ~S_IALLUGO);
        newattrs.ia_mode = (mode & S_IALLUGO) | (inode->i_mode & ~S_IALLUGO);
        newattrs.ia_valid = ATTR_MODE | ATTR_CTIME;
        newattrs.ia_valid = ATTR_MODE | ATTR_CTIME;
        inode->i_dirt = 1;
        inode->i_dirt = 1;
        return notify_change(inode, &newattrs);
        return notify_change(inode, &newattrs);
}
}
 
 
asmlinkage int sys_chmod(const char * filename, mode_t mode)
asmlinkage int sys_chmod(const char * filename, mode_t mode)
{
{
        struct inode * inode;
        struct inode * inode;
        int error;
        int error;
        struct iattr newattrs;
        struct iattr newattrs;
 
 
        error = namei(filename,&inode);
        error = namei(filename,&inode);
        if (error)
        if (error)
                return error;
                return error;
        if (IS_RDONLY(inode)) {
        if (IS_RDONLY(inode)) {
                iput(inode);
                iput(inode);
                return -EROFS;
                return -EROFS;
        }
        }
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode)) {
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode)) {
                iput(inode);
                iput(inode);
                return -EPERM;
                return -EPERM;
        }
        }
        if (mode == (mode_t) -1)
        if (mode == (mode_t) -1)
                mode = inode->i_mode;
                mode = inode->i_mode;
        newattrs.ia_mode = (mode & S_IALLUGO) | (inode->i_mode & ~S_IALLUGO);
        newattrs.ia_mode = (mode & S_IALLUGO) | (inode->i_mode & ~S_IALLUGO);
        newattrs.ia_valid = ATTR_MODE | ATTR_CTIME;
        newattrs.ia_valid = ATTR_MODE | ATTR_CTIME;
        inode->i_dirt = 1;
        inode->i_dirt = 1;
        error = notify_change(inode, &newattrs);
        error = notify_change(inode, &newattrs);
        iput(inode);
        iput(inode);
        return error;
        return error;
}
}
 
 
asmlinkage int sys_fchown(unsigned int fd, uid_t user, gid_t group)
asmlinkage int sys_fchown(unsigned int fd, uid_t user, gid_t group)
{
{
        struct inode * inode;
        struct inode * inode;
        struct file * file;
        struct file * file;
        struct iattr newattrs;
        struct iattr newattrs;
        int error;
        int error;
 
 
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
        if (fd >= NR_OPEN || !(file = current->files->fd[fd]))
                return -EBADF;
                return -EBADF;
        if (!(inode = file->f_inode))
        if (!(inode = file->f_inode))
                return -ENOENT;
                return -ENOENT;
        if (IS_RDONLY(inode))
        if (IS_RDONLY(inode))
                return -EROFS;
                return -EROFS;
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
                return -EPERM;
                return -EPERM;
        if (user == (uid_t) -1)
        if (user == (uid_t) -1)
                user = inode->i_uid;
                user = inode->i_uid;
        if (group == (gid_t) -1)
        if (group == (gid_t) -1)
                group = inode->i_gid;
                group = inode->i_gid;
        newattrs.ia_mode = inode->i_mode;
        newattrs.ia_mode = inode->i_mode;
        newattrs.ia_uid = user;
        newattrs.ia_uid = user;
        newattrs.ia_gid = group;
        newattrs.ia_gid = group;
        newattrs.ia_valid =  ATTR_UID | ATTR_GID | ATTR_CTIME;
        newattrs.ia_valid =  ATTR_UID | ATTR_GID | ATTR_CTIME;
        /*
        /*
         * If the owner has been changed, remove the setuid bit
         * If the owner has been changed, remove the setuid bit
         */
         */
        if (inode->i_mode & S_ISUID) {
        if (inode->i_mode & S_ISUID) {
                newattrs.ia_mode &= ~S_ISUID;
                newattrs.ia_mode &= ~S_ISUID;
                newattrs.ia_valid |= ATTR_MODE;
                newattrs.ia_valid |= ATTR_MODE;
        }
        }
        /*
        /*
         * If the group has been changed, remove the setgid bit
         * If the group has been changed, remove the setgid bit
         *
         *
         * Don't remove the setgid bit if no group execute bit.
         * Don't remove the setgid bit if no group execute bit.
         * This is a file marked for mandatory locking.
         * This is a file marked for mandatory locking.
         */
         */
        if (((inode->i_mode & (S_ISGID | S_IXGRP)) == (S_ISGID | S_IXGRP))) {
        if (((inode->i_mode & (S_ISGID | S_IXGRP)) == (S_ISGID | S_IXGRP))) {
                newattrs.ia_mode &= ~S_ISGID;
                newattrs.ia_mode &= ~S_ISGID;
                newattrs.ia_valid |= ATTR_MODE;
                newattrs.ia_valid |= ATTR_MODE;
        }
        }
        inode->i_dirt = 1;
        inode->i_dirt = 1;
        if (inode->i_sb && inode->i_sb->dq_op) {
        if (inode->i_sb && inode->i_sb->dq_op) {
                inode->i_sb->dq_op->initialize(inode, -1);
                inode->i_sb->dq_op->initialize(inode, -1);
                if (inode->i_sb->dq_op->transfer(inode, &newattrs, 0))
                if (inode->i_sb->dq_op->transfer(inode, &newattrs, 0))
                        return -EDQUOT;
                        return -EDQUOT;
                error = notify_change(inode, &newattrs);
                error = notify_change(inode, &newattrs);
                if (error)
                if (error)
                        inode->i_sb->dq_op->transfer(inode, &newattrs, 1);
                        inode->i_sb->dq_op->transfer(inode, &newattrs, 1);
        } else
        } else
                error = notify_change(inode, &newattrs);
                error = notify_change(inode, &newattrs);
        return error;
        return error;
}
}
 
 
asmlinkage int sys_chown(const char * filename, uid_t user, gid_t group)
asmlinkage int sys_chown(const char * filename, uid_t user, gid_t group)
{
{
        struct inode * inode;
        struct inode * inode;
        int error;
        int error;
        struct iattr newattrs;
        struct iattr newattrs;
 
 
        error = lnamei(filename,&inode);
        error = lnamei(filename,&inode);
        if (error)
        if (error)
                return error;
                return error;
        if (IS_RDONLY(inode)) {
        if (IS_RDONLY(inode)) {
                iput(inode);
                iput(inode);
                return -EROFS;
                return -EROFS;
        }
        }
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode)) {
        if (IS_IMMUTABLE(inode) || IS_APPEND(inode)) {
                iput(inode);
                iput(inode);
                return -EPERM;
                return -EPERM;
        }
        }
        if (user == (uid_t) -1)
        if (user == (uid_t) -1)
                user = inode->i_uid;
                user = inode->i_uid;
        if (group == (gid_t) -1)
        if (group == (gid_t) -1)
                group = inode->i_gid;
                group = inode->i_gid;
        newattrs.ia_mode = inode->i_mode;
        newattrs.ia_mode = inode->i_mode;
        newattrs.ia_uid = user;
        newattrs.ia_uid = user;
        newattrs.ia_gid = group;
        newattrs.ia_gid = group;
        newattrs.ia_valid =  ATTR_UID | ATTR_GID | ATTR_CTIME;
        newattrs.ia_valid =  ATTR_UID | ATTR_GID | ATTR_CTIME;
        /*
        /*
         * If the owner has been changed, remove the setuid bit
         * If the owner has been changed, remove the setuid bit
         */
         */
        if (inode->i_mode & S_ISUID) {
        if (inode->i_mode & S_ISUID) {
                newattrs.ia_mode &= ~S_ISUID;
                newattrs.ia_mode &= ~S_ISUID;
                newattrs.ia_valid |= ATTR_MODE;
                newattrs.ia_valid |= ATTR_MODE;
        }
        }
        /*
        /*
         * If the group has been changed, remove the setgid bit
         * If the group has been changed, remove the setgid bit
         *
         *
         * Don't remove the setgid bit if no group execute bit.
         * Don't remove the setgid bit if no group execute bit.
         * This is a file marked for mandatory locking.
         * This is a file marked for mandatory locking.
         */
         */
        if (((inode->i_mode & (S_ISGID | S_IXGRP)) == (S_ISGID | S_IXGRP))) {
        if (((inode->i_mode & (S_ISGID | S_IXGRP)) == (S_ISGID | S_IXGRP))) {
                newattrs.ia_mode &= ~S_ISGID;
                newattrs.ia_mode &= ~S_ISGID;
                newattrs.ia_valid |= ATTR_MODE;
                newattrs.ia_valid |= ATTR_MODE;
        }
        }
        inode->i_dirt = 1;
        inode->i_dirt = 1;
        if (inode->i_sb->dq_op) {
        if (inode->i_sb->dq_op) {
                inode->i_sb->dq_op->initialize(inode, -1);
                inode->i_sb->dq_op->initialize(inode, -1);
                if (inode->i_sb->dq_op->transfer(inode, &newattrs, 0))
                if (inode->i_sb->dq_op->transfer(inode, &newattrs, 0))
                        return -EDQUOT;
                        return -EDQUOT;
                error = notify_change(inode, &newattrs);
                error = notify_change(inode, &newattrs);
                if (error)
                if (error)
                        inode->i_sb->dq_op->transfer(inode, &newattrs, 1);
                        inode->i_sb->dq_op->transfer(inode, &newattrs, 1);
        } else
        } else
                error = notify_change(inode, &newattrs);
                error = notify_change(inode, &newattrs);
        iput(inode);
        iput(inode);
        return(error);
        return(error);
}
}
 
 
/*
/*
 * Note that while the flag value (low two bits) for sys_open means:
 * Note that while the flag value (low two bits) for sys_open means:
 *      00 - read-only
 *      00 - read-only
 *      01 - write-only
 *      01 - write-only
 *      10 - read-write
 *      10 - read-write
 *      11 - special
 *      11 - special
 * it is changed into
 * it is changed into
 *      00 - no permissions needed
 *      00 - no permissions needed
 *      01 - read-permission
 *      01 - read-permission
 *      10 - write-permission
 *      10 - write-permission
 *      11 - read-write
 *      11 - read-write
 * for the internal routines (ie open_namei()/follow_link() etc). 00 is
 * for the internal routines (ie open_namei()/follow_link() etc). 00 is
 * used by symlinks.
 * used by symlinks.
 */
 */
static int do_open(const char * filename,int flags,int mode, int fd)
static int do_open(const char * filename,int flags,int mode, int fd)
{
{
        struct inode * inode;
        struct inode * inode;
        struct file * f;
        struct file * f;
        int flag,error;
        int flag,error;
 
 
        f = get_empty_filp();
        f = get_empty_filp();
 
 
        if (!f)
        if (!f)
                return -ENFILE;
                return -ENFILE;
        f->f_flags = flag = flags;
        f->f_flags = flag = flags;
        f->f_mode = (flag+1) & O_ACCMODE;
        f->f_mode = (flag+1) & O_ACCMODE;
        if (f->f_mode)
        if (f->f_mode)
                flag++;
                flag++;
        if (flag & O_TRUNC)
        if (flag & O_TRUNC)
                flag |= 2;
                flag |= 2;
        error = open_namei(filename,flag,mode,&inode,NULL);
        error = open_namei(filename,flag,mode,&inode,NULL);
        if (error)
        if (error)
                goto cleanup_file;
                goto cleanup_file;
        if (f->f_mode & FMODE_WRITE) {
        if (f->f_mode & FMODE_WRITE) {
                error = get_write_access(inode);
                error = get_write_access(inode);
                if (error)
                if (error)
                        goto cleanup_inode;
                        goto cleanup_inode;
        }
        }
 
 
        f->f_inode = inode;
        f->f_inode = inode;
        f->f_pos = 0;
        f->f_pos = 0;
        f->f_reada = 0;
        f->f_reada = 0;
        f->f_op = NULL;
        f->f_op = NULL;
        if (inode->i_op)
        if (inode->i_op)
                f->f_op = inode->i_op->default_file_ops;
                f->f_op = inode->i_op->default_file_ops;
        if (f->f_op && f->f_op->open) {
        if (f->f_op && f->f_op->open) {
                error = f->f_op->open(inode,f);
                error = f->f_op->open(inode,f);
                if (error)
                if (error)
                        goto cleanup_all;
                        goto cleanup_all;
        }
        }
        f->f_flags &= ~(O_CREAT | O_EXCL | O_NOCTTY | O_TRUNC);
        f->f_flags &= ~(O_CREAT | O_EXCL | O_NOCTTY | O_TRUNC);
 
 
        current->files->fd[fd] = f;
        current->files->fd[fd] = f;
        return 0;
        return 0;
 
 
cleanup_all:
cleanup_all:
        if (f->f_mode & FMODE_WRITE)
        if (f->f_mode & FMODE_WRITE)
                put_write_access(inode);
                put_write_access(inode);
cleanup_inode:
cleanup_inode:
        iput(inode);
        iput(inode);
cleanup_file:
cleanup_file:
        f->f_count--;
        f->f_count--;
        return error;
        return error;
}
}
 
 
/*
/*
 * Find a empty file descriptor entry, and mark it busy
 * Find a empty file descriptor entry, and mark it busy
 */
 */
int get_unused_fd(void)
int get_unused_fd(void)
{
{
        int fd;
        int fd;
        struct files_struct * files = current->files;
        struct files_struct * files = current->files;
 
 
        fd = find_first_zero_bit(&files->open_fds, NR_OPEN);
        fd = find_first_zero_bit(&files->open_fds, NR_OPEN);
        if (fd < current->rlim[RLIMIT_NOFILE].rlim_cur) {
        if (fd < current->rlim[RLIMIT_NOFILE].rlim_cur) {
                FD_SET(fd, &files->open_fds);
                FD_SET(fd, &files->open_fds);
                FD_CLR(fd, &files->close_on_exec);
                FD_CLR(fd, &files->close_on_exec);
                return fd;
                return fd;
        }
        }
        return -EMFILE;
        return -EMFILE;
}
}
 
 
inline void put_unused_fd(int fd)
inline void put_unused_fd(int fd)
{
{
        FD_CLR(fd, &current->files->open_fds);
        FD_CLR(fd, &current->files->open_fds);
}
}
 
 
asmlinkage int sys_open(const char * filename,int flags,int mode)
asmlinkage int sys_open(const char * filename,int flags,int mode)
{
{
        char * tmp;
        char * tmp;
        int fd, error;
        int fd, error;
 
 
        fd = get_unused_fd();
        fd = get_unused_fd();
        if (fd < 0)
        if (fd < 0)
                return fd;
                return fd;
        error = getname(filename, &tmp);
        error = getname(filename, &tmp);
        if (!error) {
        if (!error) {
                error = do_open(tmp,flags,mode, fd);
                error = do_open(tmp,flags,mode, fd);
                putname(tmp);
                putname(tmp);
 
 
                if (!error)
                if (!error)
                        return fd;
                        return fd;
        }
        }
 
 
        put_unused_fd(fd);
        put_unused_fd(fd);
        return error;
        return error;
}
}
 
 
#ifndef __alpha__
#ifndef __alpha__
 
 
/*
/*
 * For backward compatibility?  Maybe this should be moved
 * For backward compatibility?  Maybe this should be moved
 * into arch/i386 instead?
 * into arch/i386 instead?
 */
 */
asmlinkage int sys_creat(const char * pathname, int mode)
asmlinkage int sys_creat(const char * pathname, int mode)
{
{
        return sys_open(pathname, O_CREAT | O_WRONLY | O_TRUNC, mode);
        return sys_open(pathname, O_CREAT | O_WRONLY | O_TRUNC, mode);
}
}
 
 
#endif
#endif
 
 
void __fput(struct file *filp, struct inode *inode)
void __fput(struct file *filp, struct inode *inode)
{
{
        if (filp->f_op && filp->f_op->release)
        if (filp->f_op && filp->f_op->release)
                filp->f_op->release(inode,filp);
                filp->f_op->release(inode,filp);
        filp->f_inode = NULL;
        filp->f_inode = NULL;
        if (filp->f_mode & FMODE_WRITE)
        if (filp->f_mode & FMODE_WRITE)
                put_write_access(inode);
                put_write_access(inode);
        iput(inode);
        iput(inode);
}
}
 
 
int close_fp(struct file *filp)
int close_fp(struct file *filp)
{
{
        struct inode *inode;
        struct inode *inode;
 
 
        if (filp->f_count == 0) {
        if (filp->f_count == 0) {
                printk("VFS: Close: file count is 0\n");
                printk("VFS: Close: file count is 0\n");
                return 0;
                return 0;
        }
        }
        inode = filp->f_inode;
        inode = filp->f_inode;
        if (inode)
        if (inode)
                locks_remove_locks(current, filp);
                locks_remove_locks(current, filp);
        fput(filp, inode);
        fput(filp, inode);
        return 0;
        return 0;
}
}
 
 
asmlinkage int sys_close(unsigned int fd)
asmlinkage int sys_close(unsigned int fd)
{
{
        int error;
        int error;
        struct file * filp;
        struct file * filp;
        struct files_struct * files;
        struct files_struct * files;
 
 
        files = current->files;
        files = current->files;
        error = -EBADF;
        error = -EBADF;
        if (fd < NR_OPEN && (filp = files->fd[fd]) != NULL) {
        if (fd < NR_OPEN && (filp = files->fd[fd]) != NULL) {
                put_unused_fd(fd);
                put_unused_fd(fd);
                FD_CLR(fd, &files->close_on_exec);
                FD_CLR(fd, &files->close_on_exec);
                files->fd[fd] = NULL;
                files->fd[fd] = NULL;
                error = close_fp(filp);
                error = close_fp(filp);
        }
        }
        return error;
        return error;
}
}
 
 
/*
/*
 * This routine simulates a hangup on the tty, to arrange that users
 * This routine simulates a hangup on the tty, to arrange that users
 * are given clean terminals at login time.
 * are given clean terminals at login time.
 */
 */
asmlinkage int sys_vhangup(void)
asmlinkage int sys_vhangup(void)
{
{
        if (!suser())
        if (!suser())
                return -EPERM;
                return -EPERM;
        /* If there is a controlling tty, hang it up */
        /* If there is a controlling tty, hang it up */
        if (current->tty)
        if (current->tty)
                tty_vhangup(current->tty);
                tty_vhangup(current->tty);
        return 0;
        return 0;
}
}
 
 

powered by: WebSVN 2.1.0

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