/* libdeps plugin for the GNU linker. Copyright (C) 2020-2021 Free Software Foundation, Inc. This file is part of the GNU Binutils. 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 "sysdep.h" #include "bfd.h" #if BFD_SUPPORTS_PLUGINS #include "plugin-api.h" #include <ctype.h> /* For isspace. */ extern enum ld_plugin_status onload (struct ld_plugin_tv *tv); /* Helper for calling plugin api message function. */ #define TV_MESSAGE if (tv_message) (*tv_message) /* Function pointers to cache hooks passed at onload time. */ static ld_plugin_register_claim_file tv_register_claim_file = 0; static ld_plugin_register_all_symbols_read tv_register_all_symbols_read = 0; static ld_plugin_register_cleanup tv_register_cleanup = 0; static ld_plugin_message tv_message = 0; static ld_plugin_add_input_library tv_add_input_library = 0; static ld_plugin_set_extra_library_path tv_set_extra_library_path = 0; /* Handle/record information received in a transfer vector entry. */ static enum ld_plugin_status parse_tv_tag (struct ld_plugin_tv *tv) { #define SETVAR(x) x = tv->tv_u.x switch (tv->tv_tag) { case LDPT_REGISTER_CLAIM_FILE_HOOK: SETVAR(tv_register_claim_file); break; case LDPT_REGISTER_ALL_SYMBOLS_READ_HOOK: SETVAR(tv_register_all_symbols_read); break; case LDPT_REGISTER_CLEANUP_HOOK: SETVAR(tv_register_cleanup); break; case LDPT_MESSAGE: SETVAR(tv_message); break; case LDPT_ADD_INPUT_LIBRARY: SETVAR(tv_add_input_library); break; case LDPT_SET_EXTRA_LIBRARY_PATH: SETVAR(tv_set_extra_library_path); break; default: break; } #undef SETVAR return LDPS_OK; } /* Defs for archive parsing. */ #define ARMAGSIZE 8 typedef struct arhdr { char ar_name[16]; char ar_date[12]; char ar_uid[6]; char ar_gid[6]; char ar_mode[8]; char ar_size[10]; char ar_fmag[2]; } arhdr; typedef struct linerec { struct linerec *next; char line[]; } linerec; #define LIBDEPS "__.LIBDEP/ " static linerec *line_head, **line_tail = &line_head; static enum ld_plugin_status get_libdeps (int fd) { arhdr ah; int len; unsigned long mlen; linerec *lr; enum ld_plugin_status rc = LDPS_NO_SYMS; lseek (fd, ARMAGSIZE, SEEK_SET); for (;;) { len = read (fd, (void *) &ah, sizeof (ah)); if (len != sizeof (ah)) break; mlen = strtoul (ah.ar_size, NULL, 10); if (!mlen || strncmp (ah.ar_name, LIBDEPS, sizeof (LIBDEPS)-1)) { lseek (fd, mlen, SEEK_CUR); continue; } lr = malloc (sizeof (linerec) + mlen); if (!lr) return LDPS_ERR; lr->next = NULL; len = read (fd, lr->line, mlen); lr->line[mlen-1] = '\0'; *line_tail = lr; line_tail = &lr->next; rc = LDPS_OK; break; } return rc; } /* Turn a string into an argvec. */ static char ** str2vec (char *in) { char **res; char *s, *first, *end; char *sq, *dq; int i; end = in + strlen (in); s = in; while (isspace (*s)) s++; first = s; i = 1; while ((s = strchr (s, ' '))) { s++; i++; } res = (char **)malloc ((i+1) * sizeof (char *)); if (!res) return res; i = 0; sq = NULL; dq = NULL; res[0] = first; for (s = first; *s; s++) { if (*s == '\\') { memmove (s, s+1, end-s-1); end--; } if (isspace (*s)) { if (sq || dq) continue; *s++ = '\0'; while (isspace (*s)) s++; if (*s) res[++i] = s; } if (*s == '\'' && !dq) { if (sq) { memmove (sq, sq+1, s-sq-1); memmove (s-2, s+1, end-s-1); end -= 2; s--; sq = NULL; } else { sq = s; } } if (*s == '"' && !sq) { if (dq) { memmove (dq, dq+1, s-dq-1); memmove (s-2, s+1, end-s-1); end -= 2; s--; dq = NULL; } else { dq = s; } } } res[++i] = NULL; return res; } static char *prevfile; /* Standard plugin API registerable hook. */ static enum ld_plugin_status onclaim_file (const struct ld_plugin_input_file *file, int *claimed) { enum ld_plugin_status rv; *claimed = 0; /* If we've already seen this file, ignore it. */ if (prevfile && !strcmp (file->name, prevfile)) return LDPS_OK; /* If it's not an archive member, ignore it. */ if (!file->offset) return LDPS_OK; if (prevfile) free (prevfile); prevfile = strdup (file->name); if (!prevfile) return LDPS_ERR; /* This hook only gets called on actual object files. * We have to examine the archive ourselves, to find * our LIBDEPS member. */ rv = get_libdeps (file->fd); if (rv == LDPS_ERR) return rv; if (rv == LDPS_OK) { linerec *lr = (linerec *)line_tail; /* Inform the user/testsuite. */ TV_MESSAGE (LDPL_INFO, "got deps for library %s: %s", file->name, lr->line); fflush (NULL); } return LDPS_OK; } /* Standard plugin API registerable hook. */ static enum ld_plugin_status onall_symbols_read (void) { linerec *lr; char **vec; enum ld_plugin_status rv = LDPS_OK; while ((lr = line_head)) { line_head = lr->next; vec = str2vec (lr->line); if (vec) { int i; for (i = 0; vec[i]; i++) { if (vec[i][0] != '-') { TV_MESSAGE (LDPL_WARNING, "ignoring libdep argument %s", vec[i]); fflush (NULL); continue; } if (vec[i][1] == 'l') rv = tv_add_input_library (vec[i]+2); else if (vec[i][1] == 'L') rv = tv_set_extra_library_path (vec[i]+2); else { TV_MESSAGE (LDPL_WARNING, "ignoring libdep argument %s", vec[i]); fflush (NULL); } if (rv != LDPS_OK) break; } free (vec); } free (lr); } line_tail = NULL; return rv; } /* Standard plugin API registerable hook. */ static enum ld_plugin_status oncleanup (void) { if (prevfile) { free (prevfile); prevfile = NULL; } if (line_head) { linerec *lr; while ((lr = line_head)) { line_head = lr->next; free (lr); } line_tail = NULL; } return LDPS_OK; } /* Standard plugin API entry point. */ enum ld_plugin_status onload (struct ld_plugin_tv *tv) { enum ld_plugin_status rv; /* This plugin requires a valid tv array. */ if (!tv) return LDPS_ERR; /* First entry should always be LDPT_MESSAGE, letting us get hold of it easily so we can send output straight away. */ if (tv[0].tv_tag == LDPT_MESSAGE) tv_message = tv[0].tv_u.tv_message; do if ((rv = parse_tv_tag (tv)) != LDPS_OK) return rv; while ((tv++)->tv_tag != LDPT_NULL); /* Register hooks. */ if (tv_register_claim_file && tv_register_all_symbols_read && tv_register_cleanup) { (*tv_register_claim_file) (onclaim_file); (*tv_register_all_symbols_read) (onall_symbols_read); (*tv_register_cleanup) (oncleanup); } fflush (NULL); return LDPS_OK; } #endif /* BFD_SUPPORTS_PLUGINS */