diff options
Diffstat (limited to 'fs')
| -rw-r--r-- | fs/proc/base.c | 94 | ||||
| -rw-r--r-- | fs/proc/namespaces.c | 4 | ||||
| -rw-r--r-- | fs/proc/uptime.c | 3 | ||||
| -rw-r--r-- | fs/timerfd.c | 3 | 
4 files changed, 104 insertions, 0 deletions
| diff --git a/fs/proc/base.c b/fs/proc/base.c index ebea9501afb8..5adc6390ac3a 100644 --- a/fs/proc/base.c +++ b/fs/proc/base.c @@ -94,6 +94,7 @@  #include <linux/sched/debug.h>  #include <linux/sched/stat.h>  #include <linux/posix-timers.h> +#include <linux/time_namespace.h>  #include <trace/events/oom.h>  #include "internal.h"  #include "fd.h" @@ -1533,6 +1534,96 @@ static const struct file_operations proc_pid_sched_autogroup_operations = {  #endif /* CONFIG_SCHED_AUTOGROUP */ +#ifdef CONFIG_TIME_NS +static int timens_offsets_show(struct seq_file *m, void *v) +{ +	struct task_struct *p; + +	p = get_proc_task(file_inode(m->file)); +	if (!p) +		return -ESRCH; +	proc_timens_show_offsets(p, m); + +	put_task_struct(p); + +	return 0; +} + +static ssize_t timens_offsets_write(struct file *file, const char __user *buf, +				    size_t count, loff_t *ppos) +{ +	struct inode *inode = file_inode(file); +	struct proc_timens_offset offsets[2]; +	char *kbuf = NULL, *pos, *next_line; +	struct task_struct *p; +	int ret, noffsets; + +	/* Only allow < page size writes at the beginning of the file */ +	if ((*ppos != 0) || (count >= PAGE_SIZE)) +		return -EINVAL; + +	/* Slurp in the user data */ +	kbuf = memdup_user_nul(buf, count); +	if (IS_ERR(kbuf)) +		return PTR_ERR(kbuf); + +	/* Parse the user data */ +	ret = -EINVAL; +	noffsets = 0; +	for (pos = kbuf; pos; pos = next_line) { +		struct proc_timens_offset *off = &offsets[noffsets]; +		int err; + +		/* Find the end of line and ensure we don't look past it */ +		next_line = strchr(pos, '\n'); +		if (next_line) { +			*next_line = '\0'; +			next_line++; +			if (*next_line == '\0') +				next_line = NULL; +		} + +		err = sscanf(pos, "%u %lld %lu", &off->clockid, +				&off->val.tv_sec, &off->val.tv_nsec); +		if (err != 3 || off->val.tv_nsec >= NSEC_PER_SEC) +			goto out; +		noffsets++; +		if (noffsets == ARRAY_SIZE(offsets)) { +			if (next_line) +				count = next_line - kbuf; +			break; +		} +	} + +	ret = -ESRCH; +	p = get_proc_task(inode); +	if (!p) +		goto out; +	ret = proc_timens_set_offset(file, p, offsets, noffsets); +	put_task_struct(p); +	if (ret) +		goto out; + +	ret = count; +out: +	kfree(kbuf); +	return ret; +} + +static int timens_offsets_open(struct inode *inode, struct file *filp) +{ +	return single_open(filp, timens_offsets_show, inode); +} + +static const struct file_operations proc_timens_offsets_operations = { +	.open		= timens_offsets_open, +	.read		= seq_read, +	.write		= timens_offsets_write, +	.llseek		= seq_lseek, +	.release	= single_release, +}; +#endif /* CONFIG_TIME_NS */ +  static ssize_t comm_write(struct file *file, const char __user *buf,  				size_t count, loff_t *offset)  { @@ -3016,6 +3107,9 @@ static const struct pid_entry tgid_base_stuff[] = {  #ifdef CONFIG_SCHED_AUTOGROUP  	REG("autogroup",  S_IRUGO|S_IWUSR, proc_pid_sched_autogroup_operations),  #endif +#ifdef CONFIG_TIME_NS +	REG("timens_offsets",  S_IRUGO|S_IWUSR, proc_timens_offsets_operations), +#endif  	REG("comm",      S_IRUGO|S_IWUSR, proc_pid_set_comm_operations),  #ifdef CONFIG_HAVE_ARCH_TRACEHOOK  	ONE("syscall",    S_IRUSR, proc_pid_syscall), diff --git a/fs/proc/namespaces.c b/fs/proc/namespaces.c index dd2b35f78b09..8b5c720fe5d7 100644 --- a/fs/proc/namespaces.c +++ b/fs/proc/namespaces.c @@ -33,6 +33,10 @@ static const struct proc_ns_operations *ns_entries[] = {  #ifdef CONFIG_CGROUPS  	&cgroupns_operations,  #endif +#ifdef CONFIG_TIME_NS +	&timens_operations, +	&timens_for_children_operations, +#endif  };  static const char *proc_ns_get_link(struct dentry *dentry, diff --git a/fs/proc/uptime.c b/fs/proc/uptime.c index a4c2791ab70b..5a1b228964fb 100644 --- a/fs/proc/uptime.c +++ b/fs/proc/uptime.c @@ -5,6 +5,7 @@  #include <linux/sched.h>  #include <linux/seq_file.h>  #include <linux/time.h> +#include <linux/time_namespace.h>  #include <linux/kernel_stat.h>  static int uptime_proc_show(struct seq_file *m, void *v) @@ -20,6 +21,8 @@ static int uptime_proc_show(struct seq_file *m, void *v)  		nsec += (__force u64) kcpustat_cpu(i).cpustat[CPUTIME_IDLE];  	ktime_get_boottime_ts64(&uptime); +	timens_add_boottime(&uptime); +  	idle.tv_sec = div_u64_rem(nsec, NSEC_PER_SEC, &rem);  	idle.tv_nsec = rem;  	seq_printf(m, "%lu.%02lu %lu.%02lu\n", diff --git a/fs/timerfd.c b/fs/timerfd.c index ac7f59a58f94..c5509d2448e3 100644 --- a/fs/timerfd.c +++ b/fs/timerfd.c @@ -26,6 +26,7 @@  #include <linux/syscalls.h>  #include <linux/compat.h>  #include <linux/rcupdate.h> +#include <linux/time_namespace.h>  struct timerfd_ctx {  	union { @@ -196,6 +197,8 @@ static int timerfd_setup(struct timerfd_ctx *ctx, int flags,  	}  	if (texp != 0) { +		if (flags & TFD_TIMER_ABSTIME) +			texp = timens_ktime_to_host(clockid, texp);  		if (isalarm(ctx)) {  			if (flags & TFD_TIMER_ABSTIME)  				alarm_start(&ctx->t.alarm, texp); | 

