4  *  Copyright (C) 1991, 1992  Linus Torvalds
 
   5  *  changes by Thomas Schoebel-Theuer
 
   8 #include <linux/module.h>
 
   9 #include <linux/time.h>
 
  11 #include <linux/string.h>
 
  12 #include <linux/capability.h>
 
  13 #include <linux/fsnotify.h>
 
  14 #include <linux/fcntl.h>
 
  15 #include <linux/quotaops.h>
 
  16 #include <linux/security.h>
 
  18 /* Taken over from the old code... */
 
  20 /* POSIX UID/GID verification for setting inode attributes. */
 
  21 int inode_change_ok(struct inode *inode, struct iattr *attr)
 
  24         unsigned int ia_valid = attr->ia_valid;
 
  26         /* If force is set do it anyway. */
 
  27         if (ia_valid & ATTR_FORCE)
 
  30         /* Make sure a caller can chown. */
 
  31         if ((ia_valid & ATTR_UID) &&
 
  32             (current->fsuid != inode->i_uid ||
 
  33              attr->ia_uid != inode->i_uid) && !capable(CAP_CHOWN))
 
  36         /* Make sure caller can chgrp. */
 
  37         if ((ia_valid & ATTR_GID) &&
 
  38             (current->fsuid != inode->i_uid ||
 
  39             (!in_group_p(attr->ia_gid) && attr->ia_gid != inode->i_gid)) &&
 
  43         /* Make sure a caller can chmod. */
 
  44         if (ia_valid & ATTR_MODE) {
 
  45                 if (!is_owner_or_cap(inode))
 
  47                 /* Also check the setgid bit! */
 
  48                 if (!in_group_p((ia_valid & ATTR_GID) ? attr->ia_gid :
 
  49                                 inode->i_gid) && !capable(CAP_FSETID))
 
  50                         attr->ia_mode &= ~S_ISGID;
 
  53         /* Check for setting the inode time. */
 
  54         if (ia_valid & (ATTR_MTIME_SET | ATTR_ATIME_SET | ATTR_TIMES_SET)) {
 
  55                 if (!is_owner_or_cap(inode))
 
  64 EXPORT_SYMBOL(inode_change_ok);
 
  66 int inode_setattr(struct inode * inode, struct iattr * attr)
 
  68         unsigned int ia_valid = attr->ia_valid;
 
  70         if (ia_valid & ATTR_SIZE &&
 
  71             attr->ia_size != i_size_read(inode)) {
 
  72                 int error = vmtruncate(inode, attr->ia_size);
 
  77         if (ia_valid & ATTR_UID)
 
  78                 inode->i_uid = attr->ia_uid;
 
  79         if (ia_valid & ATTR_GID)
 
  80                 inode->i_gid = attr->ia_gid;
 
  81         if (ia_valid & ATTR_ATIME)
 
  82                 inode->i_atime = timespec_trunc(attr->ia_atime,
 
  83                                                 inode->i_sb->s_time_gran);
 
  84         if (ia_valid & ATTR_MTIME)
 
  85                 inode->i_mtime = timespec_trunc(attr->ia_mtime,
 
  86                                                 inode->i_sb->s_time_gran);
 
  87         if (ia_valid & ATTR_CTIME)
 
  88                 inode->i_ctime = timespec_trunc(attr->ia_ctime,
 
  89                                                 inode->i_sb->s_time_gran);
 
  90         if (ia_valid & ATTR_MODE) {
 
  91                 umode_t mode = attr->ia_mode;
 
  93                 if (!in_group_p(inode->i_gid) && !capable(CAP_FSETID))
 
  97         mark_inode_dirty(inode);
 
 101 EXPORT_SYMBOL(inode_setattr);
 
 103 int notify_change(struct dentry * dentry, struct iattr * attr)
 
 105         struct inode *inode = dentry->d_inode;
 
 106         mode_t mode = inode->i_mode;
 
 109         unsigned int ia_valid = attr->ia_valid;
 
 111         if (ia_valid & (ATTR_MODE | ATTR_UID | ATTR_GID | ATTR_TIMES_SET)) {
 
 112                 if (IS_IMMUTABLE(inode) || IS_APPEND(inode))
 
 116         now = current_fs_time(inode->i_sb);
 
 118         attr->ia_ctime = now;
 
 119         if (!(ia_valid & ATTR_ATIME_SET))
 
 120                 attr->ia_atime = now;
 
 121         if (!(ia_valid & ATTR_MTIME_SET))
 
 122                 attr->ia_mtime = now;
 
 123         if (ia_valid & ATTR_KILL_PRIV) {
 
 124                 attr->ia_valid &= ~ATTR_KILL_PRIV;
 
 125                 ia_valid &= ~ATTR_KILL_PRIV;
 
 126                 error = security_inode_need_killpriv(dentry);
 
 128                         error = security_inode_killpriv(dentry);
 
 134          * We now pass ATTR_KILL_S*ID to the lower level setattr function so
 
 135          * that the function has the ability to reinterpret a mode change
 
 136          * that's due to these bits. This adds an implicit restriction that
 
 137          * no function will ever call notify_change with both ATTR_MODE and
 
 138          * ATTR_KILL_S*ID set.
 
 140         if ((ia_valid & (ATTR_KILL_SUID|ATTR_KILL_SGID)) &&
 
 141             (ia_valid & ATTR_MODE))
 
 144         if (ia_valid & ATTR_KILL_SUID) {
 
 145                 if (mode & S_ISUID) {
 
 146                         ia_valid = attr->ia_valid |= ATTR_MODE;
 
 147                         attr->ia_mode = (inode->i_mode & ~S_ISUID);
 
 150         if (ia_valid & ATTR_KILL_SGID) {
 
 151                 if ((mode & (S_ISGID | S_IXGRP)) == (S_ISGID | S_IXGRP)) {
 
 152                         if (!(ia_valid & ATTR_MODE)) {
 
 153                                 ia_valid = attr->ia_valid |= ATTR_MODE;
 
 154                                 attr->ia_mode = inode->i_mode;
 
 156                         attr->ia_mode &= ~S_ISGID;
 
 159         if (!(attr->ia_valid & ~(ATTR_KILL_SUID | ATTR_KILL_SGID)))
 
 162         if (ia_valid & ATTR_SIZE)
 
 163                 down_write(&dentry->d_inode->i_alloc_sem);
 
 165         if (inode->i_op && inode->i_op->setattr) {
 
 166                 error = security_inode_setattr(dentry, attr);
 
 168                         error = inode->i_op->setattr(dentry, attr);
 
 170                 error = inode_change_ok(inode, attr);
 
 172                         error = security_inode_setattr(dentry, attr);
 
 174                         if ((ia_valid & ATTR_UID && attr->ia_uid != inode->i_uid) ||
 
 175                             (ia_valid & ATTR_GID && attr->ia_gid != inode->i_gid))
 
 176                                 error = DQUOT_TRANSFER(inode, attr) ? -EDQUOT : 0;
 
 178                                 error = inode_setattr(inode, attr);
 
 182         if (ia_valid & ATTR_SIZE)
 
 183                 up_write(&dentry->d_inode->i_alloc_sem);
 
 186                 fsnotify_change(dentry, ia_valid);
 
 191 EXPORT_SYMBOL(notify_change);