blob: af1c0c1000ff952ad00555183224ea089db53185 [file] [log] [blame]
The Android Open Source Project1dc9e472009-03-03 19:28:35 -08001/*
2 * Copyright (C) 2008 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#ifndef _STRING_H_
29#define _STRING_H_
30
31#include <sys/cdefs.h>
32#include <stddef.h>
33#include <malloc.h>
Dan Albertdfb5ce42014-07-09 22:51:34 +000034#include <xlocale.h>
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080035
36__BEGIN_DECLS
37
Nick Kralevich1c462b72013-05-07 10:00:21 -070038extern void* memccpy(void* __restrict, const void* __restrict, int, size_t);
Nick Kralevicha6779072012-03-21 08:48:18 -070039extern void* memchr(const void *, int, size_t) __purefunc;
40extern void* memrchr(const void *, int, size_t) __purefunc;
41extern int memcmp(const void *, const void *, size_t) __purefunc;
Nick Kralevich1c462b72013-05-07 10:00:21 -070042extern void* memcpy(void* __restrict, const void* __restrict, size_t);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080043extern void* memmove(void *, const void *, size_t);
44extern void* memset(void *, int, size_t);
Nick Kralevicha6779072012-03-21 08:48:18 -070045extern void* memmem(const void *, size_t, const void *, size_t) __purefunc;
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080046
Nick Kralevicha6779072012-03-21 08:48:18 -070047extern char* strchr(const char *, int) __purefunc;
Pavel Chupin3c4b50f2013-07-26 16:50:11 +040048extern char* __strchr_chk(const char *, int, size_t);
49
Nick Kralevicha6779072012-03-21 08:48:18 -070050extern char* strrchr(const char *, int) __purefunc;
Elliott Hughes53e43292014-02-24 18:00:43 -080051extern char* __strrchr_chk(const char *, int, size_t);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080052
Nick Kralevicha6779072012-03-21 08:48:18 -070053extern size_t strlen(const char *) __purefunc;
Nick Kralevichcf870192013-05-30 16:48:53 -070054extern size_t __strlen_chk(const char *, size_t);
Nick Kralevicha6779072012-03-21 08:48:18 -070055extern int strcmp(const char *, const char *) __purefunc;
Christopher Ferris950a58e2014-04-04 14:38:18 -070056extern char* stpcpy(char* __restrict, const char* __restrict);
Nick Kralevich1c462b72013-05-07 10:00:21 -070057extern char* strcpy(char* __restrict, const char* __restrict);
58extern char* strcat(char* __restrict, const char* __restrict);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080059
Nick Kralevicha6779072012-03-21 08:48:18 -070060extern int strcasecmp(const char *, const char *) __purefunc;
61extern int strncasecmp(const char *, const char *, size_t) __purefunc;
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080062extern char* strdup(const char *);
63
Nick Kralevicha6779072012-03-21 08:48:18 -070064extern char* strstr(const char *, const char *) __purefunc;
65extern char* strcasestr(const char *haystack, const char *needle) __purefunc;
Nick Kralevich1c462b72013-05-07 10:00:21 -070066extern char* strtok(char* __restrict, const char* __restrict);
67extern char* strtok_r(char* __restrict, const char* __restrict, char** __restrict);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080068
69extern char* strerror(int);
70extern int strerror_r(int errnum, char *buf, size_t n);
71
Nick Kralevicha6779072012-03-21 08:48:18 -070072extern size_t strnlen(const char *, size_t) __purefunc;
Nick Kralevich1c462b72013-05-07 10:00:21 -070073extern char* strncat(char* __restrict, const char* __restrict, size_t);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080074extern char* strndup(const char *, size_t);
Nick Kralevicha6779072012-03-21 08:48:18 -070075extern int strncmp(const char *, const char *, size_t) __purefunc;
Christopher Ferris950a58e2014-04-04 14:38:18 -070076extern char* stpncpy(char* __restrict, const char* __restrict, size_t);
Nick Kralevich1c462b72013-05-07 10:00:21 -070077extern char* strncpy(char* __restrict, const char* __restrict, size_t);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080078
Nick Kralevich1c462b72013-05-07 10:00:21 -070079extern size_t strlcat(char* __restrict, const char* __restrict, size_t);
80extern size_t strlcpy(char* __restrict, const char* __restrict, size_t);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080081
Nick Kralevicha6779072012-03-21 08:48:18 -070082extern size_t strcspn(const char *, const char *) __purefunc;
83extern char* strpbrk(const char *, const char *) __purefunc;
Nick Kralevich1c462b72013-05-07 10:00:21 -070084extern char* strsep(char** __restrict, const char* __restrict);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080085extern size_t strspn(const char *, const char *);
86
87extern char* strsignal(int sig);
88
Nick Kralevicha6779072012-03-21 08:48:18 -070089extern int strcoll(const char *, const char *) __purefunc;
Nick Kralevich1c462b72013-05-07 10:00:21 -070090extern size_t strxfrm(char* __restrict, const char* __restrict, size_t);
The Android Open Source Project1dc9e472009-03-03 19:28:35 -080091
Dan Albertdfb5ce42014-07-09 22:51:34 +000092extern int strcoll_l(const char *, const char *, locale_t) __purefunc;
93extern size_t strxfrm_l(char* __restrict, const char* __restrict, size_t, locale_t);
94
Elliott Hughes890c8ed2013-03-22 10:58:55 -070095#if defined(__BIONIC_FORTIFY)
Nick Kralevich0a230152012-06-04 15:20:25 -070096
Christopher Ferris950a58e2014-04-04 14:38:18 -070097__errordecl(__memcpy_dest_size_error, "memcpy: prevented write past end of buffer");
98__errordecl(__memcpy_src_size_error, "memcpy: prevented read past end of buffer");
Nick Kralevichf3913b52012-07-12 15:10:03 -070099
Nick Kralevich0a230152012-06-04 15:20:25 -0700100__BIONIC_FORTIFY_INLINE
Nick Kralevich16d1af12013-06-17 14:49:19 -0700101void* memcpy(void* __restrict dest, const void* __restrict src, size_t copy_amount) {
Nick Kralevichf3913b52012-07-12 15:10:03 -0700102 char *d = (char *) dest;
103 const char *s = (const char *) src;
Nick Kralevichbd8e6742013-08-28 13:22:52 -0700104 size_t s_len = __bos0(s);
105 size_t d_len = __bos0(d);
Nick Kralevichf3913b52012-07-12 15:10:03 -0700106
107 if (__builtin_constant_p(copy_amount) && (copy_amount > d_len)) {
108 __memcpy_dest_size_error();
109 }
110
111 if (__builtin_constant_p(copy_amount) && (copy_amount > s_len)) {
112 __memcpy_src_size_error();
113 }
114
Nick Kralevichc37fc1a2012-07-13 17:49:10 -0700115 return __builtin___memcpy_chk(dest, src, copy_amount, d_len);
Nick Kralevich0a230152012-06-04 15:20:25 -0700116}
117
118__BIONIC_FORTIFY_INLINE
Nick Kralevich16d1af12013-06-17 14:49:19 -0700119void* memmove(void *dest, const void *src, size_t len) {
Nick Kralevichbd8e6742013-08-28 13:22:52 -0700120 return __builtin___memmove_chk(dest, src, len, __bos0(dest));
Nick Kralevich0a230152012-06-04 15:20:25 -0700121}
122
123__BIONIC_FORTIFY_INLINE
Christopher Ferris950a58e2014-04-04 14:38:18 -0700124char* stpcpy(char* __restrict dest, const char* __restrict src) {
125 return __builtin___stpcpy_chk(dest, src, __bos(dest));
126}
127
128__BIONIC_FORTIFY_INLINE
Nick Kralevich16d1af12013-06-17 14:49:19 -0700129char* strcpy(char* __restrict dest, const char* __restrict src) {
Nick Kralevich9020fd52013-04-30 11:31:35 -0700130 return __builtin___strcpy_chk(dest, src, __bos(dest));
Nick Kralevich0a230152012-06-04 15:20:25 -0700131}
132
Christopher Ferris950a58e2014-04-04 14:38:18 -0700133__errordecl(__stpncpy_error, "stpncpy: prevented write past end of buffer");
134extern char* __stpncpy_chk2(char* __restrict, const char* __restrict, size_t, size_t, size_t);
135
136__BIONIC_FORTIFY_INLINE
137char* stpncpy(char* __restrict dest, const char* __restrict src, size_t n) {
138 size_t bos_dest = __bos(dest);
139 size_t bos_src = __bos(src);
140 if (__builtin_constant_p(n) && (n > bos_dest)) {
141 __stpncpy_error();
142 }
143
144 if (bos_src == __BIONIC_FORTIFY_UNKNOWN_SIZE) {
145 return __builtin___stpncpy_chk(dest, src, n, bos_dest);
146 }
147
148 if (__builtin_constant_p(n) && (n <= bos_src)) {
149 return __builtin___stpncpy_chk(dest, src, n, bos_dest);
150 }
151
152 size_t slen = __builtin_strlen(src);
153 if (__builtin_constant_p(slen)) {
154 return __builtin___stpncpy_chk(dest, src, n, bos_dest);
155 }
156
157 return __stpncpy_chk2(dest, src, n, bos_dest, bos_src);
158}
159
160__errordecl(__strncpy_error, "strncpy: prevented write past end of buffer");
Nick Kralevich93501d32013-08-28 10:47:43 -0700161extern char* __strncpy_chk2(char* __restrict, const char* __restrict, size_t, size_t, size_t);
Nick Kralevich1aae9bd2013-04-29 14:07:06 -0700162
Nick Kralevich0a230152012-06-04 15:20:25 -0700163__BIONIC_FORTIFY_INLINE
Nick Kralevich16d1af12013-06-17 14:49:19 -0700164char* strncpy(char* __restrict dest, const char* __restrict src, size_t n) {
Nick Kralevich93501d32013-08-28 10:47:43 -0700165 size_t bos_dest = __bos(dest);
166 size_t bos_src = __bos(src);
167 if (__builtin_constant_p(n) && (n > bos_dest)) {
Nick Kralevich1aae9bd2013-04-29 14:07:06 -0700168 __strncpy_error();
169 }
Nick Kralevich93501d32013-08-28 10:47:43 -0700170
171 if (bos_src == __BIONIC_FORTIFY_UNKNOWN_SIZE) {
172 return __builtin___strncpy_chk(dest, src, n, bos_dest);
173 }
174
Nick Kralevichd13c2b12013-09-27 13:21:24 -0700175 if (__builtin_constant_p(n) && (n <= bos_src)) {
176 return __builtin___strncpy_chk(dest, src, n, bos_dest);
177 }
178
Nick Kralevich93501d32013-08-28 10:47:43 -0700179 size_t slen = __builtin_strlen(src);
180 if (__builtin_constant_p(slen)) {
181 return __builtin___strncpy_chk(dest, src, n, bos_dest);
182 }
183
184 return __strncpy_chk2(dest, src, n, bos_dest, bos_src);
Nick Kralevich0a230152012-06-04 15:20:25 -0700185}
186
187__BIONIC_FORTIFY_INLINE
Nick Kralevich16d1af12013-06-17 14:49:19 -0700188char* strcat(char* __restrict dest, const char* __restrict src) {
Nick Kralevich9020fd52013-04-30 11:31:35 -0700189 return __builtin___strcat_chk(dest, src, __bos(dest));
Nick Kralevich0a230152012-06-04 15:20:25 -0700190}
191
192__BIONIC_FORTIFY_INLINE
Nick Kralevich1c462b72013-05-07 10:00:21 -0700193char *strncat(char* __restrict dest, const char* __restrict src, size_t n) {
Nick Kralevich9020fd52013-04-30 11:31:35 -0700194 return __builtin___strncat_chk(dest, src, n, __bos(dest));
Nick Kralevich0a230152012-06-04 15:20:25 -0700195}
196
Nick Kralevich71a18dd2012-06-07 14:01:26 -0700197__BIONIC_FORTIFY_INLINE
Nick Kralevich16d1af12013-06-17 14:49:19 -0700198void* memset(void *s, int c, size_t n) {
Nick Kralevichbd8e6742013-08-28 13:22:52 -0700199 return __builtin___memset_chk(s, c, n, __bos0(s));
Nick Kralevich71a18dd2012-06-07 14:01:26 -0700200}
201
Nick Kralevich1c462b72013-05-07 10:00:21 -0700202extern size_t __strlcpy_real(char* __restrict, const char* __restrict, size_t)
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700203 __asm__(__USER_LABEL_PREFIX__ "strlcpy");
Christopher Ferris950a58e2014-04-04 14:38:18 -0700204__errordecl(__strlcpy_error, "strlcpy: prevented write past end of buffer");
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700205extern size_t __strlcpy_chk(char *, const char *, size_t, size_t);
206
207__BIONIC_FORTIFY_INLINE
Nick Kralevich1c462b72013-05-07 10:00:21 -0700208size_t strlcpy(char* __restrict dest, const char* __restrict src, size_t size) {
Nick Kralevich9020fd52013-04-30 11:31:35 -0700209 size_t bos = __bos(dest);
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700210
Nick Kralevich8bafa742013-06-20 12:17:44 -0700211#if !defined(__clang__)
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700212 // Compiler doesn't know destination size. Don't call __strlcpy_chk
Nick Kralevich9b6cc222012-07-13 14:46:36 -0700213 if (bos == __BIONIC_FORTIFY_UNKNOWN_SIZE) {
Nick Kralevichcb228fb2012-06-26 16:05:19 -0700214 return __strlcpy_real(dest, src, size);
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700215 }
216
217 // Compiler can prove, at compile time, that the passed in size
218 // is always <= the actual object size. Don't call __strlcpy_chk
219 if (__builtin_constant_p(size) && (size <= bos)) {
Nick Kralevichcb228fb2012-06-26 16:05:19 -0700220 return __strlcpy_real(dest, src, size);
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700221 }
222
223 // Compiler can prove, at compile time, that the passed in size
224 // is always > the actual object size. Force a compiler error.
225 if (__builtin_constant_p(size) && (size > bos)) {
226 __strlcpy_error();
227 }
Nick Kralevich8bafa742013-06-20 12:17:44 -0700228#endif /* !defined(__clang__) */
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700229
230 return __strlcpy_chk(dest, src, size, bos);
231}
232
Nick Kralevich1c462b72013-05-07 10:00:21 -0700233extern size_t __strlcat_real(char* __restrict, const char* __restrict, size_t)
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700234 __asm__(__USER_LABEL_PREFIX__ "strlcat");
Christopher Ferris950a58e2014-04-04 14:38:18 -0700235__errordecl(__strlcat_error, "strlcat: prevented write past end of buffer");
Nick Kralevich1c462b72013-05-07 10:00:21 -0700236extern size_t __strlcat_chk(char* __restrict, const char* __restrict, size_t, size_t);
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700237
238
239__BIONIC_FORTIFY_INLINE
Nick Kralevich1c462b72013-05-07 10:00:21 -0700240size_t strlcat(char* __restrict dest, const char* __restrict src, size_t size) {
Nick Kralevich9020fd52013-04-30 11:31:35 -0700241 size_t bos = __bos(dest);
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700242
Nick Kralevicha6cde392013-06-29 08:15:25 -0700243#if !defined(__clang__)
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700244 // Compiler doesn't know destination size. Don't call __strlcat_chk
Nick Kralevich9b6cc222012-07-13 14:46:36 -0700245 if (bos == __BIONIC_FORTIFY_UNKNOWN_SIZE) {
Nick Kralevichcb228fb2012-06-26 16:05:19 -0700246 return __strlcat_real(dest, src, size);
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700247 }
248
249 // Compiler can prove, at compile time, that the passed in size
250 // is always <= the actual object size. Don't call __strlcat_chk
251 if (__builtin_constant_p(size) && (size <= bos)) {
Nick Kralevichcb228fb2012-06-26 16:05:19 -0700252 return __strlcat_real(dest, src, size);
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700253 }
254
255 // Compiler can prove, at compile time, that the passed in size
256 // is always > the actual object size. Force a compiler error.
257 if (__builtin_constant_p(size) && (size > bos)) {
258 __strlcat_error();
259 }
Nick Kralevicha6cde392013-06-29 08:15:25 -0700260#endif /* !defined(__clang__) */
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700261
262 return __strlcat_chk(dest, src, size, bos);
263}
264
Nick Kralevich260bf8c2012-07-13 11:27:06 -0700265__BIONIC_FORTIFY_INLINE
266size_t strlen(const char *s) {
Nick Kralevich9020fd52013-04-30 11:31:35 -0700267 size_t bos = __bos(s);
Nick Kralevich9b6cc222012-07-13 14:46:36 -0700268
Nick Kralevich16d1af12013-06-17 14:49:19 -0700269#if !defined(__clang__)
Nick Kralevich9b6cc222012-07-13 14:46:36 -0700270 // Compiler doesn't know destination size. Don't call __strlen_chk
271 if (bos == __BIONIC_FORTIFY_UNKNOWN_SIZE) {
Nick Kralevicha44e9af2013-01-17 15:41:33 -0800272 return __builtin_strlen(s);
273 }
274
275 size_t slen = __builtin_strlen(s);
276 if (__builtin_constant_p(slen)) {
277 return slen;
Nick Kralevich260bf8c2012-07-13 11:27:06 -0700278 }
Nick Kralevich16d1af12013-06-17 14:49:19 -0700279#endif /* !defined(__clang__) */
Nick Kralevich9b6cc222012-07-13 14:46:36 -0700280
Nick Kralevich260bf8c2012-07-13 11:27:06 -0700281 return __strlen_chk(s, bos);
282}
Nick Kralevich8df49ad2012-06-13 16:57:27 -0700283
Nick Kralevich049e5832012-11-30 15:15:58 -0800284__BIONIC_FORTIFY_INLINE
285char* strchr(const char *s, int c) {
Nick Kralevich9020fd52013-04-30 11:31:35 -0700286 size_t bos = __bos(s);
Nick Kralevich049e5832012-11-30 15:15:58 -0800287
Nick Kralevich16d1af12013-06-17 14:49:19 -0700288#if !defined(__clang__)
Nick Kralevich049e5832012-11-30 15:15:58 -0800289 // Compiler doesn't know destination size. Don't call __strchr_chk
290 if (bos == __BIONIC_FORTIFY_UNKNOWN_SIZE) {
Nick Kralevicha44e9af2013-01-17 15:41:33 -0800291 return __builtin_strchr(s, c);
292 }
293
294 size_t slen = __builtin_strlen(s);
295 if (__builtin_constant_p(slen) && (slen < bos)) {
296 return __builtin_strchr(s, c);
Nick Kralevich049e5832012-11-30 15:15:58 -0800297 }
Nick Kralevich16d1af12013-06-17 14:49:19 -0700298#endif /* !defined(__clang__) */
Nick Kralevich049e5832012-11-30 15:15:58 -0800299
300 return __strchr_chk(s, c, bos);
301}
302
Nick Kralevich9a4d3052012-12-03 10:36:13 -0800303__BIONIC_FORTIFY_INLINE
304char* strrchr(const char *s, int c) {
Nick Kralevich3b2e6bc2013-04-30 14:19:23 -0700305 size_t bos = __bos(s);
Nick Kralevich9a4d3052012-12-03 10:36:13 -0800306
Nick Kralevich16d1af12013-06-17 14:49:19 -0700307#if !defined(__clang__)
Nick Kralevich9a4d3052012-12-03 10:36:13 -0800308 // Compiler doesn't know destination size. Don't call __strrchr_chk
309 if (bos == __BIONIC_FORTIFY_UNKNOWN_SIZE) {
Nick Kralevicha44e9af2013-01-17 15:41:33 -0800310 return __builtin_strrchr(s, c);
311 }
312
313 size_t slen = __builtin_strlen(s);
314 if (__builtin_constant_p(slen) && (slen < bos)) {
315 return __builtin_strrchr(s, c);
Nick Kralevich9a4d3052012-12-03 10:36:13 -0800316 }
Nick Kralevich16d1af12013-06-17 14:49:19 -0700317#endif /* !defined(__clang__) */
Nick Kralevich9a4d3052012-12-03 10:36:13 -0800318
319 return __strrchr_chk(s, c, bos);
320}
321
Nick Kralevich049e5832012-11-30 15:15:58 -0800322
Elliott Hughes890c8ed2013-03-22 10:58:55 -0700323#endif /* defined(__BIONIC_FORTIFY) */
Nick Kralevich0a230152012-06-04 15:20:25 -0700324
The Android Open Source Project1dc9e472009-03-03 19:28:35 -0800325__END_DECLS
326
327#endif /* _STRING_H_ */