| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 1 | /* | 
 | 2 |  * Copyright (C) 2013 The Android Open Source Project | 
 | 3 |  * All rights reserved. | 
 | 4 |  * | 
 | 5 |  * Redistribution and use in source and binary forms, with or without | 
 | 6 |  * modification, are permitted provided that the following conditions | 
 | 7 |  * are met: | 
 | 8 |  *  * Redistributions of source code must retain the above copyright | 
 | 9 |  *    notice, this list of conditions and the following disclaimer. | 
 | 10 |  *  * Redistributions in binary form must reproduce the above copyright | 
 | 11 |  *    notice, this list of conditions and the following disclaimer in | 
 | 12 |  *    the documentation and/or other materials provided with the | 
 | 13 |  *    distribution. | 
 | 14 |  * | 
 | 15 |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
 | 16 |  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
 | 17 |  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
 | 18 |  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | 
 | 19 |  * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | 
 | 20 |  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, | 
 | 21 |  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS | 
 | 22 |  * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED | 
 | 23 |  * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
 | 24 |  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT | 
 | 25 |  * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF | 
 | 26 |  * SUCH DAMAGE. | 
 | 27 |  */ | 
 | 28 |  | 
| Yabin Cui | 7fb680b | 2015-02-24 13:18:25 -0800 | [diff] [blame] | 29 | // This file perpetuates the mistakes of the past. | 
| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 30 |  | 
| Elliott Hughes | efbdb53 | 2014-04-07 15:17:19 -0700 | [diff] [blame] | 31 | #include <ctype.h> | 
| Elliott Hughes | d1ead2a | 2014-06-06 15:24:20 -0700 | [diff] [blame] | 32 | #include <dirent.h> | 
| Elliott Hughes | 4c5891d | 2015-02-19 22:49:44 -0800 | [diff] [blame] | 33 | #include <errno.h> | 
| Elliott Hughes | efbdb53 | 2014-04-07 15:17:19 -0700 | [diff] [blame] | 34 | #include <inttypes.h> | 
| Calin Juravle | a4eafa6 | 2014-03-10 18:10:04 +0000 | [diff] [blame] | 35 | #include <pthread.h> | 
| Dan Albert | 205dd7d | 2014-06-04 10:14:19 -0700 | [diff] [blame] | 36 | #include <signal.h> | 
| Elliott Hughes | fcac8ff | 2014-05-22 01:24:30 -0700 | [diff] [blame] | 37 | #include <stdio.h> | 
| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 38 | #include <stdlib.h> | 
| David 'Digit' Turner | 891dedb | 2014-06-13 12:28:11 +0200 | [diff] [blame] | 39 | #include <string.h> | 
| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 40 | #include <sys/resource.h> | 
| Elliott Hughes | bd3a98c | 2014-05-24 17:19:36 -0700 | [diff] [blame] | 41 | #include <sys/syscall.h> | 
| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 42 | #include <sys/time.h> | 
 | 43 | #include <sys/types.h> | 
 | 44 | #include <sys/wait.h> | 
 | 45 | #include <unistd.h> | 
| Dan Albert | 001f8f0 | 2014-06-04 09:53:06 -0700 | [diff] [blame] | 46 | #include <wchar.h> | 
| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 47 |  | 
| Josh Gao | 4956c37 | 2019-12-19 16:35:51 -0800 | [diff] [blame] | 48 | #include "platform/bionic/macros.h" | 
| Yabin Cui | 7fb680b | 2015-02-24 13:18:25 -0800 | [diff] [blame] | 49 |  | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 50 | extern "C" { | 
 | 51 |  | 
| Elliott Hughes | 5ffed9b | 2016-08-10 14:06:14 -0700 | [diff] [blame] | 52 | // LP64 doesn't need to support any legacy cruft. | 
 | 53 | #if !defined(__LP64__) | 
| Yabin Cui | 7fb680b | 2015-02-24 13:18:25 -0800 | [diff] [blame] | 54 |  | 
| Elliott Hughes | dfcb82d | 2017-05-11 15:29:03 -0700 | [diff] [blame] | 55 | // By the time any NDK-built code is running, there are plenty of threads. | 
 | 56 | int __isthreaded = 1; | 
 | 57 |  | 
| Elliott Hughes | cf34653 | 2020-07-31 10:35:03 -0700 | [diff] [blame] | 58 | // These were accidentally declared in <unistd.h> because we used to inline | 
 | 59 | // getpagesize() and __getpageshift(). Needed for backwards compatibility | 
 | 60 | // with old NDK apps. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 61 | unsigned int __page_size = PAGE_SIZE; | 
 | 62 | unsigned int __page_shift = 12; | 
| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 63 |  | 
 | 64 | // TODO: remove this backward compatibility hack (for jb-mr1 strace binaries). | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 65 | pid_t __wait4(pid_t pid, int* status, int options, struct rusage* rusage) { | 
| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 66 |   return wait4(pid, status, options, rusage); | 
 | 67 | } | 
 | 68 |  | 
