|
|
29e444 |
From 01c3d9bb14a1e90159d6999cf3469e62c0c5d4b2 Mon Sep 17 00:00:00 2001
|
|
|
29e444 |
From: Adhemerval Zanella <azanella@linux.vnet.ibm.com>
|
|
|
29e444 |
Date: Fri, 3 May 2013 15:00:31 -0500
|
|
|
29e444 |
Subject: [PATCH 27/42] PowerPC: Add time vDSO support
|
|
|
29e444 |
|
|
|
29e444 |
PowerPC kernel now provides a vDSO implementation for time syscall
|
|
|
29e444 |
(commit fcb41a2030abe0eb716ef0798035ef9562097f42). This patch changes
|
|
|
29e444 |
time syscall wrapper to use the vDSO when available. It also changes
|
|
|
29e444 |
the default non vDSO time on PowerPC to use sysdeps/posix/time.c
|
|
|
29e444 |
(since gettimeofday is a vDSO call).
|
|
|
29e444 |
(cherry picked from commit 83e7640f6bf68708ecf0b09d83c670203167271e)
|
|
|
29e444 |
---
|
|
|
29e444 |
sysdeps/unix/sysv/linux/powerpc/Versions | 1 +
|
|
|
29e444 |
sysdeps/unix/sysv/linux/powerpc/bits/libc-vdso.h | 6 ++-
|
|
|
29e444 |
sysdeps/unix/sysv/linux/powerpc/init-first.c | 4 +-
|
|
|
29e444 |
sysdeps/unix/sysv/linux/powerpc/time.c | 62 ++++++++++++++++++++++++
|
|
|
29e444 |
5 files changed, 81 insertions(+), 3 deletions(-)
|
|
|
29e444 |
create mode 100644 sysdeps/unix/sysv/linux/powerpc/time.c
|
|
|
29e444 |
|
|
|
29e444 |
diff --git a/sysdeps/unix/sysv/linux/powerpc/Versions b/sysdeps/unix/sysv/linux/powerpc/Versions
|
|
|
29e444 |
index 396a423..289c4fe 100644
|
|
|
29e444 |
--- a/sysdeps/unix/sysv/linux/powerpc/Versions
|
|
|
29e444 |
+++ b/sysdeps/unix/sysv/linux/powerpc/Versions
|
|
|
29e444 |
@@ -4,5 +4,6 @@ libc {
|
|
|
29e444 |
__vdso_clock_gettime;
|
|
|
29e444 |
__vdso_clock_getres;
|
|
|
29e444 |
__vdso_getcpu;
|
|
|
29e444 |
+ __vdso_time;
|
|
|
29e444 |
}
|
|
|
29e444 |
}
|
|
|
29e444 |
diff --git a/sysdeps/unix/sysv/linux/powerpc/bits/libc-vdso.h b/sysdeps/unix/sysv/linux/powerpc/bits/libc-vdso.h
|
|
|
29e444 |
index e4ae630..f7f635e 100644
|
|
|
29e444 |
--- a/sysdeps/unix/sysv/linux/powerpc/bits/libc-vdso.h
|
|
|
29e444 |
+++ b/sysdeps/unix/sysv/linux/powerpc/bits/libc-vdso.h
|
|
|
29e444 |
@@ -32,14 +32,16 @@ extern void *__vdso_get_tbfreq;
|
|
|
29e444 |
|
|
|
29e444 |
extern void *__vdso_getcpu;
|
|
|
29e444 |
|
|
|
29e444 |
+extern void *__vdso_time;
|
|
|
29e444 |
+
|
|
|
29e444 |
/* This macro is needed for PPC64 to return a skeleton OPD entry of a vDSO
|
|
|
29e444 |
symbol. This works because _dl_vdso_vsym always return the function
|
|
|
29e444 |
address, and no vDSO symbols use the TOC or chain pointers from the OPD
|
|
|
29e444 |
so we can allow them to be garbage. */
|
|
|
29e444 |
#if defined(__PPC64__) || defined(__powerpc64__)
|
|
|
29e444 |
-#define VDSO_IFUNC_RET(value) &value
|
|
|
29e444 |
+#define VDSO_IFUNC_RET(value) ((void *) &(value))
|
|
|
29e444 |
#else
|
|
|
29e444 |
-#define VDSO_IFUNC_RET(value) value
|
|
|
29e444 |
+#define VDSO_IFUNC_RET(value) ((void *) (value))
|
|
|
29e444 |
#endif
|
|
|
29e444 |
|
|
|
29e444 |
#endif
|
|
|
29e444 |
diff --git a/sysdeps/unix/sysv/linux/powerpc/init-first.c b/sysdeps/unix/sysv/linux/powerpc/init-first.c
|
|
|
29e444 |
index 5587e2a..3cefd9b 100644
|
|
|
29e444 |
--- a/sysdeps/unix/sysv/linux/powerpc/init-first.c
|
|
|
29e444 |
+++ b/sysdeps/unix/sysv/linux/powerpc/init-first.c
|
|
|
29e444 |
@@ -28,7 +28,7 @@ void *__vdso_clock_gettime;
|
|
|
29e444 |
void *__vdso_clock_getres;
|
|
|
29e444 |
void *__vdso_get_tbfreq;
|
|
|
29e444 |
void *__vdso_getcpu;
|
|
|
29e444 |
-
|
|
|
29e444 |
+void *__vdso_time;
|
|
|
29e444 |
|
|
|
29e444 |
static inline void
|
|
|
29e444 |
_libc_vdso_platform_setup (void)
|
|
|
29e444 |
@@ -44,6 +44,8 @@ _libc_vdso_platform_setup (void)
|
|
|
29e444 |
__vdso_get_tbfreq = _dl_vdso_vsym ("__kernel_get_tbfreq", &linux2615);
|
|
|
29e444 |
|
|
|
29e444 |
__vdso_getcpu = _dl_vdso_vsym ("__kernel_getcpu", &linux2615);
|
|
|
29e444 |
+
|
|
|
29e444 |
+ __vdso_time = _dl_vdso_vsym ("__kernel_time", &linux2615);
|
|
|
29e444 |
}
|
|
|
29e444 |
|
|
|
29e444 |
# define VDSO_SETUP _libc_vdso_platform_setup
|
|
|
29e444 |
diff --git a/sysdeps/unix/sysv/linux/powerpc/time.c b/sysdeps/unix/sysv/linux/powerpc/time.c
|
|
|
29e444 |
new file mode 100644
|
|
|
29e444 |
index 0000000..66b4eb3
|
|
|
29e444 |
--- /dev/null
|
|
|
29e444 |
+++ b/sysdeps/unix/sysv/linux/powerpc/time.c
|
|
|
29e444 |
@@ -0,0 +1,62 @@
|
|
|
29e444 |
+/* time system call for Linux/PowerPC.
|
|
|
29e444 |
+ Copyright (C) 2013 Free Software Foundation, Inc.
|
|
|
29e444 |
+ This file is part of the GNU C Library.
|
|
|
29e444 |
+
|
|
|
29e444 |
+ The GNU C Library is free software; you can redistribute it and/or
|
|
|
29e444 |
+ modify it under the terms of the GNU Lesser General Public
|
|
|
29e444 |
+ License as published by the Free Software Foundation; either
|
|
|
29e444 |
+ version 2.1 of the License, or (at your option) any later version.
|
|
|
29e444 |
+
|
|
|
29e444 |
+ The GNU C Library is distributed in the hope that it will be useful,
|
|
|
29e444 |
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
29e444 |
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
29e444 |
+ Lesser General Public License for more details.
|
|
|
29e444 |
+
|
|
|
29e444 |
+ You should have received a copy of the GNU Lesser General Public
|
|
|
29e444 |
+ License along with the GNU C Library; if not, see
|
|
|
29e444 |
+ <http://www.gnu.org/licenses/>. */
|
|
|
29e444 |
+
|
|
|
29e444 |
+#ifdef SHARED
|
|
|
29e444 |
+
|
|
|
29e444 |
+# include <time.h>
|
|
|
29e444 |
+# include <sysdep.h>
|
|
|
29e444 |
+# include <bits/libc-vdso.h>
|
|
|
29e444 |
+
|
|
|
29e444 |
+void *time_ifunc (void) asm ("time");
|
|
|
29e444 |
+
|
|
|
29e444 |
+static time_t
|
|
|
29e444 |
+time_syscall (time_t *t)
|
|
|
29e444 |
+{
|
|
|
29e444 |
+ struct timeval tv;
|
|
|
29e444 |
+ time_t result;
|
|
|
29e444 |
+
|
|
|
29e444 |
+ if (INLINE_VSYSCALL (gettimeofday, 2, &tv, NULL) < 0)
|
|
|
29e444 |
+ result = (time_t) -1;
|
|
|
29e444 |
+ else
|
|
|
29e444 |
+ result = (time_t) tv.tv_sec;
|
|
|
29e444 |
+
|
|
|
29e444 |
+ if (t != NULL)
|
|
|
29e444 |
+ *t = result;
|
|
|
29e444 |
+ return result;
|
|
|
29e444 |
+}
|
|
|
29e444 |
+
|
|
|
29e444 |
+void *
|
|
|
29e444 |
+time_ifunc (void)
|
|
|
29e444 |
+{
|
|
|
29e444 |
+ /* If the vDSO is not available we fall back to the syscall. */
|
|
|
29e444 |
+ return (__vdso_time ? VDSO_IFUNC_RET (__vdso_time)
|
|
|
29e444 |
+ : time_syscall);
|
|
|
29e444 |
+}
|
|
|
29e444 |
+asm (".type time, %gnu_indirect_function");
|
|
|
29e444 |
+
|
|
|
29e444 |
+/* This is doing "libc_hidden_def (time)" but the compiler won't
|
|
|
29e444 |
+ * let us do it in C because it doesn't know we're defining time
|
|
|
29e444 |
+ * here in this file. */
|
|
|
29e444 |
+asm (".globl __GI_time\n"
|
|
|
29e444 |
+ "__GI_time = time");
|
|
|
29e444 |
+
|
|
|
29e444 |
+#else
|
|
|
29e444 |
+
|
|
|
29e444 |
+#include <sysdeps/posix/time.c>
|
|
|
29e444 |
+
|
|
|
29e444 |
+#endif
|
|
|
29e444 |
--
|
|
|
29e444 |
1.7.11.7
|
|
|
29e444 |
|