utimes.c 5.6 KB
Newer Older
A
Alexey Dobriyan 已提交
1
#include <linux/compiler.h>
U
Ulrich Drepper 已提交
2
#include <linux/file.h>
A
Alexey Dobriyan 已提交
3 4
#include <linux/fs.h>
#include <linux/linkage.h>
5
#include <linux/mount.h>
A
Alexey Dobriyan 已提交
6
#include <linux/namei.h>
7
#include <linux/sched.h>
U
Ulrich Drepper 已提交
8
#include <linux/stat.h>
A
Alexey Dobriyan 已提交
9
#include <linux/utime.h>
10
#include <linux/syscalls.h>
A
Alexey Dobriyan 已提交
11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
#include <asm/uaccess.h>
#include <asm/unistd.h>

#ifdef __ARCH_WANT_SYS_UTIME

/*
 * sys_utime() can be implemented in user-level using sys_utimes().
 * Is this for backwards compatibility?  If so, why not move it
 * into the appropriate arch directory (for those architectures that
 * need it).
 */

/* If times==NULL, set access and modification to current time,
 * must be owner or have write permission.
 * Else, update from *times, must be owner or super user.
 */
U
Ulrich Drepper 已提交
27
asmlinkage long sys_utime(char __user *filename, struct utimbuf __user *times)
A
Alexey Dobriyan 已提交
28
{
U
Ulrich Drepper 已提交
29
	struct timespec tv[2];
A
Alexey Dobriyan 已提交
30 31

	if (times) {
U
Ulrich Drepper 已提交
32 33 34 35 36
		if (get_user(tv[0].tv_sec, &times->actime) ||
		    get_user(tv[1].tv_sec, &times->modtime))
			return -EFAULT;
		tv[0].tv_nsec = 0;
		tv[1].tv_nsec = 0;
A
Alexey Dobriyan 已提交
37
	}
U
Ulrich Drepper 已提交
38
	return do_utimes(AT_FDCWD, filename, times ? tv : NULL, 0);
A
Alexey Dobriyan 已提交
39 40 41 42
}

#endif

43 44
static bool nsec_valid(long nsec)
{
45
	if (nsec == UTIME_OMIT || nsec == UTIME_NOW)
46 47 48 49 50
		return true;

	return nsec >= 0 && nsec <= 999999999;
}

M
Miklos Szeredi 已提交
51
static int utimes_common(struct path *path, struct timespec *times)
A
Alexey Dobriyan 已提交
52 53 54
{
	int error;
	struct iattr newattrs;
M
Miklos Szeredi 已提交
55
	struct inode *inode = path->dentry->d_inode;
A
Alexey Dobriyan 已提交
56

M
Miklos Szeredi 已提交
57
	error = mnt_want_write(path->mnt);
58
	if (error)
M
Miklos Szeredi 已提交
59
		goto out;
A
Alexey Dobriyan 已提交
60

61 62 63 64
	if (times && times[0].tv_nsec == UTIME_NOW &&
		     times[1].tv_nsec == UTIME_NOW)
		times = NULL;

A
Alexey Dobriyan 已提交
65 66
	newattrs.ia_valid = ATTR_CTIME | ATTR_MTIME | ATTR_ATIME;
	if (times) {
U
Ulrich Drepper 已提交
67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
		if (times[0].tv_nsec == UTIME_OMIT)
			newattrs.ia_valid &= ~ATTR_ATIME;
		else if (times[0].tv_nsec != UTIME_NOW) {
			newattrs.ia_atime.tv_sec = times[0].tv_sec;
			newattrs.ia_atime.tv_nsec = times[0].tv_nsec;
			newattrs.ia_valid |= ATTR_ATIME_SET;
		}

		if (times[1].tv_nsec == UTIME_OMIT)
			newattrs.ia_valid &= ~ATTR_MTIME;
		else if (times[1].tv_nsec != UTIME_NOW) {
			newattrs.ia_mtime.tv_sec = times[1].tv_sec;
			newattrs.ia_mtime.tv_nsec = times[1].tv_nsec;
			newattrs.ia_valid |= ATTR_MTIME_SET;
		}
82
		/*
83 84 85
		 * Tell inode_change_ok(), that this is an explicit time
		 * update, even if neither ATTR_ATIME_SET nor ATTR_MTIME_SET
		 * were used.
86
		 */
87
		newattrs.ia_valid |= ATTR_TIMES_SET;
88 89 90 91 92 93
	} else {
		/*
		 * If times is NULL (or both times are UTIME_NOW),
		 * then we need to check permissions, because
		 * inode_change_ok() won't do it.
		 */
A
Alexey Dobriyan 已提交
94 95
		error = -EACCES;
                if (IS_IMMUTABLE(inode))
96
			goto mnt_drop_write_and_out;
A
Alexey Dobriyan 已提交
97

98
		if (!is_owner_or_cap(inode)) {
99 100 101
			error = permission(inode, MAY_WRITE, NULL);
			if (error)
				goto mnt_drop_write_and_out;
102
		}
A
Alexey Dobriyan 已提交
103 104
	}
	mutex_lock(&inode->i_mutex);
M
Miklos Szeredi 已提交
105
	error = notify_change(path->dentry, &newattrs);
A
Alexey Dobriyan 已提交
106
	mutex_unlock(&inode->i_mutex);
M
Miklos Szeredi 已提交
107

108
mnt_drop_write_and_out:
M
Miklos Szeredi 已提交
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165
	mnt_drop_write(path->mnt);
out:
	return error;
}

