diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2012-06-01 13:34:35 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2012-06-01 13:34:35 -0400 |
commit | 1193755ac6328ad240ba987e6ec41d5e8baf0680 (patch) | |
tree | 40bf847d7e3ebaa57b107151d14e6cd1d280cc6d /mm/filemap.c | |
parent | 4edebed86690eb8db9af3ab85baf4a34e73266cc (diff) | |
parent | 0ef97dcfce4179a2eba046b855ee2f91d6f1b414 (diff) |
Merge branch 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/viro/vfs
Pull vfs changes from Al Viro.
"A lot of misc stuff. The obvious groups:
* Miklos' atomic_open series; kills the damn abuse of
->d_revalidate() by NFS, which was the major stumbling block for
all work in that area.
* ripping security_file_mmap() and dealing with deadlocks in the
area; sanitizing the neighborhood of vm_mmap()/vm_munmap() in
general.
* ->encode_fh() switched to saner API; insane fake dentry in
mm/cleancache.c gone.
* assorted annotations in fs (endianness, __user)
* parts of Artem's ->s_dirty work (jff2 and reiserfs parts)
* ->update_time() work from Josef.
* other bits and pieces all over the place.
Normally it would've been in two or three pull requests, but
signal.git stuff had eaten a lot of time during this cycle ;-/"
Fix up trivial conflicts in Documentation/filesystems/vfs.txt (the
'truncate_range' inode method was removed by the VM changes, the VFS
update adds an 'update_time()' method), and in fs/btrfs/ulist.[ch] (due
to sparse fix added twice, with other changes nearby).
* 'for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/viro/vfs: (95 commits)
nfs: don't open in ->d_revalidate
vfs: retry last component if opening stale dentry
vfs: nameidata_to_filp(): don't throw away file on error
vfs: nameidata_to_filp(): inline __dentry_open()
vfs: do_dentry_open(): don't put filp
vfs: split __dentry_open()
vfs: do_last() common post lookup
vfs: do_last(): add audit_inode before open
vfs: do_last(): only return EISDIR for O_CREAT
vfs: do_last(): check LOOKUP_DIRECTORY
vfs: do_last(): make ENOENT exit RCU safe
vfs: make follow_link check RCU safe
vfs: do_last(): use inode variable
vfs: do_last(): inline walk_component()
vfs: do_last(): make exit RCU safe
vfs: split do_lookup()
Btrfs: move over to use ->update_time
fs: introduce inode operation ->update_time
reiserfs: get rid of resierfs_sync_super
reiserfs: mark the superblock as dirty a bit later
...
Diffstat (limited to 'mm/filemap.c')
-rw-r--r-- | mm/filemap.c | 69 |
1 files changed, 3 insertions, 66 deletions
diff --git a/mm/filemap.c b/mm/filemap.c index 64b48f934b89..a4a5260b0279 100644 --- a/mm/filemap.c +++ b/mm/filemap.c | |||
@@ -1899,71 +1899,6 @@ struct page *read_cache_page(struct address_space *mapping, | |||
1899 | } | 1899 | } |
1900 | EXPORT_SYMBOL(read_cache_page); | 1900 | EXPORT_SYMBOL(read_cache_page); |
1901 | 1901 | ||
1902 | /* | ||
1903 | * The logic we want is | ||
1904 | * | ||
1905 | * if suid or (sgid and xgrp) | ||
1906 | * remove privs | ||
1907 | */ | ||
1908 | int should_remove_suid(struct dentry *dentry) | ||
1909 | { | ||
1910 | umode_t mode = dentry->d_inode->i_mode; | ||
1911 | int kill = 0; | ||
1912 | |||
1913 | /* suid always must be killed */ | ||
1914 | if (unlikely(mode & S_ISUID)) | ||
1915 | kill = ATTR_KILL_SUID; | ||
1916 | |||
1917 | /* | ||
1918 | * sgid without any exec bits is just a mandatory locking mark; leave | ||
1919 | * it alone. If some exec bits are set, it's a real sgid; kill it. | ||
1920 | */ | ||
1921 | if (unlikely((mode & S_ISGID) && (mode & S_IXGRP))) | ||
1922 | kill |= ATTR_KILL_SGID; | ||
1923 | |||
1924 | if (unlikely(kill && !capable(CAP_FSETID) && S_ISREG(mode))) | ||
1925 | return kill; | ||
1926 | |||
1927 | return 0; | ||
1928 | } | ||
1929 | EXPORT_SYMBOL(should_remove_suid); | ||
1930 | |||
1931 | static int __remove_suid(struct dentry *dentry, int kill) | ||
1932 | { | ||
1933 | struct iattr newattrs; | ||
1934 | |||
1935 | newattrs.ia_valid = ATTR_FORCE | kill; | ||
1936 | return notify_change(dentry, &newattrs); | ||
1937 | } | ||
1938 | |||
1939 | int file_remove_suid(struct file *file) | ||
1940 | { | ||
1941 | struct dentry *dentry = file->f_path.dentry; | ||
1942 | struct inode *inode = dentry->d_inode; | ||
1943 | int killsuid; | ||
1944 | int killpriv; | ||
1945 | int error = 0; | ||
1946 | |||
1947 | /* Fast path for nothing security related */ | ||
1948 | if (IS_NOSEC(inode)) | ||
1949 | return 0; | ||
1950 | |||
1951 | killsuid = should_remove_suid(dentry); | ||
1952 | killpriv = security_inode_need_killpriv(dentry); | ||
1953 | |||
1954 | if (killpriv < 0) | ||
1955 | return killpriv; | ||
1956 | if (killpriv) | ||
1957 | error = security_inode_killpriv(dentry); | ||
1958 | if (!error && killsuid) | ||
1959 | error = __remove_suid(dentry, killsuid); | ||
1960 | if (!error && (inode->i_sb->s_flags & MS_NOSEC)) | ||
1961 | inode->i_flags |= S_NOSEC; | ||
1962 | |||
1963 | return error; | ||
1964 | } | ||
1965 | EXPORT_SYMBOL(file_remove_suid); | ||
1966 | |||
1967 | static size_t __iovec_copy_from_user_inatomic(char *vaddr, | 1902 | static size_t __iovec_copy_from_user_inatomic(char *vaddr, |
1968 | const struct iovec *iov, size_t base, size_t bytes) | 1903 | const struct iovec *iov, size_t base, size_t bytes) |
1969 | { | 1904 | { |
@@ -2489,7 +2424,9 @@ ssize_t __generic_file_aio_write(struct kiocb *iocb, const struct iovec *iov, | |||
2489 | if (err) | 2424 | if (err) |
2490 | goto out; | 2425 | goto out; |
2491 | 2426 | ||
2492 | file_update_time(file); | 2427 | err = file_update_time(file); |
2428 | if (err) | ||
2429 | goto out; | ||
2493 | 2430 | ||
2494 | /* coalesce the iovecs and go direct-to-BIO for O_DIRECT */ | 2431 | /* coalesce the iovecs and go direct-to-BIO for O_DIRECT */ |
2495 | if (unlikely(file->f_flags & O_DIRECT)) { | 2432 | if (unlikely(file->f_flags & O_DIRECT)) { |