| Elliott Hughes | 0620925 | 2013-11-06 16:20:54 -0800 | [diff] [blame] | 69 | // TODO: does anything still need this? | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 70 | int __open() { | 
| Elliott Hughes | 567a8de | 2013-10-24 17:14:55 -0700 | [diff] [blame] | 71 |   abort(); | 
 | 72 | } | 
 | 73 |  | 
| Elliott Hughes | 0620925 | 2013-11-06 16:20:54 -0800 | [diff] [blame] | 74 | // TODO: does anything still need this? | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 75 | void** __get_tls() { | 
| Christopher Ferris | c5d3a43 | 2019-09-25 17:50:36 -0700 | [diff] [blame] | 76 | #include "platform/bionic/tls.h" | 
| Elliott Hughes | 0620925 | 2013-11-06 16:20:54 -0800 | [diff] [blame] | 77 |   return __get_tls(); | 
 | 78 | } | 
 | 79 |  | 
| Elliott Hughes | 152b9de | 2014-03-10 15:54:40 -0700 | [diff] [blame] | 80 | // This non-standard function was in our <string.h> for some reason. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 81 | void memswap(void* m1, void* m2, size_t n) { | 
| Elliott Hughes | 152b9de | 2014-03-10 15:54:40 -0700 | [diff] [blame] | 82 |   char* p = reinterpret_cast<char*>(m1); | 
 | 83 |   char* p_end = p + n; | 
 | 84 |   char* q = reinterpret_cast<char*>(m2); | 
 | 85 |   while (p < p_end) { | 
 | 86 |     char tmp = *p; | 
 | 87 |     *p = *q; | 
 | 88 |     *q = tmp; | 
 | 89 |     p++; | 
 | 90 |     q++; | 
 | 91 |   } | 
 | 92 | } | 
 | 93 |  | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 94 | int pthread_attr_setstackaddr(pthread_attr_t*, void*) { | 
| Calin Juravle | a4eafa6 | 2014-03-10 18:10:04 +0000 | [diff] [blame] | 95 |   // This was removed from POSIX.1-2008, and is not implemented on bionic. | 
 | 96 |   // Needed for ABI compatibility with the NDK. | 
 | 97 |   return ENOSYS; | 
 | 98 | } | 
 | 99 |  | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 100 | int pthread_attr_getstackaddr(const pthread_attr_t* attr, void** stack_addr) { | 
| Calin Juravle | a4eafa6 | 2014-03-10 18:10:04 +0000 | [diff] [blame] | 101 |   // This was removed from POSIX.1-2008. | 
 | 102 |   // Needed for ABI compatibility with the NDK. | 
 | 103 |   *stack_addr = (char*)attr->stack_base + attr->stack_size; | 
 | 104 |   return 0; | 
 | 105 | } | 
 | 106 |  | 
