mirror of
https://sourceware.org/git/glibc.git
synced 2024-11-27 11:43:34 +08:00
ARM: Consolidate NPTL/non versions of vfork
This commit is contained in:
parent
4f02e2b8f9
commit
5675da1e84
11
ChangeLog
11
ChangeLog
@ -1,3 +1,14 @@
|
|||||||
|
2014-05-14 Roland McGrath <roland@hack.frob.com>
|
||||||
|
|
||||||
|
* sysdeps/unix/sysv/linux/arm/vfork.S: Include <tcb-offsets.h>.
|
||||||
|
(__vfork): Incorporate save/restore of PID from nptl/vfork.S here.
|
||||||
|
(__libc_vfork): New strong alias.
|
||||||
|
* sysdeps/unix/sysv/linux/arm/nptl/vfork.S: File removed.
|
||||||
|
* sysdeps/unix/sysv/linux/arm/nptl/pt-vfork.S: File removed.
|
||||||
|
* nptl/pt-vfork.c: New file.
|
||||||
|
* nptl/Versions (libc: GLIBC_PRIVATE): Add __libc_vfork.
|
||||||
|
(libpthread: GLIBC_2.20): New version set (empty).
|
||||||
|
|
||||||
2014-05-14 Will Newton <will.newton@linaro.org>
|
2014-05-14 Will Newton <will.newton@linaro.org>
|
||||||
|
|
||||||
* stdlib/gmp-impl.h: Test USE_STACK_ALLOC #ifdef
|
* stdlib/gmp-impl.h: Test USE_STACK_ALLOC #ifdef
|
||||||
|
@ -30,6 +30,7 @@ libc {
|
|||||||
__libc_alloca_cutoff;
|
__libc_alloca_cutoff;
|
||||||
# Internal libc interface to libpthread
|
# Internal libc interface to libpthread
|
||||||
__libc_dl_error_tsd;
|
__libc_dl_error_tsd;
|
||||||
|
__libc_vfork;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -257,6 +258,9 @@ libpthread {
|
|||||||
pthread_setattr_default_np;
|
pthread_setattr_default_np;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
GLIBC_2.20 {
|
||||||
|
}
|
||||||
|
|
||||||
GLIBC_PRIVATE {
|
GLIBC_PRIVATE {
|
||||||
__pthread_initialize_minimal;
|
__pthread_initialize_minimal;
|
||||||
__pthread_clock_gettime; __pthread_clock_settime;
|
__pthread_clock_gettime; __pthread_clock_settime;
|
||||||
|
77
nptl/pt-vfork.c
Normal file
77
nptl/pt-vfork.c
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
/* vfork ABI-compatibility entry points for libpthread.
|
||||||
|
Copyright (C) 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 <unistd.h>
|
||||||
|
#include <shlib-compat.h>
|
||||||
|
|
||||||
|
/* libpthread used to have its own vfork implementation that differed
|
||||||
|
from libc's only in having a pointless micro-optimization. There
|
||||||
|
is no longer any use to having a separate copy in libpthread, but
|
||||||
|
the historical ABI requires it. For static linking, there is no
|
||||||
|
need to provide anything here--the libc version will be linked in.
|
||||||
|
For shared library ABI compatibility, there must be __vfork and
|
||||||
|
vfork symbols in libpthread.so; so we define them using IFUNC to
|
||||||
|
redirect to the libc function. */
|
||||||
|
|
||||||
|
#if (SHLIB_COMPAT (libpthread, GLIBC_2_0, GLIBC_2_20) \
|
||||||
|
|| SHLIB_COMPAT (libpthread, GLIBC_2_1_2, GLIBC_2_20))
|
||||||
|
|
||||||
|
extern __typeof (vfork) __libc_vfork; /* Defined in libc. */
|
||||||
|
|
||||||
|
# ifdef HAVE_IFUNC
|
||||||
|
|
||||||
|
attribute_hidden __attribute__ ((used))
|
||||||
|
__typeof (vfork) *
|
||||||
|
vfork_ifunc (void)
|
||||||
|
{
|
||||||
|
return &__libc_vfork;
|
||||||
|
}
|
||||||
|
|
||||||
|
# ifdef HAVE_ASM_SET_DIRECTIVE
|
||||||
|
# define DEFINE_VFORK(name) \
|
||||||
|
asm (".set " #name ", vfork_ifunc\n" \
|
||||||
|
".globl " #name "\n" \
|
||||||
|
".type " #name ", %gnu_indirect_function");
|
||||||
|
# else
|
||||||
|
# define DEFINE_VFORK(name) \
|
||||||
|
asm (#name " = vfork_ifunc\n" \
|
||||||
|
".globl " #name "\n" \
|
||||||
|
".type " #name ", %gnu_indirect_function");
|
||||||
|
# endif
|
||||||
|
|
||||||
|
# else
|
||||||
|
|
||||||
|
attribute_hidden
|
||||||
|
pid_t
|
||||||
|
vfork_compat (void)
|
||||||
|
{
|
||||||
|
return __libc_vfork ();
|
||||||
|
}
|
||||||
|
|
||||||
|
# define DEFINE_VFORK(name) weak_alias (vfork_compat, name)
|
||||||
|
|
||||||
|
# endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if SHLIB_COMPAT (libpthread, GLIBC_2_0, GLIBC_2_20)
|
||||||
|
DEFINE_VFORK (vfork)
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if SHLIB_COMPAT (libpthread, GLIBC_2_1_2, GLIBC_2_20)
|
||||||
|
DEFINE_VFORK (__vfork)
|
||||||
|
#endif
|
@ -1,34 +0,0 @@
|
|||||||
/* 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 <tcb-offsets.h>
|
|
||||||
|
|
||||||
/* Save the PID value. */
|
|
||||||
#define SAVE_PID \
|
|
||||||
GET_TLS (r2); \
|
|
||||||
NEGOFF_ADJ_BASE2 (r2, r0, PID_OFFSET); /* Save the TLS addr in r2. */ \
|
|
||||||
ldr r3, NEGOFF_OFF1 (r2, PID_OFFSET); /* Load the saved PID. */ \
|
|
||||||
rsb r0, r3, #0; /* Negate it. */ \
|
|
||||||
str r0, NEGOFF_OFF1 (r2, PID_OFFSET); /* Store the temp PID. */
|
|
||||||
|
|
||||||
/* Restore the old PID value in the parent. */
|
|
||||||
#define RESTORE_PID \
|
|
||||||
cmp r0, #0; /* If we are the parent... */ \
|
|
||||||
it ne; \
|
|
||||||
strne r3, NEGOFF_OFF1 (r2, PID_OFFSET); /* restore the saved PID. */
|
|
||||||
|
|
||||||
#include "../vfork.S"
|
|
@ -1,36 +0,0 @@
|
|||||||
/* 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 <tcb-offsets.h>
|
|
||||||
|
|
||||||
/* Save the PID value. */
|
|
||||||
#define SAVE_PID \
|
|
||||||
GET_TLS (r2); \
|
|
||||||
NEGOFF_ADJ_BASE2 (r2, r0, PID_OFFSET); /* Save the TLS addr in r2. */ \
|
|
||||||
ldr r3, NEGOFF_OFF1 (r2, PID_OFFSET); /* Load the saved PID. */ \
|
|
||||||
rsbs r0, r3, #0; /* Negate it. */ \
|
|
||||||
it eq; \
|
|
||||||
moveq r0, #0x80000000; /* Use 0x80000000 if it was 0. */ \
|
|
||||||
str r0, NEGOFF_OFF1 (r2, PID_OFFSET); /* Store the temp PID. */
|
|
||||||
|
|
||||||
/* Restore the old PID value in the parent. */
|
|
||||||
#define RESTORE_PID \
|
|
||||||
cmp r0, #0; /* If we are the parent... */ \
|
|
||||||
it ne; \
|
|
||||||
strne r3, NEGOFF_OFF1 (r2, PID_OFFSET); /* restore the saved PID. */
|
|
||||||
|
|
||||||
#include "../vfork.S"
|
|
@ -19,6 +19,8 @@
|
|||||||
#include <sysdep.h>
|
#include <sysdep.h>
|
||||||
#define _ERRNO_H 1
|
#define _ERRNO_H 1
|
||||||
#include <bits/errno.h>
|
#include <bits/errno.h>
|
||||||
|
#include <tcb-offsets.h>
|
||||||
|
|
||||||
|
|
||||||
/* Clone the calling process, but without copying the whole address space.
|
/* Clone the calling process, but without copying the whole address space.
|
||||||
The calling process is suspended until the new process exits or is
|
The calling process is suspended until the new process exits or is
|
||||||
@ -26,9 +28,16 @@
|
|||||||
and the process ID of the new process to the old process. */
|
and the process ID of the new process to the old process. */
|
||||||
|
|
||||||
ENTRY (__vfork)
|
ENTRY (__vfork)
|
||||||
#ifdef SAVE_PID
|
/* Save the PID value. */
|
||||||
SAVE_PID
|
GET_TLS (r2)
|
||||||
#endif
|
NEGOFF_ADJ_BASE2 (r2, r0, PID_OFFSET) /* Save the TLS addr in r2. */
|
||||||
|
ldr r3, NEGOFF_OFF1 (r2, PID_OFFSET) /* Load the saved PID. */
|
||||||
|
rsbs r0, r3, #0 /* Negate it, and test for zero. */
|
||||||
|
/* Use 0x80000000 if it was 0. See raise.c for how this is used. */
|
||||||
|
it eq
|
||||||
|
moveq r0, #0x80000000
|
||||||
|
str r0, NEGOFF_OFF1 (r2, PID_OFFSET) /* Store the temp PID. */
|
||||||
|
|
||||||
/* The DO_CALL macro saves r7 on the stack, to enable generation
|
/* The DO_CALL macro saves r7 on the stack, to enable generation
|
||||||
of ARM unwind info. Since the stack is initially shared between
|
of ARM unwind info. Since the stack is initially shared between
|
||||||
parent and child of vfork, that saved value could be corrupted.
|
parent and child of vfork, that saved value could be corrupted.
|
||||||
@ -46,10 +55,13 @@ ENTRY (__vfork)
|
|||||||
add sp, sp, #4
|
add sp, sp, #4
|
||||||
cfi_adjust_cfa_offset (-4)
|
cfi_adjust_cfa_offset (-4)
|
||||||
mov r7, ip
|
mov r7, ip
|
||||||
cfi_restore (r7);
|
cfi_restore (r7)
|
||||||
#ifdef RESTORE_PID
|
|
||||||
RESTORE_PID
|
/* Restore the old PID value in the parent. */
|
||||||
#endif
|
cmp r0, #0 /* If we are the parent... */
|
||||||
|
it ne
|
||||||
|
strne r3, NEGOFF_OFF1 (r2, PID_OFFSET) /* restore the saved PID. */
|
||||||
|
|
||||||
cmn a1, #4096
|
cmn a1, #4096
|
||||||
it cc
|
it cc
|
||||||
RETINSTR(cc, lr)
|
RETINSTR(cc, lr)
|
||||||
@ -59,3 +71,4 @@ PSEUDO_END (__vfork)
|
|||||||
libc_hidden_def (__vfork)
|
libc_hidden_def (__vfork)
|
||||||
|
|
||||||
weak_alias (__vfork, vfork)
|
weak_alias (__vfork, vfork)
|
||||||
|
strong_alias (__vfork, __libc_vfork)
|
||||||
|
Loading…
Reference in New Issue
Block a user