1 /* Minimal replacements for basic facilities used in the dynamic linker.
2    Copyright (C) 1995-2022 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    <https://www.gnu.org/licenses/>.  */
18 
19 #include <assert.h>
20 #include <ldsodefs.h>
21 #include <dl-irel.h>
22 #include <dl-hash.h>
23 #include <dl-sym-post.h>
24 #include <_itoa.h>
25 #include <dl-minimal-malloc.h>
26 #include <stdio.h>
27 #include <unistd.h>
28 #include <errno.h>
29 
30 /* The rtld startup code calls __rtld_malloc_init_stubs after the
31   first self-relocation to adjust the pointers to the minimal
32   implementation below.  Before the final relocation,
33   __rtld_malloc_init_real is called to replace the pointers with the
34   real implementation.  */
35 __typeof (calloc) *__rtld_calloc attribute_relro;
36 __typeof (free) *__rtld_free attribute_relro;
37 __typeof (malloc) *__rtld_malloc attribute_relro;
38 __typeof (realloc) *__rtld_realloc attribute_relro;
39 
40 void
__rtld_malloc_init_stubs(void)41 __rtld_malloc_init_stubs (void)
42 {
43   __rtld_calloc = &__minimal_calloc;
44   __rtld_free = &__minimal_free;
45   __rtld_malloc = &__minimal_malloc;
46   __rtld_realloc = &__minimal_realloc;
47 }
48 
49 bool
__rtld_malloc_is_complete(void)50 __rtld_malloc_is_complete (void)
51 {
52   /* The caller assumes that there is an active malloc.  */
53   assert (__rtld_malloc != NULL);
54   return __rtld_malloc != &__minimal_malloc;
55 }
56 
57 /* Lookup NAME at VERSION in the scope of MATCH.  */
58 static void *
lookup_malloc_symbol(struct link_map * main_map,const char * name,struct r_found_version * version)59 lookup_malloc_symbol (struct link_map *main_map, const char *name,
60 		      struct r_found_version *version)
61 {
62 
63   const ElfW(Sym) *ref = NULL;
64   lookup_t result = _dl_lookup_symbol_x (name, main_map, &ref,
65 					 main_map->l_scope,
66 					 version, 0, 0, NULL);
67 
68   assert (ELFW(ST_TYPE) (ref->st_info) != STT_TLS);
69   void *value = DL_SYMBOL_ADDRESS (result, ref);
70 
71   return _dl_sym_post (result, ref, value, 0, main_map);
72 }
73 
74 void
__rtld_malloc_init_real(struct link_map * main_map)75 __rtld_malloc_init_real (struct link_map *main_map)
76 {
77   /* We cannot use relocations and initializers for this because the
78      changes made by __rtld_malloc_init_stubs break REL-style
79      (non-RELA) relocations that depend on the previous pointer
80      contents.  Also avoid direct relocation depedencies for the
81      malloc symbols so this function can be called before the final
82      rtld relocation (which enables RELRO, after which the pointer
83      variables cannot be written to).  */
84 
85   struct r_found_version version;
86   version.name = symbol_version_string (libc, GLIBC_2_0);
87   version.hidden = 0;
88   version.hash = _dl_elf_hash (version.name);
89   version.filename = NULL;
90 
91   void *new_calloc = lookup_malloc_symbol (main_map, "calloc", &version);
92   void *new_free = lookup_malloc_symbol (main_map, "free", &version);
93   void *new_malloc = lookup_malloc_symbol (main_map, "malloc", &version);
94   void *new_realloc = lookup_malloc_symbol (main_map, "realloc", &version);
95 
96   /* Update the pointers in one go, so that any internal allocations
97      performed by lookup_malloc_symbol see a consistent
98      implementation.  */
99   __rtld_calloc = new_calloc;
100   __rtld_free = new_free;
101   __rtld_malloc = new_malloc;
102   __rtld_realloc = new_realloc;
103 }
104 
105 
106 /* Avoid signal frobnication in setjmp/longjmp.  Keeps things smaller.  */
107 
108 #include <setjmp.h>
109 
110 int weak_function
__sigjmp_save(sigjmp_buf env,int savemask)111 __sigjmp_save (sigjmp_buf env, int savemask __attribute__ ((unused)))
112 {
113   env[0].__mask_was_saved = 0;
114   return 0;
115 }
116 
117 /* Define our own version of the internal function used by strerror.  We
118    only provide the messages for some common errors.  This avoids pulling
119    in the whole error list.  */
120 
121 char * weak_function
__strerror_r(int errnum,char * buf,size_t buflen)122 __strerror_r (int errnum, char *buf, size_t buflen)
123 {
124   char *msg;
125 
126   switch (errnum)
127     {
128     case ENOMEM:
129       msg = (char *) "Cannot allocate memory";
130       break;
131     case EINVAL:
132       msg = (char *) "Invalid argument";
133       break;
134     case ENOENT:
135       msg = (char *) "No such file or directory";
136       break;
137     case EPERM:
138       msg = (char *) "Operation not permitted";
139       break;
140     case EIO:
141       msg = (char *) "Input/output error";
142       break;
143     case EACCES:
144       msg = (char *) "Permission denied";
145       break;
146     default:
147       /* No need to check buffer size, all calls in the dynamic linker
148 	 provide enough space.  */
149       buf[buflen - 1] = '\0';
150       msg = _itoa (errnum, buf + buflen - 1, 10, 0);
151       msg = memcpy (msg - (sizeof ("Error ") - 1), "Error ",
152 		    sizeof ("Error ") - 1);
153       break;
154     }
155 
156   return msg;
157 }
158 
159 void
__libc_fatal(const char * message)160 __libc_fatal (const char *message)
161 {
162   _dl_fatal_printf ("%s", message);
163 }
rtld_hidden_def(__libc_fatal)164 rtld_hidden_def (__libc_fatal)
165 
166 void
167 __attribute__ ((noreturn))
168 __chk_fail (void)
169 {
170   _exit (127);
171 }
rtld_hidden_def(__chk_fail)172 rtld_hidden_def (__chk_fail)
173 
174 #ifndef NDEBUG
175 /* Define (weakly) our own assert failure function which doesn't use stdio.
176    If we are linked into the user program (-ldl), the normal __assert_fail
177    defn can override this one.  */
178 
179 void weak_function
180 __assert_fail (const char *assertion,
181 	       const char *file, unsigned int line, const char *function)
182 {
183   _dl_fatal_printf ("\
184 Inconsistency detected by ld.so: %s: %u: %s%sAssertion `%s' failed!\n",
185 		    file, line, function ?: "", function ? ": " : "",
186 		    assertion);
187 
188 }
189 # ifndef NO_RTLD_HIDDEN
rtld_hidden_weak(__assert_fail)190 rtld_hidden_weak (__assert_fail)
191 # endif
192 
193 void weak_function
194 __assert_perror_fail (int errnum,
195 		      const char *file, unsigned int line,
196 		      const char *function)
197 {
198   char errbuf[400];
199   _dl_fatal_printf ("\
200 Inconsistency detected by ld.so: %s: %u: %s%sUnexpected error: %s.\n",
201 		    file, line, function ?: "", function ? ": " : "",
202 		    __strerror_r (errnum, errbuf, sizeof errbuf));
203 
204 }
205 # ifndef NO_RTLD_HIDDEN
rtld_hidden_weak(__assert_perror_fail)206 rtld_hidden_weak (__assert_perror_fail)
207 # endif
208 #endif
209 
210 #undef _itoa
211 /* We always use _itoa instead of _itoa_word in ld.so since the former
212    also has to be present and it is never about speed when these
213    functions are used.  */
214 char *
215 _itoa (unsigned long long int value, char *buflim, unsigned int base,
216        int upper_case)
217 {
218   assert (! upper_case);
219 
220   do
221     *--buflim = _itoa_lower_digits[value % base];
222   while ((value /= base) != 0);
223 
224   return buflim;
225 }
226 
227 /* The '_itoa_lower_digits' variable in libc.so is able to handle bases
228    up to 36.  We don't need this here.  */
229 const char _itoa_lower_digits[16] = "0123456789abcdef";
rtld_hidden_data_def(_itoa_lower_digits)230 rtld_hidden_data_def (_itoa_lower_digits)
231 
232 /* The following is not a complete strsep implementation.  It cannot
233    handle empty delimiter strings.  But this isn't necessary for the
234    execution of ld.so.  */
235 #undef strsep
236 #undef __strsep
237 char *
238 __strsep (char **stringp, const char *delim)
239 {
240   char *begin;
241 
242   assert (delim[0] != '\0');
243 
244   begin = *stringp;
245   if (begin != NULL)
246     {
247       char *end = begin;
248 
249       while (*end != '\0' || (end = NULL))
250 	{
251 	  const char *dp = delim;
252 
253 	  do
254 	    if (*dp == *end)
255 	      break;
256 	  while (*++dp != '\0');
257 
258 	  if (*dp != '\0')
259 	    {
260 	      *end++ = '\0';
261 	      break;
262 	    }
263 
264 	  ++end;
265 	}
266 
267       *stringp = end;
268     }
269 
270   return begin;
271 }
272 weak_alias (__strsep, strsep)
273 strong_alias (__strsep, __strsep_g)
274