2019-06-21 10:52:29 +01:00
|
|
|
// SPDX-License-Identifier: GPL-2.0
|
|
|
|
/*
|
|
|
|
* Generic userspace implementations of gettimeofday() and similar.
|
|
|
|
*/
|
2025-07-01 10:58:06 +02:00
|
|
|
#include <vdso/auxclock.h>
|
2019-06-21 10:52:29 +01:00
|
|
|
#include <vdso/datapage.h>
|
|
|
|
#include <vdso/helpers.h>
|
|
|
|
|
2025-02-04 13:05:37 +01:00
|
|
|
/* Bring in default accessors */
|
|
|
|
#include <vdso/vsyscall.h>
|
|
|
|
|
2024-03-25 08:40:06 +02:00
|
|
|
#ifndef vdso_calc_ns
|
2024-03-25 08:40:05 +02:00
|
|
|
|
|
|
|
#ifdef VDSO_DELTA_NOMASK
|
2024-04-09 09:26:39 +03:00
|
|
|
# define VDSO_DELTA_MASK(vd) ULLONG_MAX
|
2024-03-25 08:40:05 +02:00
|
|
|
#else
|
2024-03-25 08:40:06 +02:00
|
|
|
# define VDSO_DELTA_MASK(vd) (vd->mask)
|
|
|
|
#endif
|
|
|
|
|
2024-03-25 08:40:11 +02:00
|
|
|
#ifdef CONFIG_GENERIC_VDSO_OVERFLOW_PROTECT
|
2025-03-03 12:11:13 +01:00
|
|
|
static __always_inline bool vdso_delta_ok(const struct vdso_clock *vc, u64 delta)
|
2024-03-25 08:40:11 +02:00
|
|
|
{
|
2025-03-03 12:11:13 +01:00
|
|
|
return delta < vc->max_cycles;
|
2024-03-25 08:40:11 +02:00
|
|
|
}
|
|
|
|
#else
|
2025-03-03 12:11:13 +01:00
|
|
|
static __always_inline bool vdso_delta_ok(const struct vdso_clock *vc, u64 delta)
|
2024-03-25 08:40:11 +02:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2024-03-25 08:40:06 +02:00
|
|
|
#ifndef vdso_shift_ns
|
|
|
|
static __always_inline u64 vdso_shift_ns(u64 ns, u32 shift)
|
|
|
|
{
|
|
|
|
return ns >> shift;
|
|
|
|
}
|
2024-03-25 08:40:05 +02:00
|
|
|
#endif
|
|
|
|
|
2019-06-26 12:02:00 +02:00
|
|
|
/*
|
|
|
|
* Default implementation which works for all sane clocksources. That
|
|
|
|
* obviously excludes x86/TSC.
|
|
|
|
*/
|
2025-03-03 12:11:13 +01:00
|
|
|
static __always_inline u64 vdso_calc_ns(const struct vdso_clock *vc, u64 cycles, u64 base)
|
2019-06-26 12:02:00 +02:00
|
|
|
{
|
2025-03-03 12:11:13 +01:00
|
|
|
u64 delta = (cycles - vc->cycle_last) & VDSO_DELTA_MASK(vc);
|
2019-06-26 12:02:00 +02:00
|
|
|
|
2025-03-03 12:11:13 +01:00
|
|
|
if (likely(vdso_delta_ok(vc, delta)))
|
|
|
|
return vdso_shift_ns((delta * vc->mult) + base, vc->shift);
|
2024-03-25 08:40:11 +02:00
|
|
|
|
2025-03-03 12:11:13 +01:00
|
|
|
return mul_u64_u32_add_u64_shr(delta, vc->mult, base, vc->shift);
|
lib/vdso: Allow architectures to override the ns shift operation
On powerpc/32, GCC (8.1) generates pretty bad code for the ns >>= vd->shift
operation taking into account that the shift is always <= 32 and the upper
part of the result is likely to be zero. GCC makes reversed assumptions
considering the shift to be likely >= 32 and the upper part to be like not
zero.
unsigned long long shift(unsigned long long x, unsigned char s)
{
return x >> s;
}
results in:
00000018 <shift>:
18: 35 25 ff e0 addic. r9,r5,-32
1c: 41 80 00 10 blt 2c <shift+0x14>
20: 7c 64 4c 30 srw r4,r3,r9
24: 38 60 00 00 li r3,0
28: 4e 80 00 20 blr
2c: 54 69 08 3c rlwinm r9,r3,1,0,30
30: 21 45 00 1f subfic r10,r5,31
34: 7c 84 2c 30 srw r4,r4,r5
38: 7d 29 50 30 slw r9,r9,r10
3c: 7c 63 2c 30 srw r3,r3,r5
40: 7d 24 23 78 or r4,r9,r4
44: 4e 80 00 20 blr
Even when forcing the shift to be smaller than 32 with an &= 31, it still
considers the shift as likely >= 32.
Move the default shift implementation into an inline which can be redefined
in architecture code via a macro.
[ tglx: Made the shift argument u32 and removed the __arch prefix ]
Signed-off-by: Christophe Leroy <christophe.leroy@c-s.fr>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: Vincenzo Frascino <vincenzo.frascino@arm.com>
Reviewed-by: Vincenzo Frascino <vincenzo.frascino@arm.com>
Link: https://lore.kernel.org/r/b3d449de856982ed060a71e6ace8eeca4654e685.1580399657.git.christophe.leroy@c-s.fr
Link: https://lkml.kernel.org/r/20200207124403.857649978@linutronix.de
2020-02-07 13:39:03 +01:00
|
|
|
}
|
2024-03-25 08:40:06 +02:00
|
|
|
#endif /* vdso_calc_ns */
|
lib/vdso: Allow architectures to override the ns shift operation
On powerpc/32, GCC (8.1) generates pretty bad code for the ns >>= vd->shift
operation taking into account that the shift is always <= 32 and the upper
part of the result is likely to be zero. GCC makes reversed assumptions
considering the shift to be likely >= 32 and the upper part to be like not
zero.
unsigned long long shift(unsigned long long x, unsigned char s)
{
return x >> s;
}
results in:
00000018 <shift>:
18: 35 25 ff e0 addic. r9,r5,-32
1c: 41 80 00 10 blt 2c <shift+0x14>
20: 7c 64 4c 30 srw r4,r3,r9
24: 38 60 00 00 li r3,0
28: 4e 80 00 20 blr
2c: 54 69 08 3c rlwinm r9,r3,1,0,30
30: 21 45 00 1f subfic r10,r5,31
34: 7c 84 2c 30 srw r4,r4,r5
38: 7d 29 50 30 slw r9,r9,r10
3c: 7c 63 2c 30 srw r3,r3,r5
40: 7d 24 23 78 or r4,r9,r4
44: 4e 80 00 20 blr
Even when forcing the shift to be smaller than 32 with an &= 31, it still
considers the shift as likely >= 32.
Move the default shift implementation into an inline which can be redefined
in architecture code via a macro.
[ tglx: Made the shift argument u32 and removed the __arch prefix ]
Signed-off-by: Christophe Leroy <christophe.leroy@c-s.fr>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Tested-by: Vincenzo Frascino <vincenzo.frascino@arm.com>
Reviewed-by: Vincenzo Frascino <vincenzo.frascino@arm.com>
Link: https://lore.kernel.org/r/b3d449de856982ed060a71e6ace8eeca4654e685.1580399657.git.christophe.leroy@c-s.fr
Link: https://lkml.kernel.org/r/20200207124403.857649978@linutronix.de
2020-02-07 13:39:03 +01:00
|
|
|
|
2020-02-07 13:38:50 +01:00
|
|
|
#ifndef __arch_vdso_hres_capable
|
|
|
|
static inline bool __arch_vdso_hres_capable(void)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-02-07 13:39:02 +01:00
|
|
|
#ifndef vdso_clocksource_ok
|
2025-03-03 12:11:13 +01:00
|
|
|
static inline bool vdso_clocksource_ok(const struct vdso_clock *vc)
|
2020-02-07 13:39:02 +01:00
|
|
|
{
|
2025-03-03 12:11:13 +01:00
|
|
|
return vc->clock_mode != VDSO_CLOCKMODE_NONE;
|
2020-02-07 13:39:02 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2020-06-06 23:51:16 +02:00
|
|
|
#ifndef vdso_cycles_ok
|
|
|
|
static inline bool vdso_cycles_ok(u64 cycles)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2025-07-01 10:58:01 +02:00
|
|
|
static __always_inline bool vdso_clockid_valid(clockid_t clock)
|
|
|
|
{
|
|
|
|
/* Check for negative values or invalid clocks */
|
2025-07-01 10:58:06 +02:00
|
|
|
return likely((u32) clock <= CLOCK_AUX_LAST);
|
2025-07-01 10:58:01 +02:00
|
|
|
}
|
|
|
|
|
2025-07-01 10:58:02 +02:00
|
|
|
/*
|
|
|
|
* Must not be invoked within the sequence read section as a race inside
|
|
|
|
* that loop could result in __iter_div_u64_rem() being extremely slow.
|
|
|
|
*/
|
|
|
|
static __always_inline void vdso_set_timespec(struct __kernel_timespec *ts, u64 sec, u64 ns)
|
|
|
|
{
|
|
|
|
ts->tv_sec = sec + __iter_div_u64_rem(ns, NSEC_PER_SEC, &ns);
|
|
|
|
ts->tv_nsec = ns;
|
|
|
|
}
|
|
|
|
|
2025-07-01 10:58:03 +02:00
|
|
|
static __always_inline
|
|
|
|
bool vdso_get_timestamp(const struct vdso_time_data *vd, const struct vdso_clock *vc,
|
|
|
|
unsigned int clkidx, u64 *sec, u64 *ns)
|
|
|
|
{
|
|
|
|
const struct vdso_timestamp *vdso_ts = &vc->basetime[clkidx];
|
|
|
|
u64 cycles;
|
|
|
|
|
|
|
|
if (unlikely(!vdso_clocksource_ok(vc)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
cycles = __arch_get_hw_counter(vc->clock_mode, vd);
|
|
|
|
if (unlikely(!vdso_cycles_ok(cycles)))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
*ns = vdso_calc_ns(vc, cycles, vdso_ts->nsec);
|
|
|
|
*sec = vdso_ts->sec;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-11-12 01:27:09 +00:00
|
|
|
#ifdef CONFIG_TIME_NS
|
2025-02-04 13:05:37 +01:00
|
|
|
|
|
|
|
#ifdef CONFIG_GENERIC_VDSO_DATA_STORE
|
|
|
|
static __always_inline
|
|
|
|
const struct vdso_time_data *__arch_get_vdso_u_timens_data(const struct vdso_time_data *vd)
|
|
|
|
{
|
|
|
|
return (void *)vd + PAGE_SIZE;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_GENERIC_VDSO_DATA_STORE */
|
|
|
|
|
2025-03-03 12:11:10 +01:00
|
|
|
static __always_inline
|
2025-07-01 10:58:00 +02:00
|
|
|
bool do_hres_timens(const struct vdso_time_data *vdns, const struct vdso_clock *vcns,
|
|
|
|
clockid_t clk, struct __kernel_timespec *ts)
|
2019-11-12 01:27:09 +00:00
|
|
|
{
|
2025-03-03 12:11:10 +01:00
|
|
|
const struct vdso_time_data *vd = __arch_get_vdso_u_timens_data(vdns);
|
|
|
|
const struct timens_offset *offs = &vcns->offset[clk];
|
2025-03-03 12:11:21 +01:00
|
|
|
const struct vdso_clock *vc = vd->clock_data;
|
2019-11-12 01:27:09 +00:00
|
|
|
u32 seq;
|
|
|
|
s64 sec;
|
2025-07-01 10:58:03 +02:00
|
|
|
u64 ns;
|
2019-11-12 01:27:09 +00:00
|
|
|
|
|
|
|
if (clk != CLOCK_MONOTONIC_RAW)
|
2025-03-03 12:11:10 +01:00
|
|
|
vc = &vc[CS_HRES_COARSE];
|
2019-11-12 01:27:09 +00:00
|
|
|
else
|
2025-03-03 12:11:10 +01:00
|
|
|
vc = &vc[CS_RAW];
|
2019-11-12 01:27:09 +00:00
|
|
|
|
|
|
|
do {
|
2025-03-03 12:11:10 +01:00
|
|
|
seq = vdso_read_begin(vc);
|
2020-02-07 13:38:59 +01:00
|
|
|
|
2025-07-01 10:58:03 +02:00
|
|
|
if (!vdso_get_timestamp(vd, vc, clk, &sec, &ns))
|
2025-07-01 10:58:00 +02:00
|
|
|
return false;
|
2025-03-03 12:11:10 +01:00
|
|
|
} while (unlikely(vdso_read_retry(vc, seq)));
|
2019-11-12 01:27:09 +00:00
|
|
|
|
|
|
|
/* Add the namespace offset */
|
|
|
|
sec += offs->sec;
|
|
|
|
ns += offs->nsec;
|
|
|
|
|
2025-07-01 10:58:02 +02:00
|
|
|
vdso_set_timespec(ts, sec, ns);
|
2019-11-12 01:27:09 +00:00
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
return true;
|
2019-11-12 01:27:09 +00:00
|
|
|
}
|
|
|
|
#else
|
2021-03-31 16:48:45 +00:00
|
|
|
static __always_inline
|
2025-02-04 13:05:50 +01:00
|
|
|
const struct vdso_time_data *__arch_get_vdso_u_timens_data(const struct vdso_time_data *vd)
|
2019-11-12 01:27:09 +00:00
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2025-03-03 12:11:10 +01:00
|
|
|
static __always_inline
|
2025-07-01 10:58:00 +02:00
|
|
|
bool do_hres_timens(const struct vdso_time_data *vdns, const struct vdso_clock *vcns,
|
|
|
|
clockid_t clk, struct __kernel_timespec *ts)
|
2019-11-12 01:27:09 +00:00
|
|
|
{
|
2025-07-01 10:58:00 +02:00
|
|
|
return false;
|
2019-11-12 01:27:09 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2025-03-03 12:11:09 +01:00
|
|
|
static __always_inline
|
2025-07-01 10:58:00 +02:00
|
|
|
bool do_hres(const struct vdso_time_data *vd, const struct vdso_clock *vc,
|
|
|
|
clockid_t clk, struct __kernel_timespec *ts)
|
2019-06-21 10:52:29 +01:00
|
|
|
{
|
2025-07-01 10:58:03 +02:00
|
|
|
u64 sec, ns;
|
2019-06-21 10:52:29 +01:00
|
|
|
u32 seq;
|
|
|
|
|
2020-02-07 13:38:50 +01:00
|
|
|
/* Allows to compile the high resolution parts out */
|
|
|
|
if (!__arch_vdso_hres_capable())
|
2025-07-01 10:58:00 +02:00
|
|
|
return false;
|
2020-02-07 13:38:50 +01:00
|
|
|
|
2019-06-21 10:52:29 +01:00
|
|
|
do {
|
2019-11-12 01:27:09 +00:00
|
|
|
/*
|
2024-07-01 16:47:54 +02:00
|
|
|
* Open coded function vdso_read_begin() to handle
|
|
|
|
* VDSO_CLOCKMODE_TIMENS. Time namespace enabled tasks have a
|
2025-03-03 12:11:09 +01:00
|
|
|
* special VVAR page installed which has vc->seq set to 1 and
|
|
|
|
* vc->clock_mode set to VDSO_CLOCKMODE_TIMENS. For non time
|
2024-07-01 16:47:54 +02:00
|
|
|
* namespace affected tasks this does not affect performance
|
2025-03-03 12:11:09 +01:00
|
|
|
* because if vc->seq is odd, i.e. a concurrent update is in
|
|
|
|
* progress the extra check for vc->clock_mode is just a few
|
|
|
|
* extra instructions while spin waiting for vc->seq to become
|
2019-11-12 01:27:09 +00:00
|
|
|
* even again.
|
|
|
|
*/
|
2025-03-03 12:11:09 +01:00
|
|
|
while (unlikely((seq = READ_ONCE(vc->seq)) & 1)) {
|
2019-11-12 01:27:09 +00:00
|
|
|
if (IS_ENABLED(CONFIG_TIME_NS) &&
|
2025-03-03 12:11:09 +01:00
|
|
|
vc->clock_mode == VDSO_CLOCKMODE_TIMENS)
|
2025-03-03 12:11:10 +01:00
|
|
|
return do_hres_timens(vd, vc, clk, ts);
|
2019-11-12 01:27:09 +00:00
|
|
|
cpu_relax();
|
|
|
|
}
|
|
|
|
smp_rmb();
|
|
|
|
|
2025-07-01 10:58:03 +02:00
|
|
|
if (!vdso_get_timestamp(vd, vc, clk, &sec, &ns))
|
2025-07-01 10:58:00 +02:00
|
|
|
return false;
|
2025-03-03 12:11:09 +01:00
|
|
|
} while (unlikely(vdso_read_retry(vc, seq)));
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2025-07-01 10:58:02 +02:00
|
|
|
vdso_set_timespec(ts, sec, ns);
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
return true;
|
2019-06-21 10:52:29 +01:00
|
|
|
}
|
|
|
|
|
2019-11-12 01:27:09 +00:00
|
|
|
#ifdef CONFIG_TIME_NS
|
2025-03-03 12:11:12 +01:00
|
|
|
static __always_inline
|
2025-07-01 10:58:00 +02:00
|
|
|
bool do_coarse_timens(const struct vdso_time_data *vdns, const struct vdso_clock *vcns,
|
|
|
|
clockid_t clk, struct __kernel_timespec *ts)
|
2019-11-12 01:27:09 +00:00
|
|
|
{
|
2025-02-04 13:05:50 +01:00
|
|
|
const struct vdso_time_data *vd = __arch_get_vdso_u_timens_data(vdns);
|
2025-03-03 12:11:12 +01:00
|
|
|
const struct timens_offset *offs = &vcns->offset[clk];
|
2025-03-03 12:11:21 +01:00
|
|
|
const struct vdso_clock *vc = vd->clock_data;
|
2025-03-03 12:11:12 +01:00
|
|
|
const struct vdso_timestamp *vdso_ts;
|
2019-11-12 01:27:09 +00:00
|
|
|
u64 nsec;
|
|
|
|
s64 sec;
|
|
|
|
s32 seq;
|
|
|
|
|
2025-03-03 12:11:12 +01:00
|
|
|
vdso_ts = &vc->basetime[clk];
|
|
|
|
|
2019-11-12 01:27:09 +00:00
|
|
|
do {
|
2025-03-03 12:11:12 +01:00
|
|
|
seq = vdso_read_begin(vc);
|
2019-11-12 01:27:09 +00:00
|
|
|
sec = vdso_ts->sec;
|
|
|
|
nsec = vdso_ts->nsec;
|
2025-03-03 12:11:12 +01:00
|
|
|
} while (unlikely(vdso_read_retry(vc, seq)));
|
2019-11-12 01:27:09 +00:00
|
|
|
|
|
|
|
/* Add the namespace offset */
|
|
|
|
sec += offs->sec;
|
|
|
|
nsec += offs->nsec;
|
|
|
|
|
2025-07-01 10:58:02 +02:00
|
|
|
vdso_set_timespec(ts, sec, nsec);
|
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
return true;
|
2019-11-12 01:27:09 +00:00
|
|
|
}
|
|
|
|
#else
|
2025-03-03 12:11:12 +01:00
|
|
|
static __always_inline
|
2025-07-01 10:58:00 +02:00
|
|
|
bool do_coarse_timens(const struct vdso_time_data *vdns, const struct vdso_clock *vcns,
|
|
|
|
clockid_t clk, struct __kernel_timespec *ts)
|
2019-11-12 01:27:09 +00:00
|
|
|
{
|
2025-07-01 10:58:00 +02:00
|
|
|
return false;
|
2019-11-12 01:27:09 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2025-03-03 12:11:11 +01:00
|
|
|
static __always_inline
|
2025-07-01 10:58:00 +02:00
|
|
|
bool do_coarse(const struct vdso_time_data *vd, const struct vdso_clock *vc,
|
|
|
|
clockid_t clk, struct __kernel_timespec *ts)
|
2019-06-21 10:52:29 +01:00
|
|
|
{
|
2025-03-03 12:11:11 +01:00
|
|
|
const struct vdso_timestamp *vdso_ts = &vc->basetime[clk];
|
2019-06-21 10:52:29 +01:00
|
|
|
u32 seq;
|
|
|
|
|
|
|
|
do {
|
2019-11-12 01:27:09 +00:00
|
|
|
/*
|
2024-07-01 16:47:54 +02:00
|
|
|
* Open coded function vdso_read_begin() to handle
|
|
|
|
* VDSO_CLOCK_TIMENS. See comment in do_hres().
|
2019-11-12 01:27:09 +00:00
|
|
|
*/
|
2025-03-03 12:11:11 +01:00
|
|
|
while ((seq = READ_ONCE(vc->seq)) & 1) {
|
2019-11-12 01:27:09 +00:00
|
|
|
if (IS_ENABLED(CONFIG_TIME_NS) &&
|
2025-03-03 12:11:11 +01:00
|
|
|
vc->clock_mode == VDSO_CLOCKMODE_TIMENS)
|
2025-03-03 12:11:12 +01:00
|
|
|
return do_coarse_timens(vd, vc, clk, ts);
|
2019-11-12 01:27:09 +00:00
|
|
|
cpu_relax();
|
|
|
|
}
|
|
|
|
smp_rmb();
|
|
|
|
|
2019-06-21 10:52:29 +01:00
|
|
|
ts->tv_sec = vdso_ts->sec;
|
|
|
|
ts->tv_nsec = vdso_ts->nsec;
|
2025-03-03 12:11:11 +01:00
|
|
|
} while (unlikely(vdso_read_retry(vc, seq)));
|
2019-12-23 14:31:07 +00:00
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
return true;
|
2019-06-21 10:52:29 +01:00
|
|
|
}
|
|
|
|
|
2025-07-01 10:58:06 +02:00
|
|
|
static __always_inline
|
|
|
|
bool do_aux(const struct vdso_time_data *vd, clockid_t clock, struct __kernel_timespec *ts)
|
|
|
|
{
|
|
|
|
const struct vdso_clock *vc;
|
|
|
|
u32 seq, idx;
|
|
|
|
u64 sec, ns;
|
|
|
|
|
|
|
|
if (!IS_ENABLED(CONFIG_POSIX_AUX_CLOCKS))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
idx = clock - CLOCK_AUX;
|
|
|
|
vc = &vd->aux_clock_data[idx];
|
|
|
|
|
|
|
|
do {
|
|
|
|
/*
|
|
|
|
* Open coded function vdso_read_begin() to handle
|
|
|
|
* VDSO_CLOCK_TIMENS. See comment in do_hres().
|
|
|
|
*/
|
|
|
|
while ((seq = READ_ONCE(vc->seq)) & 1) {
|
|
|
|
if (IS_ENABLED(CONFIG_TIME_NS) && vc->clock_mode == VDSO_CLOCKMODE_TIMENS) {
|
|
|
|
vd = __arch_get_vdso_u_timens_data(vd);
|
|
|
|
vc = &vd->aux_clock_data[idx];
|
|
|
|
/* Re-read from the real time data page */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
cpu_relax();
|
|
|
|
}
|
|
|
|
smp_rmb();
|
|
|
|
|
|
|
|
/* Auxclock disabled? */
|
|
|
|
if (vc->clock_mode == VDSO_CLOCKMODE_NONE)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!vdso_get_timestamp(vd, vc, VDSO_BASE_AUX, &sec, &ns))
|
|
|
|
return false;
|
|
|
|
} while (unlikely(vdso_read_retry(vc, seq)));
|
|
|
|
|
|
|
|
vdso_set_timespec(ts, sec, ns);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
static __always_inline bool
|
2025-02-04 13:05:50 +01:00
|
|
|
__cvdso_clock_gettime_common(const struct vdso_time_data *vd, clockid_t clock,
|
2020-02-07 13:39:04 +01:00
|
|
|
struct __kernel_timespec *ts)
|
2019-06-21 10:52:29 +01:00
|
|
|
{
|
2025-03-03 12:11:21 +01:00
|
|
|
const struct vdso_clock *vc = vd->clock_data;
|
2019-06-21 10:52:29 +01:00
|
|
|
u32 msk;
|
|
|
|
|
2025-07-01 10:58:01 +02:00
|
|
|
if (!vdso_clockid_valid(clock))
|
2025-07-01 10:58:00 +02:00
|
|
|
return false;
|
2019-06-21 10:52:29 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Convert the clockid to a bitmask and use it to check which
|
|
|
|
* clocks are handled in the VDSO directly.
|
|
|
|
*/
|
|
|
|
msk = 1U << clock;
|
2019-12-23 14:31:07 +00:00
|
|
|
if (likely(msk & VDSO_HRES))
|
2025-03-03 12:11:08 +01:00
|
|
|
vc = &vc[CS_HRES_COARSE];
|
2019-12-23 14:31:07 +00:00
|
|
|
else if (msk & VDSO_COARSE)
|
2025-03-03 12:11:11 +01:00
|
|
|
return do_coarse(vd, &vc[CS_HRES_COARSE], clock, ts);
|
2019-12-23 14:31:07 +00:00
|
|
|
else if (msk & VDSO_RAW)
|
2025-03-03 12:11:08 +01:00
|
|
|
vc = &vc[CS_RAW];
|
2025-07-01 10:58:06 +02:00
|
|
|
else if (msk & VDSO_AUX)
|
|
|
|
return do_aux(vd, clock, ts);
|
2019-11-12 01:26:51 +00:00
|
|
|
else
|
2025-07-01 10:58:00 +02:00
|
|
|
return false;
|
2019-12-23 14:31:07 +00:00
|
|
|
|
2025-03-03 12:11:09 +01:00
|
|
|
return do_hres(vd, vc, clock, ts);
|
2019-07-28 15:12:53 +02:00
|
|
|
}
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2019-07-28 15:12:53 +02:00
|
|
|
static __maybe_unused int
|
2025-02-04 13:05:50 +01:00
|
|
|
__cvdso_clock_gettime_data(const struct vdso_time_data *vd, clockid_t clock,
|
2020-02-07 13:39:04 +01:00
|
|
|
struct __kernel_timespec *ts)
|
2019-07-28 15:12:53 +02:00
|
|
|
{
|
2025-07-01 10:58:00 +02:00
|
|
|
bool ok;
|
2019-07-28 15:12:53 +02:00
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
ok = __cvdso_clock_gettime_common(vd, clock, ts);
|
|
|
|
|
|
|
|
if (unlikely(!ok))
|
2019-07-28 15:12:53 +02:00
|
|
|
return clock_gettime_fallback(clock, ts);
|
|
|
|
return 0;
|
2019-06-21 10:52:29 +01:00
|
|
|
}
|
|
|
|
|
2020-02-07 13:39:04 +01:00
|
|
|
static __maybe_unused int
|
|
|
|
__cvdso_clock_gettime(clockid_t clock, struct __kernel_timespec *ts)
|
|
|
|
{
|
2025-02-04 13:05:37 +01:00
|
|
|
return __cvdso_clock_gettime_data(__arch_get_vdso_u_time_data(), clock, ts);
|
2020-02-07 13:39:04 +01:00
|
|
|
}
|
|
|
|
|
2019-08-30 14:58:56 +01:00
|
|
|
#ifdef BUILD_VDSO32
|
2019-06-21 10:52:29 +01:00
|
|
|
static __maybe_unused int
|
2025-02-04 13:05:50 +01:00
|
|
|
__cvdso_clock_gettime32_data(const struct vdso_time_data *vd, clockid_t clock,
|
2020-02-07 13:39:04 +01:00
|
|
|
struct old_timespec32 *res)
|
2019-06-21 10:52:29 +01:00
|
|
|
{
|
|
|
|
struct __kernel_timespec ts;
|
2025-07-01 10:58:00 +02:00
|
|
|
bool ok;
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
ok = __cvdso_clock_gettime_common(vd, clock, &ts);
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
if (unlikely(!ok))
|
2019-07-30 11:38:50 +02:00
|
|
|
return clock_gettime32_fallback(clock, res);
|
2019-07-28 15:12:53 +02:00
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
/* For ok == true */
|
2019-08-30 14:58:59 +01:00
|
|
|
res->tv_sec = ts.tv_sec;
|
|
|
|
res->tv_nsec = ts.tv_nsec;
|
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
return 0;
|
2019-06-21 10:52:29 +01:00
|
|
|
}
|
2020-02-07 13:39:04 +01:00
|
|
|
|
|
|
|
static __maybe_unused int
|
|
|
|
__cvdso_clock_gettime32(clockid_t clock, struct old_timespec32 *res)
|
|
|
|
{
|
2025-02-04 13:05:37 +01:00
|
|
|
return __cvdso_clock_gettime32_data(__arch_get_vdso_u_time_data(), clock, res);
|
2020-02-07 13:39:04 +01:00
|
|
|
}
|
2019-08-30 14:58:56 +01:00
|
|
|
#endif /* BUILD_VDSO32 */
|
2019-06-21 10:52:29 +01:00
|
|
|
|
|
|
|
static __maybe_unused int
|
2025-02-04 13:05:50 +01:00
|
|
|
__cvdso_gettimeofday_data(const struct vdso_time_data *vd,
|
2020-02-07 13:39:04 +01:00
|
|
|
struct __kernel_old_timeval *tv, struct timezone *tz)
|
2019-06-21 10:52:29 +01:00
|
|
|
{
|
2025-03-03 12:11:21 +01:00
|
|
|
const struct vdso_clock *vc = vd->clock_data;
|
2019-06-21 10:52:29 +01:00
|
|
|
|
|
|
|
if (likely(tv != NULL)) {
|
|
|
|
struct __kernel_timespec ts;
|
|
|
|
|
2025-07-01 10:58:00 +02:00
|
|
|
if (!do_hres(vd, &vc[CS_HRES_COARSE], CLOCK_REALTIME, &ts))
|
2019-06-21 10:52:29 +01:00
|
|
|
return gettimeofday_fallback(tv, tz);
|
|
|
|
|
|
|
|
tv->tv_sec = ts.tv_sec;
|
|
|
|
tv->tv_usec = (u32)ts.tv_nsec / NSEC_PER_USEC;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (unlikely(tz != NULL)) {
|
2019-11-12 01:27:09 +00:00
|
|
|
if (IS_ENABLED(CONFIG_TIME_NS) &&
|
2025-03-03 12:11:08 +01:00
|
|
|
vc->clock_mode == VDSO_CLOCKMODE_TIMENS)
|
2025-02-04 13:05:50 +01:00
|
|
|
vd = __arch_get_vdso_u_timens_data(vd);
|
2019-11-12 01:27:09 +00:00
|
|
|
|
2019-06-21 10:52:29 +01:00
|
|
|
tz->tz_minuteswest = vd[CS_HRES_COARSE].tz_minuteswest;
|
|
|
|
tz->tz_dsttime = vd[CS_HRES_COARSE].tz_dsttime;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2020-02-07 13:39:04 +01:00
|
|
|
static __maybe_unused int
|
|
|
|
__cvdso_gettimeofday(struct __kernel_old_timeval *tv, struct timezone *tz)
|
|
|
|
{
|
2025-02-04 13:05:37 +01:00
|
|
|
return __cvdso_gettimeofday_data(__arch_get_vdso_u_time_data(), tv, tz);
|
2020-02-07 13:39:04 +01:00
|
|
|
}
|
|
|
|
|
2019-06-21 10:52:29 +01:00
|
|
|
#ifdef VDSO_HAS_TIME
|
2020-02-07 13:39:04 +01:00
|
|
|
static __maybe_unused __kernel_old_time_t
|
2025-02-04 13:05:50 +01:00
|
|
|
__cvdso_time_data(const struct vdso_time_data *vd, __kernel_old_time_t *time)
|
2019-06-21 10:52:29 +01:00
|
|
|
{
|
2025-03-03 12:11:21 +01:00
|
|
|
const struct vdso_clock *vc = vd->clock_data;
|
2019-11-12 01:27:09 +00:00
|
|
|
__kernel_old_time_t t;
|
|
|
|
|
2020-02-07 13:39:01 +01:00
|
|
|
if (IS_ENABLED(CONFIG_TIME_NS) &&
|
2025-03-03 12:11:08 +01:00
|
|
|
vc->clock_mode == VDSO_CLOCKMODE_TIMENS) {
|
2025-02-04 13:05:50 +01:00
|
|
|
vd = __arch_get_vdso_u_timens_data(vd);
|
2025-03-03 12:11:21 +01:00
|
|
|
vc = vd->clock_data;
|
2025-03-03 12:11:08 +01:00
|
|
|
}
|
2019-11-12 01:27:09 +00:00
|
|
|
|
2025-03-03 12:11:08 +01:00
|
|
|
t = READ_ONCE(vc[CS_HRES_COARSE].basetime[CLOCK_REALTIME].sec);
|
2019-06-21 10:52:29 +01:00
|
|
|
|
|
|
|
if (time)
|
|
|
|
*time = t;
|
|
|
|
|
|
|
|
return t;
|
|
|
|
}
|
2020-02-07 13:39:04 +01:00
|
|
|
|
|
|
|
static __maybe_unused __kernel_old_time_t __cvdso_time(__kernel_old_time_t *time)
|
|
|
|
{
|
2025-02-04 13:05:37 +01:00
|
|
|
return __cvdso_time_data(__arch_get_vdso_u_time_data(), time);
|
2020-02-07 13:39:04 +01:00
|
|
|
}
|
2019-06-21 10:52:29 +01:00
|
|
|
#endif /* VDSO_HAS_TIME */
|
|
|
|
|
|
|
|
#ifdef VDSO_HAS_CLOCK_GETRES
|
|
|
|
static __maybe_unused
|
2025-07-01 10:57:59 +02:00
|
|
|
bool __cvdso_clock_getres_common(const struct vdso_time_data *vd, clockid_t clock,
|
|
|
|
struct __kernel_timespec *res)
|
2019-06-21 10:52:29 +01:00
|
|
|
{
|
2025-03-03 12:11:21 +01:00
|
|
|
const struct vdso_clock *vc = vd->clock_data;
|
2019-06-21 10:52:29 +01:00
|
|
|
u32 msk;
|
2019-07-28 15:12:53 +02:00
|
|
|
u64 ns;
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2025-07-01 10:58:01 +02:00
|
|
|
if (!vdso_clockid_valid(clock))
|
2025-07-01 10:57:59 +02:00
|
|
|
return false;
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2020-02-07 13:39:01 +01:00
|
|
|
if (IS_ENABLED(CONFIG_TIME_NS) &&
|
2025-03-03 12:11:08 +01:00
|
|
|
vc->clock_mode == VDSO_CLOCKMODE_TIMENS)
|
2025-02-04 13:05:50 +01:00
|
|
|
vd = __arch_get_vdso_u_timens_data(vd);
|
2019-11-12 01:27:09 +00:00
|
|
|
|
2019-06-21 10:52:29 +01:00
|
|
|
/*
|
|
|
|
* Convert the clockid to a bitmask and use it to check which
|
|
|
|
* clocks are handled in the VDSO directly.
|
|
|
|
*/
|
|
|
|
msk = 1U << clock;
|
2019-12-23 14:31:09 +00:00
|
|
|
if (msk & (VDSO_HRES | VDSO_RAW)) {
|
2019-06-21 10:52:29 +01:00
|
|
|
/*
|
|
|
|
* Preserves the behaviour of posix_get_hrtimer_res().
|
|
|
|
*/
|
2025-03-03 12:11:21 +01:00
|
|
|
ns = READ_ONCE(vd->hrtimer_res);
|
2019-06-21 10:52:29 +01:00
|
|
|
} else if (msk & VDSO_COARSE) {
|
|
|
|
/*
|
|
|
|
* Preserves the behaviour of posix_get_coarse_res().
|
|
|
|
*/
|
|
|
|
ns = LOW_RES_NSEC;
|
2025-07-01 10:58:06 +02:00
|
|
|
} else if (msk & VDSO_AUX) {
|
|
|
|
ns = aux_clock_resolution_ns();
|
2019-06-21 10:52:29 +01:00
|
|
|
} else {
|
2025-07-01 10:57:59 +02:00
|
|
|
return false;
|
2019-06-21 10:52:29 +01:00
|
|
|
}
|
|
|
|
|
2019-10-21 12:07:15 +02:00
|
|
|
if (likely(res)) {
|
|
|
|
res->tv_sec = 0;
|
|
|
|
res->tv_nsec = ns;
|
|
|
|
}
|
2025-07-01 10:57:59 +02:00
|
|
|
return true;
|
2019-07-28 15:12:53 +02:00
|
|
|
}
|
|
|
|
|
2019-11-28 11:17:19 +00:00
|
|
|
static __maybe_unused
|
2025-02-04 13:05:50 +01:00
|
|
|
int __cvdso_clock_getres_data(const struct vdso_time_data *vd, clockid_t clock,
|
2020-02-07 13:39:04 +01:00
|
|
|
struct __kernel_timespec *res)
|
2019-07-28 15:12:53 +02:00
|
|
|
{
|
2025-07-01 10:57:59 +02:00
|
|
|
bool ok;
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2025-07-01 10:57:59 +02:00
|
|
|
ok = __cvdso_clock_getres_common(vd, clock, res);
|
|
|
|
|
|
|
|
if (unlikely(!ok))
|
2019-07-28 15:12:53 +02:00
|
|
|
return clock_getres_fallback(clock, res);
|
|
|
|
return 0;
|
2019-06-21 10:52:29 +01:00
|
|
|
}
|
|
|
|
|
2020-02-07 13:39:04 +01:00
|
|
|
static __maybe_unused
|
|
|
|
int __cvdso_clock_getres(clockid_t clock, struct __kernel_timespec *res)
|
|
|
|
{
|
2025-02-04 13:05:37 +01:00
|
|
|
return __cvdso_clock_getres_data(__arch_get_vdso_u_time_data(), clock, res);
|
2020-02-07 13:39:04 +01:00
|
|
|
}
|
|
|
|
|
2019-08-30 14:58:56 +01:00
|
|
|
#ifdef BUILD_VDSO32
|
2019-06-21 10:52:29 +01:00
|
|
|
static __maybe_unused int
|
2025-02-04 13:05:50 +01:00
|
|
|
__cvdso_clock_getres_time32_data(const struct vdso_time_data *vd, clockid_t clock,
|
2020-02-07 13:39:04 +01:00
|
|
|
struct old_timespec32 *res)
|
2019-06-21 10:52:29 +01:00
|
|
|
{
|
|
|
|
struct __kernel_timespec ts;
|
2025-07-01 10:57:59 +02:00
|
|
|
bool ok;
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2025-07-01 10:57:59 +02:00
|
|
|
ok = __cvdso_clock_getres_common(vd, clock, &ts);
|
2019-07-30 11:38:50 +02:00
|
|
|
|
2025-07-01 10:57:59 +02:00
|
|
|
if (unlikely(!ok))
|
2019-07-30 11:38:50 +02:00
|
|
|
return clock_getres32_fallback(clock, res);
|
2019-06-21 10:52:29 +01:00
|
|
|
|
2019-08-30 14:58:59 +01:00
|
|
|
if (likely(res)) {
|
2019-06-21 10:52:29 +01:00
|
|
|
res->tv_sec = ts.tv_sec;
|
|
|
|
res->tv_nsec = ts.tv_nsec;
|
|
|
|
}
|
2025-07-01 10:57:59 +02:00
|
|
|
return 0;
|
2019-06-21 10:52:29 +01:00
|
|
|
}
|
2020-02-07 13:39:04 +01:00
|
|
|
|
|
|
|
static __maybe_unused int
|
|
|
|
__cvdso_clock_getres_time32(clockid_t clock, struct old_timespec32 *res)
|
|
|
|
{
|
2025-02-04 13:05:37 +01:00
|
|
|
return __cvdso_clock_getres_time32_data(__arch_get_vdso_u_time_data(),
|
2020-02-07 13:39:04 +01:00
|
|
|
clock, res);
|
|
|
|
}
|
2019-08-30 14:58:56 +01:00
|
|
|
#endif /* BUILD_VDSO32 */
|
2019-06-21 10:52:29 +01:00
|
|
|
#endif /* VDSO_HAS_CLOCK_GETRES */
|