| Elliott Hughes | efbdb53 | 2014-04-07 15:17:19 -0700 | [diff] [blame] | 107 | // Non-standard cruft that should only ever have been in system/core/toolbox. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 108 | char* strtotimeval(const char* str, struct timeval* ts) { | 
| Elliott Hughes | efbdb53 | 2014-04-07 15:17:19 -0700 | [diff] [blame] | 109 |   char* s; | 
 | 110 |   ts->tv_sec = strtoumax(str, &s, 10); | 
 | 111 |  | 
 | 112 |   long fractional_seconds = 0; | 
 | 113 |   if (*s == '.') { | 
 | 114 |     s++; | 
 | 115 |     int count = 0; | 
 | 116 |  | 
 | 117 |     // Read up to 6 digits (microseconds). | 
 | 118 |     while (*s && isdigit(*s)) { | 
 | 119 |       if (++count < 7) { | 
 | 120 |         fractional_seconds = fractional_seconds*10 + (*s - '0'); | 
 | 121 |       } | 
 | 122 |       s++; | 
 | 123 |     } | 
 | 124 |  | 
 | 125 |     for (; count < 6; count++) { | 
 | 126 |       fractional_seconds *= 10; | 
 | 127 |     } | 
 | 128 |   } | 
 | 129 |  | 
 | 130 |   ts->tv_usec = fractional_seconds; | 
 | 131 |   return s; | 
 | 132 | } | 
 | 133 |  | 
| Elliott Hughes | eae5902 | 2014-04-22 17:56:42 -0700 | [diff] [blame] | 134 | static inline int digitval(int ch) { | 
 | 135 |   unsigned d; | 
 | 136 |  | 
 | 137 |   d = (unsigned)(ch - '0'); | 
 | 138 |   if (d < 10) return (int)d; | 
 | 139 |  | 
 | 140 |   d = (unsigned)(ch - 'a'); | 
 | 141 |   if (d < 6) return (int)(d+10); | 
 | 142 |  | 
 | 143 |   d = (unsigned)(ch - 'A'); | 
 | 144 |   if (d < 6) return (int)(d+10); | 
 | 145 |  | 
 | 146 |   return -1; | 
 | 147 | } | 
 | 148 |  | 
 | 149 | // This non-standard function was in our <inttypes.h> for some reason. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 150 | uintmax_t strntoumax(const char *nptr, char **endptr, int base, size_t n) { | 
| Elliott Hughes | eae5902 | 2014-04-22 17:56:42 -0700 | [diff] [blame] | 151 |   const unsigned char*  p   = (const unsigned char *)nptr; | 
 | 152 |   const unsigned char*  end = p + n; | 
 | 153 |   int                   minus = 0; | 
 | 154 |   uintmax_t             v = 0; | 
 | 155 |   int                   d; | 
 | 156 |  | 
 | 157 |   while (p < end && isspace(*p)) { | 
 | 158 |     p++; | 
 | 159 |   } | 
 | 160 |  | 
 | 161 |   if (p < end) { | 
 | 162 |     char c = p[0]; | 
 | 163 |     if (c == '-' || c == '+') { | 
 | 164 |       minus = (c == '-'); | 
 | 165 |       p++; | 
 | 166 |     } | 
 | 167 |   } | 
 | 168 |  | 
 | 169 |   if (base == 0) { | 
 | 170 |     if (p+2 < end && p[0] == '0' && (p[1] == 'x' || p[1] == 'X')) { | 
 | 171 |       p += 2; | 
 | 172 |       base = 16; | 
 | 173 |     } else if (p+1 < end && p[0] == '0') { | 
 | 174 |       p   += 1; | 
 | 175 |       base = 8; | 
 | 176 |     } else { | 
 | 177 |       base = 10; | 
 | 178 |     } | 
 | 179 |   } else if (base == 16) { | 
 | 180 |     if (p+2 < end && p[0] == '0' && (p[1] == 'x' || p[1] == 'X')) { | 
 | 181 |       p += 2; | 
 | 182 |     } | 
 | 183 |   } | 
 | 184 |  | 
 | 185 |   while (p < end && (d = digitval(*p)) >= 0 && d < base) { | 
 | 186 |     v = v*base + d; | 
 | 187 |     p += 1; | 
 | 188 |   } | 
 | 189 |  | 
 | 190 |   if (endptr) { | 
 | 191 |     *endptr = (char*) p; | 
 | 192 |   } | 
 | 193 |  | 
 | 194 |   return minus ? -v : v; | 
 | 195 | } | 
 | 196 |  | 
 | 197 | // This non-standard function was in our <inttypes.h> for some reason. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 198 | intmax_t strntoimax(const char* nptr, char** endptr, int base, size_t n) { | 
| Elliott Hughes | eae5902 | 2014-04-22 17:56:42 -0700 | [diff] [blame] | 199 |   return (intmax_t) strntoumax(nptr, endptr, base, n); | 
 | 200 | } | 
 | 201 |  | 
