Whamcloud - gitweb
- ext2_obd.c --- fix the bugs in read/write for Linux 2.4.3
[fs/lustre-release.git] / lustre / obdfs / file.c
1 /*
2  *  linux/fs/ext2/file.c
3  *
4  * This code is issued under the GNU General Public License.
5  * See the file COPYING in this distribution
6  *
7  * Copyright (C) 1992, 1993, 1994, 1995
8  * Remy Card (card@masi.ibp.fr)
9  * Laboratoire MASI - Institut Blaise Pascal
10  * Universite Pierre et Marie Curie (Paris VI)
11  *
12  *  from
13  *
14  *  linux/fs/minix/file.c
15  *
16  *  Copyright (C) 1991, 1992  Linus Torvalds
17  *
18  *  ext2 fs regular file handling primitives
19  *
20  *  64-bit file support on 64-bit platforms by Jakub Jelinek
21  *      (jj@sunsite.ms.mff.cuni.cz)
22  */
23
24 #include <asm/uaccess.h>
25 #include <asm/system.h>
26
27 #include <linux/errno.h>
28 #include <linux/fs.h>
29 #include <linux/fcntl.h>
30 #include <linux/sched.h>
31 #include <linux/stat.h>
32 #include <linux/locks.h>
33 #include <linux/mm.h>
34 #include <linux/pagemap.h>
35 #include <linux/smp_lock.h>
36
37 #include <linux/obd_support.h>
38 #include <linux/obdfs.h>
39
40 static inline void obdfs_remove_suid(struct inode *inode)
41 {
42         unsigned int mode;
43
44         /* set S_IGID if S_IXGRP is set, and always set S_ISUID */
45         mode = (inode->i_mode & S_IXGRP)*(S_ISGID/S_IXGRP) | S_ISUID;
46
47         /* was any of the uid bits set? */
48         mode &= inode->i_mode;
49         if (mode && !capable(CAP_FSETID)) {
50                 inode->i_mode &= ~mode;
51                 mark_inode_dirty(inode);
52         }
53 }
54
55 /*
56  * Write to a file (through the page cache).
57  */
58 static ssize_t
59 obdfs_file_write(struct file *file, const char *buf, size_t count, loff_t *ppos)
60 {
61         ssize_t retval;
62         CDEBUG(D_INFO, "Writing inode %ld, %d bytes, offset %ld\n",
63                file->f_dentry->d_inode->i_ino, count, (long)*ppos);
64
65         retval = generic_file_write(file, buf, count, ppos);
66         CDEBUG(D_INFO, "Wrote %d\n", retval);
67         if (retval > 0) {
68                 struct inode *inode = file->f_dentry->d_inode;
69                 obdfs_remove_suid(inode);
70                 inode->i_ctime = inode->i_mtime = CURRENT_TIME;
71                 mark_inode_dirty(inode);
72         }
73         EXIT;
74         return retval;
75 }
76
77 struct file_operations obdfs_file_operations = {
78         read: generic_file_read,      /* read */
79         write: obdfs_file_write,       /* write  */
80         mmap: generic_file_mmap,      /* mmap */
81 };
82
83 extern int obdfs_notify_change(struct dentry *de, struct iattr *attr);
84 struct inode_operations obdfs_file_inode_operations = {
85         create: obdfs_create,
86         lookup: obdfs_lookup,
87         link: obdfs_link,
88         unlink: obdfs_unlink,
89         symlink: obdfs_symlink,
90         mkdir: obdfs_mkdir,
91         rmdir: obdfs_rmdir,
92         mknod: obdfs_mknod,
93         rename: obdfs_rename,
94         truncate: obdfs_truncate,
95         setattr: obdfs_notify_change
96 };
97