Updated to fedora-glibc-20070731T1624
[glibc.git] / elf / ldconfig.c
blob7692e3a8afe5225f6f3eac45bf64ed8f47ccd446
1 /* Copyright (C) 1999-2004, 2005, 2006, 2007 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
3 Contributed by Andreas Jaeger <aj@suse.de>, 1999.
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published
7 by the Free Software Foundation; version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software Foundation,
17 Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
19 #define PROCINFO_CLASS static
20 #include <alloca.h>
21 #include <argp.h>
22 #include <dirent.h>
23 #include <elf.h>
24 #include <error.h>
25 #include <errno.h>
26 #include <inttypes.h>
27 #include <libintl.h>
28 #include <locale.h>
29 #include <stdbool.h>
30 #include <stdio.h>
31 #include <stdio_ext.h>
32 #include <stdlib.h>
33 #include <string.h>
34 #include <unistd.h>
35 #include <sys/fcntl.h>
36 #include <sys/mman.h>
37 #include <sys/stat.h>
38 #include <sys/types.h>
39 #include <glob.h>
40 #include <libgen.h>
42 #include <ldconfig.h>
43 #include <dl-cache.h>
45 #include <dl-procinfo.h>
47 #ifdef _DL_FIRST_PLATFORM
48 # define _DL_FIRST_EXTRA (_DL_FIRST_PLATFORM + _DL_PLATFORMS_COUNT)
49 #else
50 # define _DL_FIRST_EXTRA _DL_HWCAP_COUNT
51 #endif
53 #ifndef LD_SO_CONF
54 # define LD_SO_CONF SYSCONFDIR "/ld.so.conf"
55 #endif
57 /* Get libc version number. */
58 #include <version.h>
60 #define PACKAGE _libc_intl_domainname
62 static const struct
64 const char *name;
65 int flag;
66 } lib_types[] =
68 {"libc4", FLAG_LIBC4},
69 {"libc5", FLAG_ELF_LIBC5},
70 {"libc6", FLAG_ELF_LIBC6},
71 {"glibc2", FLAG_ELF_LIBC6}
75 /* List of directories to handle. */
76 struct dir_entry
78 char *path;
79 int flag;
80 ino64_t ino;
81 dev_t dev;
82 struct dir_entry *next;
85 /* The list is unsorted, contains no duplicates. Entries are added at
86 the end. */
87 static struct dir_entry *dir_entries;
89 /* Flags for different options. */
90 /* Print Cache. */
91 static int opt_print_cache;
93 /* Be verbose. */
94 int opt_verbose;
96 /* Format to support. */
97 /* 0: only libc5/glibc2; 1: both; 2: only glibc 2.2. */
98 int opt_format = 1;
100 /* Build cache. */
101 static int opt_build_cache = 1;
103 /* Generate links. */
104 static int opt_link = 1;
106 /* Only process directories specified on the command line. */
107 static int opt_only_cline;
109 /* Path to root for chroot. */
110 static char *opt_chroot;
112 /* Manually link given shared libraries. */
113 static int opt_manual_link;
115 /* Cache file to use. */
116 static char *cache_file;
118 /* Configuration file. */
119 static const char *config_file;
121 /* Mask to use for important hardware capabilities. */
122 static unsigned long int hwcap_mask = HWCAP_IMPORTANT;
124 /* Configuration-defined capabilities defined in kernel vDSOs. */
125 static const char *hwcap_extra[64 - _DL_FIRST_EXTRA];
127 /* Name and version of program. */
128 static void print_version (FILE *stream, struct argp_state *state);
129 void (*argp_program_version_hook) (FILE *, struct argp_state *)
130 = print_version;
132 /* Definitions of arguments for argp functions. */
133 static const struct argp_option options[] =
135 { "print-cache", 'p', NULL, 0, N_("Print cache"), 0},
136 { "verbose", 'v', NULL, 0, N_("Generate verbose messages"), 0},
137 { NULL, 'N', NULL, 0, N_("Don't build cache"), 0},
138 { NULL, 'X', NULL, 0, N_("Don't generate links"), 0},
139 { NULL, 'r', N_("ROOT"), 0, N_("Change to and use ROOT as root directory"), 0},
140 { NULL, 'C', N_("CACHE"), 0, N_("Use CACHE as cache file"), 0},
141 { NULL, 'f', N_("CONF"), 0, N_("Use CONF as configuration file"), 0},
142 { NULL, 'n', NULL, 0, N_("Only process directories specified on the command line. Don't build cache."), 0},
143 { NULL, 'l', NULL, 0, N_("Manually link individual libraries."), 0},
144 { "format", 'c', N_("FORMAT"), 0, N_("Format to use: new, old or compat (default)"), 0},
145 { NULL, 0, NULL, 0, NULL, 0 }
148 #define PROCINFO_CLASS static
149 #include <dl-procinfo.c>
151 /* Short description of program. */
152 static const char doc[] = N_("Configure Dynamic Linker Run Time Bindings.");
154 /* Prototype for option handler. */
155 static error_t parse_opt (int key, char *arg, struct argp_state *state);
157 /* Data structure to communicate with argp functions. */
158 static struct argp argp =
160 options, parse_opt, NULL, doc, NULL, NULL, NULL
163 /* Check if string corresponds to an important hardware capability or
164 a platform. */
165 static int
166 is_hwcap_platform (const char *name)
168 int hwcap_idx = _dl_string_hwcap (name);
170 if (hwcap_idx != -1 && ((1 << hwcap_idx) & hwcap_mask))
171 return 1;
173 hwcap_idx = _dl_string_platform (name);
174 if (hwcap_idx != -1)
175 return 1;
177 for (hwcap_idx = _DL_FIRST_EXTRA; hwcap_idx < 64; ++hwcap_idx)
178 if (hwcap_extra[hwcap_idx - _DL_FIRST_EXTRA] != NULL
179 && !strcmp (name, hwcap_extra[hwcap_idx - _DL_FIRST_EXTRA]))
180 return 1;
182 return 0;
185 /* Get hwcap (including platform) encoding of path. */
186 static uint64_t
187 path_hwcap (const char *path)
189 char *str = xstrdup (path);
190 char *ptr;
191 uint64_t hwcap = 0;
192 uint64_t h;
194 size_t len;
196 len = strlen (str);
197 if (str[len] == '/')
198 str[len] = '\0';
200 /* Search pathname from the end and check for hwcap strings. */
201 for (;;)
203 ptr = strrchr (str, '/');
205 if (ptr == NULL)
206 break;
208 h = _dl_string_hwcap (ptr + 1);
210 if (h == (uint64_t) -1)
212 h = _dl_string_platform (ptr + 1);
213 if (h == (uint64_t) -1)
215 for (h = _DL_FIRST_EXTRA; h < 64; ++h)
216 if (hwcap_extra[h - _DL_FIRST_EXTRA] != NULL
217 && !strcmp (ptr + 1, hwcap_extra[h - _DL_FIRST_EXTRA]))
218 break;
219 if (h == 64)
220 break;
223 hwcap += 1ULL << h;
225 /* Search the next part of the path. */
226 *ptr = '\0';
229 free (str);
230 return hwcap;
233 /* Handle program arguments. */
234 static error_t
235 parse_opt (int key, char *arg, struct argp_state *state)
237 switch (key)
239 case 'C':
240 cache_file = arg;
241 break;
242 case 'f':
243 config_file = arg;
244 break;
245 case 'l':
246 opt_manual_link = 1;
247 break;
248 case 'N':
249 opt_build_cache = 0;
250 break;
251 case 'n':
252 opt_build_cache = 0;
253 opt_only_cline = 1;
254 break;
255 case 'p':
256 opt_print_cache = 1;
257 break;
258 case 'r':
259 opt_chroot = arg;
260 break;
261 case 'v':
262 opt_verbose = 1;
263 break;
264 case 'X':
265 opt_link = 0;
266 break;
267 case 'c':
268 if (strcmp (arg, "old") == 0)
269 opt_format = 0;
270 else if (strcmp (arg, "compat") == 0)
271 opt_format = 1;
272 else if (strcmp (arg, "new") == 0)
273 opt_format = 2;
274 break;
275 default:
276 return ARGP_ERR_UNKNOWN;
279 return 0;
282 /* Print the version information. */
283 static void
284 print_version (FILE *stream, struct argp_state *state)
286 fprintf (stream, "ldconfig (GNU %s) %s\n", PACKAGE, VERSION);
287 fprintf (stream, gettext ("\
288 Copyright (C) %s Free Software Foundation, Inc.\n\
289 This is free software; see the source for copying conditions. There is NO\n\
290 warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\
291 "), "2007");
292 fprintf (stream, gettext ("Written by %s.\n"),
293 "Andreas Jaeger");
296 /* Add a single directory entry. */
297 static void
298 add_single_dir (struct dir_entry *entry, int verbose)
300 struct dir_entry *ptr, *prev;
302 ptr = dir_entries;
303 prev = ptr;
304 while (ptr != NULL)
306 /* Check for duplicates. */
307 if (ptr->ino == entry->ino && ptr->dev == entry->dev)
309 if (opt_verbose && verbose)
310 error (0, 0, _("Path `%s' given more than once"), entry->path);
311 /* Use the newer information. */
312 ptr->flag = entry->flag;
313 free (entry->path);
314 free (entry);
315 break;
317 prev = ptr;
318 ptr = ptr->next;
320 /* Is this the first entry? */
321 if (ptr == NULL && dir_entries == NULL)
322 dir_entries = entry;
323 else if (ptr == NULL)
324 prev->next = entry;
327 /* Add one directory to the list of directories to process. */
328 static void
329 add_dir (const char *line)
331 unsigned int i;
332 struct dir_entry *entry = xmalloc (sizeof (struct dir_entry));
333 entry->next = NULL;
335 /* Search for an '=' sign. */
336 entry->path = xstrdup (line);
337 char *equal_sign = strchr (entry->path, '=');
338 if (equal_sign)
340 *equal_sign = '\0';
341 ++equal_sign;
342 entry->flag = FLAG_ANY;
343 for (i = 0; i < sizeof (lib_types) / sizeof (lib_types[0]); ++i)
344 if (strcmp (equal_sign, lib_types[i].name) == 0)
346 entry->flag = lib_types[i].flag;
347 break;
349 if (entry->flag == FLAG_ANY)
350 error (0, 0, _("%s is not a known library type"), equal_sign);
352 else
354 entry->flag = FLAG_ANY;
357 /* Canonify path: for now only remove leading and trailing
358 whitespace and the trailing slashes slashes. */
359 i = strlen (entry->path) - 1;
361 while (isspace (entry->path[i]) && i > 0)
362 entry->path[i--] = '\0';
364 while (entry->path[i] == '/' && i > 0)
365 entry->path[i--] = '\0';
367 char *path = entry->path;
368 if (opt_chroot)
369 path = chroot_canon (opt_chroot, path);
371 struct stat64 stat_buf;
372 if (path == NULL || stat64 (path, &stat_buf))
374 if (opt_verbose)
375 error (0, errno, _("Can't stat %s"), entry->path);
376 free (entry->path);
377 free (entry);
379 else
381 entry->ino = stat_buf.st_ino;
382 entry->dev = stat_buf.st_dev;
384 add_single_dir (entry, 1);
387 if (opt_chroot)
388 free (path);
392 static int
393 chroot_stat (const char *real_path, const char *path, struct stat64 *st)
395 int ret;
396 char *canon_path;
398 if (!opt_chroot)
399 return stat64 (real_path, st);
401 ret = lstat64 (real_path, st);
402 if (ret || !S_ISLNK (st->st_mode))
403 return ret;
405 canon_path = chroot_canon (opt_chroot, path);
406 if (canon_path == NULL)
407 return -1;
409 ret = stat64 (canon_path, st);
410 free (canon_path);
411 return ret;
414 /* Create a symbolic link from soname to libname in directory path. */
415 static void
416 create_links (const char *real_path, const char *path, const char *libname,
417 const char *soname)
419 char *full_libname, *full_soname;
420 char *real_full_libname, *real_full_soname;
421 struct stat64 stat_lib, stat_so, lstat_so;
422 int do_link = 1;
423 int do_remove = 1;
424 /* XXX: The logics in this function should be simplified. */
426 /* Get complete path. */
427 full_libname = alloca (strlen (path) + strlen (libname) + 2);
428 full_soname = alloca (strlen (path) + strlen (soname) + 2);
429 sprintf (full_libname, "%s/%s", path, libname);
430 sprintf (full_soname, "%s/%s", path, soname);
431 if (opt_chroot)
433 real_full_libname = alloca (strlen (real_path) + strlen (libname) + 2);
434 real_full_soname = alloca (strlen (real_path) + strlen (soname) + 2);
435 sprintf (real_full_libname, "%s/%s", real_path, libname);
436 sprintf (real_full_soname, "%s/%s", real_path, soname);
438 else
440 real_full_libname = full_libname;
441 real_full_soname = full_soname;
444 /* Does soname already exist and point to the right library? */
445 if (chroot_stat (real_full_soname, full_soname, &stat_so) == 0)
447 if (chroot_stat (real_full_libname, full_libname, &stat_lib))
449 error (0, 0, _("Can't stat %s\n"), full_libname);
450 return;
452 if (stat_lib.st_dev == stat_so.st_dev
453 && stat_lib.st_ino == stat_so.st_ino)
454 /* Link is already correct. */
455 do_link = 0;
456 else if (lstat64 (full_soname, &lstat_so) == 0
457 && !S_ISLNK (lstat_so.st_mode))
459 error (0, 0, _("%s is not a symbolic link\n"), full_soname);
460 do_link = 0;
461 do_remove = 0;
464 else if (lstat64 (real_full_soname, &lstat_so) != 0
465 || !S_ISLNK (lstat_so.st_mode))
466 /* Unless it is a stale symlink, there is no need to remove. */
467 do_remove = 0;
469 if (opt_verbose)
470 printf ("\t%s -> %s", soname, libname);
472 if (do_link && opt_link)
474 /* Remove old link. */
475 if (do_remove)
476 if (unlink (real_full_soname))
478 error (0, 0, _("Can't unlink %s"), full_soname);
479 do_link = 0;
481 /* Create symbolic link. */
482 if (do_link && symlink (libname, real_full_soname))
484 error (0, 0, _("Can't link %s to %s"), full_soname, libname);
485 do_link = 0;
487 if (opt_verbose)
489 if (do_link)
490 fputs (_(" (changed)\n"), stdout);
491 else
492 fputs (_(" (SKIPPED)\n"), stdout);
495 else if (opt_verbose)
496 fputs ("\n", stdout);
499 /* Manually link the given library. */
500 static void
501 manual_link (char *library)
503 char *path;
504 char *real_path;
505 char *real_library;
506 char *libname;
507 char *soname;
508 struct stat64 stat_buf;
509 int flag;
510 unsigned int osversion;
512 /* Prepare arguments for create_links call. Split library name in
513 directory and filename first. Since path is allocated, we've got
514 to be careful to free at the end. */
515 path = xstrdup (library);
516 libname = strrchr (path, '/');
518 if (libname)
520 /* Successfully split names. Check if path is just "/" to avoid
521 an empty path. */
522 if (libname == path)
524 libname = library + 1;
525 path = xrealloc (path, 2);
526 strcpy (path, "/");
528 else
530 *libname = '\0';
531 ++libname;
534 else
536 /* There's no path, construct one. */
537 libname = library;
538 path = xrealloc (path, 2);
539 strcpy (path, ".");
542 if (opt_chroot)
544 real_path = chroot_canon (opt_chroot, path);
545 if (real_path == NULL)
547 error (0, errno, _("Can't find %s"), path);
548 free (path);
549 return;
551 real_library = alloca (strlen (real_path) + strlen (libname) + 2);
552 sprintf (real_library, "%s/%s", real_path, libname);
554 else
556 real_path = path;
557 real_library = library;
560 /* Do some sanity checks first. */
561 if (lstat64 (real_library, &stat_buf))
563 error (0, errno, _("Cannot lstat %s"), library);
564 free (path);
565 return;
567 /* We don't want links here! */
568 else if (!S_ISREG (stat_buf.st_mode))
570 error (0, 0, _("Ignored file %s since it is not a regular file."),
571 library);
572 free (path);
573 return;
575 if (process_file (real_library, library, libname, &flag, &osversion,
576 &soname, 0))
578 error (0, 0, _("No link created since soname could not be found for %s"),
579 library);
580 free (path);
581 return;
583 create_links (real_path, path, libname, soname);
584 free (soname);
585 free (path);
589 /* Read a whole directory and search for libraries.
590 The purpose is two-fold:
591 - search for libraries which will be added to the cache
592 - create symbolic links to the soname for each library
594 This has to be done separatly for each directory.
596 To keep track of which libraries to add to the cache and which
597 links to create, we save a list of all libraries.
599 The algorithm is basically:
600 for all libraries in the directory do
601 get soname of library
602 if soname is already in list
603 if new library is newer, replace entry
604 otherwise ignore this library
605 otherwise add library to list
607 For example, if the two libraries libxy.so.1.1 and libxy.so.1.2
608 exist and both have the same soname, e.g. libxy.so, a symbolic link
609 is created from libxy.so.1.2 (the newer one) to libxy.so.
610 libxy.so.1.2 and libxy.so are added to the cache - but not
611 libxy.so.1.1. */
613 /* Information for one library. */
614 struct dlib_entry
616 char *name;
617 char *soname;
618 int flag;
619 int is_link;
620 unsigned int osversion;
621 struct dlib_entry *next;
625 static void
626 search_dir (const struct dir_entry *entry)
628 DIR *dir;
629 struct dirent64 *direntry;
630 char *file_name, *dir_name, *real_file_name, *real_name;
631 int file_name_len, real_file_name_len, len;
632 char *soname;
633 struct dlib_entry *dlibs;
634 struct dlib_entry *dlib_ptr;
635 struct stat64 lstat_buf, stat_buf;
636 int is_link, is_dir;
637 uint64_t hwcap = path_hwcap (entry->path);
638 unsigned int osversion;
640 file_name_len = PATH_MAX;
641 file_name = alloca (file_name_len);
643 dlibs = NULL;
645 if (opt_verbose)
647 if (hwcap != 0)
648 printf ("%s: (hwcap: %#.16" PRIx64 ")\n", entry->path, hwcap);
649 else
650 printf ("%s:\n", entry->path);
653 if (opt_chroot)
655 dir_name = chroot_canon (opt_chroot, entry->path);
656 real_file_name_len = PATH_MAX;
657 real_file_name = alloca (real_file_name_len);
659 else
661 dir_name = entry->path;
662 real_file_name_len = 0;
663 real_file_name = file_name;
666 if (dir_name == NULL || (dir = opendir (dir_name)) == NULL)
668 if (opt_verbose)
669 error (0, errno, _("Can't open directory %s"), entry->path);
670 if (opt_chroot && dir_name)
671 free (dir_name);
672 return;
675 while ((direntry = readdir64 (dir)) != NULL)
677 int flag;
678 #ifdef _DIRENT_HAVE_D_TYPE
679 /* We only look at links and regular files. */
680 if (direntry->d_type != DT_UNKNOWN
681 && direntry->d_type != DT_LNK
682 && direntry->d_type != DT_REG
683 && direntry->d_type != DT_DIR)
684 continue;
685 #endif /* _DIRENT_HAVE_D_TYPE */
686 /* Does this file look like a shared library or is it a hwcap
687 subdirectory? The dynamic linker is also considered as
688 shared library. */
689 if (((strncmp (direntry->d_name, "lib", 3) != 0
690 && strncmp (direntry->d_name, "ld-", 3) != 0)
691 || strstr (direntry->d_name, ".so") == NULL)
692 && (
693 #ifdef _DIRENT_HAVE_D_TYPE
694 direntry->d_type == DT_REG ||
695 #endif
696 !is_hwcap_platform (direntry->d_name)))
697 continue;
698 len = strlen (direntry->d_name);
699 /* Skip temporary files created by the prelink program. Files with
700 names like these are never really DSOs we want to look at. */
701 if (len >= sizeof (".#prelink#") - 1)
703 if (strcmp (direntry->d_name + len - sizeof (".#prelink#") + 1,
704 ".#prelink#") == 0)
705 continue;
706 if (len >= sizeof (".#prelink#.XXXXXX") - 1
707 && memcmp (direntry->d_name + len - sizeof (".#prelink#.XXXXXX")
708 + 1, ".#prelink#.", sizeof (".#prelink#.") - 1) == 0)
709 continue;
711 len += strlen (entry->path) + 2;
712 if (len > file_name_len)
714 file_name_len = len;
715 file_name = alloca (file_name_len);
716 if (!opt_chroot)
717 real_file_name = file_name;
719 sprintf (file_name, "%s/%s", entry->path, direntry->d_name);
720 if (opt_chroot)
722 len = strlen (dir_name) + strlen (direntry->d_name) + 2;
723 if (len > real_file_name_len)
725 real_file_name_len = len;
726 real_file_name = alloca (real_file_name_len);
728 sprintf (real_file_name, "%s/%s", dir_name, direntry->d_name);
730 #ifdef _DIRENT_HAVE_D_TYPE
731 if (direntry->d_type != DT_UNKNOWN)
732 lstat_buf.st_mode = DTTOIF (direntry->d_type);
733 else
734 #endif
735 if (__builtin_expect (lstat64 (real_file_name, &lstat_buf), 0))
737 error (0, errno, _("Cannot lstat %s"), file_name);
738 continue;
741 is_link = S_ISLNK (lstat_buf.st_mode);
742 if (is_link)
744 /* In case of symlink, we check if the symlink refers to
745 a directory. */
746 if (__builtin_expect (stat64 (real_file_name, &stat_buf), 0))
748 if (opt_verbose)
749 error (0, errno, _("Cannot stat %s"), file_name);
751 /* Remove stale symlinks. */
752 if (strstr (direntry->d_name, ".so."))
753 unlink (real_file_name);
754 continue;
756 is_dir = S_ISDIR (stat_buf.st_mode);
758 else
759 is_dir = S_ISDIR (lstat_buf.st_mode);
761 if (is_dir && is_hwcap_platform (direntry->d_name))
763 /* Handle subdirectory later. */
764 struct dir_entry *new_entry;
766 new_entry = xmalloc (sizeof (struct dir_entry));
767 new_entry->path = xstrdup (file_name);
768 new_entry->flag = entry->flag;
769 new_entry->next = NULL;
770 if (is_link)
772 new_entry->ino = stat_buf.st_ino;
773 new_entry->dev = stat_buf.st_dev;
775 else
777 #ifdef _DIRENT_HAVE_D_TYPE
778 /* We have filled in lstat only #ifndef
779 _DIRENT_HAVE_D_TYPE. Fill it in if needed. */
780 if (direntry->d_type != DT_UNKNOWN
781 && __builtin_expect (lstat64 (real_file_name, &lstat_buf),
784 error (0, errno, _("Cannot lstat %s"), file_name);
785 free (new_entry->path);
786 free (new_entry);
787 continue;
789 #endif
791 new_entry->ino = lstat_buf.st_ino;
792 new_entry->dev = lstat_buf.st_dev;
794 add_single_dir (new_entry, 0);
795 continue;
797 else if (!S_ISREG (lstat_buf.st_mode) && !is_link)
798 continue;
800 if (opt_chroot && is_link)
802 real_name = chroot_canon (opt_chroot, file_name);
803 if (real_name == NULL)
805 if (strstr (file_name, ".so") == NULL)
806 error (0, 0, _("Input file %s not found.\n"), file_name);
807 continue;
810 else
811 real_name = real_file_name;
813 if (process_file (real_name, file_name, direntry->d_name, &flag,
814 &osversion, &soname, is_link))
816 if (real_name != real_file_name)
817 free (real_name);
818 continue;
822 /* A link may just point to itself. */
823 if (is_link)
825 /* If the path the link points to isn't its soname and it is not
826 .so symlink for ld(1) only, we treat it as a normal file. */
827 const char *real_base_name = basename (real_file_name);
829 if (strcmp (real_base_name, soname) != 0)
831 len = strlen (real_base_name);
832 if (len < strlen (".so")
833 || strcmp (real_base_name + len - strlen (".so"), ".so") != 0
834 || strncmp (real_base_name, soname, len) != 0)
835 is_link = 0;
839 if (real_name != real_file_name)
840 free (real_name);
842 if (is_link)
844 free (soname);
845 soname = xstrdup (direntry->d_name);
848 if (flag == FLAG_ELF
849 && (entry->flag == FLAG_ELF_LIBC5
850 || entry->flag == FLAG_ELF_LIBC6))
851 flag = entry->flag;
852 /* Some sanity checks to print warnings. */
853 if (opt_verbose)
855 if (flag == FLAG_ELF_LIBC5 && entry->flag != FLAG_ELF_LIBC5
856 && entry->flag != FLAG_ANY)
857 error (0, 0, _("libc5 library %s in wrong directory"), file_name);
858 if (flag == FLAG_ELF_LIBC6 && entry->flag != FLAG_ELF_LIBC6
859 && entry->flag != FLAG_ANY)
860 error (0, 0, _("libc6 library %s in wrong directory"), file_name);
861 if (flag == FLAG_LIBC4 && entry->flag != FLAG_LIBC4
862 && entry->flag != FLAG_ANY)
863 error (0, 0, _("libc4 library %s in wrong directory"), file_name);
866 /* Add library to list. */
867 for (dlib_ptr = dlibs; dlib_ptr != NULL; dlib_ptr = dlib_ptr->next)
869 /* Is soname already in list? */
870 if (strcmp (dlib_ptr->soname, soname) == 0)
872 /* Prefer a file to a link, otherwise check which one
873 is newer. */
874 if ((!is_link && dlib_ptr->is_link)
875 || (is_link == dlib_ptr->is_link
876 && _dl_cache_libcmp (dlib_ptr->name, direntry->d_name) < 0))
878 /* It's newer - add it. */
879 /* Flag should be the same - sanity check. */
880 if (dlib_ptr->flag != flag)
882 if (dlib_ptr->flag == FLAG_ELF
883 && (flag == FLAG_ELF_LIBC5 || flag == FLAG_ELF_LIBC6))
884 dlib_ptr->flag = flag;
885 else if ((dlib_ptr->flag == FLAG_ELF_LIBC5
886 || dlib_ptr->flag == FLAG_ELF_LIBC6)
887 && flag == FLAG_ELF)
888 dlib_ptr->flag = flag;
889 else
890 error (0, 0, _("libraries %s and %s in directory %s have same soname but different type."),
891 dlib_ptr->name, direntry->d_name, entry->path);
893 free (dlib_ptr->name);
894 dlib_ptr->osversion = osversion;
895 dlib_ptr->name = xstrdup (direntry->d_name);
896 dlib_ptr->is_link = is_link;
898 /* Don't add this library, abort loop. */
899 /* Also free soname, since it's dynamically allocated. */
900 free (soname);
901 break;
904 /* Add the library if it's not already in. */
905 if (dlib_ptr == NULL)
907 dlib_ptr = (struct dlib_entry *)xmalloc (sizeof (struct dlib_entry));
908 dlib_ptr->name = xstrdup (direntry->d_name);
909 dlib_ptr->flag = flag;
910 dlib_ptr->osversion = osversion;
911 dlib_ptr->soname = soname;
912 dlib_ptr->is_link = is_link;
913 /* Add at head of list. */
914 dlib_ptr->next = dlibs;
915 dlibs = dlib_ptr;
919 closedir (dir);
921 /* Now dlibs contains a list of all libs - add those to the cache
922 and created all symbolic links. */
923 for (dlib_ptr = dlibs; dlib_ptr != NULL; dlib_ptr = dlib_ptr->next)
925 /* Don't create links to links. */
926 if (dlib_ptr->is_link == 0)
927 create_links (dir_name, entry->path, dlib_ptr->name,
928 dlib_ptr->soname);
929 if (opt_build_cache)
930 add_to_cache (entry->path, dlib_ptr->soname, dlib_ptr->flag,
931 dlib_ptr->osversion, hwcap);
934 /* Free all resources. */
935 while (dlibs)
937 dlib_ptr = dlibs;
938 free (dlib_ptr->soname);
939 free (dlib_ptr->name);
940 dlibs = dlibs->next;
941 free (dlib_ptr);
944 if (opt_chroot && dir_name)
945 free (dir_name);
948 /* Search through all libraries. */
949 static void
950 search_dirs (void)
952 struct dir_entry *entry;
954 for (entry = dir_entries; entry != NULL; entry = entry->next)
955 search_dir (entry);
957 /* Free all allocated memory. */
958 while (dir_entries)
960 entry = dir_entries;
961 dir_entries = dir_entries->next;
962 free (entry->path);
963 free (entry);
968 static void parse_conf_include (const char *config_file, unsigned int lineno,
969 const char *prefix, bool do_chroot,
970 const char *pattern);
972 /* Parse configuration file. */
973 static void
974 parse_conf (const char *filename, const char *prefix, bool do_chroot)
976 FILE *file = NULL;
977 char *line = NULL;
978 const char *canon;
979 size_t len = 0;
980 unsigned int lineno;
981 size_t prefix_len = prefix ? strlen (prefix) : 0;
983 if (do_chroot && opt_chroot)
985 canon = chroot_canon (opt_chroot, filename);
986 if (canon)
987 file = fopen (canon, "r");
988 else
989 canon = filename;
991 else
993 canon = filename;
994 file = fopen (filename, "r");
997 if (file == NULL)
999 error (0, errno, _("Can't open configuration file %s"), canon);
1000 if (canon != filename)
1001 free ((char *) canon);
1002 return;
1005 /* No threads use this stream. */
1006 __fsetlocking (file, FSETLOCKING_BYCALLER);
1008 if (canon != filename)
1009 free ((char *) canon);
1011 lineno = 0;
1014 ssize_t n = getline (&line, &len, file);
1015 if (n < 0)
1016 break;
1018 ++lineno;
1019 if (line[n - 1] == '\n')
1020 line[n - 1] = '\0';
1022 /* Because the file format does not know any form of quoting we
1023 can search forward for the next '#' character and if found
1024 make it terminating the line. */
1025 *strchrnul (line, '#') = '\0';
1027 /* Remove leading whitespace. NUL is no whitespace character. */
1028 char *cp = line;
1029 while (isspace (*cp))
1030 ++cp;
1032 /* If the line is blank it is ignored. */
1033 if (cp[0] == '\0')
1034 continue;
1036 if (!strncmp (cp, "include", 7) && isblank (cp[7]))
1038 char *dir;
1039 cp += 8;
1040 while ((dir = strsep (&cp, " \t")) != NULL)
1041 if (dir[0] != '\0')
1042 parse_conf_include (filename, lineno, prefix, do_chroot, dir);
1044 else if (prefix != NULL)
1046 size_t cp_len = strlen (cp);
1047 char new_cp [prefix_len + cp_len + 1];
1048 memcpy (mempcpy (new_cp, prefix, prefix_len), cp, cp_len + 1);
1049 add_dir (new_cp);
1051 else if (!strncasecmp (cp, "hwcap", 5) && isblank (cp[5]))
1053 cp += 6;
1054 char *p, *name = NULL;
1055 unsigned long int n = strtoul (cp, &cp, 0);
1056 if (cp != NULL && isblank (*cp))
1057 while ((p = strsep (&cp, " \t")) != NULL)
1058 if (p[0] != '\0')
1060 if (name == NULL)
1061 name = p;
1062 else
1064 name = NULL;
1065 break;
1068 if (name == NULL)
1070 error (EXIT_FAILURE, 0, _("%s:%u: bad syntax in hwcap line"),
1071 filename, lineno);
1072 break;
1074 if (n >= (64 - _DL_FIRST_EXTRA))
1075 error (EXIT_FAILURE, 0,
1076 _("%s:%u: hwcap index %lu above maximum %u"),
1077 filename, lineno, n, 64 - _DL_FIRST_EXTRA - 1);
1078 if (hwcap_extra[n] == NULL)
1080 for (unsigned long int h = 0; h < (64 - _DL_FIRST_EXTRA); ++h)
1081 if (hwcap_extra[h] != NULL && !strcmp (name, hwcap_extra[h]))
1082 error (EXIT_FAILURE, 0,
1083 _("%s:%u: hwcap index %lu already defined as %s"),
1084 filename, lineno, h, name);
1085 hwcap_extra[n] = xstrdup (name);
1087 else
1089 if (strcmp (name, hwcap_extra[n]))
1090 error (EXIT_FAILURE, 0,
1091 _("%s:%u: hwcap index %lu already defined as %s"),
1092 filename, lineno, n, hwcap_extra[n]);
1093 if (opt_verbose)
1094 error (0, 0, _("%s:%u: duplicate hwcap %lu %s"),
1095 filename, lineno, n, name);
1098 else
1099 add_dir (cp);
1101 while (!feof_unlocked (file));
1103 /* Free buffer and close file. */
1104 free (line);
1105 fclose (file);
1108 /* Handle one word in an `include' line, a glob pattern of additional
1109 config files to read. */
1110 static void
1111 parse_conf_include (const char *config_file, unsigned int lineno,
1112 const char *prefix, bool do_chroot, const char *pattern)
1114 if (opt_chroot && pattern[0] != '/')
1115 error (EXIT_FAILURE, 0,
1116 _("need absolute file name for configuration file when using -r"));
1118 char *copy = NULL;
1119 if (pattern[0] != '/' && strchr (config_file, '/') != NULL)
1121 if (asprintf (&copy, "%s/%s", dirname (strdupa (config_file)),
1122 pattern) < 0)
1123 error (EXIT_FAILURE, 0, _("memory exhausted"));
1124 pattern = copy;
1127 glob64_t gl;
1128 int result;
1129 if (do_chroot && opt_chroot)
1131 char *canon = chroot_canon (opt_chroot, pattern);
1132 result = glob64 (canon ?: pattern, 0, NULL, &gl);
1133 free (canon);
1135 else
1136 result = glob64 (pattern, 0, NULL, &gl);
1138 switch (result)
1140 case 0:
1141 for (size_t i = 0; i < gl.gl_pathc; ++i)
1142 parse_conf (gl.gl_pathv[i], prefix, false);
1143 globfree64 (&gl);
1144 break;
1146 case GLOB_NOMATCH:
1147 break;
1149 case GLOB_NOSPACE:
1150 errno = ENOMEM;
1151 case GLOB_ABORTED:
1152 if (opt_verbose)
1153 error (0, errno, _("%s:%u: cannot read directory %s"),
1154 config_file, lineno, pattern);
1155 break;
1157 default:
1158 abort ();
1159 break;
1162 if (copy)
1163 free (copy);
1166 /* Honour LD_HWCAP_MASK. */
1167 static void
1168 set_hwcap (void)
1170 char *mask = getenv ("LD_HWCAP_MASK");
1172 if (mask)
1173 hwcap_mask = strtoul (mask, NULL, 0);
1178 main (int argc, char **argv)
1180 /* Set locale via LC_ALL. */
1181 setlocale (LC_ALL, "");
1183 /* Set the text message domain. */
1184 textdomain (_libc_intl_domainname);
1186 arch_startup (argc, argv);
1188 /* Parse and process arguments. */
1189 int remaining;
1190 argp_parse (&argp, argc, argv, 0, &remaining, NULL);
1192 /* Remaining arguments are additional directories if opt_manual_link
1193 is not set. */
1194 if (remaining != argc && !opt_manual_link)
1196 int i;
1197 for (i = remaining; i < argc; ++i)
1198 if (opt_build_cache && argv[i][0] != '/')
1199 error (EXIT_FAILURE, 0,
1200 _("relative path `%s' used to build cache"),
1201 argv[i]);
1202 else
1203 add_dir (argv[i]);
1206 hwcap_extra[63 - _DL_FIRST_EXTRA] = "tls";
1208 set_hwcap ();
1210 if (opt_chroot)
1212 /* Normalize the path a bit, we might need it for printing later. */
1213 char *endp = rawmemchr (opt_chroot, '\0');
1214 while (endp > opt_chroot && endp[-1] == '/')
1215 --endp;
1216 *endp = '\0';
1217 if (endp == opt_chroot)
1218 opt_chroot = NULL;
1220 if (opt_chroot)
1222 /* It is faster to use chroot if we can. */
1223 if (!chroot (opt_chroot))
1225 if (chdir ("/"))
1226 error (EXIT_FAILURE, errno, _("Can't chdir to /"));
1227 opt_chroot = NULL;
1232 if (cache_file == NULL)
1234 cache_file = alloca (strlen (LD_SO_CACHE) + 1);
1235 strcpy (cache_file, LD_SO_CACHE);
1238 if (config_file == NULL)
1239 config_file = LD_SO_CONF;
1241 if (opt_print_cache)
1243 if (opt_chroot)
1245 char *p = chroot_canon (opt_chroot, cache_file);
1246 if (p == NULL)
1247 error (EXIT_FAILURE, errno, _("Can't open cache file %s\n"),
1248 cache_file);
1249 cache_file = p;
1251 print_cache (cache_file);
1252 if (opt_chroot)
1253 free (cache_file);
1254 exit (0);
1257 if (opt_chroot)
1259 /* Canonicalize the directory name of cache_file, not cache_file,
1260 because we'll rename a temporary cache file to it. */
1261 char *p = strrchr (cache_file, '/');
1262 char *canon = chroot_canon (opt_chroot,
1263 p ? (*p = '\0', cache_file) : "/");
1265 if (canon == NULL)
1267 error (EXIT_FAILURE, errno,
1268 _("Can't open cache file directory %s\n"),
1269 p ? cache_file : "/");
1272 if (p)
1273 ++p;
1274 else
1275 p = cache_file;
1277 cache_file = alloca (strlen (canon) + strlen (p) + 2);
1278 sprintf (cache_file, "%s/%s", canon, p);
1279 free (canon);
1282 if (opt_manual_link)
1284 /* Link all given libraries manually. */
1285 int i;
1287 for (i = remaining; i < argc; ++i)
1288 manual_link (argv[i]);
1290 exit (0);
1294 if (opt_build_cache)
1295 init_cache ();
1297 if (!opt_only_cline)
1299 parse_conf (config_file, NULL, true);
1301 /* Always add the standard search paths. */
1302 add_system_dir (SLIBDIR);
1303 if (strcmp (SLIBDIR, LIBDIR))
1304 add_system_dir (LIBDIR);
1306 add_arch_dirs (config_file);
1309 search_dirs ();
1311 if (opt_build_cache)
1312 save_cache (cache_file);
1314 return 0;