elf: Add l_soname accessor function for DT_SONAME values

It's not necessary to introduce temporaries because the compiler
is able to evaluate l_soname just once in constracts like:

  l_soname (l) != NULL && strcmp (l_soname (l), LIBC_SO) != 0
This commit is contained in:
Florian Weimer 2025-02-02 20:10:09 +01:00
parent aa1bf89039
commit 749310c61b
6 changed files with 46 additions and 55 deletions

View File

@ -1421,10 +1421,8 @@ cannot enable executable stack as shared object requires");
/* When we profile the SONAME might be needed for something else but
loading. Add it right away. */
if (__glibc_unlikely (GLRO(dl_profile) != NULL)
&& l->l_info[DT_SONAME] != NULL)
add_name_to_object (l, ((const char *) D_PTR (l, l_info[DT_STRTAB])
+ l->l_info[DT_SONAME]->d_un.d_val));
if (__glibc_unlikely (GLRO(dl_profile) != NULL) && l_soname (l) != NULL)
add_name_to_object (l, l_soname (l));
#else
/* Audit modules only exist when linking is dynamic so ORIGNAME
cannot be non-NULL. */
@ -1434,9 +1432,7 @@ cannot enable executable stack as shared object requires");
/* If we have newly loaded libc.so, update the namespace
description. */
if (GL(dl_ns)[nsid].libc_map == NULL
&& l->l_info[DT_SONAME] != NULL
&& strcmp (((const char *) D_PTR (l, l_info[DT_STRTAB])
+ l->l_info[DT_SONAME]->d_un.d_val), LIBC_SO) == 0)
&& l_soname (l) != NULL && strcmp (l_soname(l), LIBC_SO) == 0)
GL(dl_ns)[nsid].libc_map = l;
/* _dl_close can only eventually undo the module ID assignment (via
@ -1903,19 +1899,12 @@ _dl_lookup_map (Lmid_t nsid, const char *name)
continue;
if (!_dl_name_match_p (name, l))
{
const char *soname;
if (__glibc_likely (l->l_soname_added)
|| l->l_info[DT_SONAME] == NULL)
continue;
soname = ((const char *) D_PTR (l, l_info[DT_STRTAB])
+ l->l_info[DT_SONAME]->d_un.d_val);
if (strcmp (name, soname) != 0)
if (__glibc_likely (l->l_soname_added) || l_soname (l) == NULL
|| strcmp (name, l_soname (l)) != 0)
continue;
/* We have a match on a new name -- cache it. */
add_name_to_object (l, soname);
add_name_to_object (l, l_soname (l));
l->l_soname_added = 1;
}

View File

@ -617,9 +617,7 @@ dl_open_worker_begin (void *a)
Perform partial initialization in this case. This must
come after the symbol versioning initialization in
_dl_check_map_versions. */
if (map->l_info[DT_SONAME] != NULL
&& strcmp (((const char *) D_PTR (map, l_info[DT_STRTAB])
+ map->l_info[DT_SONAME]->d_un.d_val), LD_SO) == 0)
if (l_soname (map) != NULL && strcmp (l_soname (map), LD_SO) == 0)
__rtld_static_init (map);
#endif
}

View File