| Elliott Hughes | fcac8ff | 2014-05-22 01:24:30 -0700 | [diff] [blame] | 202 | // POSIX calls this dprintf, but LP32 Android had fdprintf instead. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 203 | int fdprintf(int fd, const char* fmt, ...) { | 
| Elliott Hughes | fcac8ff | 2014-05-22 01:24:30 -0700 | [diff] [blame] | 204 |   va_list ap; | 
 | 205 |   va_start(ap, fmt); | 
 | 206 |   int rc = vdprintf(fd, fmt, ap); | 
 | 207 |   va_end(ap); | 
 | 208 |   return rc; | 
 | 209 | } | 
 | 210 |  | 
 | 211 | // POSIX calls this vdprintf, but LP32 Android had fdprintf instead. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 212 | int vfdprintf(int fd, const char* fmt, va_list ap) { | 
| Elliott Hughes | fcac8ff | 2014-05-22 01:24:30 -0700 | [diff] [blame] | 213 |   return vdprintf(fd, fmt, ap); | 
 | 214 | } | 
 | 215 |  | 
| Elliott Hughes | b30aff4 | 2014-05-28 19:35:33 +0000 | [diff] [blame] | 216 | #define __futex_wake __real_futex_wake | 
 | 217 | #define __futex_wait __real_futex_wait | 
 | 218 | #include "private/bionic_futex.h" | 
 | 219 | #undef __futex_wake | 
 | 220 | #undef __futex_wait | 
| Elliott Hughes | bd3a98c | 2014-05-24 17:19:36 -0700 | [diff] [blame] | 221 |  | 
 | 222 | // This used to be in <sys/atomics.h>. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 223 | int __futex_wake(volatile void* ftx, int count) { | 
| Elliott Hughes | b30aff4 | 2014-05-28 19:35:33 +0000 | [diff] [blame] | 224 |   return __real_futex_wake(ftx, count); | 
| Elliott Hughes | bd3a98c | 2014-05-24 17:19:36 -0700 | [diff] [blame] | 225 | } | 
 | 226 |  | 
 | 227 | // This used to be in <sys/atomics.h>. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 228 | int __futex_wait(volatile void* ftx, int value, const struct timespec* timeout) { | 
| Elliott Hughes | b30aff4 | 2014-05-28 19:35:33 +0000 | [diff] [blame] | 229 |   return __real_futex_wait(ftx, value, timeout); | 
| Elliott Hughes | bd3a98c | 2014-05-24 17:19:36 -0700 | [diff] [blame] | 230 | } | 
 | 231 |  | 
| Anthony King | 0017073 | 2014-05-24 16:47:14 +0000 | [diff] [blame] | 232 | // Unity's libmono uses this. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 233 | int tkill(pid_t tid, int sig) { | 
| Anthony King | 0017073 | 2014-05-24 16:47:14 +0000 | [diff] [blame] | 234 |   return syscall(__NR_tkill, tid, sig); | 
 | 235 | } | 
 | 236 |  | 
| Elliott Hughes | 1628eb1 | 2014-08-06 10:47:33 -0700 | [diff] [blame] | 237 | // This was removed from POSIX 2008. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 238 | wchar_t* wcswcs(wchar_t* haystack, wchar_t* needle) { | 
| Dan Albert | 001f8f0 | 2014-06-04 09:53:06 -0700 | [diff] [blame] | 239 |   return wcsstr(haystack, needle); | 
 | 240 | } | 
 | 241 |  | 
| Dan Albert | 205dd7d | 2014-06-04 10:14:19 -0700 | [diff] [blame] | 242 | // This was removed from POSIX 2008. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 243 | sighandler_t bsd_signal(int signum, sighandler_t handler) { | 
| Dan Albert | 205dd7d | 2014-06-04 10:14:19 -0700 | [diff] [blame] | 244 |   return signal(signum, handler); | 
 | 245 | } | 
 | 246 |  | 
