d98720e07f
The IFUNC selector for gettimeofday runs before _libc_vdso_platform_setup where __vdso_gettimeofday is set. The selector then sets __gettimeofday (the internal version used within GLIBC) to use the system call version instead of the vDSO one. This patch changes the check if vDSO is available to get its value directly instead of rely on __vdso_gettimeofday. This patch changes it by getting the vDSO value directly. It fixes BZ#16431.
84 lines
2.6 KiB
C
84 lines
2.6 KiB
C
/* Copyright (C) 2005-2014 Free Software Foundation, Inc.
|
|
This file is part of the GNU C Library.
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU Lesser General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2.1 of the License, or (at your option) any later version.
|
|
|
|
The GNU C Library is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with the GNU C Library; if not, see
|
|
<http://www.gnu.org/licenses/>. */
|
|
|
|
|
|
#include <sys/time.h>
|
|
|
|
#ifdef SHARED
|
|
|
|
# include <dl-vdso.h>
|
|
# include <bits/libc-vdso.h>
|
|
# include <dl-machine.h>
|
|
|
|
void *gettimeofday_ifunc (void) __asm__ ("__gettimeofday");
|
|
|
|
static int
|
|
__gettimeofday_syscall (struct timeval *tv, struct timezone *tz)
|
|
{
|
|
return INLINE_SYSCALL (gettimeofday, 2, tv, tz);
|
|
}
|
|
|
|
void *
|
|
gettimeofday_ifunc (void)
|
|
{
|
|
PREPARE_VERSION (linux2615, "LINUX_2.6.15", 123718565);
|
|
|
|
/* If the vDSO is not available we fall back syscall. */
|
|
void *vdso_gettimeofday = _dl_vdso_vsym ("__kernel_gettimeofday", &linux2615);
|
|
return (vdso_gettimeofday ? VDSO_IFUNC_RET (vdso_gettimeofday)
|
|
: (void*)__gettimeofday_syscall);
|
|
}
|
|
asm (".type __gettimeofday, %gnu_indirect_function");
|
|
|
|
/* This is doing "libc_hidden_def (__gettimeofday)" but the compiler won't
|
|
let us do it in C because it doesn't know we're defining __gettimeofday
|
|
here in this file. */
|
|
asm (".globl __GI___gettimeofday");
|
|
|
|
/* __GI___gettimeofday is defined as hidden and for ppc32 it enables the
|
|
compiler make a local call (symbol@local) for internal GLIBC usage. It
|
|
means the PLT won't be used and the ifunc resolver will be called directly.
|
|
For ppc64 a call to a function in another translation unit might use a
|
|
different toc pointer thus disallowing direct branchess and making internal
|
|
ifuncs calls safe. */
|
|
#ifdef __powerpc64__
|
|
asm ("__GI___gettimeofday = __gettimeofday");
|
|
#else
|
|
int
|
|
__gettimeofday_vsyscall (struct timeval *tv, struct timezone *tz)
|
|
{
|
|
return INLINE_VSYSCALL (gettimeofday, 2, tv, tz);
|
|
}
|
|
asm ("__GI___gettimeofday = __gettimeofday_vsyscall");
|
|
#endif
|
|
|
|
#else
|
|
|
|
# include <sysdep.h>
|
|
# include <errno.h>
|
|
|
|
int
|
|
__gettimeofday (struct timeval *tv, struct timezone *tz)
|
|
{
|
|
return INLINE_SYSCALL (gettimeofday, 2, tv, tz);
|
|
}
|
|
libc_hidden_def (__gettimeofday)
|
|
|
|
#endif
|
|
weak_alias (__gettimeofday, gettimeofday)
|
|
libc_hidden_weak (gettimeofday)
|