1 /* Minimal replacements for basic facilities used in the dynamic linker.
2 Copyright (C) 1995-1998,2000-2002,2004,2005 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
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.
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.
15 You should have received a copy of the GNU Lesser General Public
16 License along with the GNU C Library; if not, write to the Free
17 Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
26 #include <sys/param.h>
27 #include <sys/types.h>
29 #include <stdio-common/_itoa.h>
33 /* Minimal `malloc' allocator for use while loading shared libraries.
34 No block is ever freed. */
36 static void *alloc_ptr
, *alloc_end
, *alloc_last_block
;
38 /* Declarations of global functions. */
39 extern void weak_function
free (void *ptr
);
40 extern void * weak_function
realloc (void *ptr
, size_t n
);
41 extern unsigned long int weak_function
__strtoul_internal (const char *nptr
,
45 extern unsigned long int weak_function
strtoul (const char *nptr
,
46 char **endptr
, int base
);
49 /* Allocate an aligned memory block. */
51 __libc_memalign (size_t align
, size_t n
)
54 #define _dl_zerofd (-1)
56 extern int _dl_zerofd
;
59 _dl_zerofd
= _dl_sysdep_open_zero_fill ();
65 /* Consume any unused space in the last page of our data segment. */
66 extern int _end attribute_hidden
;
68 alloc_end
= (void *) 0 + (((alloc_ptr
- (void *) 0)
69 + GLRO(dl_pagesize
) - 1)
70 & ~(GLRO(dl_pagesize
) - 1));
73 /* Make sure the allocation pointer is ideally aligned. */
74 alloc_ptr
= (void *) 0 + (((alloc_ptr
- (void *) 0) + align
- 1)
77 if (alloc_ptr
+ n
>= alloc_end
)
79 /* Insufficient space left; allocate another page. */
81 size_t nup
= (n
+ GLRO(dl_pagesize
) - 1) & ~(GLRO(dl_pagesize
) - 1);
82 page
= __mmap (0, nup
, PROT_READ
|PROT_WRITE
,
83 MAP_ANON
|MAP_PRIVATE
, _dl_zerofd
, 0);
84 assert (page
!= MAP_FAILED
);
85 if (page
!= alloc_end
)
87 alloc_end
= page
+ nup
;
90 alloc_last_block
= (void *) alloc_ptr
;
92 return alloc_last_block
;
98 return __libc_memalign (sizeof (double), n
);
101 /* We use this function occasionally since the real implementation may
102 be optimized when it can assume the memory it returns already is
105 calloc (size_t nmemb
, size_t size
)
107 /* New memory from the trivial malloc above is always already cleared.
108 (We make sure that's true in the rare occasion it might not be,
109 by clearing memory in free, below.) */
110 return malloc (nmemb
* size
);
113 /* This will rarely be called. */
117 /* We can free only the last block allocated. */
118 if (ptr
== alloc_last_block
)
120 /* Since this is rare, we clear the freed block here
121 so that calloc can presume malloc returns cleared memory. */
122 memset (alloc_last_block
, '\0', alloc_ptr
- alloc_last_block
);
123 alloc_ptr
= alloc_last_block
;
127 /* This is only called with the most recent block returned by malloc. */
129 realloc (void *ptr
, size_t n
)
134 assert (ptr
== alloc_last_block
);
135 alloc_ptr
= alloc_last_block
;
141 /* Avoid signal frobnication in setjmp/longjmp. Keeps things smaller. */
146 __sigjmp_save (sigjmp_buf env
, int savemask
__attribute__ ((unused
)))
148 env
[0].__mask_was_saved
= 0;
152 /* Define our own version of the internal function used by strerror. We
153 only provide the messages for some common errors. This avoids pulling
154 in the whole error list. */
157 __strerror_r (int errnum
, char *buf
, size_t buflen
)
164 msg
= (char *) "Cannot allocate memory";
167 msg
= (char *) "Invalid argument";
170 msg
= (char *) "No such file or directory";
173 msg
= (char *) "Operation not permitted";
176 msg
= (char *) "Input/output error";
179 msg
= (char *) "Permission denied";
182 /* No need to check buffer size, all calls in the dynamic linker
183 provide enough space. */
184 buf
[buflen
- 1] = '\0';
185 msg
= _itoa (errnum
, buf
+ buflen
- 1, 10, 0);
186 msg
= memcpy (msg
- (sizeof ("Error ") - 1), "Error ",
187 sizeof ("Error ") - 1);
196 /* Define (weakly) our own assert failure function which doesn't use stdio.
197 If we are linked into the user program (-ldl), the normal __assert_fail
198 defn can override this one. */
201 __assert_fail (const char *assertion
,
202 const char *file
, unsigned int line
, const char *function
)
205 Inconsistency detected by ld.so: %s: %u: %s%sAssertion `%s' failed!\n",
206 file
, line
, function
?: "", function
? ": " : "",
210 rtld_hidden_weak(__assert_fail
)
213 __assert_perror_fail (int errnum
,
214 const char *file
, unsigned int line
,
215 const char *function
)
219 Inconsistency detected by ld.so: %s: %u: %s%sUnexpected error: %s.\n",
220 file
, line
, function
?: "", function
? ": " : "",
221 __strerror_r (errnum
, errbuf
, sizeof errbuf
));
224 rtld_hidden_weak (__assert_perror_fail
)
227 unsigned long int weak_function
228 __strtoul_internal (const char *nptr
, char **endptr
, int base
, int group
)
230 unsigned long int result
= 0;
233 while (*nptr
== ' ' || *nptr
== '\t')
241 else if (*nptr
== '+')
244 if (*nptr
< '0' || *nptr
> '9')
247 *endptr
= (char *) nptr
;
255 if (nptr
[1] == 'x' || nptr
[1] == 'X')
264 while (*nptr
>= '0' && *nptr
<= '9')
266 unsigned long int digval
= *nptr
- '0';
267 if (result
> LONG_MAX
/ 10
268 || (result
== ULONG_MAX
/ 10 && digval
> ULONG_MAX
% 10))
272 *endptr
= (char *) nptr
;
281 *endptr
= (char *) nptr
;
282 return result
* sign
;
286 /* We always use _itoa instead of _itoa_word in ld.so since the former
287 also has to be present and it is never about speed when these
288 functions are used. */
290 _itoa (value
, buflim
, base
, upper_case
)
291 unsigned long long int value
;
296 extern const char INTUSE(_itoa_lower_digits
)[] attribute_hidden
;
298 assert (! upper_case
);
301 *--buflim
= INTUSE(_itoa_lower_digits
)[value
% base
];
302 while ((value
/= base
) != 0);
308 /* The following is not a complete strsep implementation. It cannot
309 handle empty delimiter strings. But this isn't necessary for the
310 execution of ld.so. */
314 __strsep (char **stringp
, const char *delim
)
318 assert (delim
[0] != '\0');
325 while (*end
!= '\0' || (end
= NULL
))
327 const char *dp
= delim
;
332 while (*++dp
!= '\0');
348 weak_alias (__strsep
, strsep
)
349 strong_alias (__strsep
, __strsep_g
)
352 __attribute__ ((noreturn
))
357 rtld_hidden_def (__chk_fail
)
359 /* The '_itoa_lower_digits' variable in libc.so is able to handle bases
360 up to 36. We don't need this here. */
361 const char INTUSE(_itoa_lower_digits
)[16] attribute_hidden
362 = "0123456789abcdef";