@ -1055,13 +1055,9 @@ static void
rtld_chain_load (struct link_map *main_map, char *argv0)
{
/* The dynamic loader run against itself. */
const char *rtld_soname
= ((const char *) D_PTR (&_dl_rtld_map, l_info[DT_STRTAB])
+ _dl_rtld_map.l_info[DT_SONAME]->d_un.d_val);
if (main_map->l_info[DT_SONAME] != NULL
&& strcmp (rtld_soname,
((const char *) D_PTR (main_map, l_info[DT_STRTAB])
+ main_map->l_info[DT_SONAME]->d_un.d_val)) == 0)
const char *rtld_soname = l_soname (&_dl_rtld_map);
if (l_soname (main_map) != NULL
&& strcmp (rtld_soname, l_soname (main_map)) == 0)
_dl_fatal_printf ("%s: loader cannot load itself\n", rtld_soname);
/* With DT_NEEDED dependencies, the executable is dynamically
@ -1632,20 +1628,20 @@ dl_main (const ElfW(Phdr) *phdr,
/* If the current libname is different from the SONAME, add the
latter as well. */
if (_dl_rtld_map.l_info[DT_SONAME] != NULL
&& strcmp (_dl_rtld_map.l_libname->name,
(const char *) D_PTR (&_dl_rtld_map, l_info[DT_STRTAB])
+ _dl_rtld_map.l_info[DT_SONAME]->d_un.d_val) != 0)
{
static struct libname_list newname;
newname.name = ((char *) D_PTR (&_dl_rtld_map, l_info[DT_STRTAB])
+ _dl_rtld_map.l_info[DT_SONAME]->d_un.d_ptr);
newname.next = NULL;
newname.dont_free = 1;
{
const char *soname = l_soname (&_dl_rtld_map);
if (soname != NULL
&& strcmp (_dl_rtld_map.l_libname->name, soname) != 0)
{
static struct libname_list newname;
newname.name = soname;
newname.next = NULL;
newname.dont_free = 1;
assert (_dl_rtld_map.l_libname->next == NULL);
_dl_rtld_map.l_libname->next = &newname;
}
assert (_dl_rtld_map.l_libname->next == NULL);
_dl_rtld_map.l_libname->next = &newname;
}
}
/* The ld.so must be relocated since otherwise loading audit modules
will fail since they reuse the very same ld.so. */
assert (_dl_rtld_map.l_relocated);
@ -1658,10 +1654,8 @@ dl_main (const ElfW(Phdr) *phdr,
/* If the main map is libc.so, update the base namespace to
refer to this map. If libc.so is loaded later, this happens
in _dl_map_object_from_fd. */
if (main_map->l_info[DT_SONAME] != NULL
&& (strcmp (((const char *) D_PTR (main_map, l_info[DT_STRTAB])
+ main_map->l_info[DT_SONAME]->d_un.d_val), LIBC_SO)
== 0))
if (l_soname (main_map) != NULL
&& strcmp (l_soname (main_map), LIBC_SO) == 0)
GL(dl_ns)[LM_ID_BASE].libc_map = main_map;
/* Set up our cache of pointers into the hash table. */

View File

@ -76,13 +76,14 @@ setup_vdso (struct link_map *main_map __attribute__ ((unused)),
/* Now that we have the info handy, use the DSO image's soname
so this object can be looked up by name. */
if (l->l_info[DT_SONAME] != NULL)
{
char *dsoname = ((char *) D_PTR (l, l_info[DT_STRTAB])
+ l->l_info[DT_SONAME]->d_un.d_val);
l->l_libname->name = dsoname;
l->l_name = dsoname;
}
{
const char *dsoname = l_soname (l);
if (dsoname != NULL)
{
l->l_libname->name = dsoname;
l->l_name = (char *) dsoname;
}
}
/* Add the vDSO to the object list. */
_dl_add_to_namespace_list (l, LM_ID_BASE);

View File

@ -530,10 +530,7 @@ load_shobj (const char *name)
printf ("string table: %p\n", result->dynstrtab);
/* Determine the soname. */
if (map->l_info[DT_SONAME] == NULL)
result->soname = NULL;
else
result->soname = result->dynstrtab + map->l_info[DT_SONAME]->d_un.d_val;
result->soname = l_soname (map);
if (do_test && result->soname != NULL)
printf ("soname: %s\n", result->soname);

View File

@ -88,6 +88,18 @@ dl_relocate_ld (const struct link_map *l)
#define D_PTR(map, i) \
((map)->i->d_un.d_ptr + (dl_relocate_ld (map) ? 0 : (map)->l_addr))
/* Returns the soname string if the link map has a DT_SONAME tag, or
NULL if it does not. */
static inline const char *
l_soname (const struct link_map *l)
{
if (l->l_info[DT_SONAME] == NULL)
return NULL;
else
return ((const char *) D_PTR (l, l_info[DT_STRTAB])
+ l->l_info[DT_SONAME]->d_un.d_val);
}
/* Result of the lookup functions and how to retrieve the base address. */
typedef struct link_map *lookup_t;
#define LOOKUP_VALUE(map) map