| Elliott Hughes | 76f8916 | 2015-01-26 13:34:58 -0800 | [diff] [blame] | 247 | // This was removed from POSIX 2008. | 
| Logan Chien | b33952c | 2019-08-27 20:50:24 -0700 | [diff] [blame] | 248 | #undef bcopy | 
 | 249 | void bcopy(const void* src, void* dst, size_t n) { | 
| Rohit Agrawal | d51a0b0 | 2015-12-05 12:39:54 -0800 | [diff] [blame] | 250 |   memmove(dst, src, n); | 
| Elliott Hughes | 76f8916 | 2015-01-26 13:34:58 -0800 | [diff] [blame] | 251 | } | 
 | 252 |  | 
| Elliott Hughes | 01d5b94 | 2016-03-02 17:18:18 -0800 | [diff] [blame] | 253 | // This was removed from POSIX 2008. | 
| Logan Chien | b33952c | 2019-08-27 20:50:24 -0700 | [diff] [blame] | 254 | #undef bzero | 
 | 255 | void bzero(void* dst, size_t n) { | 
| Elliott Hughes | 01d5b94 | 2016-03-02 17:18:18 -0800 | [diff] [blame] | 256 |   memset(dst, 0, n); | 
 | 257 | } | 
 | 258 |  | 
| Dan Albert | 205dd7d | 2014-06-04 10:14:19 -0700 | [diff] [blame] | 259 | // sysv_signal() was never in POSIX. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 260 | extern "C++" sighandler_t _signal(int signum, sighandler_t handler, int flags); | 
 | 261 | sighandler_t sysv_signal(int signum, sighandler_t handler) { | 
| Dan Albert | 205dd7d | 2014-06-04 10:14:19 -0700 | [diff] [blame] | 262 |   return _signal(signum, handler, SA_RESETHAND); | 
 | 263 | } | 
 | 264 |  | 
| Elliott Hughes | 3d5cb30 | 2014-06-06 11:44:55 -0700 | [diff] [blame] | 265 | // This is a system call that was never in POSIX. Use readdir(3) instead. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 266 | int __getdents64(unsigned int, dirent*, unsigned int); | 
 | 267 | int getdents(unsigned int fd, dirent* dirp, unsigned int count) { | 
| Elliott Hughes | 3d5cb30 | 2014-06-06 11:44:55 -0700 | [diff] [blame] | 268 |   return __getdents64(fd, dirp, count); | 
 | 269 | } | 
 | 270 |  | 
| Elliott Hughes | bffbfee | 2014-06-06 20:41:42 -0700 | [diff] [blame] | 271 | // This is a BSDism that we never implemented correctly. Used by Firefox. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 272 | int issetugid() { | 
| Elliott Hughes | bffbfee | 2014-06-06 20:41:42 -0700 | [diff] [blame] | 273 |   return 0; | 
 | 274 | } | 
 | 275 |  | 
| Dan Albert | 8229ae4 | 2014-06-13 16:04:41 -0700 | [diff] [blame] | 276 | // This was removed from POSIX 2004. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 277 | pid_t wait3(int* status, int options, struct rusage* rusage) { | 
| Dan Albert | 8229ae4 | 2014-06-13 16:04:41 -0700 | [diff] [blame] | 278 |   return wait4(-1, status, options, rusage); | 
 | 279 | } | 
 | 280 |  | 
| Dan Albert | 462abab | 2014-06-13 16:51:24 -0700 | [diff] [blame] | 281 | // This was removed from POSIX 2004. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 282 | int getdtablesize() { | 
| Dan Albert | 462abab | 2014-06-13 16:51:24 -0700 | [diff] [blame] | 283 |   struct rlimit r; | 
 | 284 |  | 
 | 285 |   if (getrlimit(RLIMIT_NOFILE, &r) < 0) { | 
 | 286 |     return sysconf(_SC_OPEN_MAX); | 
 | 287 |   } | 
 | 288 |  | 
 | 289 |   return r.rlim_cur; | 
 | 290 | } | 
 | 291 |  | 
| Elliott Hughes | bb46afd | 2015-12-04 18:03:12 -0800 | [diff] [blame] | 292 | // A leaked BSD stdio implementation detail that's now a no-op. | 
| Elliott Hughes | bb46afd | 2015-12-04 18:03:12 -0800 | [diff] [blame] | 293 | void __sinit() {} | 
 | 294 | int __sdidinit = 1; | 
