1 /*
2 * Copyright (C) 2017 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 _UNISTD_H_
29 #error "Never include this file directly; instead, include <unistd.h>"
30 #endif
31
32 char* __getcwd_chk(char*, size_t, size_t) __INTRODUCED_IN(24);
33
34 ssize_t __pread_chk(int, void*, size_t, off_t, size_t) __INTRODUCED_IN(23);
35 ssize_t __pread_real(int, void*, size_t, off_t) __RENAME(pread);
36
37 ssize_t __pread64_chk(int, void*, size_t, off64_t, size_t) __INTRODUCED_IN(23);
38 ssize_t __pread64_real(int, void*, size_t, off64_t) __RENAME(pread64);
39
40 ssize_t __pwrite_chk(int, const void*, size_t, off_t, size_t) __INTRODUCED_IN(24);
41 ssize_t __pwrite_real(int, const void*, size_t, off_t) __RENAME(pwrite);
42
43 ssize_t __pwrite64_chk(int, const void*, size_t, off64_t, size_t) __INTRODUCED_IN(24);
44 ssize_t __pwrite64_real(int, const void*, size_t, off64_t) __RENAME(pwrite64);
45
46 ssize_t __read_chk(int, void*, size_t, size_t) __INTRODUCED_IN(21);
47 ssize_t __write_chk(int, const void*, size_t, size_t) __INTRODUCED_IN(24);
48 ssize_t __readlink_chk(const char*, char*, size_t, size_t) __INTRODUCED_IN(23);
49 ssize_t __readlinkat_chk(int dirfd, const char*, char*, size_t, size_t) __INTRODUCED_IN(23);
50
51 #if defined(__BIONIC_FORTIFY)
52
53 #if defined(__USE_FILE_OFFSET64)
54 #define __PREAD_PREFIX(x) __pread64_ ## x
55 #define __PWRITE_PREFIX(x) __pwrite64_ ## x
56 #else
57 #define __PREAD_PREFIX(x) __pread_ ## x
58 #define __PWRITE_PREFIX(x) __pwrite_ ## x
59 #endif
60
61 #define __error_if_overflows_ssizet(what, fn) \
62 __clang_error_if((what) > SSIZE_MAX, "in call to '" #fn "', '" #what "' must be <= SSIZE_MAX")
63
64 #define __error_if_overflows_objectsize(what, objsize, fn) \
65 __clang_error_if(__bos_unevaluated_lt((objsize), (what)), \
66 "in call to '" #fn "', '" #what "' bytes overflows the given object")
67
68 #define __bos_trivially_ge_no_overflow(bos_val, index) \
69 ((__bos_dynamic_check_impl_and((bos_val), >=, (index), (bos_val) <= SSIZE_MAX) && \
70 __builtin_constant_p(index) && (index) <= SSIZE_MAX))
71
72 __BIONIC_FORTIFY_INLINE
getcwd(char * const __pass_object_size buf,size_t size)73 char* getcwd(char* const __pass_object_size buf, size_t size)
74 __overloadable
75 __error_if_overflows_objectsize(size, __bos(buf), getcwd) {
76 #if __ANDROID_API__ >= 24 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
77 size_t bos = __bos(buf);
78
79 if (!__bos_trivially_ge(bos, size)) {
80 return __getcwd_chk(buf, size, bos);
81 }
82 #endif
83 return __call_bypassing_fortify(getcwd)(buf, size);
84 }
85
86 #if !defined(__USE_FILE_OFFSET64)
87 __BIONIC_FORTIFY_INLINE
pread(int fd,void * const __pass_object_size0 buf,size_t count,off_t offset)88 ssize_t pread(int fd, void* const __pass_object_size0 buf, size_t count, off_t offset)
89 __overloadable
90 __error_if_overflows_ssizet(count, pread)
91 __error_if_overflows_objectsize(count, __bos0(buf), pread) {
92 #if __ANDROID_API__ >= 23 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
93 size_t bos = __bos0(buf);
94
95 if (!__bos_trivially_ge_no_overflow(bos, count)) {
96 return __PREAD_PREFIX(chk)(fd, buf, count, offset, bos);
97 }
98 #endif
99 return __PREAD_PREFIX(real)(fd, buf, count, offset);
100 }
101 #endif /* !defined(__USE_FILE_OFFSET64) */
102
103 __BIONIC_FORTIFY_INLINE
pread64(int fd,void * const __pass_object_size0 buf,size_t count,off64_t offset)104 ssize_t pread64(int fd, void* const __pass_object_size0 buf, size_t count, off64_t offset)
105 __overloadable
106 __error_if_overflows_ssizet(count, pread64)
107 __error_if_overflows_objectsize(count, __bos0(buf), pread64) {
108 #if __ANDROID_API__ >= 23 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
109 size_t bos = __bos0(buf);
110
111 if (!__bos_trivially_ge_no_overflow(bos, count)) {
112 return __pread64_chk(fd, buf, count, offset, bos);
113 }
114 #endif
115 return __pread64_real(fd, buf, count, offset);
116 }
117
118 #if !defined(__USE_FILE_OFFSET64)
119 __BIONIC_FORTIFY_INLINE
pwrite(int fd,const void * const __pass_object_size0 buf,size_t count,off_t offset)120 ssize_t pwrite(int fd, const void* const __pass_object_size0 buf, size_t count, off_t offset)
121 __overloadable
122 __error_if_overflows_ssizet(count, pwrite)
123 __error_if_overflows_objectsize(count, __bos0(buf), pwrite) {
124 #if __ANDROID_API__ >= 24 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
125 size_t bos = __bos0(buf);
126
127 if (!__bos_trivially_ge_no_overflow(bos, count)) {
128 return __PWRITE_PREFIX(chk)(fd, buf, count, offset, bos);
129 }
130 #endif
131 return __PWRITE_PREFIX(real)(fd, buf, count, offset);
132 }
133 #endif /* !defined(__USE_FILE_OFFSET64) */
134
135 __BIONIC_FORTIFY_INLINE
pwrite64(int fd,const void * const __pass_object_size0 buf,size_t count,off64_t offset)136 ssize_t pwrite64(int fd, const void* const __pass_object_size0 buf, size_t count, off64_t offset)
137 __overloadable
138 __error_if_overflows_ssizet(count, pwrite64)
139 __error_if_overflows_objectsize(count, __bos0(buf), pwrite64) {
140 #if __ANDROID_API__ >= 24 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
141 size_t bos = __bos0(buf);
142
143 if (!__bos_trivially_ge_no_overflow(bos, count)) {
144 return __pwrite64_chk(fd, buf, count, offset, bos);
145 }
146 #endif
147 return __pwrite64_real(fd, buf, count, offset);
148 }
149
150 __BIONIC_FORTIFY_INLINE
read(int fd,void * const __pass_object_size0 buf,size_t count)151 ssize_t read(int fd, void* const __pass_object_size0 buf, size_t count)
152 __overloadable
153 __error_if_overflows_ssizet(count, read)
154 __error_if_overflows_objectsize(count, __bos0(buf), read) {
155 #if __ANDROID_API__ >= 21 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
156 size_t bos = __bos0(buf);
157
158 if (!__bos_trivially_ge_no_overflow(bos, count)) {
159 return __read_chk(fd, buf, count, bos);
160 }
161 #endif
162 return __call_bypassing_fortify(read)(fd, buf, count);
163 }
164
165 __BIONIC_FORTIFY_INLINE
write(int fd,const void * const __pass_object_size0 buf,size_t count)166 ssize_t write(int fd, const void* const __pass_object_size0 buf, size_t count)
167 __overloadable
168 __error_if_overflows_ssizet(count, write)
169 __error_if_overflows_objectsize(count, __bos0(buf), write) {
170 #if __ANDROID_API__ >= 24 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
171 size_t bos = __bos0(buf);
172
173 if (!__bos_trivially_ge_no_overflow(bos, count)) {
174 return __write_chk(fd, buf, count, bos);
175 }
176 #endif
177 return __call_bypassing_fortify(write)(fd, buf, count);
178 }
179
180 __BIONIC_FORTIFY_INLINE
readlink(const char * path,char * const __pass_object_size buf,size_t size)181 ssize_t readlink(const char* path, char* const __pass_object_size buf, size_t size)
182 __overloadable
183 __error_if_overflows_ssizet(size, readlink)
184 __error_if_overflows_objectsize(size, __bos(buf), readlink) {
185 #if __ANDROID_API__ >= 23 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
186 size_t bos = __bos(buf);
187
188 if (!__bos_trivially_ge_no_overflow(bos, size)) {
189 return __readlink_chk(path, buf, size, bos);
190 }
191 #endif
192 return __call_bypassing_fortify(readlink)(path, buf, size);
193 }
194
195 #if __ANDROID_API__ >= 21
196 __BIONIC_FORTIFY_INLINE
readlinkat(int dirfd,const char * path,char * const __pass_object_size buf,size_t size)197 ssize_t readlinkat(int dirfd, const char* path, char* const __pass_object_size buf, size_t size)
198 __overloadable
199 __error_if_overflows_ssizet(size, readlinkat)
200 __error_if_overflows_objectsize(size, __bos(buf), readlinkat) {
201 #if __ANDROID_API__ >= 23 && __BIONIC_FORTIFY_RUNTIME_CHECKS_ENABLED
202 size_t bos = __bos(buf);
203
204 if (!__bos_trivially_ge_no_overflow(bos, size)) {
205 return __readlinkat_chk(dirfd, path, buf, size, bos);
206 }
207 #endif
208 return __call_bypassing_fortify(readlinkat)(dirfd, path, buf, size);
209 }
210 #endif /* __ANDROID_API__ >= 21 */
211
212 #undef __bos_trivially_ge_no_overflow
213 #undef __enable_if_no_overflow_ssizet
214 #undef __error_if_overflows_objectsize
215 #undef __error_if_overflows_ssizet
216 #undef __PREAD_PREFIX
217 #undef __PWRITE_PREFIX
218 #endif /* defined(__BIONIC_FORTIFY) */
219