| xf.li | bdd93d5 | 2023-05-12 07:10:14 -0700 | [diff] [blame] | 1 | /* Operating system support for run-time dynamic linker.  Generic Unix version. | 
 | 2 |    Copyright (C) 1995-2016 Free Software Foundation, Inc. | 
 | 3 |    This file is part of the GNU C Library. | 
 | 4 |  | 
 | 5 |    The GNU C Library is free software; you can redistribute it and/or | 
 | 6 |    modify it under the terms of the GNU Lesser General Public | 
 | 7 |    License as published by the Free Software Foundation; either | 
 | 8 |    version 2.1 of the License, or (at your option) any later version. | 
 | 9 |  | 
 | 10 |    The GNU C Library 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 GNU | 
 | 13 |    Lesser General Public License for more details. | 
 | 14 |  | 
 | 15 |    You should have received a copy of the GNU Lesser General Public | 
 | 16 |    License along with the GNU C Library; if not, see | 
 | 17 |    <http://www.gnu.org/licenses/>.  */ | 
 | 18 |  | 
 | 19 | /* We conditionalize the whole of this file rather than simply eliding it | 
 | 20 |    from the static build, because other sysdeps/ versions of this file | 
 | 21 |    might define things needed by a static build.  */ | 
 | 22 |  | 
 | 23 | #ifdef SHARED | 
 | 24 |  | 
 | 25 | #include <assert.h> | 
 | 26 | #include <elf.h> | 
 | 27 | #include <errno.h> | 
 | 28 | #include <fcntl.h> | 
 | 29 | #include <libintl.h> | 
 | 30 | #include <stdlib.h> | 
 | 31 | #include <string.h> | 
 | 32 | #include <unistd.h> | 
 | 33 | #include <sys/types.h> | 
 | 34 | #include <sys/stat.h> | 
 | 35 | #include <sys/mman.h> | 
 | 36 | #include <ldsodefs.h> | 
 | 37 | #include <_itoa.h> | 
 | 38 | #include <fpu_control.h> | 
 | 39 |  | 
 | 40 | #include <entry.h> | 
 | 41 | #include <dl-machine.h> | 
 | 42 | #include <dl-procinfo.h> | 
 | 43 | #include <dl-osinfo.h> | 
 | 44 | #include <hp-timing.h> | 
 | 45 | #include <tls.h> | 
 | 46 |  | 
 | 47 | extern char **_environ attribute_hidden; | 
 | 48 | extern char _end[] attribute_hidden; | 
 | 49 |  | 
 | 50 | /* Protect SUID program against misuse of file descriptors.  */ | 
 | 51 | extern void __libc_check_standard_fds (void); | 
 | 52 |  | 
 | 53 | #ifdef NEED_DL_BASE_ADDR | 
 | 54 | ElfW(Addr) _dl_base_addr; | 
 | 55 | #endif | 
 | 56 | int __libc_enable_secure attribute_relro = 0; | 
 | 57 | rtld_hidden_data_def (__libc_enable_secure) | 
 | 58 | int __libc_multiple_libcs = 0;	/* Defining this here avoids the inclusion | 
 | 59 | 				   of init-first.  */ | 
 | 60 | /* This variable contains the lowest stack address ever used.  */ | 
 | 61 | void *__libc_stack_end attribute_relro = NULL; | 
 | 62 | rtld_hidden_data_def(__libc_stack_end) | 
 | 63 | void *_dl_random attribute_relro = NULL; | 
 | 64 |  | 
 | 65 | #ifndef DL_FIND_ARG_COMPONENTS | 
 | 66 | # define DL_FIND_ARG_COMPONENTS(cookie, argc, argv, envp, auxp)	\ | 
 | 67 |   do {									      \ | 
 | 68 |     void **_tmp;							      \ | 
 | 69 |     (argc) = *(long int *) cookie;					      \ | 
 | 70 |     (argv) = (char **) ((long int *) cookie + 1);			      \ | 
 | 71 |     (envp) = (argv) + (argc) + 1;					      \ | 
 | 72 |     for (_tmp = (void **) (envp); *_tmp; ++_tmp)			      \ | 
 | 73 |       continue;								      \ | 
 | 74 |     (auxp) = (void *) ++_tmp;						      \ | 
 | 75 |   } while (0) | 
 | 76 | #endif | 
 | 77 |  | 
 | 78 | #ifndef DL_STACK_END | 
 | 79 | # define DL_STACK_END(cookie) ((void *) (cookie)) | 
 | 80 | #endif | 
 | 81 |  | 
 | 82 | ElfW(Addr) | 
 | 83 | _dl_sysdep_start (void **start_argptr, | 
 | 84 | 		  void (*dl_main) (const ElfW(Phdr) *phdr, ElfW(Word) phnum, | 
 | 85 | 				   ElfW(Addr) *user_entry, ElfW(auxv_t) *auxv)) | 
 | 86 | { | 
 | 87 |   const ElfW(Phdr) *phdr = NULL; | 
 | 88 |   ElfW(Word) phnum = 0; | 
 | 89 |   ElfW(Addr) user_entry; | 
 | 90 |   ElfW(auxv_t) *av; | 
 | 91 | #ifdef HAVE_AUX_SECURE | 
 | 92 | # define set_seen(tag) (tag)	/* Evaluate for the side effects.  */ | 
 | 93 | # define set_seen_secure() ((void) 0) | 
 | 94 | #else | 
 | 95 |   uid_t uid = 0; | 
 | 96 |   gid_t gid = 0; | 
 | 97 |   unsigned int seen = 0; | 
 | 98 | # define set_seen_secure() (seen = -1) | 
 | 99 | # ifdef HAVE_AUX_XID | 
 | 100 | #  define set_seen(tag) (tag)	/* Evaluate for the side effects.  */ | 
 | 101 | # else | 
 | 102 | #  define M(type) (1 << (type)) | 
 | 103 | #  define set_seen(tag) seen |= M ((tag)->a_type) | 
 | 104 | # endif | 
 | 105 | #endif | 
 | 106 | #ifdef NEED_DL_SYSINFO | 
 | 107 |   uintptr_t new_sysinfo = 0; | 
 | 108 | #endif | 
 | 109 |  | 
 | 110 |   __libc_stack_end = DL_STACK_END (start_argptr); | 
 | 111 |   DL_FIND_ARG_COMPONENTS (start_argptr, _dl_argc, _dl_argv, _environ, | 
 | 112 | 			  GLRO(dl_auxv)); | 
 | 113 |  | 
 | 114 |   user_entry = (ElfW(Addr)) ENTRY_POINT; | 
 | 115 |   GLRO(dl_platform) = NULL; /* Default to nothing known about the platform.  */ | 
 | 116 |  | 
 | 117 |   for (av = GLRO(dl_auxv); av->a_type != AT_NULL; set_seen (av++)) | 
 | 118 |     switch (av->a_type) | 
 | 119 |       { | 
 | 120 |       case AT_PHDR: | 
 | 121 | 	phdr = (void *) av->a_un.a_val; | 
 | 122 | 	break; | 
 | 123 |       case AT_PHNUM: | 
 | 124 | 	phnum = av->a_un.a_val; | 
 | 125 | 	break; | 
 | 126 |       case AT_PAGESZ: | 
 | 127 | 	GLRO(dl_pagesize) = av->a_un.a_val; | 
 | 128 | 	break; | 
 | 129 |       case AT_ENTRY: | 
 | 130 | 	user_entry = av->a_un.a_val; | 
 | 131 | 	break; | 
 | 132 | #ifdef NEED_DL_BASE_ADDR | 
 | 133 |       case AT_BASE: | 
 | 134 | 	_dl_base_addr = av->a_un.a_val; | 
 | 135 | 	break; | 
 | 136 | #endif | 
 | 137 | #ifndef HAVE_AUX_SECURE | 
 | 138 |       case AT_UID: | 
 | 139 |       case AT_EUID: | 
 | 140 | 	uid ^= av->a_un.a_val; | 
 | 141 | 	break; | 
 | 142 |       case AT_GID: | 
 | 143 |       case AT_EGID: | 
 | 144 | 	gid ^= av->a_un.a_val; | 
 | 145 | 	break; | 
 | 146 | #endif | 
 | 147 |       case AT_SECURE: | 
 | 148 | #ifndef HAVE_AUX_SECURE | 
 | 149 | 	seen = -1; | 
 | 150 | #endif | 
 | 151 | 	__libc_enable_secure = av->a_un.a_val; | 
 | 152 | 	break; | 
 | 153 |       case AT_PLATFORM: | 
 | 154 | 	GLRO(dl_platform) = (void *) av->a_un.a_val; | 
 | 155 | 	break; | 
 | 156 |       case AT_HWCAP: | 
 | 157 | 	GLRO(dl_hwcap) = (unsigned long int) av->a_un.a_val; | 
 | 158 | 	break; | 
 | 159 |       case AT_HWCAP2: | 
 | 160 | 	GLRO(dl_hwcap2) = (unsigned long int) av->a_un.a_val; | 
 | 161 | 	break; | 
 | 162 |       case AT_CLKTCK: | 
 | 163 | 	GLRO(dl_clktck) = av->a_un.a_val; | 
 | 164 | 	break; | 
 | 165 |       case AT_FPUCW: | 
 | 166 | 	GLRO(dl_fpu_control) = av->a_un.a_val; | 
 | 167 | 	break; | 
 | 168 | #ifdef NEED_DL_SYSINFO | 
 | 169 |       case AT_SYSINFO: | 
 | 170 | 	new_sysinfo = av->a_un.a_val; | 
 | 171 | 	break; | 
 | 172 | #endif | 
 | 173 | #ifdef NEED_DL_SYSINFO_DSO | 
 | 174 |       case AT_SYSINFO_EHDR: | 
 | 175 | 	GLRO(dl_sysinfo_dso) = (void *) av->a_un.a_val; | 
 | 176 | 	break; | 
 | 177 | #endif | 
 | 178 |       case AT_RANDOM: | 
 | 179 | 	_dl_random = (void *) av->a_un.a_val; | 
 | 180 | 	break; | 
 | 181 | #ifdef DL_PLATFORM_AUXV | 
 | 182 |       DL_PLATFORM_AUXV | 
 | 183 | #endif | 
 | 184 |       } | 
 | 185 |  | 
 | 186 | #ifndef HAVE_AUX_SECURE | 
 | 187 |   if (seen != -1) | 
 | 188 |     { | 
 | 189 |       /* Fill in the values we have not gotten from the kernel through the | 
 | 190 | 	 auxiliary vector.  */ | 
 | 191 | # ifndef HAVE_AUX_XID | 
 | 192 | #  define SEE(UID, var, uid) \ | 
 | 193 |    if ((seen & M (AT_##UID)) == 0) var ^= __get##uid () | 
 | 194 |       SEE (UID, uid, uid); | 
 | 195 |       SEE (EUID, uid, euid); | 
 | 196 |       SEE (GID, gid, gid); | 
 | 197 |       SEE (EGID, gid, egid); | 
 | 198 | # endif | 
 | 199 |  | 
 | 200 |       /* If one of the two pairs of IDs does not match this is a setuid | 
 | 201 | 	 or setgid run.  */ | 
 | 202 |       __libc_enable_secure = uid | gid; | 
 | 203 |     } | 
 | 204 | #endif | 
 | 205 |  | 
 | 206 | #ifndef HAVE_AUX_PAGESIZE | 
 | 207 |   if (GLRO(dl_pagesize) == 0) | 
 | 208 |     GLRO(dl_pagesize) = __getpagesize (); | 
 | 209 | #endif | 
 | 210 |  | 
 | 211 | #ifdef NEED_DL_SYSINFO | 
 | 212 |   if (new_sysinfo != 0) | 
 | 213 |     { | 
 | 214 | # ifdef NEED_DL_SYSINFO_DSO | 
 | 215 |       /* Only set the sysinfo value if we also have the vsyscall DSO.  */ | 
 | 216 |       if (GLRO(dl_sysinfo_dso) != 0) | 
 | 217 | # endif | 
 | 218 |         GLRO(dl_sysinfo) = new_sysinfo; | 
 | 219 |     } | 
 | 220 | #endif | 
 | 221 |  | 
 | 222 | #ifdef DL_SYSDEP_INIT | 
 | 223 |   DL_SYSDEP_INIT; | 
 | 224 | #endif | 
 | 225 |  | 
 | 226 | #ifdef DL_PLATFORM_INIT | 
 | 227 |   DL_PLATFORM_INIT; | 
 | 228 | #endif | 
 | 229 |  | 
 | 230 |   /* Determine the length of the platform name.  */ | 
 | 231 |   if (GLRO(dl_platform) != NULL) | 
 | 232 |     GLRO(dl_platformlen) = strlen (GLRO(dl_platform)); | 
 | 233 |  | 
 | 234 |   if (__sbrk (0) == _end) | 
 | 235 |     /* The dynamic linker was run as a program, and so the initial break | 
 | 236 |        starts just after our bss, at &_end.  The malloc in dl-minimal.c | 
 | 237 |        will consume the rest of this page, so tell the kernel to move the | 
 | 238 |        break up that far.  When the user program examines its break, it | 
 | 239 |        will see this new value and not clobber our data.  */ | 
 | 240 |     __sbrk (GLRO(dl_pagesize) | 
 | 241 | 	    - ((_end - (char *) 0) & (GLRO(dl_pagesize) - 1))); | 
 | 242 |  | 
 | 243 |   /* If this is a SUID program we make sure that FDs 0, 1, and 2 are | 
 | 244 |      allocated.  If necessary we are doing it ourself.  If it is not | 
 | 245 |      possible we stop the program.  */ | 
 | 246 |   if (__builtin_expect (__libc_enable_secure, 0)) | 
 | 247 |     __libc_check_standard_fds (); | 
 | 248 |  | 
 | 249 |   (*dl_main) (phdr, phnum, &user_entry, GLRO(dl_auxv)); | 
 | 250 |   return user_entry; | 
 | 251 | } | 
 | 252 |  | 
 | 253 | void | 
 | 254 | internal_function | 
 | 255 | _dl_sysdep_start_cleanup (void) | 
 | 256 | { | 
 | 257 | } | 
 | 258 |  | 
 | 259 | void | 
 | 260 | internal_function | 
 | 261 | _dl_show_auxv (void) | 
 | 262 | { | 
 | 263 |   char buf[64]; | 
 | 264 |   ElfW(auxv_t) *av; | 
 | 265 |  | 
 | 266 |   /* Terminate string.  */ | 
 | 267 |   buf[63] = '\0'; | 
 | 268 |  | 
 | 269 |   /* The following code assumes that the AT_* values are encoded | 
 | 270 |      starting from 0 with AT_NULL, 1 for AT_IGNORE, and all other values | 
 | 271 |      close by (otherwise the array will be too large).  In case we have | 
 | 272 |      to support a platform where these requirements are not fulfilled | 
 | 273 |      some alternative implementation has to be used.  */ | 
 | 274 |   for (av = GLRO(dl_auxv); av->a_type != AT_NULL; ++av) | 
 | 275 |     { | 
 | 276 |       static const struct | 
 | 277 |       { | 
 | 278 | 	const char label[17]; | 
 | 279 | 	enum { unknown = 0, dec, hex, str, ignore } form : 8; | 
 | 280 |       } auxvars[] = | 
 | 281 | 	{ | 
 | 282 | 	  [AT_EXECFD - 2] =		{ "EXECFD:       ", dec }, | 
 | 283 | 	  [AT_EXECFN - 2] =		{ "EXECFN:       ", str }, | 
 | 284 | 	  [AT_PHDR - 2] =		{ "PHDR:         0x", hex }, | 
 | 285 | 	  [AT_PHENT - 2] =		{ "PHENT:        ", dec }, | 
 | 286 | 	  [AT_PHNUM - 2] =		{ "PHNUM:        ", dec }, | 
 | 287 | 	  [AT_PAGESZ - 2] =		{ "PAGESZ:       ", dec }, | 
 | 288 | 	  [AT_BASE - 2] =		{ "BASE:         0x", hex }, | 
 | 289 | 	  [AT_FLAGS - 2] =		{ "FLAGS:        0x", hex }, | 
 | 290 | 	  [AT_ENTRY - 2] =		{ "ENTRY:        0x", hex }, | 
 | 291 | 	  [AT_NOTELF - 2] =		{ "NOTELF:       ", hex }, | 
 | 292 | 	  [AT_UID - 2] =		{ "UID:          ", dec }, | 
 | 293 | 	  [AT_EUID - 2] =		{ "EUID:         ", dec }, | 
 | 294 | 	  [AT_GID - 2] =		{ "GID:          ", dec }, | 
 | 295 | 	  [AT_EGID - 2] =		{ "EGID:         ", dec }, | 
 | 296 | 	  [AT_PLATFORM - 2] =		{ "PLATFORM:     ", str }, | 
 | 297 | 	  [AT_HWCAP - 2] =		{ "HWCAP:        ", hex }, | 
 | 298 | 	  [AT_CLKTCK - 2] =		{ "CLKTCK:       ", dec }, | 
 | 299 | 	  [AT_FPUCW - 2] =		{ "FPUCW:        ", hex }, | 
 | 300 | 	  [AT_DCACHEBSIZE - 2] =	{ "DCACHEBSIZE:  0x", hex }, | 
 | 301 | 	  [AT_ICACHEBSIZE - 2] =	{ "ICACHEBSIZE:  0x", hex }, | 
 | 302 | 	  [AT_UCACHEBSIZE - 2] =	{ "UCACHEBSIZE:  0x", hex }, | 
 | 303 | 	  [AT_IGNOREPPC - 2] =		{ "IGNOREPPC", ignore }, | 
 | 304 | 	  [AT_SECURE - 2] =		{ "SECURE:       ", dec }, | 
 | 305 | 	  [AT_BASE_PLATFORM - 2] =	{ "BASE_PLATFORM:", str }, | 
 | 306 | 	  [AT_SYSINFO - 2] =		{ "SYSINFO:      0x", hex }, | 
 | 307 | 	  [AT_SYSINFO_EHDR - 2] =	{ "SYSINFO_EHDR: 0x", hex }, | 
 | 308 | 	  [AT_RANDOM - 2] =		{ "RANDOM:       0x", hex }, | 
 | 309 | 	  [AT_HWCAP2 - 2] =		{ "HWCAP2:       0x", hex }, | 
 | 310 | 	}; | 
 | 311 |       unsigned int idx = (unsigned int) (av->a_type - 2); | 
 | 312 |  | 
 | 313 |       if ((unsigned int) av->a_type < 2u | 
 | 314 | 	  || (idx < sizeof (auxvars) / sizeof (auxvars[0]) | 
 | 315 | 	      && auxvars[idx].form == ignore)) | 
 | 316 | 	continue; | 
 | 317 |  | 
 | 318 |       assert (AT_NULL == 0); | 
 | 319 |       assert (AT_IGNORE == 1); | 
 | 320 |  | 
 | 321 |       if (av->a_type == AT_HWCAP || av->a_type == AT_HWCAP2) | 
 | 322 | 	{ | 
 | 323 | 	  /* These are handled in a special way per platform.  */ | 
 | 324 | 	  if (_dl_procinfo (av->a_type, av->a_un.a_val) == 0) | 
 | 325 | 	    continue; | 
 | 326 | 	} | 
 | 327 |  | 
 | 328 |       if (idx < sizeof (auxvars) / sizeof (auxvars[0]) | 
 | 329 | 	  && auxvars[idx].form != unknown) | 
 | 330 | 	{ | 
 | 331 | 	  const char *val = (char *) av->a_un.a_val; | 
 | 332 |  | 
 | 333 | 	  if (__builtin_expect (auxvars[idx].form, dec) == dec) | 
 | 334 | 	    val = _itoa ((unsigned long int) av->a_un.a_val, | 
 | 335 | 			 buf + sizeof buf - 1, 10, 0); | 
 | 336 | 	  else if (__builtin_expect (auxvars[idx].form, hex) == hex) | 
 | 337 | 	    val = _itoa ((unsigned long int) av->a_un.a_val, | 
 | 338 | 			 buf + sizeof buf - 1, 16, 0); | 
 | 339 |  | 
 | 340 | 	  _dl_printf ("AT_%s%s\n", auxvars[idx].label, val); | 
 | 341 |  | 
 | 342 | 	  continue; | 
 | 343 | 	} | 
 | 344 |  | 
 | 345 |       /* Unknown value: print a generic line.  */ | 
 | 346 |       char buf2[17]; | 
 | 347 |       buf2[sizeof (buf2) - 1] = '\0'; | 
 | 348 |       const char *val2 = _itoa ((unsigned long int) av->a_un.a_val, | 
 | 349 | 				buf2 + sizeof buf2 - 1, 16, 0); | 
 | 350 |       const char *val =  _itoa ((unsigned long int) av->a_type, | 
 | 351 | 				buf + sizeof buf - 1, 16, 0); | 
 | 352 |       _dl_printf ("AT_??? (0x%s): 0x%s\n", val, val2); | 
 | 353 |     } | 
 | 354 | } | 
 | 355 |  | 
 | 356 | #endif |