2019-06-03 05:44:50 +00:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
2012-03-05 11:49:31 +00:00
|
|
|
/*
|
2019-04-15 09:49:34 +00:00
|
|
|
* VDSO implementations.
|
2012-03-05 11:49:31 +00:00
|
|
|
*
|
|
|
|
* Copyright (C) 2012 ARM Limited
|
|
|
|
*
|
|
|
|
* Author: Will Deacon <will.deacon@arm.com>
|
|
|
|
*/
|
|
|
|
|
2016-08-15 06:45:46 +00:00
|
|
|
#include <linux/cache.h>
|
2012-03-05 11:49:31 +00:00
|
|
|
#include <linux/clocksource.h>
|
|
|
|
#include <linux/elf.h>
|
|
|
|
#include <linux/err.h>
|
|
|
|
#include <linux/errno.h>
|
|
|
|
#include <linux/gfp.h>
|
2016-08-15 06:45:46 +00:00
|
|
|
#include <linux/kernel.h>
|
2012-03-05 11:49:31 +00:00
|
|
|
#include <linux/mm.h>
|
|
|
|
#include <linux/sched.h>
|
|
|
|
#include <linux/signal.h>
|
|
|
|
#include <linux/slab.h>
|
2012-10-16 10:44:53 +00:00
|
|
|
#include <linux/timekeeper_internal.h>
|
2012-03-05 11:49:31 +00:00
|
|
|
#include <linux/vmalloc.h>
|
|
|
|
|
|
|
|
#include <asm/cacheflush.h>
|
|
|
|
#include <asm/signal32.h>
|
|
|
|
#include <asm/vdso.h>
|
|
|
|
#include <asm/vdso_datapage.h>
|
|
|
|
|
2017-06-06 04:52:30 +00:00
|
|
|
extern char vdso_start[], vdso_end[];
|
2016-08-15 06:45:46 +00:00
|
|
|
static unsigned long vdso_pages __ro_after_init;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* The vDSO data page.
|
|
|
|
*/
|
|
|
|
static union {
|
|
|
|
struct vdso_data data;
|
|
|
|
u8 page[PAGE_SIZE];
|
|
|
|
} vdso_data_store __page_aligned_data;
|
|
|
|
struct vdso_data *vdso_data = &vdso_data_store.data;
|
|
|
|
|
|
|
|
#ifdef CONFIG_COMPAT
|
|
|
|
/*
|
|
|
|
* Create and map the vectors page for AArch32 tasks.
|
|
|
|
*/
|
2019-04-15 09:49:34 +00:00
|
|
|
#define C_VECTORS 0
|
|
|
|
#define C_SIGPAGE 1
|
|
|
|
#define C_PAGES (C_SIGPAGE + 1)
|
|
|
|
static struct page *aarch32_vdso_pages[C_PAGES] __ro_after_init;
|
|
|
|
static const struct vm_special_mapping aarch32_vdso_spec[C_PAGES] = {
|
|
|
|
{
|
|
|
|
.name = "[vectors]", /* ABI */
|
|
|
|
.pages = &aarch32_vdso_pages[C_VECTORS],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "[sigpage]", /* ABI */
|
|
|
|
.pages = &aarch32_vdso_pages[C_SIGPAGE],
|
|
|
|
},
|
|
|
|
};
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:36 +00:00
|
|
|
static int aarch32_alloc_kuser_vdso_page(void)
|
2012-03-05 11:49:31 +00:00
|
|
|
{
|
|
|
|
extern char __kuser_helper_start[], __kuser_helper_end[];
|
|
|
|
int kuser_sz = __kuser_helper_end - __kuser_helper_start;
|
2019-04-15 09:49:36 +00:00
|
|
|
unsigned long vdso_page;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:37 +00:00
|
|
|
if (!IS_ENABLED(CONFIG_KUSER_HELPERS))
|
|
|
|
return 0;
|
|
|
|
|
2019-04-15 09:49:36 +00:00
|
|
|
vdso_page = get_zeroed_page(GFP_ATOMIC);
|
|
|
|
if (!vdso_page)
|
2019-04-15 09:49:34 +00:00
|
|
|
return -ENOMEM;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:36 +00:00
|
|
|
memcpy((void *)(vdso_page + 0x1000 - kuser_sz), __kuser_helper_start,
|
2019-04-15 09:49:34 +00:00
|
|
|
kuser_sz);
|
2019-04-15 09:49:36 +00:00
|
|
|
aarch32_vdso_pages[C_VECTORS] = virt_to_page(vdso_page);
|
|
|
|
flush_dcache_page(aarch32_vdso_pages[C_VECTORS]);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int __init aarch32_alloc_vdso_pages(void)
|
|
|
|
{
|
|
|
|
extern char __aarch32_sigret_code_start[], __aarch32_sigret_code_end[];
|
|
|
|
int sigret_sz = __aarch32_sigret_code_end - __aarch32_sigret_code_start;
|
|
|
|
unsigned long sigpage;
|
|
|
|
int ret;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:36 +00:00
|
|
|
sigpage = get_zeroed_page(GFP_ATOMIC);
|
|
|
|
if (!sigpage)
|
|
|
|
return -ENOMEM;
|
2019-04-15 09:49:34 +00:00
|
|
|
|
2019-04-15 09:49:36 +00:00
|
|
|
memcpy((void *)sigpage, __aarch32_sigret_code_start, sigret_sz);
|
|
|
|
aarch32_vdso_pages[C_SIGPAGE] = virt_to_page(sigpage);
|
|
|
|
flush_dcache_page(aarch32_vdso_pages[C_SIGPAGE]);
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:36 +00:00
|
|
|
ret = aarch32_alloc_kuser_vdso_page();
|
|
|
|
if (ret)
|
|
|
|
free_page(sigpage);
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:36 +00:00
|
|
|
return ret;
|
2012-03-05 11:49:31 +00:00
|
|
|
}
|
2019-04-15 09:49:34 +00:00
|
|
|
arch_initcall(aarch32_alloc_vdso_pages);
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:34 +00:00
|
|
|
static int aarch32_kuser_helpers_setup(struct mm_struct *mm)
|
2012-03-05 11:49:31 +00:00
|
|
|
{
|
2019-04-15 09:49:34 +00:00
|
|
|
void *ret;
|
2014-07-09 18:22:12 +00:00
|
|
|
|
2019-04-15 09:49:37 +00:00
|
|
|
if (!IS_ENABLED(CONFIG_KUSER_HELPERS))
|
|
|
|
return 0;
|
|
|
|
|
2019-04-15 09:49:34 +00:00
|
|
|
/*
|
|
|
|
* Avoid VM_MAYWRITE for compatibility with arch/arm/, where it's
|
|
|
|
* not safe to CoW the page containing the CPU exception vectors.
|
|
|
|
*/
|
|
|
|
ret = _install_special_mapping(mm, AARCH32_VECTORS_BASE, PAGE_SIZE,
|
|
|
|
VM_READ | VM_EXEC |
|
|
|
|
VM_MAYREAD | VM_MAYEXEC,
|
|
|
|
&aarch32_vdso_spec[C_VECTORS]);
|
|
|
|
|
|
|
|
return PTR_ERR_OR_ZERO(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int aarch32_sigreturn_setup(struct mm_struct *mm)
|
|
|
|
{
|
|
|
|
unsigned long addr;
|
2014-07-09 18:22:12 +00:00
|
|
|
void *ret;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:34 +00:00
|
|
|
addr = get_unmapped_area(NULL, 0, PAGE_SIZE, 0, 0);
|
|
|
|
if (IS_ERR_VALUE(addr)) {
|
|
|
|
ret = ERR_PTR(addr);
|
|
|
|
goto out;
|
|
|
|
}
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:34 +00:00
|
|
|
/*
|
|
|
|
* VM_MAYWRITE is required to allow gdb to Copy-on-Write and
|
|
|
|
* set breakpoints.
|
|
|
|
*/
|
2014-07-09 18:22:12 +00:00
|
|
|
ret = _install_special_mapping(mm, addr, PAGE_SIZE,
|
2019-04-15 09:49:34 +00:00
|
|
|
VM_READ | VM_EXEC | VM_MAYREAD |
|
|
|
|
VM_MAYWRITE | VM_MAYEXEC,
|
|
|
|
&aarch32_vdso_spec[C_SIGPAGE]);
|
|
|
|
if (IS_ERR(ret))
|
|
|
|
goto out;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:34 +00:00
|
|
|
mm->context.vdso = (void *)addr;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-15 09:49:34 +00:00
|
|
|
out:
|
2014-07-09 18:22:12 +00:00
|
|
|
return PTR_ERR_OR_ZERO(ret);
|
2012-03-05 11:49:31 +00:00
|
|
|
}
|
2019-04-15 09:49:34 +00:00
|
|
|
|
|
|
|
int aarch32_setup_additional_pages(struct linux_binprm *bprm, int uses_interp)
|
|
|
|
{
|
|
|
|
struct mm_struct *mm = current->mm;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (down_write_killable(&mm->mmap_sem))
|
|
|
|
return -EINTR;
|
|
|
|
|
|
|
|
ret = aarch32_kuser_helpers_setup(mm);
|
|
|
|
if (ret)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
ret = aarch32_sigreturn_setup(mm);
|
|
|
|
|
|
|
|
out:
|
|
|
|
up_write(&mm->mmap_sem);
|
|
|
|
return ret;
|
|
|
|
}
|
2012-03-05 11:49:31 +00:00
|
|
|
#endif /* CONFIG_COMPAT */
|
|
|
|
|
2017-07-26 17:07:37 +00:00
|
|
|
static int vdso_mremap(const struct vm_special_mapping *sm,
|
|
|
|
struct vm_area_struct *new_vma)
|
|
|
|
{
|
|
|
|
unsigned long new_size = new_vma->vm_end - new_vma->vm_start;
|
|
|
|
unsigned long vdso_size = vdso_end - vdso_start;
|
|
|
|
|
|
|
|
if (vdso_size != new_size)
|
|
|
|
return -EINVAL;
|
|
|
|
|
|
|
|
current->mm->context.vdso = (void *)new_vma->vm_start;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-08-15 06:45:46 +00:00
|
|
|
static struct vm_special_mapping vdso_spec[2] __ro_after_init = {
|
|
|
|
{
|
|
|
|
.name = "[vvar]",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.name = "[vdso]",
|
2017-07-26 17:07:37 +00:00
|
|
|
.mremap = vdso_mremap,
|
2016-08-15 06:45:46 +00:00
|
|
|
},
|
|
|
|
};
|
2014-07-09 18:22:12 +00:00
|
|
|
|
2012-03-05 11:49:31 +00:00
|
|
|
static int __init vdso_init(void)
|
|
|
|
{
|
2014-02-11 22:28:42 +00:00
|
|
|
int i;
|
2016-08-15 06:45:46 +00:00
|
|
|
struct page **vdso_pagelist;
|
2017-01-10 21:35:49 +00:00
|
|
|
unsigned long pfn;
|
2014-02-11 22:28:42 +00:00
|
|
|
|
2017-06-06 04:52:30 +00:00
|
|
|
if (memcmp(vdso_start, "\177ELF", 4)) {
|
2014-02-11 22:28:42 +00:00
|
|
|
pr_err("vDSO is not a valid ELF object!\n");
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2017-06-06 04:52:30 +00:00
|
|
|
vdso_pages = (vdso_end - vdso_start) >> PAGE_SHIFT;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
|
|
|
/* Allocate the vDSO pagelist, plus a page for the data. */
|
2014-02-11 22:28:42 +00:00
|
|
|
vdso_pagelist = kcalloc(vdso_pages + 1, sizeof(struct page *),
|
2012-03-05 11:49:31 +00:00
|
|
|
GFP_KERNEL);
|
2014-02-11 22:28:42 +00:00
|
|
|
if (vdso_pagelist == NULL)
|
2012-03-05 11:49:31 +00:00
|
|
|
return -ENOMEM;
|
|
|
|
|
2014-07-09 18:22:13 +00:00
|
|
|
/* Grab the vDSO data page. */
|
2017-01-10 21:35:49 +00:00
|
|
|
vdso_pagelist[0] = phys_to_page(__pa_symbol(vdso_data));
|
|
|
|
|
2014-07-09 18:22:13 +00:00
|
|
|
|
2012-03-05 11:49:31 +00:00
|
|
|
/* Grab the vDSO code pages. */
|
2017-06-06 04:52:30 +00:00
|
|
|
pfn = sym_to_pfn(vdso_start);
|
2017-01-10 21:35:49 +00:00
|
|
|
|
2014-02-11 22:28:42 +00:00
|
|
|
for (i = 0; i < vdso_pages; i++)
|
2017-01-10 21:35:49 +00:00
|
|
|
vdso_pagelist[i + 1] = pfn_to_page(pfn + i);
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2016-08-15 06:45:46 +00:00
|
|
|
vdso_spec[0].pages = &vdso_pagelist[0];
|
|
|
|
vdso_spec[1].pages = &vdso_pagelist[1];
|
2014-07-09 18:22:12 +00:00
|
|
|
|
2014-02-11 22:28:42 +00:00
|
|
|
return 0;
|
2012-03-05 11:49:31 +00:00
|
|
|
}
|
|
|
|
arch_initcall(vdso_init);
|
|
|
|
|
|
|
|
int arch_setup_additional_pages(struct linux_binprm *bprm,
|
|
|
|
int uses_interp)
|
|
|
|
{
|
|
|
|
struct mm_struct *mm = current->mm;
|
2014-07-09 18:22:11 +00:00
|
|
|
unsigned long vdso_base, vdso_text_len, vdso_mapping_len;
|
2014-07-09 18:22:12 +00:00
|
|
|
void *ret;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2014-07-09 18:22:11 +00:00
|
|
|
vdso_text_len = vdso_pages << PAGE_SHIFT;
|
2012-03-05 11:49:31 +00:00
|
|
|
/* Be sure to map the data page */
|
2014-07-09 18:22:11 +00:00
|
|
|
vdso_mapping_len = vdso_text_len + PAGE_SIZE;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2016-05-23 23:25:54 +00:00
|
|
|
if (down_write_killable(&mm->mmap_sem))
|
|
|
|
return -EINTR;
|
2012-03-05 11:49:31 +00:00
|
|
|
vdso_base = get_unmapped_area(NULL, 0, vdso_mapping_len, 0, 0);
|
|
|
|
if (IS_ERR_VALUE(vdso_base)) {
|
2014-07-09 18:22:12 +00:00
|
|
|
ret = ERR_PTR(vdso_base);
|
2012-03-05 11:49:31 +00:00
|
|
|
goto up_fail;
|
|
|
|
}
|
2014-07-09 18:22:13 +00:00
|
|
|
ret = _install_special_mapping(mm, vdso_base, PAGE_SIZE,
|
|
|
|
VM_READ|VM_MAYREAD,
|
2014-07-09 18:22:12 +00:00
|
|
|
&vdso_spec[0]);
|
|
|
|
if (IS_ERR(ret))
|
2014-07-09 18:22:11 +00:00
|
|
|
goto up_fail;
|
|
|
|
|
2014-07-09 18:22:13 +00:00
|
|
|
vdso_base += PAGE_SIZE;
|
|
|
|
mm->context.vdso = (void *)vdso_base;
|
|
|
|
ret = _install_special_mapping(mm, vdso_base, vdso_text_len,
|
|
|
|
VM_READ|VM_EXEC|
|
|
|
|
VM_MAYREAD|VM_MAYWRITE|VM_MAYEXEC,
|
2014-07-09 18:22:12 +00:00
|
|
|
&vdso_spec[1]);
|
|
|
|
if (IS_ERR(ret))
|
2012-03-05 11:49:31 +00:00
|
|
|
goto up_fail;
|
|
|
|
|
2014-07-09 18:22:13 +00:00
|
|
|
|
2012-03-05 11:49:31 +00:00
|
|
|
up_write(&mm->mmap_sem);
|
2014-07-09 18:22:11 +00:00
|
|
|
return 0;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2014-07-09 18:22:11 +00:00
|
|
|
up_fail:
|
|
|
|
mm->context.vdso = NULL;
|
|
|
|
up_write(&mm->mmap_sem);
|
2014-07-09 18:22:12 +00:00
|
|
|
return PTR_ERR(ret);
|
2012-03-05 11:49:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Update the vDSO data page to keep in sync with kernel timekeeping.
|
|
|
|
*/
|
2012-10-16 10:44:53 +00:00
|
|
|
void update_vsyscall(struct timekeeper *tk)
|
2012-03-05 11:49:31 +00:00
|
|
|
{
|
2016-09-22 08:35:18 +00:00
|
|
|
u32 use_syscall = !tk->tkr_mono.clock->archdata.vdso_direct;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
|
|
|
++vdso_data->tb_seq_count;
|
|
|
|
smp_wmb();
|
|
|
|
|
|
|
|
vdso_data->use_syscall = use_syscall;
|
2015-08-08 02:03:23 +00:00
|
|
|
vdso_data->xtime_coarse_sec = tk->xtime_sec;
|
|
|
|
vdso_data->xtime_coarse_nsec = tk->tkr_mono.xtime_nsec >>
|
|
|
|
tk->tkr_mono.shift;
|
2014-02-03 19:48:52 +00:00
|
|
|
vdso_data->wtm_clock_sec = tk->wall_to_monotonic.tv_sec;
|
|
|
|
vdso_data->wtm_clock_nsec = tk->wall_to_monotonic.tv_nsec;
|
2012-03-05 11:49:31 +00:00
|
|
|
|
2019-04-16 16:14:30 +00:00
|
|
|
/* Read without the seqlock held by clock_getres() */
|
|
|
|
WRITE_ONCE(vdso_data->hrtimer_res, hrtimer_resolution);
|
|
|
|
|
2012-03-05 11:49:31 +00:00
|
|
|
if (!use_syscall) {
|
arm64: Add support for CLOCK_MONOTONIC_RAW in clock_gettime() vDSO
So far the arm64 clock_gettime() vDSO implementation only supported
the following clocks, falling back to the syscall for the others:
- CLOCK_REALTIME{,_COARSE}
- CLOCK_MONOTONIC{,_COARSE}
This patch adds support for the CLOCK_MONOTONIC_RAW clock, taking
advantage of the recent refactoring of the vDSO time functions. Like
the non-_COARSE clocks, this only works when the "arch_sys_counter"
clocksource is in use (allowing us to read the current time from the
virtual counter register), otherwise we also have to fall back to the
syscall.
Most of the data is shared with CLOCK_MONOTONIC, and the algorithm is
similar. The reference implementation in kernel/time/timekeeping.c
shows that:
- CLOCK_MONOTONIC = tk->wall_to_monotonic + tk->xtime_sec +
timekeeping_get_ns(&tk->tkr_mono)
- CLOCK_MONOTONIC_RAW = tk->raw_time + timekeeping_get_ns(&tk->tkr_raw)
- tkr_mono and tkr_raw are identical (in particular, same
clocksource), except these members:
* mult (only mono's multiplier is NTP-adjusted)
* xtime_nsec (always 0 for raw)
Therefore, tk->raw_time and tkr_raw->mult are now also stored in the
vDSO data page.
Cc: Ali Saidi <ali.saidi@arm.com>
Signed-off-by: Kevin Brodsky <kevin.brodsky@arm.com>
Reviewed-by: Dave Martin <dave.martin@arm.com>
Acked-by: Will Deacon <will.deacon@arm.com>
Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
2016-07-12 10:24:00 +00:00
|
|
|
/* tkr_mono.cycle_last == tkr_raw.cycle_last */
|
2015-03-19 09:09:06 +00:00
|
|
|
vdso_data->cs_cycle_last = tk->tkr_mono.cycle_last;
|
2017-05-23 00:20:20 +00:00
|
|
|
vdso_data->raw_time_sec = tk->raw_sec;
|
|
|
|
vdso_data->raw_time_nsec = tk->tkr_raw.xtime_nsec;
|
2012-10-16 10:44:53 +00:00
|
|
|
vdso_data->xtime_clock_sec = tk->xtime_sec;
|
2015-03-19 09:09:06 +00:00
|
|
|
vdso_data->xtime_clock_nsec = tk->tkr_mono.xtime_nsec;
|
arm64: Add support for CLOCK_MONOTONIC_RAW in clock_gettime() vDSO
So far the arm64 clock_gettime() vDSO implementation only supported
the following clocks, falling back to the syscall for the others:
- CLOCK_REALTIME{,_COARSE}
- CLOCK_MONOTONIC{,_COARSE}
This patch adds support for the CLOCK_MONOTONIC_RAW clock, taking
advantage of the recent refactoring of the vDSO time functions. Like
the non-_COARSE clocks, this only works when the "arch_sys_counter"
clocksource is in use (allowing us to read the current time from the
virtual counter register), otherwise we also have to fall back to the
syscall.
Most of the data is shared with CLOCK_MONOTONIC, and the algorithm is
similar. The reference implementation in kernel/time/timekeeping.c
shows that:
- CLOCK_MONOTONIC = tk->wall_to_monotonic + tk->xtime_sec +
timekeeping_get_ns(&tk->tkr_mono)
- CLOCK_MONOTONIC_RAW = tk->raw_time + timekeeping_get_ns(&tk->tkr_raw)
- tkr_mono and tkr_raw are identical (in particular, same
clocksource), except these members:
* mult (only mono's multiplier is NTP-adjusted)
* xtime_nsec (always 0 for raw)
Therefore, tk->raw_time and tkr_raw->mult are now also stored in the
vDSO data page.
Cc: Ali Saidi <ali.saidi@arm.com>
Signed-off-by: Kevin Brodsky <kevin.brodsky@arm.com>
Reviewed-by: Dave Martin <dave.martin@arm.com>
Acked-by: Will Deacon <will.deacon@arm.com>
Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
2016-07-12 10:24:00 +00:00
|
|
|
vdso_data->cs_mono_mult = tk->tkr_mono.mult;
|
|
|
|
vdso_data->cs_raw_mult = tk->tkr_raw.mult;
|
|
|
|
/* tkr_mono.shift == tkr_raw.shift */
|
2015-03-19 09:09:06 +00:00
|
|
|
vdso_data->cs_shift = tk->tkr_mono.shift;
|
2012-03-05 11:49:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
smp_wmb();
|
|
|
|
++vdso_data->tb_seq_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
void update_vsyscall_tz(void)
|
|
|
|
{
|
|
|
|
vdso_data->tz_minuteswest = sys_tz.tz_minuteswest;
|
|
|
|
vdso_data->tz_dsttime = sys_tz.tz_dsttime;
|
|
|
|
}
|