| 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" |