/*
 * do_utimes - change times on filename or file descriptor
 * @dfd: open file descriptor, -1 or AT_FDCWD
 * @filename: path name or NULL
 * @times: new times or NULL
 * @flags: zero or more flags (only AT_SYMLINK_NOFOLLOW for the moment)
 *
 * If filename is NULL and dfd refers to an open file, then operate on
 * the file.  Otherwise look up filename, possibly using dfd as a
 * starting point.
 *
 * If times==NULL, set access and modification to current time,
 * must be owner or have write permission.
 * Else, update from *times, must be owner or super user.
 */
long do_utimes(int dfd, char __user *filename, struct timespec *times, int flags)
{
	int error = -EINVAL;

	if (times && (!nsec_valid(times[0].tv_nsec) ||
		      !nsec_valid(times[1].tv_nsec))) {
		goto out;
	}

	if (flags & ~AT_SYMLINK_NOFOLLOW)
		goto out;

	if (filename == NULL && dfd != AT_FDCWD) {
		struct file *file;

		if (flags & AT_SYMLINK_NOFOLLOW)
			goto out;

		file = fget(dfd);
		error = -EBADF;
		if (!file)
			goto out;

		error = utimes_common(&file->f_path, times);
		fput(file);
	} else {
		struct nameidata nd;
		int lookup_flags = 0;

		if (!(flags & AT_SYMLINK_NOFOLLOW))
			lookup_flags |= LOOKUP_FOLLOW;

		error = __user_walk_fd(dfd, filename, lookup_flags, &nd);
		if (error)
			goto out;

		error = utimes_common(&nd.path, times);
J
Jan Blunck 已提交
166
		path_put(&nd.path);
M
Miklos Szeredi 已提交
167 168
	}

A
Alexey Dobriyan 已提交
169 170 171 172
out:
	return error;
}

U
Ulrich Drepper 已提交
173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189
asmlinkage long sys_utimensat(int dfd, char __user *filename, struct timespec __user *utimes, int flags)
{
	struct timespec tstimes[2];

	if (utimes) {
		if (copy_from_user(&tstimes, utimes, sizeof(tstimes)))
			return -EFAULT;

		/* Nothing to do, we must not even check the path.  */
		if (tstimes[0].tv_nsec == UTIME_OMIT &&
		    tstimes[1].tv_nsec == UTIME_OMIT)
			return 0;
	}

	return do_utimes(dfd, filename, utimes ? tstimes : NULL, flags);
}

A
Alexey Dobriyan 已提交
190 191 192
asmlinkage long sys_futimesat(int dfd, char __user *filename, struct timeval __user *utimes)
{
	struct timeval times[2];
U
Ulrich Drepper 已提交
193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212
	struct timespec tstimes[2];

	if (utimes) {
		if (copy_from_user(&times, utimes, sizeof(times)))
			return -EFAULT;

		/* This test is needed to catch all invalid values.  If we
		   would test only in do_utimes we would miss those invalid
		   values truncated by the multiplication with 1000.  Note
		   that we also catch UTIME_{NOW,OMIT} here which are only
		   valid for utimensat.  */
		if (times[0].tv_usec >= 1000000 || times[0].tv_usec < 0 ||
		    times[1].tv_usec >= 1000000 || times[1].tv_usec < 0)
			return -EINVAL;

		tstimes[0].tv_sec = times[0].tv_sec;
		tstimes[0].tv_nsec = 1000 * times[0].tv_usec;
		tstimes[1].tv_sec = times[1].tv_sec;
		tstimes[1].tv_nsec = 1000 * times[1].tv_usec;
	}
A
Alexey Dobriyan 已提交
213

U
Ulrich Drepper 已提交
214
	return do_utimes(dfd, filename, utimes ? tstimes : NULL, 0);
A
Alexey Dobriyan 已提交
215 216 217 218 219 220
}

asmlinkage long sys_utimes(char __user *filename, struct timeval __user *utimes)
{
	return sys_futimesat(AT_FDCWD, filename, utimes);
}