Blame elf/dl-fptr.c

Packit 6c4009
/* Manage function descriptors.  Generic version.
Packit 6c4009
   Copyright (C) 1999-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <libintl.h>
Packit 6c4009
#include <unistd.h>
Packit 6c4009
#include <string.h>
Packit 6c4009
#include <sys/param.h>
Packit 6c4009
#include <sys/mman.h>
Packit 6c4009
#include <link.h>
Packit 6c4009
#include <ldsodefs.h>
Packit 6c4009
#include <elf/dynamic-link.h>
Packit 6c4009
#include <dl-fptr.h>
Packit 6c4009
#include <dl-unmap-segments.h>
Packit 6c4009
#include <atomic.h>
Packit 6c4009
Packit 6c4009
#ifndef ELF_MACHINE_BOOT_FPTR_TABLE_LEN
Packit 6c4009
/* ELF_MACHINE_BOOT_FPTR_TABLE_LEN should be greater than the number of
Packit 6c4009
   dynamic symbols in ld.so.  */
Packit 6c4009
# define ELF_MACHINE_BOOT_FPTR_TABLE_LEN 256
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#ifndef ELF_MACHINE_LOAD_ADDRESS
Packit 6c4009
# error "ELF_MACHINE_LOAD_ADDRESS is not defined."
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
#ifndef COMPARE_AND_SWAP
Packit 6c4009
# define COMPARE_AND_SWAP(ptr, old, new) \
Packit 6c4009
  (catomic_compare_and_exchange_bool_acq (ptr, new, old) == 0)
Packit 6c4009
#endif
Packit 6c4009
Packit 6c4009
ElfW(Addr) _dl_boot_fptr_table [ELF_MACHINE_BOOT_FPTR_TABLE_LEN];
Packit 6c4009
Packit 6c4009
static struct local
Packit 6c4009
  {
Packit 6c4009
    struct fdesc_table *root;
Packit 6c4009
    struct fdesc *free_list;
Packit 6c4009
    unsigned int npages;		/* # of pages to allocate */
Packit 6c4009
    /* the next to members MUST be consecutive! */
Packit 6c4009
    struct fdesc_table boot_table;
Packit 6c4009
    struct fdesc boot_fdescs[1024];
Packit 6c4009
  }
Packit 6c4009
local =
Packit 6c4009
  {
Packit 6c4009
    .root = &local.boot_table,
Packit 6c4009
    .npages = 2,
Packit 6c4009
    .boot_table =
Packit 6c4009
      {
Packit 6c4009
	.len = sizeof (local.boot_fdescs) / sizeof (local.boot_fdescs[0]),
Packit 6c4009
	.first_unused = 0
Packit 6c4009
      }
Packit 6c4009
  };
Packit 6c4009
Packit 6c4009
/* Create a new fdesc table and return a pointer to the first fdesc
Packit 6c4009
   entry.  The fdesc lock must have been acquired already.  */
