/* libdeps plugin for the GNU linker. Copyright (C) 2020-2025 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 /* 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; size_t amt; 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; } amt = mlen + sizeof (linerec); if (amt <= mlen) return LDPS_ERR; lr = malloc (amt); 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; } /* Parse arguments in-place as contiguous C-strings and return the number of arguments. */ static int parse_libdep (char *str) { char *src, *dst; char quote; int narg; src = dst = str; for (; isspace ((unsigned char) *src); ++src) ; if (*src == '\0') return 0; narg = 1; quote = 0; while (*src) { if (*src == '\'' || *src == '\"') { if (!quote) quote = *src++; else if (*src == quote) { ++src; quote = 0; } else *dst++ = *src++; } else if (!quote && isspace ((unsigned char) *src)) { ++narg; ++src; *dst++ = '\0'; for (; isspace ((unsigned char) *src); ++src); } else *dst++ = *src++; } *dst = '\0'; if (quote) { TV_MESSAGE (LDPL_WARNING, "libdep syntax error: unterminated quoted string"); return 0; } return narg; } 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; int nargs; char const *arg; enum ld_plugin_status rv = LDPS_OK; while ((lr = line_head)) { line_head = lr->next; nargs = parse_libdep (lr->line); arg = lr->line; int i; for (i = 0; i < nargs; i++, arg = strchr (arg, '\0') + 1) { if (arg[0] != '-') { TV_MESSAGE (LDPL_WARNING, "ignoring libdep argument %s", arg); fflush (NULL); continue; } if (arg[1] == 'l') rv = tv_add_input_library (arg + 2); else if (arg[1] == 'L') rv = tv_set_extra_library_path (arg + 2); else { TV_MESSAGE (LDPL_WARNING, "ignoring libdep argument %s", arg); fflush (NULL); } if (rv != LDPS_OK) break; } 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 */