| Elliott Hughes | bb46afd | 2015-12-04 18:03:12 -0800 | [diff] [blame] | 295 |  | 
| Elliott Hughes | 1628eb1 | 2014-08-06 10:47:33 -0700 | [diff] [blame] | 296 | // Only used by ftime, which was removed from POSIX 2008. | 
| Dan Albert | ac64675 | 2014-06-05 02:10:49 +0000 | [diff] [blame] | 297 | struct timeb { | 
 | 298 |   time_t          time; | 
 | 299 |   unsigned short  millitm; | 
 | 300 |   short           timezone; | 
 | 301 |   short           dstflag; | 
 | 302 | }; | 
 | 303 |  | 
 | 304 | // This was removed from POSIX 2008. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 305 | int ftime(struct timeb* tb) { | 
| Dan Albert | ac64675 | 2014-06-05 02:10:49 +0000 | [diff] [blame] | 306 |   struct timeval  tv; | 
 | 307 |   struct timezone tz; | 
 | 308 |  | 
 | 309 |   if (gettimeofday(&tv, &tz) < 0) | 
 | 310 |     return -1; | 
 | 311 |  | 
 | 312 |   tb->time    = tv.tv_sec; | 
 | 313 |   tb->millitm = (tv.tv_usec + 500) / 1000; | 
 | 314 |  | 
 | 315 |   if (tb->millitm == 1000) { | 
 | 316 |     ++tb->time; | 
 | 317 |     tb->millitm = 0; | 
 | 318 |   } | 
 | 319 |  | 
 | 320 |   tb->timezone = tz.tz_minuteswest; | 
 | 321 |   tb->dstflag  = tz.tz_dsttime; | 
 | 322 |  | 
 | 323 |   return 0; | 
 | 324 | } | 
 | 325 |  | 
| David 'Digit' Turner | 891dedb | 2014-06-13 12:28:11 +0200 | [diff] [blame] | 326 | // This was removed from POSIX 2008. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 327 | char* index(const char* str, int ch) { | 
| George Burgess IV | bd3d208 | 2017-04-04 17:34:02 -0700 | [diff] [blame] | 328 |   return const_cast<char*>(strchr(str, ch)); | 
| David 'Digit' Turner | 891dedb | 2014-06-13 12:28:11 +0200 | [diff] [blame] | 329 | } | 
 | 330 |  | 
| Elliott Hughes | 5dea472 | 2014-09-03 15:53:11 -0700 | [diff] [blame] | 331 | // This was removed from BSD. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 332 | void arc4random_stir(void) { | 
| Elliott Hughes | 5dea472 | 2014-09-03 15:53:11 -0700 | [diff] [blame] | 333 |   // The current implementation stirs itself as needed. | 
 | 334 | } | 
 | 335 |  | 
| Elliott Hughes | fc82973 | 2014-09-08 10:25:33 -0700 | [diff] [blame] | 336 | // This was removed from BSD. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 337 | void arc4random_addrandom(unsigned char*, int) { | 
| Elliott Hughes | fc82973 | 2014-09-08 10:25:33 -0700 | [diff] [blame] | 338 |   // The current implementation adds randomness as needed. | 
 | 339 | } | 
 | 340 |  | 
| Christopher Ferris | f903558 | 2014-09-05 16:39:22 -0700 | [diff] [blame] | 341 | // Old versions of the NDK did not export malloc_usable_size, but did | 
 | 342 | // export dlmalloc_usable_size. We are moving away from dlmalloc in L | 
 | 343 | // so make this call malloc_usable_size. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 344 | size_t dlmalloc_usable_size(void* ptr) { | 
| Christopher Ferris | f903558 | 2014-09-05 16:39:22 -0700 | [diff] [blame] | 345 |   return malloc_usable_size(ptr); | 
 | 346 | } | 
 | 347 |  | 
| Elliott Hughes | 0f001b6 | 2014-09-12 11:35:05 -0700 | [diff] [blame] | 348 | // In L we added a public pthread_gettid_np, but some apps were using the private API. | 
| Dimitry Ivanov | bba3954 | 2016-01-21 22:25:32 +0000 | [diff] [blame] | 349 | pid_t __pthread_gettid(pthread_t t) { | 
| Elliott Hughes | 0f001b6 | 2014-09-12 11:35:05 -0700 | [diff] [blame] | 350 |   return pthread_gettid_np(t); | 
 | 351 | } | 
 | 352 |  | 
