2017-01-01 00:14:16 +00:00
|
|
|
/* Copyright (C) 1998-2017 Free Software Foundation, Inc.
|
1998-03-06 17:21:43 +00:00
|
|
|
This file is part of the GNU C Library.
|
|
|
|
|
|
|
|
The GNU C Library is free software; you can redistribute it and/or
|
2001-07-06 04:58:11 +00:00
|
|
|
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.
|
1998-03-06 17:21:43 +00:00
|
|
|
|
|
|
|
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
|
2001-07-06 04:58:11 +00:00
|
|
|
Lesser General Public License for more details.
|
1998-03-06 17:21:43 +00:00
|
|
|
|
2001-07-06 04:58:11 +00:00
|
|
|
You should have received a copy of the GNU Lesser General Public
|
2012-02-09 23:18:22 +00:00
|
|
|
License along with the GNU C Library; if not, see
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
1998-03-06 17:21:43 +00:00
|
|
|
|
2013-03-28 23:15:48 +00:00
|
|
|
#include <assert.h>
|
1998-03-08 15:26:29 +00:00
|
|
|
#include <stdlib.h>
|
2002-12-01 22:14:57 +00:00
|
|
|
#include <stdio.h>
|
1998-03-08 15:26:29 +00:00
|
|
|
#include <unistd.h>
|
2000-03-23 20:31:46 +00:00
|
|
|
#include <ldsodefs.h>
|
2014-05-13 16:49:20 +00:00
|
|
|
#include <exit-thread.h>
|
2016-12-31 18:02:17 +00:00
|
|
|
#include <libc-internal.h>
|
|
|
|
|
|
|
|
#include <elf/dl-tunables.h>
|
1998-03-08 15:26:29 +00:00
|
|
|
|
1998-03-18 14:42:25 +00:00
|
|
|
extern void __libc_init_first (int argc, char **argv, char **envp);
|
|
|
|
|
|
|
|
extern int __libc_multiple_libcs;
|
|
|
|
|
2002-09-15 02:37:32 +00:00
|
|
|
#include <tls.h>
|
2000-07-06 21:16:06 +00:00
|
|
|
#ifndef SHARED
|
2002-12-01 22:14:57 +00:00
|
|
|
# include <dl-osinfo.h>
|
2005-06-26 18:14:26 +00:00
|
|
|
# ifndef THREAD_SET_STACK_GUARD
|
|
|
|
/* Only exported for architectures that don't store the stack guard canary
|
|
|
|
in thread local area. */
|
|
|
|
uintptr_t __stack_chk_guard attribute_relro;
|
|
|
|
# endif
|
2013-09-23 04:52:09 +00:00
|
|
|
# ifndef THREAD_SET_POINTER_GUARD
|
|
|
|
/* Only exported for architectures that don't store the pointer guard
|
|
|
|
value in thread local area. */
|
|
|
|
uintptr_t __pointer_chk_guard_local
|
|
|
|
attribute_relro attribute_hidden __attribute__ ((nocommon));
|
|
|
|
# endif
|
2000-07-06 21:16:06 +00:00
|
|
|
#endif
|
|
|
|
|
2003-02-14 23:31:11 +00:00
|
|
|
#ifdef HAVE_PTR_NTHREADS
|
|
|
|
/* We need atomic operations. */
|
|
|
|
# include <atomic.h>
|
|
|
|
#endif
|
|
|
|
|
2000-11-20 09:16:41 +00:00
|
|
|
|
2012-08-21 22:01:27 +00:00
|
|
|
#ifndef SHARED
|
|
|
|
# include <link.h>
|
|
|
|
# include <dl-irel.h>
|
|
|
|
|
|
|
|
# ifdef ELF_MACHINE_IRELA
|
|
|
|
# define IREL_T ElfW(Rela)
|
|
|
|
# define IPLT_START __rela_iplt_start
|
|
|
|
# define IPLT_END __rela_iplt_end
|
|
|
|
# define IREL elf_irela
|
|
|
|
# elif defined ELF_MACHINE_IREL
|
|
|
|
# define IREL_T ElfW(Rel)
|
|
|
|
# define IPLT_START __rel_iplt_start
|
|
|
|
# define IPLT_END __rel_iplt_end
|
|
|
|
# define IREL elf_irel
|
|
|
|
# endif
|
|
|
|
|
|
|
|
static void
|
|
|
|
apply_irel (void)
|
|
|
|
{
|
2012-08-22 19:53:38 +00:00
|
|
|
# ifdef IREL
|
|
|
|
/* We use weak references for these so that we'll still work with a linker
|
|
|
|
that doesn't define them. Such a linker doesn't support IFUNC at all
|
|
|
|
and so uses won't work, but a statically-linked program that doesn't
|
|
|
|
use any IFUNC symbols won't have a problem. */
|
|
|
|
extern const IREL_T IPLT_START[] __attribute__ ((weak));
|
|
|
|
extern const IREL_T IPLT_END[] __attribute__ ((weak));
|
2012-08-21 22:01:27 +00:00
|
|
|
for (const IREL_T *ipltent = IPLT_START; ipltent < IPLT_END; ++ipltent)
|
|
|
|
IREL (ipltent);
|
2012-08-22 19:53:38 +00:00
|
|
|
# endif
|
2012-08-21 22:01:27 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2003-03-15 23:09:11 +00:00
|
|
|
#ifdef LIBC_START_MAIN
|
2004-02-20 06:07:55 +00:00
|
|
|
# ifdef LIBC_START_DISABLE_INLINE
|
|
|
|
# define STATIC static
|
|
|
|
# else
|
|
|
|
# define STATIC static inline __attribute__ ((always_inline))
|
|
|
|
# endif
|
2003-03-15 23:09:11 +00:00
|
|
|
#else
|
|
|
|
# define STATIC
|
2013-02-14 13:12:02 +00:00
|
|
|
# define LIBC_START_MAIN __libc_start_main
|
2003-03-15 23:09:11 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef MAIN_AUXVEC_ARG
|
2003-04-12 00:58:26 +00:00
|
|
|
/* main gets passed a pointer to the auxiliary. */
|
|
|
|
# define MAIN_AUXVEC_DECL , void *
|
|
|
|
# define MAIN_AUXVEC_PARAM , auxvec
|
|
|
|
#else
|
|
|
|
# define MAIN_AUXVEC_DECL
|
|
|
|
# define MAIN_AUXVEC_PARAM
|
2003-03-15 23:09:11 +00:00
|
|
|
#endif
|
|
|
|
|
2017-05-31 01:08:32 +00:00
|
|
|
#ifndef ARCH_INIT_CPU_FEATURES
|
|
|
|
# define ARCH_INIT_CPU_FEATURES()
|
|
|
|
#endif
|
|
|
|
|
2003-04-12 00:58:26 +00:00
|
|
|
STATIC int LIBC_START_MAIN (int (*main) (int, char **, char **
|
|
|
|
MAIN_AUXVEC_DECL),
|
2003-03-15 23:09:11 +00:00
|
|
|
int argc,
|
2013-02-13 23:30:40 +00:00
|
|
|
char **argv,
|
2003-03-15 23:09:11 +00:00
|
|
|
#ifdef LIBC_START_MAIN_AUXVEC_ARG
|
2013-02-13 23:30:40 +00:00
|
|
|
ElfW(auxv_t) *auxvec,
|
2003-03-15 23:09:11 +00:00
|
|
|
#endif
|
|
|
|
__typeof (main) init,
|
|
|
|
void (*fini) (void),
|
|
|
|
void (*rtld_fini) (void),
|
2013-02-13 23:30:40 +00:00
|
|
|
void *stack_end)
|
2000-11-20 09:16:41 +00:00
|
|
|
__attribute__ ((noreturn));
|
1999-07-09 16:18:17 +00:00
|
|
|
|
2005-01-06 22:40:27 +00:00
|
|
|
|
2005-02-14 21:25:50 +00:00
|
|
|
/* Note: the fini parameter is ignored here for shared library. It
|
|
|
|
is registered with __cxa_atexit. This had the disadvantage that
|
|
|
|
finalizers were called in more than one place. */
|
2003-03-15 23:09:11 +00:00
|
|
|
STATIC int
|
2003-04-12 00:58:26 +00:00
|
|
|
LIBC_START_MAIN (int (*main) (int, char **, char ** MAIN_AUXVEC_DECL),
|
2013-02-13 23:30:40 +00:00
|
|
|
int argc, char **argv,
|
2003-03-15 23:09:11 +00:00
|
|
|
#ifdef LIBC_START_MAIN_AUXVEC_ARG
|
2013-02-13 23:30:40 +00:00
|
|
|
ElfW(auxv_t) *auxvec,
|
2003-03-15 23:09:11 +00:00
|
|
|
#endif
|
|
|
|
__typeof (main) init,
|
|
|
|
void (*fini) (void),
|
2013-02-13 23:30:40 +00:00
|
|
|
void (*rtld_fini) (void), void *stack_end)
|
1998-03-06 17:21:43 +00:00
|
|
|
{
|
2002-09-15 02:37:32 +00:00
|
|
|
/* Result of the 'main' function. */
|
|
|
|
int result;
|
|
|
|
|
|
|
|
__libc_multiple_libcs = &_dl_starting_up && !_dl_starting_up;
|
1998-06-07 14:06:56 +00:00
|
|
|
|
2005-01-04 07:11:05 +00:00
|
|
|
#ifndef SHARED
|
2013-02-13 23:30:40 +00:00
|
|
|
char **ev = &argv[argc + 1];
|
2005-02-16 09:39:23 +00:00
|
|
|
|
2013-02-13 23:30:40 +00:00
|
|
|
__environ = ev;
|
2000-06-13 07:33:12 +00:00
|
|
|
|
2004-01-14 00:24:36 +00:00
|
|
|
/* Store the lowest stack address. This is done in ld.so if this is
|
|
|
|
the code for the DSO. */
|
1998-06-07 14:06:56 +00:00
|
|
|
__libc_stack_end = stack_end;
|
1998-03-18 14:42:25 +00:00
|
|
|
|
2002-07-23 23:12:01 +00:00
|
|
|
# ifdef HAVE_AUX_VECTOR
|
|
|
|
/* First process the auxiliary vector since we need to find the
|
|
|
|
program header to locate an eventually present PT_TLS entry. */
|
2003-03-15 23:09:11 +00:00
|
|
|
# ifndef LIBC_START_MAIN_AUXVEC_ARG
|
2013-02-13 23:30:40 +00:00
|
|
|
ElfW(auxv_t) *auxvec;
|
2003-03-15 23:09:11 +00:00
|
|
|
{
|
2013-02-13 23:30:40 +00:00
|
|
|
char **evp = ev;
|
2003-03-18 01:56:11 +00:00
|
|
|
while (*evp++ != NULL)
|
|
|
|
;
|
2013-02-13 23:30:40 +00:00
|
|
|
auxvec = (ElfW(auxv_t) *) evp;
|
2003-03-15 23:09:11 +00:00
|
|
|
}
|
|
|
|
# endif
|
|
|
|
_dl_aux_init (auxvec);
|
2013-03-28 23:15:48 +00:00
|
|
|
if (GL(dl_phdr) == NULL)
|
2002-07-23 23:12:01 +00:00
|
|
|
# endif
|
2013-03-28 23:15:48 +00:00
|
|
|
{
|
|
|
|
/* Starting from binutils-2.23, the linker will define the
|
|
|
|
magic symbol __ehdr_start to point to our own ELF header
|
|
|
|
if it is visible in a segment that also includes the phdrs.
|
|
|
|
So we can set up _dl_phdr and _dl_phnum even without any
|
|
|
|
information from auxv. */
|
|
|
|
|
2013-04-25 15:09:19 +00:00
|
|
|
extern const ElfW(Ehdr) __ehdr_start
|
|
|
|
__attribute__ ((weak, visibility ("hidden")));
|
2013-03-28 23:15:48 +00:00
|
|
|
if (&__ehdr_start != NULL)
|
|
|
|
{
|
|
|
|
assert (__ehdr_start.e_phentsize == sizeof *GL(dl_phdr));
|
|
|
|
GL(dl_phdr) = (const void *) &__ehdr_start + __ehdr_start.e_phoff;
|
|
|
|
GL(dl_phnum) = __ehdr_start.e_phnum;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-31 18:02:17 +00:00
|
|
|
/* Initialize very early so that tunables can use it. */
|
|
|
|
__libc_init_secure ();
|
|
|
|
|
|
|
|
__tunables_init (__environ);
|
|
|
|
|
2017-05-31 01:08:32 +00:00
|
|
|
ARCH_INIT_CPU_FEATURES ();
|
|
|
|
|
2012-08-21 22:01:27 +00:00
|
|
|
/* Perform IREL{,A} relocations. */
|
|
|
|
apply_irel ();
|
2009-11-24 19:17:06 +00:00
|
|
|
|
2016-12-26 09:08:34 +00:00
|
|
|
/* The stack guard goes into the TCB, so initialize it early. */
|
|
|
|
__libc_setup_tls ();
|
1999-07-09 16:18:17 +00:00
|
|
|
|
2005-06-26 18:14:26 +00:00
|
|
|
/* Set up the stack checker's canary. */
|
2009-01-11 04:40:39 +00:00
|
|
|
uintptr_t stack_chk_guard = _dl_setup_stack_chk_guard (_dl_random);
|
2007-08-12 19:03:54 +00:00
|
|
|
# ifdef THREAD_SET_STACK_GUARD
|
2005-06-26 18:14:26 +00:00
|
|
|
THREAD_SET_STACK_GUARD (stack_chk_guard);
|
2007-08-12 19:03:54 +00:00
|
|
|
# else
|
2005-06-26 18:14:26 +00:00
|
|
|
__stack_chk_guard = stack_chk_guard;
|
2007-08-12 19:03:54 +00:00
|
|
|
# endif
|
2013-09-23 04:52:09 +00:00
|
|
|
|
2016-12-26 09:08:34 +00:00
|
|
|
# ifdef DL_SYSDEP_OSCHECK
|
|
|
|
if (!__libc_multiple_libcs)
|
|
|
|
{
|
|
|
|
/* This needs to run to initiliaze _dl_osversion before TLS
|
|
|
|
setup might check it. */
|
|
|
|
DL_SYSDEP_OSCHECK (__libc_fatal);
|
|
|
|
}
|
|
|
|
# endif
|
|
|
|
|
|
|
|
/* Initialize libpthread if linked in. */
|
|
|
|
if (__pthread_initialize_minimal != NULL)
|
|
|
|
__pthread_initialize_minimal ();
|
|
|
|
|
2013-09-23 04:52:09 +00:00
|
|
|
/* Set up the pointer guard value. */
|
|
|
|
uintptr_t pointer_chk_guard = _dl_setup_pointer_guard (_dl_random,
|
|
|
|
stack_chk_guard);
|
|
|
|
# ifdef THREAD_SET_POINTER_GUARD
|
|
|
|
THREAD_SET_POINTER_GUARD (pointer_chk_guard);
|
|
|
|
# else
|
|
|
|
__pointer_chk_guard_local = pointer_chk_guard;
|
|
|
|
# endif
|
|
|
|
|
2007-08-12 19:03:54 +00:00
|
|
|
#endif
|
|
|
|
|
1998-03-08 15:26:29 +00:00
|
|
|
/* Register the destructor of the dynamic linker if there is any. */
|
2014-02-10 13:45:42 +00:00
|
|
|
if (__glibc_likely (rtld_fini != NULL))
|
2001-02-26 17:53:15 +00:00
|
|
|
__cxa_atexit ((void (*) (void *)) rtld_fini, NULL, NULL);
|
1998-03-08 15:26:29 +00:00
|
|
|
|
2005-02-14 21:25:50 +00:00
|
|
|
#ifndef SHARED
|
2000-03-30 16:30:49 +00:00
|
|
|
/* Call the initializer of the libc. This is only needed here if we
|
|
|
|
are compiling for the static library in which case we haven't
|
|
|
|
run the constructors in `_dl_start_user'. */
|
1998-03-18 14:42:25 +00:00
|
|
|
__libc_init_first (argc, argv, __environ);
|
|
|
|
|
2005-02-14 21:25:50 +00:00
|
|
|
/* Register the destructor of the program, if any. */
|
|
|
|
if (fini)
|
|
|
|
__cxa_atexit ((void (*) (void *)) fini, NULL, NULL);
|
|
|
|
|
2003-05-06 05:55:00 +00:00
|
|
|
/* Some security at this point. Prevent starting a SUID binary where
|
|
|
|
the standard file descriptors are not opened. We have to do this
|
|
|
|
only for statically linked applications since otherwise the dynamic
|
|
|
|
loader did the work already. */
|
|
|
|
if (__builtin_expect (__libc_enable_secure, 0))
|
|
|
|
__libc_check_standard_fds ();
|
|
|
|
#endif
|
|
|
|
|
1998-04-06 19:08:46 +00:00
|
|
|
/* Call the initializer of the program, if any. */
|
2000-03-31 05:50:02 +00:00
|
|
|
#ifdef SHARED
|
2004-03-05 10:29:47 +00:00
|
|
|
if (__builtin_expect (GLRO(dl_debug_mask) & DL_DEBUG_IMPCALLS, 0))
|
2004-03-06 08:19:29 +00:00
|
|
|
GLRO(dl_debug_printf) ("\ninitialize program: %s\n\n", argv[0]);
|
1998-03-18 14:42:25 +00:00
|
|
|
#endif
|
1998-04-06 19:08:46 +00:00
|
|
|
if (init)
|
2005-07-07 23:00:02 +00:00
|
|
|
(*init) (argc, argv, __environ MAIN_AUXVEC_PARAM);
|
1998-03-08 15:26:29 +00:00
|
|
|
|
2005-01-06 22:40:27 +00:00
|
|
|
#ifdef SHARED
|
|
|
|
/* Auditing checkpoint: we have a new object. */
|
2014-02-10 13:45:42 +00:00
|
|
|
if (__glibc_unlikely (GLRO(dl_naudit) > 0))
|
2005-01-06 22:40:27 +00:00
|
|
|
{
|
|
|
|
struct audit_ifaces *afct = GLRO(dl_audit);
|
|
|
|
struct link_map *head = GL(dl_ns)[LM_ID_BASE]._ns_loaded;
|
|
|
|
for (unsigned int cnt = 0; cnt < GLRO(dl_naudit); ++cnt)
|
|
|
|
{
|
|
|
|
if (afct->preinit != NULL)
|
|
|
|
afct->preinit (&head->l_audit[cnt].cookie);
|
|
|
|
|
|
|
|
afct = afct->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2000-03-31 05:50:02 +00:00
|
|
|
#ifdef SHARED
|
2014-02-10 13:45:42 +00:00
|
|
|
if (__glibc_unlikely (GLRO(dl_debug_mask) & DL_DEBUG_IMPCALLS))
|
2004-03-06 08:19:29 +00:00
|
|
|
GLRO(dl_debug_printf) ("\ntransferring control: %s\n\n", argv[0]);
|
1998-03-18 14:42:25 +00:00
|
|
|
#endif
|
|
|
|
|
2014-04-11 16:43:58 +00:00
|
|
|
#ifndef SHARED
|
|
|
|
_dl_debug_initialize (0, LM_ID_BASE);
|
|
|
|
#endif
|
2003-04-12 00:58:26 +00:00
|
|
|
#ifdef HAVE_CLEANUP_JMP_BUF
|
|
|
|
/* Memory for the cancellation buffer. */
|
|
|
|
struct pthread_unwind_buf unwind_buf;
|
|
|
|
|
|
|
|
int not_first_call;
|
|
|
|
not_first_call = setjmp ((struct __jmp_buf_tag *) unwind_buf.cancel_jmp_buf);
|
2014-02-10 13:45:42 +00:00
|
|
|
if (__glibc_likely (! not_first_call))
|
2002-09-15 02:37:32 +00:00
|
|
|
{
|
2003-04-12 00:58:26 +00:00
|
|
|
struct pthread *self = THREAD_SELF;
|
2002-09-15 02:37:32 +00:00
|
|
|
|
2003-04-12 00:58:26 +00:00
|
|
|
/* Store old info. */
|
|
|
|
unwind_buf.priv.data.prev = THREAD_GETMEM (self, cleanup_jmp_buf);
|
|
|
|
unwind_buf.priv.data.cleanup = THREAD_GETMEM (self, cleanup);
|
2003-03-15 23:09:11 +00:00
|
|
|
|
2003-04-12 00:58:26 +00:00
|
|
|
/* Store the new cleanup handler info. */
|
|
|
|
THREAD_SETMEM (self, cleanup_jmp_buf, &unwind_buf);
|
|
|
|
|
|
|
|
/* Run the program. */
|
|
|
|
result = main (argc, argv, __environ MAIN_AUXVEC_PARAM);
|
2002-09-15 02:37:32 +00:00
|
|
|
}
|
|
|
|
else
|
2003-02-14 23:31:11 +00:00
|
|
|
{
|
2004-03-31 01:47:34 +00:00
|
|
|
/* Remove the thread-local data. */
|
|
|
|
# ifdef SHARED
|
2007-01-17 08:37:26 +00:00
|
|
|
PTHFCT_CALL (ptr__nptl_deallocate_tsd, ());
|
2004-03-31 01:47:34 +00:00
|
|
|
# else
|
|
|
|
extern void __nptl_deallocate_tsd (void) __attribute ((weak));
|
|
|
|
__nptl_deallocate_tsd ();
|
|
|
|
# endif
|
|
|
|
|
2003-02-14 23:31:11 +00:00
|
|
|
/* One less thread. Decrement the counter. If it is zero we
|
|
|
|
terminate the entire process. */
|
|
|
|
result = 0;
|
2003-04-12 00:58:26 +00:00
|
|
|
# ifdef SHARED
|
2007-01-17 08:37:26 +00:00
|
|
|
unsigned int *ptr = __libc_pthread_functions.ptr_nthreads;
|
2012-11-04 20:46:30 +00:00
|
|
|
# ifdef PTR_DEMANGLE
|
2007-01-17 08:37:26 +00:00
|
|
|
PTR_DEMANGLE (ptr);
|
2012-11-04 20:46:30 +00:00
|
|
|
# endif
|
2003-04-12 00:58:26 +00:00
|
|
|
# else
|
[BZ #284, BZ #721]
* intl/dcigettext.c (_nl_find_msg): Add a cast.
* nis/nis_clone_dir.c (nis_clone_directory): Use char * for ADDR.
* nis/nis_clone_obj.c (nis_clone_object): Likewise.
* nis/nis_clone_res.c (nis_clone_result): Likewise.
* resolv/nss_dns/dns-network.c (getanswer_r): Use const unsigned char *
for END_OF_MESSAGE and CP.
* resolv/res_send.c (send_dg): Add else branch for case impossible
unless `poll' is buggy.
* crypt/crypt_util.c (__setkey_r): Add a cast.
* locale/programs/linereader.c (get_toplvl_escape): Use size_t for
NBYTES, and unsigned char * for BYTES.
* locale/programs/charmap.c (charmap_new_char): Use size_t and
unsighed char * for NBYTES, BYTES parameters.
* sysdeps/generic/dl-hash.h (_dl_elf_hash): Take const char * argument
and cast it.
* sysdeps/i386/i686/dl-hash.h (_dl_elf_hash): Likewise.
* sunrpc/create_xid.c (_create_xid): Don't use unsigned long for RES.
* sunrpc/svcauth_des.c (_svcauth_des): Fix cast type.
* sunrpc/auth_des.c (authdes_create): Don't use u_char for PKEY_DATA.
(authdes_marshal): Don't use unsigned int for LEN.
* sunrpc/xdr.c (xdr_hyper): Don't use unsigned long for T2.
(xdr_u_hyper): Likewise.
(xdr_u_short): Don't use u_long for L.
* sunrpc/xdr_intXX_t.c (xdr_int64_t): Don't use uint32_t for T2.
* inet/rexec.c (rexec_af): Use socklen_t.
* sunrpc/key_call.c (getkeyserv_handle): Likewise.
* sunrpc/rtime.c (rtime): Likewise.
* resolv/res_send.c (send_vc, send_dg): Likewise.
* nis/nis_callback.c (__nis_create_callback): Likewise.
* sysdeps/generic/libc-start.c: Use unsigned int for nthreads ptr.
* sysdeps/posix/getaddrinfo.c (gaih_inet): Fix type of ADDR local.
* libio/libio.h (_IO_BE): Add parenthesis around EXPR.
* intl/dcigettext.c (INTVARDEF, INTUSE): Macros removed.
(_nl_default_dirname): Use libc_hidden_data_def instead of INTVARDEF.
(libc_freeres_fn, DCIGETTEXT): Don't use INTUSE.
* intl/bindtextdom.c (INTUSE): Macro removed.
(_nl_default_dirname): Use libc_hidden_proto.
(set_binding_values): Don't use INTUSE.
* include/libintl.h (_libc_intl_domainname_internal): Decl removed.
(_libc_intl_domainname): Use libc_hidden_proto.
* posix/regex_internal.h (gettext): Remove INTUSE on it.
* locale/SYS_libc.c (_libc_intl_domainname): Use libc_hidden_data_def
rather than INTDEF.
* include/libintl.h (_): Don't use *_internal name.
* ctype/ctype-extn.c (__ctype_tolower, __ctype_toupper): Use int32_t,
not uint32_t.
* locale/lc-ctype.c (_nl_postload_ctype): Likewise for assignments.
* iconv/gconv_open.c (__gconv_open): Remove useless cast.
[BZ #721]
* sysdeps/i386/dl-machine.h (ELF_MACHINE_NO_RELA): Define this outside
of [RESOLVE_MAP].
* sysdeps/sh/dl-machine.h (ELF_MACHINE_NO_REL): Likewise.
* sysdeps/powerpc/powerpc32/dl-machine.h
(elf_machine_rel, elf_machine_rel_relative): Removed.
* sysdeps/powerpc/powerpc64/dl-machine.h
(elf_machine_rel, elf_machine_rel_relative): Removed.
2005-02-03 Alexandre Oliva <aoliva@redhat.com>
[BZ #721]
* elf/dynamic-link.h: Don't declare nested auto functions that are
not going to be defined.
2004-07-23 Jakub Jelinek <jakub@redhat.com>
[BZ #284]
* include/features.h (_POSIX_SOURCE, _POSIX_C_SOURCE): Define
if _XOPEN_SOURCE >= 500 even if __STRICT_ANSI__ is defined.
2005-02-16 Roland McGrath <roland@redhat.com>
2005-02-17 01:19:55 +00:00
|
|
|
extern unsigned int __nptl_nthreads __attribute ((weak));
|
|
|
|
unsigned int *const ptr = &__nptl_nthreads;
|
2003-04-12 00:58:26 +00:00
|
|
|
# endif
|
2003-02-14 23:31:11 +00:00
|
|
|
|
|
|
|
if (! atomic_decrement_and_test (ptr))
|
|
|
|
/* Not much left to do but to exit the thread, not the process. */
|
2014-05-13 16:49:20 +00:00
|
|
|
__exit_thread ();
|
2003-02-14 23:31:11 +00:00
|
|
|
}
|
2003-04-12 00:58:26 +00:00
|
|
|
#else
|
|
|
|
/* Nothing fancy, just call the function. */
|
|
|
|
result = main (argc, argv, __environ MAIN_AUXVEC_PARAM);
|
2002-09-15 02:37:32 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
exit (result);
|
1998-03-06 17:21:43 +00:00
|
|
|
}
|