mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-06 04:46:16 +01:00
aec5785e7b
The changes made in commitbc59e28bf6
(issue #3399) fixed introspection of the GThread API. However, they introduced a trampoline in every threading function. So with those changes applied, the disassembly of `g_mutex_lock()` (for example) was: ``` 0x7ffff7f038b0 <g_mutex_lock> jmp 0x7ffff7f2f440 <g_mutex_lock_impl> 0x7ffff7f038b5 data16 cs nopw 0x0(%rax,%rax,1) ``` i.e. It jumps straight to the `_impl` function, even with an optimised build. Since `g_mutex_lock()` (and various other GThread functions) are frequently run hot paths, this additional `jmp` to a function which has ended up in a different code page is a slowdown which we’d rather avoid. So, this commit reworks things to define all the `_impl` functions as `G_ALWAYS_INLINE static inline` (which typically expands to `__attribute__((__always_inline__)) static inline`), and to move them into the same compilation unit as `gthread.c` so that they can be inlined without the need for link-time optimisation to be enabled. It makes the code a little less readable, but not much worse than what commitbc59e28bf6
already did. And perhaps the addition of the `inline` decorations to all the `_impl` functions will make it a bit clearer what their intended purpose is (platform-specific implementations). After applying this commit, the disassembly of `g_mutex_lock()` successfully contains the inlining for me: ``` => 0x00007ffff7f03d80 <+0>: xor %eax,%eax 0x00007ffff7f03d82 <+2>: mov $0x1,%edx 0x00007ffff7f03d87 <+7>: lock cmpxchg %edx,(%rdi) 0x00007ffff7f03d8b <+11>: jne 0x7ffff7f03d8e <g_mutex_lock+14> 0x00007ffff7f03d8d <+13>: ret 0x00007ffff7f03d8e <+14>: jmp 0x7ffff7f03610 <g_mutex_lock_slowpath> ``` I considered making a similar change to the other APIs touched in #3399 (GContentType, GAppInfo, GSpawn), but they are all much less performance critical, so it’s probably not worth making their code more complex for that sake. Signed-off-by: Philip Withnall <pwithnall@gnome.org> Fixes: #3417
179 lines
8.8 KiB
C
179 lines
8.8 KiB
C
/* GLIB - Library of useful routines for C programming
|
||
*
|
||
* gthreadprivate.h - GLib internal thread system related declarations.
|
||
*
|
||
* Copyright (C) 2003 Sebastian Wilhelmi
|
||
*
|
||
* SPDX-License-Identifier: LGPL-2.1-or-later
|
||
*
|
||
* This 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.
|
||
*
|
||
* This 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 this library; if not, see <http://www.gnu.org/licenses/>.
|
||
*/
|
||
|
||
#ifndef __G_THREADPRIVATE_H__
|
||
#define __G_THREADPRIVATE_H__
|
||
|
||
#include "config.h"
|
||
|
||
#include "deprecated/gthread.h"
|
||
|
||
typedef struct _GRealThread GRealThread;
|
||
struct _GRealThread
|
||
{
|
||
GThread thread;
|
||
|
||
gint ref_count;
|
||
gboolean ours;
|
||
gchar *name;
|
||
gpointer retval;
|
||
};
|
||
|
||
/* system thread implementation (gthread-posix.c, gthread-win32.c) */
|
||
|
||
#if defined(HAVE_FUTEX) || defined(HAVE_FUTEX_TIME64)
|
||
#include <errno.h>
|
||
#include <linux/futex.h>
|
||
#include <sys/syscall.h>
|
||
#include <unistd.h>
|
||
|
||
#ifndef FUTEX_WAIT_PRIVATE
|
||
#define FUTEX_WAIT_PRIVATE FUTEX_WAIT
|
||
#define FUTEX_WAKE_PRIVATE FUTEX_WAKE
|
||
#endif
|
||
|
||
/* Wrapper macro to call `futex_time64` and/or `futex` with simple
|
||
* parameters and without returning the return value.
|
||
*
|
||
* We expect futex to sometimes return EAGAIN due to the race
|
||
* between the caller checking the current value and deciding to
|
||
* do the futex op. To avoid splattering errno on success, we
|
||
* restore the original errno if EAGAIN is seen. See also:
|
||
* https://gitlab.gnome.org/GNOME/glib/-/issues/3034
|
||
*
|
||
* If the `futex_time64` syscall does not exist (`ENOSYS`), we retry again
|
||
* with the normal `futex` syscall. This can happen if newer kernel headers
|
||
* are used than the kernel that is actually running.
|
||
*
|
||
* The `futex_time64` syscall is also skipped in favour of `futex` if the
|
||
* Android runtime’s API level is lower than 30, as it’s blocked by seccomp
|
||
* there and using it will cause the app to be terminated:
|
||
* https://android-review.googlesource.com/c/platform/bionic/+/1094758
|
||
* https://github.com/aosp-mirror/platform_bionic/commit/ee7bc3002dc3127faac110167d28912eb0e86a20
|
||
*
|
||
* This must not be called with a timeout parameter as that differs
|
||
* in size between the two syscall variants!
|
||
*/
|
||
#if defined(HAVE_FUTEX) && defined(HAVE_FUTEX_TIME64)
|
||
#if defined(__ANDROID__)
|
||
#define g_futex_simple(uaddr, futex_op, ...) \
|
||
G_STMT_START \
|
||
{ \
|
||
int saved_errno = errno; \
|
||
int res = 0; \
|
||
if (__builtin_available (android 30, *)) \
|
||
{ \
|
||
res = syscall (__NR_futex_time64, uaddr, (gsize) futex_op, __VA_ARGS__); \
|
||
if (res < 0 && errno == ENOSYS) \
|
||
{ \
|
||
errno = saved_errno; \
|
||
res = syscall (__NR_futex, uaddr, (gsize) futex_op, __VA_ARGS__); \
|
||
} \
|
||
} \
|
||
else \
|
||
{ \
|
||
res = syscall (__NR_futex, uaddr, (gsize) futex_op, __VA_ARGS__); \
|
||
} \
|
||
if (res < 0 && errno == EAGAIN) \
|
||
{ \
|
||
errno = saved_errno; \
|
||
} \
|
||
} \
|
||
G_STMT_END
|
||
#else
|
||
#define g_futex_simple(uaddr, futex_op, ...) \
|
||
G_STMT_START \
|
||
{ \
|
||
int saved_errno = errno; \
|
||
int res = syscall (__NR_futex_time64, uaddr, (gsize) futex_op, __VA_ARGS__); \
|
||
if (res < 0 && errno == ENOSYS) \
|
||
{ \
|
||
errno = saved_errno; \
|
||
res = syscall (__NR_futex, uaddr, (gsize) futex_op, __VA_ARGS__); \
|
||
} \
|
||
if (res < 0 && errno == EAGAIN) \
|
||
{ \
|
||
errno = saved_errno; \
|
||
} \
|
||
} \
|
||
G_STMT_END
|
||
#endif /* defined(__ANDROID__) */
|
||
#elif defined(HAVE_FUTEX_TIME64)
|
||
#define g_futex_simple(uaddr, futex_op, ...) \
|
||
G_STMT_START \
|
||
{ \
|
||
int saved_errno = errno; \
|
||
int res = syscall (__NR_futex_time64, uaddr, (gsize) futex_op, __VA_ARGS__); \
|
||
if (res < 0 && errno == EAGAIN) \
|
||
{ \
|
||
errno = saved_errno; \
|
||
} \
|
||
} \
|
||
G_STMT_END
|
||
#elif defined(HAVE_FUTEX)
|
||
#define g_futex_simple(uaddr, futex_op, ...) \
|
||
G_STMT_START \
|
||
{ \
|
||
int saved_errno = errno; \
|
||
int res = syscall (__NR_futex, uaddr, (gsize) futex_op, __VA_ARGS__); \
|
||
if (res < 0 && errno == EAGAIN) \
|
||
{ \
|
||
errno = saved_errno; \
|
||
} \
|
||
} \
|
||
G_STMT_END
|
||
#else /* !defined(HAVE_FUTEX) && !defined(HAVE_FUTEX_TIME64) */
|
||
#error "Neither __NR_futex nor __NR_futex_time64 are available"
|
||
#endif /* defined(HAVE_FUTEX) && defined(HAVE_FUTEX_TIME64) */
|
||
|
||
#endif
|
||
|
||
void g_system_thread_wait (GRealThread *thread);
|
||
|
||
GRealThread *g_system_thread_new (GThreadFunc proxy,
|
||
gulong stack_size,
|
||
const char *name,
|
||
GThreadFunc func,
|
||
gpointer data,
|
||
GError **error);
|
||
void g_system_thread_free (GRealThread *thread);
|
||
|
||
G_NORETURN void g_system_thread_exit (void);
|
||
void g_system_thread_set_name (const gchar *name);
|
||
|
||
/* gthread.c */
|
||
GThread *g_thread_new_internal (const gchar *name,
|
||
GThreadFunc proxy,
|
||
GThreadFunc func,
|
||
gpointer data,
|
||
gsize stack_size,
|
||
GError **error);
|
||
|
||
gpointer g_thread_proxy (gpointer thread);
|
||
|
||
guint g_thread_n_created (void);
|
||
|
||
gpointer g_private_set_alloc0 (GPrivate *key,
|
||
gsize size);
|
||
|
||
#endif /* __G_THREADPRIVATE_H__ */
|