mirror of
https://github.com/Fishwaldo/linux-bl808.git
synced 2025-06-17 20:25:19 +00:00
fs/proc: Introduce /proc/pid/timens_offsets
API to set time namespace offsets for children processes, i.e.: echo "$clockid $offset_sec $offset_nsec" > /proc/self/timens_offsets Co-developed-by: Dmitry Safonov <dima@arista.com> Signed-off-by: Andrei Vagin <avagin@gmail.com> Signed-off-by: Dmitry Safonov <dima@arista.com> Signed-off-by: Thomas Gleixner <tglx@linutronix.de> Link: https://lore.kernel.org/r/20191112012724.250792-28-dima@arista.com
This commit is contained in:
parent
70ddf65184
commit
04a8682a71
3 changed files with 205 additions and 0 deletions
|
@ -8,6 +8,7 @@
|
|||
#include <linux/user_namespace.h>
|
||||
#include <linux/sched/signal.h>
|
||||
#include <linux/sched/task.h>
|
||||
#include <linux/seq_file.h>
|
||||
#include <linux/proc_ns.h>
|
||||
#include <linux/export.h>
|
||||
#include <linux/time.h>
|
||||
|
@ -334,6 +335,106 @@ static struct user_namespace *timens_owner(struct ns_common *ns)
|
|||
return to_time_ns(ns)->user_ns;
|
||||
}
|
||||
|
||||
static void show_offset(struct seq_file *m, int clockid, struct timespec64 *ts)
|
||||
{
|
||||
seq_printf(m, "%d %lld %ld\n", clockid, ts->tv_sec, ts->tv_nsec);
|
||||
}
|
||||
|
||||
void proc_timens_show_offsets(struct task_struct *p, struct seq_file *m)
|
||||
{
|
||||
struct ns_common *ns;
|
||||
struct time_namespace *time_ns;
|
||||
|
||||
ns = timens_for_children_get(p);
|
||||
if (!ns)
|
||||
return;
|
||||
time_ns = to_time_ns(ns);
|
||||
|
||||
show_offset(m, CLOCK_MONOTONIC, &time_ns->offsets.monotonic);
|
||||
show_offset(m, CLOCK_BOOTTIME, &time_ns->offsets.boottime);
|
||||
put_time_ns(time_ns);
|
||||
}
|
||||
|
||||
int proc_timens_set_offset(struct file *file, struct task_struct *p,
|
||||
struct proc_timens_offset *offsets, int noffsets)
|
||||
{
|
||||
struct ns_common *ns;
|
||||
struct time_namespace *time_ns;
|
||||
struct timespec64 tp;
|
||||
int i, err;
|
||||
|
||||
ns = timens_for_children_get(p);
|
||||
if (!ns)
|
||||
return -ESRCH;
|
||||
time_ns = to_time_ns(ns);
|
||||
|
||||
if (!file_ns_capable(file, time_ns->user_ns, CAP_SYS_TIME)) {
|
||||
put_time_ns(time_ns);
|
||||
return -EPERM;
|
||||
}
|
||||
|
||||
for (i = 0; i < noffsets; i++) {
|
||||
struct proc_timens_offset *off = &offsets[i];
|
||||
|
||||
switch (off->clockid) {
|
||||
case CLOCK_MONOTONIC:
|
||||
ktime_get_ts64(&tp);
|
||||
break;
|
||||
case CLOCK_BOOTTIME:
|
||||
ktime_get_boottime_ts64(&tp);
|
||||
break;
|
||||
default:
|
||||
err = -EINVAL;
|
||||
goto out;
|
||||
}
|
||||
|
||||
err = -ERANGE;
|
||||
|
||||
if (off->val.tv_sec > KTIME_SEC_MAX ||
|
||||
off->val.tv_sec < -KTIME_SEC_MAX)
|
||||
goto out;
|
||||
|
||||
tp = timespec64_add(tp, off->val);
|
||||
/*
|
||||
* KTIME_SEC_MAX is divided by 2 to be sure that KTIME_MAX is
|
||||
* still unreachable.
|
||||
*/
|
||||
if (tp.tv_sec < 0 || tp.tv_sec > KTIME_SEC_MAX / 2)
|
||||
goto out;
|
||||
}
|
||||
|
||||
mutex_lock(&offset_lock);
|
||||
if (time_ns->frozen_offsets) {
|
||||
err = -EACCES;
|
||||
goto out_unlock;
|
||||
}
|
||||
|
||||
err = 0;
|
||||
/* Don't report errors after this line */
|
||||
for (i = 0; i < noffsets; i++) {
|
||||
struct proc_timens_offset *off = &offsets[i];
|
||||
struct timespec64 *offset = NULL;
|
||||
|
||||
switch (off->clockid) {
|
||||
case CLOCK_MONOTONIC:
|
||||
offset = &time_ns->offsets.monotonic;
|
||||
break;
|
||||
case CLOCK_BOOTTIME:
|
||||
offset = &time_ns->offsets.boottime;
|
||||
break;
|
||||
}
|
||||
|
||||
*offset = off->val;
|
||||
}
|
||||
|
||||
out_unlock:
|
||||
mutex_unlock(&offset_lock);
|
||||
out:
|
||||
put_time_ns(time_ns);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
const struct proc_ns_operations timens_operations = {
|
||||
.name = "time",
|
||||
.type = CLONE_NEWTIME,
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue