2012-10-05 17:20:39 +00:00
|
|
|
/* Read the dynamic section at DYN and fill in INFO with indices DT_*.
|
2017-01-01 00:14:16 +00:00
|
|
|
Copyright (C) 2012-2017 Free Software Foundation, Inc.
|
2012-10-05 17:20:39 +00:00
|
|
|
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/>. */
|
|
|
|
|
2017-03-02 16:28:41 +00:00
|
|
|
/* This file is included multiple times and therefore lacks a header
|
|
|
|
file inclusion guard. */
|
Narrowing the visibility of libc-internal.h even further.
posix/wordexp-test.c used libc-internal.h for PTR_ALIGN_DOWN; similar
to what was done with libc-diag.h, I have split the definitions of
cast_to_integer, ALIGN_UP, ALIGN_DOWN, PTR_ALIGN_UP, and PTR_ALIGN_DOWN
to a new header, libc-pointer-arith.h.
It then occurred to me that the remaining declarations in libc-internal.h
are mostly to do with early initialization, and probably most of the
files including it, even in the core code, don't need it anymore. Indeed,
only 19 files actually need what remains of libc-internal.h. 23 others
need libc-diag.h instead, and 12 need libc-pointer-arith.h instead.
No file needs more than one of them, and 16 don't need any of them!
So, with this patch, libc-internal.h stops including libc-diag.h as
well as losing the pointer arithmetic macros, and all including files
are adjusted.
* include/libc-pointer-arith.h: New file. Define
cast_to_integer, ALIGN_UP, ALIGN_DOWN, PTR_ALIGN_UP, and
PTR_ALIGN_DOWN here.
* include/libc-internal.h: Definitions of above macros
moved from here. Don't include libc-diag.h anymore either.
* posix/wordexp-test.c: Include stdint.h and libc-pointer-arith.h.
Don't include libc-internal.h.
* debug/pcprofile.c, elf/dl-tunables.c, elf/soinit.c, io/openat.c
* io/openat64.c, misc/ptrace.c, nptl/pthread_clock_gettime.c
* nptl/pthread_clock_settime.c, nptl/pthread_cond_common.c
* string/strcoll_l.c, sysdeps/nacl/brk.c
* sysdeps/unix/clock_settime.c
* sysdeps/unix/sysv/linux/i386/get_clockfreq.c
* sysdeps/unix/sysv/linux/ia64/get_clockfreq.c
* sysdeps/unix/sysv/linux/powerpc/get_clockfreq.c
* sysdeps/unix/sysv/linux/sparc/sparc64/get_clockfreq.c:
Don't include libc-internal.h.
* elf/get-dynamic-info.h, iconv/loop.c
* iconvdata/iso-2022-cn-ext.c, locale/weight.h, locale/weightwc.h
* misc/reboot.c, nis/nis_table.c, nptl_db/thread_dbP.h
* nscd/connections.c, resolv/res_send.c, soft-fp/fmadf4.c
* soft-fp/fmasf4.c, soft-fp/fmatf4.c, stdio-common/vfscanf.c
* sysdeps/ieee754/dbl-64/e_lgamma_r.c
* sysdeps/ieee754/dbl-64/k_rem_pio2.c
* sysdeps/ieee754/flt-32/e_lgammaf_r.c
* sysdeps/ieee754/flt-32/k_rem_pio2f.c
* sysdeps/ieee754/ldbl-128/k_tanl.c
* sysdeps/ieee754/ldbl-128ibm/k_tanl.c
* sysdeps/ieee754/ldbl-96/e_lgammal_r.c
* sysdeps/ieee754/ldbl-96/k_tanl.c, sysdeps/nptl/futex-internal.h:
Include libc-diag.h instead of libc-internal.h.
* elf/dl-load.c, elf/dl-reloc.c, locale/programs/locarchive.c
* nptl/nptl-init.c, string/strcspn.c, string/strspn.c
* malloc/malloc.c, sysdeps/i386/nptl/tls.h
* sysdeps/nacl/dl-map-segments.h, sysdeps/x86_64/atomic-machine.h
* sysdeps/unix/sysv/linux/spawni.c
* sysdeps/x86_64/nptl/tls.h:
Include libc-pointer-arith.h instead of libc-internal.h.
* elf/get-dynamic-info.h, sysdeps/nacl/dl-map-segments.h
* sysdeps/x86_64/atomic-machine.h:
Add multiple include guard.
2017-02-27 01:17:52 +00:00
|
|
|
|
2012-10-05 17:20:39 +00:00
|
|
|
#include <assert.h>
|
Narrowing the visibility of libc-internal.h even further.
posix/wordexp-test.c used libc-internal.h for PTR_ALIGN_DOWN; similar
to what was done with libc-diag.h, I have split the definitions of
cast_to_integer, ALIGN_UP, ALIGN_DOWN, PTR_ALIGN_UP, and PTR_ALIGN_DOWN
to a new header, libc-pointer-arith.h.
It then occurred to me that the remaining declarations in libc-internal.h
are mostly to do with early initialization, and probably most of the
files including it, even in the core code, don't need it anymore. Indeed,
only 19 files actually need what remains of libc-internal.h. 23 others
need libc-diag.h instead, and 12 need libc-pointer-arith.h instead.
No file needs more than one of them, and 16 don't need any of them!
So, with this patch, libc-internal.h stops including libc-diag.h as
well as losing the pointer arithmetic macros, and all including files
are adjusted.
* include/libc-pointer-arith.h: New file. Define
cast_to_integer, ALIGN_UP, ALIGN_DOWN, PTR_ALIGN_UP, and
PTR_ALIGN_DOWN here.
* include/libc-internal.h: Definitions of above macros
moved from here. Don't include libc-diag.h anymore either.
* posix/wordexp-test.c: Include stdint.h and libc-pointer-arith.h.
Don't include libc-internal.h.
* debug/pcprofile.c, elf/dl-tunables.c, elf/soinit.c, io/openat.c
* io/openat64.c, misc/ptrace.c, nptl/pthread_clock_gettime.c
* nptl/pthread_clock_settime.c, nptl/pthread_cond_common.c
* string/strcoll_l.c, sysdeps/nacl/brk.c
* sysdeps/unix/clock_settime.c
* sysdeps/unix/sysv/linux/i386/get_clockfreq.c
* sysdeps/unix/sysv/linux/ia64/get_clockfreq.c
* sysdeps/unix/sysv/linux/powerpc/get_clockfreq.c
* sysdeps/unix/sysv/linux/sparc/sparc64/get_clockfreq.c:
Don't include libc-internal.h.
* elf/get-dynamic-info.h, iconv/loop.c
* iconvdata/iso-2022-cn-ext.c, locale/weight.h, locale/weightwc.h
* misc/reboot.c, nis/nis_table.c, nptl_db/thread_dbP.h
* nscd/connections.c, resolv/res_send.c, soft-fp/fmadf4.c
* soft-fp/fmasf4.c, soft-fp/fmatf4.c, stdio-common/vfscanf.c
* sysdeps/ieee754/dbl-64/e_lgamma_r.c
* sysdeps/ieee754/dbl-64/k_rem_pio2.c
* sysdeps/ieee754/flt-32/e_lgammaf_r.c
* sysdeps/ieee754/flt-32/k_rem_pio2f.c
* sysdeps/ieee754/ldbl-128/k_tanl.c
* sysdeps/ieee754/ldbl-128ibm/k_tanl.c
* sysdeps/ieee754/ldbl-96/e_lgammal_r.c
* sysdeps/ieee754/ldbl-96/k_tanl.c, sysdeps/nptl/futex-internal.h:
Include libc-diag.h instead of libc-internal.h.
* elf/dl-load.c, elf/dl-reloc.c, locale/programs/locarchive.c
* nptl/nptl-init.c, string/strcspn.c, string/strspn.c
* malloc/malloc.c, sysdeps/i386/nptl/tls.h
* sysdeps/nacl/dl-map-segments.h, sysdeps/x86_64/atomic-machine.h
* sysdeps/unix/sysv/linux/spawni.c
* sysdeps/x86_64/nptl/tls.h:
Include libc-pointer-arith.h instead of libc-internal.h.
* elf/get-dynamic-info.h, sysdeps/nacl/dl-map-segments.h
* sysdeps/x86_64/atomic-machine.h:
Add multiple include guard.
2017-02-27 01:17:52 +00:00
|
|
|
#include <libc-diag.h>
|
2012-10-05 17:20:39 +00:00
|
|
|
|
|
|
|
#ifndef RESOLVE_MAP
|
|
|
|
static
|
|
|
|
#else
|
|
|
|
auto
|
|
|
|
#endif
|
|
|
|
inline void __attribute__ ((unused, always_inline))
|
|
|
|
elf_get_dynamic_info (struct link_map *l, ElfW(Dyn) *temp)
|
|
|
|
{
|
|
|
|
ElfW(Dyn) *dyn = l->l_ld;
|
|
|
|
ElfW(Dyn) **info;
|
|
|
|
#if __ELF_NATIVE_CLASS == 32
|
|
|
|
typedef Elf32_Word d_tag_utype;
|
|
|
|
#elif __ELF_NATIVE_CLASS == 64
|
|
|
|
typedef Elf64_Xword d_tag_utype;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef RTLD_BOOTSTRAP
|
|
|
|
if (dyn == NULL)
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
info = l->l_info;
|
|
|
|
|
|
|
|
while (dyn->d_tag != DT_NULL)
|
|
|
|
{
|
|
|
|
if ((d_tag_utype) dyn->d_tag < DT_NUM)
|
|
|
|
info[dyn->d_tag] = dyn;
|
|
|
|
else if (dyn->d_tag >= DT_LOPROC &&
|
|
|
|
dyn->d_tag < DT_LOPROC + DT_THISPROCNUM)
|
2014-12-19 21:23:40 +00:00
|
|
|
{
|
|
|
|
/* This does not violate the array bounds of l->l_info, but
|
|
|
|
gcc 4.6 on sparc somehow does not see this. */
|
|
|
|
DIAG_PUSH_NEEDS_COMMENT;
|
|
|
|
DIAG_IGNORE_NEEDS_COMMENT (4.6,
|
|
|
|
"-Warray-bounds");
|
|
|
|
info[dyn->d_tag - DT_LOPROC + DT_NUM] = dyn;
|
|
|
|
DIAG_POP_NEEDS_COMMENT;
|
|
|
|
}
|
2012-10-05 17:20:39 +00:00
|
|
|
else if ((d_tag_utype) DT_VERSIONTAGIDX (dyn->d_tag) < DT_VERSIONTAGNUM)
|
|
|
|
info[VERSYMIDX (dyn->d_tag)] = dyn;
|
|
|
|
else if ((d_tag_utype) DT_EXTRATAGIDX (dyn->d_tag) < DT_EXTRANUM)
|
|
|
|
info[DT_EXTRATAGIDX (dyn->d_tag) + DT_NUM + DT_THISPROCNUM
|
|
|
|
+ DT_VERSIONTAGNUM] = dyn;
|
|
|
|
else if ((d_tag_utype) DT_VALTAGIDX (dyn->d_tag) < DT_VALNUM)
|
|
|
|
info[DT_VALTAGIDX (dyn->d_tag) + DT_NUM + DT_THISPROCNUM
|
|
|
|
+ DT_VERSIONTAGNUM + DT_EXTRANUM] = dyn;
|
|
|
|
else if ((d_tag_utype) DT_ADDRTAGIDX (dyn->d_tag) < DT_ADDRNUM)
|
|
|
|
info[DT_ADDRTAGIDX (dyn->d_tag) + DT_NUM + DT_THISPROCNUM
|
|
|
|
+ DT_VERSIONTAGNUM + DT_EXTRANUM + DT_VALNUM] = dyn;
|
|
|
|
++dyn;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define DL_RO_DYN_TEMP_CNT 8
|
|
|
|
|
|
|
|
#ifndef DL_RO_DYN_SECTION
|
|
|
|
/* Don't adjust .dynamic unnecessarily. */
|
|
|
|
if (l->l_addr != 0)
|
|
|
|
{
|
|
|
|
ElfW(Addr) l_addr = l->l_addr;
|
|
|
|
int cnt = 0;
|
|
|
|
|
|
|
|
# define ADJUST_DYN_INFO(tag) \
|
|
|
|
do \
|
|
|
|
if (info[tag] != NULL) \
|
|
|
|
{ \
|
|
|
|
if (temp) \
|
|
|
|
{ \
|
|
|
|
temp[cnt].d_tag = info[tag]->d_tag; \
|
|
|
|
temp[cnt].d_un.d_ptr = info[tag]->d_un.d_ptr + l_addr; \
|
|
|
|
info[tag] = temp + cnt++; \
|
|
|
|
} \
|
|
|
|
else \
|
|
|
|
info[tag]->d_un.d_ptr += l_addr; \
|
|
|
|
} \
|
|
|
|
while (0)
|
|
|
|
|
|
|
|
ADJUST_DYN_INFO (DT_HASH);
|
|
|
|
ADJUST_DYN_INFO (DT_PLTGOT);
|
|
|
|
ADJUST_DYN_INFO (DT_STRTAB);
|
|
|
|
ADJUST_DYN_INFO (DT_SYMTAB);
|
|
|
|
# if ! ELF_MACHINE_NO_RELA
|
|
|
|
ADJUST_DYN_INFO (DT_RELA);
|
|
|
|
# endif
|
|
|
|
# if ! ELF_MACHINE_NO_REL
|
|
|
|
ADJUST_DYN_INFO (DT_REL);
|
|
|
|
# endif
|
|
|
|
ADJUST_DYN_INFO (DT_JMPREL);
|
|
|
|
ADJUST_DYN_INFO (VERSYMIDX (DT_VERSYM));
|
|
|
|
ADJUST_DYN_INFO (DT_ADDRTAGIDX (DT_GNU_HASH) + DT_NUM + DT_THISPROCNUM
|
|
|
|
+ DT_VERSIONTAGNUM + DT_EXTRANUM + DT_VALNUM);
|
|
|
|
# undef ADJUST_DYN_INFO
|
|
|
|
assert (cnt <= DL_RO_DYN_TEMP_CNT);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (info[DT_PLTREL] != NULL)
|
|
|
|
{
|
|
|
|
#if ELF_MACHINE_NO_RELA
|
|
|
|
assert (info[DT_PLTREL]->d_un.d_val == DT_REL);
|
|
|
|
#elif ELF_MACHINE_NO_REL
|
|
|
|
assert (info[DT_PLTREL]->d_un.d_val == DT_RELA);
|
|
|
|
#else
|
|
|
|
assert (info[DT_PLTREL]->d_un.d_val == DT_REL
|
|
|
|
|| info[DT_PLTREL]->d_un.d_val == DT_RELA);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
#if ! ELF_MACHINE_NO_RELA
|
|
|
|
if (info[DT_RELA] != NULL)
|
|
|
|
assert (info[DT_RELAENT]->d_un.d_val == sizeof (ElfW(Rela)));
|
|
|
|
# endif
|
|
|
|
# if ! ELF_MACHINE_NO_REL
|
|
|
|
if (info[DT_REL] != NULL)
|
|
|
|
assert (info[DT_RELENT]->d_un.d_val == sizeof (ElfW(Rel)));
|
|
|
|
#endif
|
|
|
|
#ifdef RTLD_BOOTSTRAP
|
|
|
|
/* Only the bind now flags are allowed. */
|
|
|
|
assert (info[VERSYMIDX (DT_FLAGS_1)] == NULL
|
|
|
|
|| (info[VERSYMIDX (DT_FLAGS_1)]->d_un.d_val & ~DF_1_NOW) == 0);
|
|
|
|
assert (info[DT_FLAGS] == NULL
|
|
|
|
|| (info[DT_FLAGS]->d_un.d_val & ~DF_BIND_NOW) == 0);
|
|
|
|
/* Flags must not be set for ld.so. */
|
2013-12-05 13:47:34 +00:00
|
|
|
assert (info[DT_RUNPATH] == NULL);
|
|
|
|
assert (info[DT_RPATH] == NULL);
|
2012-10-05 17:20:39 +00:00
|
|
|
#else
|
|
|
|
if (info[DT_FLAGS] != NULL)
|
|
|
|
{
|
|
|
|
/* Flags are used. Translate to the old form where available.
|
|
|
|
Since these l_info entries are only tested for NULL pointers it
|
|
|
|
is ok if they point to the DT_FLAGS entry. */
|
|
|
|
l->l_flags = info[DT_FLAGS]->d_un.d_val;
|
|
|
|
|
|
|
|
if (l->l_flags & DF_SYMBOLIC)
|
|
|
|
info[DT_SYMBOLIC] = info[DT_FLAGS];
|
|
|
|
if (l->l_flags & DF_TEXTREL)
|
|
|
|
info[DT_TEXTREL] = info[DT_FLAGS];
|
|
|
|
if (l->l_flags & DF_BIND_NOW)
|
|
|
|
info[DT_BIND_NOW] = info[DT_FLAGS];
|
|
|
|
}
|
|
|
|
if (info[VERSYMIDX (DT_FLAGS_1)] != NULL)
|
|
|
|
{
|
|
|
|
l->l_flags_1 = info[VERSYMIDX (DT_FLAGS_1)]->d_un.d_val;
|
|
|
|
|
2012-11-30 02:09:12 +00:00
|
|
|
/* Only DT_1_SUPPORTED_MASK bits are supported, and we would like
|
|
|
|
to assert this, but we can't. Users have been setting
|
|
|
|
unsupported DF_1_* flags for a long time and glibc has ignored
|
|
|
|
them. Therefore to avoid breaking existing applications the
|
|
|
|
best we can do is add a warning during debugging with the
|
|
|
|
intent of notifying the user of the problem. */
|
|
|
|
if (__builtin_expect (GLRO(dl_debug_mask) & DL_DEBUG_FILES, 0)
|
|
|
|
&& l->l_flags_1 & ~DT_1_SUPPORTED_MASK)
|
|
|
|
_dl_debug_printf ("\nWARNING: Unsupported flag value(s) of 0x%x in DT_FLAGS_1.\n",
|
|
|
|
l->l_flags_1 & ~DT_1_SUPPORTED_MASK);
|
2012-11-21 14:33:19 +00:00
|
|
|
|
2012-10-05 17:20:39 +00:00
|
|
|
if (l->l_flags_1 & DF_1_NOW)
|
|
|
|
info[DT_BIND_NOW] = info[VERSYMIDX (DT_FLAGS_1)];
|
|
|
|
}
|
|
|
|
if (info[DT_RUNPATH] != NULL)
|
|
|
|
/* If both RUNPATH and RPATH are given, the latter is ignored. */
|
|
|
|
info[DT_RPATH] = NULL;
|
|
|
|
#endif
|
|
|
|
}
|