| Christopher Ferris | f9554a1 | 2015-06-05 17:12:17 -0700 | [diff] [blame] | 353 | // Older versions of apportable used dlmalloc directly instead of malloc, | 
| Christopher Ferris | f183f95 | 2014-10-08 22:48:20 -0700 | [diff] [blame] | 354 | // so export this compatibility shim that simply calls malloc. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 355 | void* dlmalloc(size_t size) { | 
| Christopher Ferris | f183f95 | 2014-10-08 22:48:20 -0700 | [diff] [blame] | 356 |   return malloc(size); | 
 | 357 | } | 
 | 358 |  | 
| Ryan Prichard | 16455b5 | 2019-01-18 01:00:59 -0800 | [diff] [blame] | 359 | } // extern "C" | 
 | 360 |  | 
| Yabin Cui | 2f836d4 | 2015-03-18 14:14:02 -0700 | [diff] [blame] | 361 | #define __get_thread __real_get_thread | 
 | 362 | #include "pthread_internal.h" | 
 | 363 | #undef __get_thread | 
| Ryan Prichard | 16455b5 | 2019-01-18 01:00:59 -0800 | [diff] [blame] | 364 |  | 
 | 365 | extern "C" { | 
 | 366 |  | 
| Yabin Cui | 2f836d4 | 2015-03-18 14:14:02 -0700 | [diff] [blame] | 367 | // Various third-party apps contain a backport of our pthread_rwlock implementation that uses this. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 368 | pthread_internal_t* __get_thread() { | 
| Yabin Cui | 2f836d4 | 2015-03-18 14:14:02 -0700 | [diff] [blame] | 369 |   return __real_get_thread(); | 
 | 370 | } | 
 | 371 |  | 
| Christopher Ferris | 9978a9a | 2015-10-29 18:11:32 -0700 | [diff] [blame] | 372 | // This one exists only for the LP32 NDK and is not present anywhere else. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 373 | extern long __set_errno_internal(int); | 
 | 374 | long __set_errno(int n) { | 
| Christopher Ferris | 9978a9a | 2015-10-29 18:11:32 -0700 | [diff] [blame] | 375 |   return __set_errno_internal(n); | 
 | 376 | } | 
 | 377 |  | 
| Christopher Ferris | f9554a1 | 2015-06-05 17:12:17 -0700 | [diff] [blame] | 378 | // Since dlmalloc_inspect_all and dlmalloc_trim are exported for systems | 
 | 379 | // that use dlmalloc, be consistent and export them everywhere. | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 380 | void dlmalloc_inspect_all(void (*)(void*, void*, size_t, void*), void*) { | 
| Christopher Ferris | f9554a1 | 2015-06-05 17:12:17 -0700 | [diff] [blame] | 381 | } | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 382 | int dlmalloc_trim(size_t) { | 
| Elliott Hughes | fb8fd50 | 2015-10-12 17:53:48 -0700 | [diff] [blame] | 383 |     return 0; | 
 | 384 | } | 
| Elliott Hughes | fb8fd50 | 2015-10-12 17:53:48 -0700 | [diff] [blame] | 385 |  | 
| Elliott Hughes | 53cf348 | 2016-08-09 13:06:41 -0700 | [diff] [blame] | 386 | // LP32's <stdio.h> had putw (but not getw). | 
| Elliott Hughes | 53cf348 | 2016-08-09 13:06:41 -0700 | [diff] [blame] | 387 | int putw(int value, FILE* fp) { | 
 | 388 |     return fwrite(&value, sizeof(value), 1, fp) == 1 ? 0 : EOF; | 
 | 389 | } | 
| Elliott Hughes | 5ffed9b | 2016-08-10 14:06:14 -0700 | [diff] [blame] | 390 |  | 
 | 391 | #endif // !defined (__LP64__) | 
| Elliott Hughes | 53cf348 | 2016-08-09 13:06:41 -0700 | [diff] [blame] | 392 |  | 
| Elliott Hughes | cfd5a46 | 2015-12-04 15:57:51 -0800 | [diff] [blame] | 393 | } // extern "C" |