+++ /dev/null
-/* Program to read the IL symbol table.
- Copyright (C) 2008-2019 Free Software Foundation, Inc.
- Contributed by Rafael Avila de Espindola (espindola@google.com).
-
-This program is free software; you can redistribute it and/or modify
-it under the terms of the GNU General Public License as published by
-the Free Software Foundation; either version 3 of the License, or
-(at your option) any later version.
-
-This program 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 General Public License for more details.
-
-You should have received a copy of the GNU General Public License
-along with this program; if not, write to the Free Software
-Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA. */
-
-#include <fcntl.h>
-#include <assert.h>
-#include <dlfcn.h>
-#include <stdio.h>
-#include <inttypes.h>
-#include <stdlib.h>
-#include <string.h>
-
-#include "plugin-api.h"
-#include "../gcc/lto/common.h"
-
-/* The presence of gelf.h is checked by the toplevel configure script. */
-# include <gelf.h>
-
-static ld_plugin_claim_file_handler claim_file_handler;
-static ld_plugin_all_symbols_read_handler all_symbols_read_handler;
-static ld_plugin_cleanup_handler cleanup_handler;
-static void *plugin_handle;
-
-struct file_handle {
- unsigned nsyms;
- struct ld_plugin_symbol *syms;
-};
-
-static struct file_handle **all_file_handles = NULL;
-static unsigned int num_file_handles;
-
-/* Write NSYMS symbols from file HANDLE in SYMS. */
-
-static enum ld_plugin_status
-get_symbols (const void *handle, int nsyms, struct ld_plugin_symbol *syms)
-{
- unsigned i;
- struct file_handle *h = (struct file_handle *) handle;
- assert (h->nsyms == nsyms);
-
- for (i = 0; i < nsyms; i++)
- syms[i] = h->syms[i];
-
- return LDPS_OK;
-}
-
-/* Register HANDLER as the callback for notifying the plugin that all symbols
- have been read. */
-
-static enum ld_plugin_status
-register_all_symbols_read (ld_plugin_all_symbols_read_handler handler)
-{
- all_symbols_read_handler = handler;
- return LDPS_OK;
-}
-
-/* Register HANDLER as the callback for claiming a file. */
-
-static enum ld_plugin_status
-register_claim_file(ld_plugin_claim_file_handler handler)
-{
- claim_file_handler = handler;
- return LDPS_OK;
-}
-
-/* Register HANDLER as the callback to removing temporary files. */
-
-static enum ld_plugin_status
-register_cleanup (ld_plugin_cleanup_handler handler)
-{
- cleanup_handler = handler;
- return LDPS_OK;
-}
-
-/* For a file identified by HANDLE, add NSYMS symbols from SYMS. */
-
-static enum ld_plugin_status
-add_symbols (void *handle, int nsyms,
- const struct ld_plugin_symbol *syms)
-{
- int i;
- struct file_handle *h = (struct file_handle *) handle;
- h->nsyms = nsyms;
- h->syms = calloc (nsyms, sizeof (struct ld_plugin_symbol));
- assert (h->syms);
-
- for (i = 0; i < nsyms; i++)
- {
- h->syms[i] = syms[i];
- h->syms[i].name = strdup (h->syms[i].name);
- if (h->syms[i].version)
- h->syms[i].version = strdup (h->syms[i].version);
- if (h->syms[i].comdat_key)
- h->syms[i].comdat_key = strdup (h->syms[i].comdat_key);
- }
-
- return LDPS_OK;
-}
-
-struct ld_plugin_tv tv[] = {
- {LDPT_REGISTER_CLAIM_FILE_HOOK,
- {.tv_register_claim_file = register_claim_file}
- },
- {LDPT_ADD_SYMBOLS,
- {.tv_add_symbols = add_symbols}
- },
-
- {LDPT_REGISTER_ALL_SYMBOLS_READ_HOOK,
- {.tv_register_all_symbols_read = register_all_symbols_read}
- },
- {LDPT_GET_SYMBOLS,
- {.tv_get_symbols = get_symbols}
- },
- {LDPT_REGISTER_CLEANUP_HOOK,
- {.tv_register_cleanup = register_cleanup}
- },
- {0, {0}}
-};
-
-/* Load a plugin from a file named NAME. */
-
-static void
-load_plugin (const char *name)
-{
- ld_plugin_onload onload;
- plugin_handle = dlopen (name, RTLD_LAZY);
-
- assert (plugin_handle != NULL);
- onload = dlsym (plugin_handle, "onload");
- assert (onload);
- onload (tv);
- assert (claim_file_handler);
-}
-
-/* Send object to the plugin. The file (archive or object) name is NAME.
- FD is an open file descriptor. The object data starts at OFFSET and is
- FILESIZE bytes long. */
-
-static void
-register_object (const char *name, int fd, off_t offset, off_t filesize)
-{
- int claimed;
- struct ld_plugin_input_file file;
- void *handle;
-
- num_file_handles++;
- all_file_handles = realloc (all_file_handles, num_file_handles
- * sizeof (struct file_handle *));
- assert (all_file_handles);
-
- all_file_handles[num_file_handles - 1] = calloc (1,
- sizeof (struct file_handle));
- handle = all_file_handles[num_file_handles - 1];
- assert (handle);
-
- file.name = (char *) name;
- file.fd = fd;
- file.offset = offset;
- file.filesize = filesize;
-
- file.handle = handle;
-
- claim_file_handler (&file, &claimed);
-}
-
-/* Send file named NAME to the plugin. */
-
-static void
-register_file (const char *name)
-{
- int fd = open (name, O_RDONLY);
- Elf *elf;
-
- assert (fd >= 0);
-
- elf = elf_begin (fd, ELF_C_READ, NULL);
- assert (elf);
-
- Elf_Kind kind = elf_kind (elf);
-
- assert (kind == ELF_K_ELF || kind == ELF_K_AR);
-
- if (kind == ELF_K_AR)
- {
- Elf *member = elf_begin (fd, ELF_C_READ, elf);
- while (member)
- {
- Elf_Arhdr *h = elf_getarhdr (member);
- assert (h);
-
- if (h->ar_name[0] != '/')
- {
- off_t offset = elf_getbase (member);
- register_object (name, fd, offset, h->ar_size);
- }
-
- Elf_Cmd cmd = elf_next (member);
- elf_end (member);
- member = elf_begin (fd, cmd, elf);
- }
- }
- else /* Single File */
- register_object (name, fd, 0, 0);
-
- elf_end (elf);
-}
-
-/* Fake symbol resolution for testing. */
-
-static void
-resolve (void)
-{
- unsigned j;
- for (j = 0; j < num_file_handles; j++)
- {
- struct file_handle *handle = all_file_handles[j];
- unsigned int nsyms = handle->nsyms;
- struct ld_plugin_symbol *syms = handle->syms;
- unsigned i;
- for (i = 0; i < nsyms; i++)
- {
- switch (syms[i].def)
- {
- case LDPK_DEF:
- case LDPK_WEAKDEF:
- case LDPK_COMMON:
- syms[i].resolution = LDPR_PREVAILING_DEF;
- break;
- case LDPK_UNDEF:
- case LDPK_WEAKUNDEF:
- syms[i].resolution = LDPR_RESOLVED_IR;
- break;
- }
- }
- }
-}
-
-/* Print all symbol information. */
-
-static void
-print (void)
-{
- unsigned j;
- for (j = 0; j < num_file_handles; j++)
- {
- struct file_handle *handle = all_file_handles[j];
- unsigned int nsyms = handle->nsyms;
- struct ld_plugin_symbol *syms = handle->syms;
- unsigned i;
- for (i = 0; i < nsyms; i++)
- {
- printf("name: %s; ", syms[i].name);
- if (syms[i].version)
- printf("version: %s;", syms[i].version);
- else
- printf("not versioned; ");
- printf("kind: %s; ", lto_kind_str[syms[i].def]);
- printf("visibility: %s; ", lto_visibility_str[syms[i].visibility]);
- printf("size: %" PRId64 "; ", syms[i].size);
- if (syms[i].comdat_key)
- printf("comdat_key: %s; ", syms[i].comdat_key);
- else
- printf("no comdat_key; ");
- printf ("resolution: %s\n", lto_resolution_str[syms[i].resolution]);
- }
- }
-}
-
-/* Unload the plugin. */
-
-static void
-unload_plugin (void)
-{
- unsigned err = dlclose (plugin_handle);
- assert (err == 0);
- claim_file_handler = 0;
- all_symbols_read_handler = 0;
-}
-
-/* Free all memory allocated by us that hasn't been freed yet. */
-
-static void
-free_all (void)
-{
- unsigned j;
- for (j = 0; j < num_file_handles; j++)
- {
- struct file_handle *handle = all_file_handles[j];
- unsigned int nsyms = handle->nsyms;
- struct ld_plugin_symbol *syms = handle->syms;
- unsigned i;
- for (i = 0; i < nsyms; i++)
- {
- free (syms[i].name);
- syms[i].name = 0;
- if (syms[i].version)
- {
- free (syms[i].version);
- syms[i].version = 0;
- }
- if (syms[i].comdat_key)
- {
- free (syms[i].comdat_key);
- syms[i].comdat_key = 0;
- }
- }
- free (syms);
- handle->syms = NULL;
- handle->nsyms = 0;
- free (all_file_handles[j]);
- all_file_handles[j] = NULL;
- }
-
- free (all_file_handles);
- all_file_handles = NULL;
- num_file_handles = 0;
-}
-
-int
-main(int argc, char *argv[])
-{
- const char *plugin;
- unsigned int i;
- assert (argc >= 3);
- plugin = argv[1];
-
- load_plugin (plugin);
-
- for (i = 2; i < argc; i++)
- register_file (argv[i]);
-
- resolve ();
-
- print ();
-
- all_symbols_read_handler ();
-
- free_all ();
-
- cleanup_handler ();
-
- unload_plugin ();
-
- return 0;
-}