Packit 6c4009
Packit 6c4009
static struct fdesc_table *
Packit 6c4009
new_fdesc_table (struct local *l, size_t *size)
Packit 6c4009
{
Packit 6c4009
  size_t old_npages = l->npages;
Packit 6c4009
  size_t new_npages = old_npages + old_npages;
Packit 6c4009
  struct fdesc_table *new_table;
Packit 6c4009
Packit 6c4009
  /* If someone has just created a new table, we return NULL to tell
Packit 6c4009
     the caller to use the new table.  */
Packit 6c4009
  if (! COMPARE_AND_SWAP (&l->npages, old_npages, new_npages))
Packit 6c4009
    return (struct fdesc_table *) NULL;
Packit 6c4009
Packit 6c4009
  *size = old_npages * GLRO(dl_pagesize);
Packit 6c4009
  new_table = __mmap (NULL, *size,
Packit 6c4009
		      PROT_READ | PROT_WRITE, MAP_ANON | MAP_PRIVATE, -1, 0);
Packit 6c4009
  if (new_table == MAP_FAILED)
Packit 6c4009
    _dl_signal_error (errno, NULL, NULL,
Packit 6c4009
		      N_("cannot map pages for fdesc table"));
Packit 6c4009
Packit 6c4009
  new_table->len
Packit 6c4009
    = (*size - sizeof (*new_table)) / sizeof (struct fdesc);
Packit 6c4009
  new_table->first_unused = 1;
Packit 6c4009
  return new_table;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
static ElfW(Addr)
Packit 6c4009
make_fdesc (ElfW(Addr) ip, ElfW(Addr) gp)
Packit 6c4009
{
Packit 6c4009
  struct fdesc *fdesc = NULL;
Packit 6c4009
  struct fdesc_table *root;
Packit 6c4009
  unsigned int old;
Packit 6c4009
  struct local *l;
Packit 6c4009
Packit 6c4009
  ELF_MACHINE_LOAD_ADDRESS (l, local);
Packit 6c4009
Packit 6c4009
 retry:
Packit 6c4009
  root = l->root;
Packit 6c4009
  while (1)
Packit 6c4009
    {
Packit 6c4009
      old = root->first_unused;
Packit 6c4009
      if (old >= root->len)
Packit 6c4009
	break;
Packit 6c4009
      else if (COMPARE_AND_SWAP (&root->first_unused, old, old + 1))
Packit 6c4009
	{
Packit 6c4009
	  fdesc = &root->fdesc[old];
Packit 6c4009
	  goto install;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  if (l->free_list)
Packit 6c4009
    {
Packit 6c4009
      /* Get it from free-list.  */
Packit 6c4009
      do
Packit 6c4009
	{
Packit 6c4009
	  fdesc = l->free_list;
Packit 6c4009
	  if (fdesc == NULL)
Packit 6c4009
	    goto retry;
Packit 6c4009
	}
Packit 6c4009
      while (! COMPARE_AND_SWAP ((ElfW(Addr) *) &l->free_list,
Packit 6c4009
				 (ElfW(Addr)) fdesc, fdesc->ip));
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      /* Create a new fdesc table.  */
Packit 6c4009
      size_t size;
Packit 6c4009
      struct fdesc_table *new_table = new_fdesc_table (l, &size);
Packit 6c4009
Packit 6c4009
      if (new_table == NULL)
Packit 6c4009
	goto retry;
Packit 6c4009
Packit 6c4009
      new_table->next = root;
Packit 6c4009
      if (! COMPARE_AND_SWAP ((ElfW(Addr) *) &l->root,
Packit 6c4009
			      (ElfW(Addr)) root,
Packit 6c4009
			      (ElfW(Addr)) new_table))
Packit 6c4009
	{
Packit 6c4009
	  /* Someone has just installed a new table. Return NULL to
Packit 6c4009
	     tell the caller to use the new table.  */
Packit 6c4009
	  __munmap (new_table, size);
Packit 6c4009
	  goto retry;
Packit 6c4009
	}
Packit 6c4009
Packit 6c4009
      /* Note that the first entry was reserved while allocating the
Packit 6c4009
	 memory for the new page.  */
Packit 6c4009
      fdesc = &new_table->fdesc[0];
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
 install:
Packit 6c4009
  fdesc->ip = ip;
Packit 6c4009
  fdesc->gp = gp;
Packit 6c4009
Packit 6c4009
  return (ElfW(Addr)) fdesc;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
static inline ElfW(Addr) * __attribute__ ((always_inline))
Packit 6c4009
make_fptr_table (struct link_map *map)
Packit 6c4009
{
Packit 6c4009
  const ElfW(Sym) *symtab
Packit 6c4009
    = (const void *) D_PTR (map, l_info[DT_SYMTAB]);
Packit 6c4009
  const char *strtab = (const void *) D_PTR (map, l_info[DT_STRTAB]);
Packit 6c4009
  ElfW(Addr) *fptr_table;
Packit 6c4009
  size_t size;
Packit 6c4009
  size_t len;
Packit 6c4009
Packit 6c4009
  /* XXX Apparently the only way to find out the size of the dynamic
Packit 6c4009
     symbol section is to assume that the string table follows right
Packit 6c4009
     afterwards...  */
Packit 6c4009
  len = ((strtab - (char *) symtab)
Packit 6c4009
	 / map->l_info[DT_SYMENT]->d_un.d_val);
Packit 6c4009
  size = ((len * sizeof (fptr_table[0]) + GLRO(dl_pagesize) - 1)
Packit 6c4009
	  & -GLRO(dl_pagesize));
Packit 6c4009
  /* XXX We don't support here in the moment systems without MAP_ANON.
Packit 6c4009
     There probably are none for IA-64.  In case this is proven wrong
Packit 6c4009
     we will have to open /dev/null here and use the file descriptor
Packit 6c4009
     instead of the hard-coded -1.  */
Packit 6c4009
  fptr_table = __mmap (NULL, size,
Packit 6c4009
		       PROT_READ | PROT_WRITE, MAP_ANON | MAP_PRIVATE,
Packit 6c4009
		       -1, 0);
Packit 6c4009
  if (fptr_table == MAP_FAILED)
Packit 6c4009
    _dl_signal_error (errno, NULL, NULL,
Packit 6c4009
		      N_("cannot map pages for fptr table"));
Packit 6c4009
Packit 6c4009
  if (COMPARE_AND_SWAP ((ElfW(Addr) *) &map->l_mach.fptr_table,
Packit 6c4009
			(ElfW(Addr)) NULL, (ElfW(Addr)) fptr_table))
Packit 6c4009
    map->l_mach.fptr_table_len = len;
Packit 6c4009
  else
Packit 6c4009
    __munmap (fptr_table, len * sizeof (fptr_table[0]));
Packit 6c4009
Packit 6c4009
  return map->l_mach.fptr_table;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
ElfW(Addr)
Packit 6c4009
_dl_make_fptr (struct link_map *map, const ElfW(Sym) *sym,
Packit 6c4009
	       ElfW(Addr) ip)
Packit 6c4009
{
Packit 6c4009
  ElfW(Addr) *ftab = map->l_mach.fptr_table;
Packit 6c4009
  const ElfW(Sym) *symtab;
Packit 6c4009
  Elf_Symndx symidx;
Packit 6c4009
  struct local *l;
Packit 6c4009
Packit 6c4009
  if (__glibc_unlikely (ftab == NULL))
Packit 6c4009
    ftab = make_fptr_table (map);
Packit 6c4009
Packit 6c4009
  symtab = (const void *) D_PTR (map, l_info[DT_SYMTAB]);
Packit 6c4009
  symidx = sym - symtab;
Packit 6c4009
Packit 6c4009
  if (symidx >= map->l_mach.fptr_table_len)
Packit 6c4009
    _dl_signal_error (0, NULL, NULL,
Packit 6c4009
		      N_("internal error: symidx out of range of fptr table"));
Packit 6c4009
Packit 6c4009
  while (ftab[symidx] == 0)
Packit 6c4009
    {
Packit 6c4009
      /* GOT has already been relocated in elf_get_dynamic_info -
Packit 6c4009
	 don't try to relocate it again.  */
Packit 6c4009
      ElfW(Addr) fdesc
Packit 6c4009
	= make_fdesc (ip, map->l_info[DT_PLTGOT]->d_un.d_ptr);
Packit 6c4009
Packit 6c4009
      if (__builtin_expect (COMPARE_AND_SWAP (&ftab[symidx], (ElfW(Addr)) NULL,
Packit 6c4009
					      fdesc), 1))
Packit 6c4009
	{
Packit 6c4009
	  /* Noone has updated the entry and the new function
Packit 6c4009
	     descriptor has been installed.  */
Packit 6c4009
#if 0
Packit 6c4009
	  const char *strtab
Packit 6c4009
	    = (const void *) D_PTR (map, l_info[DT_STRTAB]);
Packit 6c4009
Packit 6c4009
	  ELF_MACHINE_LOAD_ADDRESS (l, local);
Packit 6c4009
	  if (l->root != &l->boot_table
Packit 6c4009
	      || l->boot_table.first_unused > 20)
Packit 6c4009
	    _dl_debug_printf ("created fdesc symbol `%s' at %lx\n",
Packit 6c4009
			      strtab + sym->st_name, ftab[symidx]);
Packit 6c4009
#endif
Packit 6c4009
	  break;
Packit 6c4009
	}
Packit 6c4009
      else
Packit 6c4009
	{
Packit 6c4009
	  /* We created a duplicated function descriptor. We put it on
Packit 6c4009
	     free-list.  */
Packit 6c4009
	  struct fdesc *f = (struct fdesc *) fdesc;
Packit 6c4009
Packit 6c4009
	  ELF_MACHINE_LOAD_ADDRESS (l, local);
Packit 6c4009
Packit 6c4009
	  do
Packit 6c4009
	    f->ip = (ElfW(Addr)) l->free_list;
Packit 6c4009
	  while (! COMPARE_AND_SWAP ((ElfW(Addr) *) &l->free_list,
Packit 6c4009
				     f->ip, fdesc));
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return ftab[symidx];
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
void
Packit 6c4009
_dl_unmap (struct link_map *map)
Packit 6c4009
{
Packit 6c4009
  ElfW(Addr) *ftab = map->l_mach.fptr_table;
Packit 6c4009
  struct fdesc *head = NULL, *tail = NULL;
Packit 6c4009
  size_t i;
Packit 6c4009
Packit 6c4009
  _dl_unmap_segments (map);
Packit 6c4009
Packit 6c4009
  if (ftab == NULL)
Packit 6c4009
    return;
Packit 6c4009
Packit 6c4009
  /* String together the fdesc structures that are being freed.  */
Packit 6c4009
  for (i = 0; i < map->l_mach.fptr_table_len; ++i)
Packit 6c4009
    {
Packit 6c4009
      if (ftab[i])
Packit 6c4009
	{
Packit 6c4009
	  *(struct fdesc **) ftab[i] = head;
Packit 6c4009
	  head = (struct fdesc *) ftab[i];
Packit 6c4009
	  if (tail == NULL)
Packit 6c4009
	    tail = head;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  /* Prepend the new list to the free_list: */
Packit 6c4009
  if (tail)
Packit 6c4009
    do
Packit 6c4009
      tail->ip = (ElfW(Addr)) local.free_list;
Packit 6c4009
    while (! COMPARE_AND_SWAP ((ElfW(Addr) *) &local.free_list,
Packit 6c4009
			       tail->ip, (ElfW(Addr)) head));
Packit 6c4009
Packit 6c4009
  __munmap (ftab, (map->l_mach.fptr_table_len
Packit 6c4009
		   * sizeof (map->l_mach.fptr_table[0])));
Packit 6c4009
Packit 6c4009
  map->l_mach.fptr_table = NULL;
Packit 6c4009
}
Packit 6c4009
Packit 6c4009
Packit 6c4009
ElfW(Addr)
Packit 6c4009
_dl_lookup_address (const void *address)
Packit 6c4009
{
Packit 6c4009
  ElfW(Addr) addr = (ElfW(Addr)) address;
Packit 6c4009
  struct fdesc_table *t;
Packit 6c4009
  unsigned long int i;
Packit 6c4009
Packit 6c4009
  for (t = local.root; t != NULL; t = t->next)
Packit 6c4009
    {
Packit 6c4009
      i = (struct fdesc *) addr - &t->fdesc[0];
Packit 6c4009
      if (i < t->first_unused && addr == (ElfW(Addr)) &t->fdesc[i])
Packit 6c4009
	{
Packit 6c4009
	  addr = t->fdesc[i].ip;
Packit 6c4009
	  break;
Packit 6c4009
	}
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return addr;
Packit 6c4009
}