xref: /freebsd/contrib/llvm-project/compiler-rt/lib/dfsan/dfsan_custom.cpp (revision 278d6950943a9fec2bddb037b547c04a847c54ba)
1 //===-- dfsan_custom.cpp --------------------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 //
9 // This file is a part of DataFlowSanitizer.
10 //
11 // This file defines the custom functions listed in done_abilist.txt.
12 //===----------------------------------------------------------------------===//
13 
14 #include <arpa/inet.h>
15 #include <assert.h>
16 #include <ctype.h>
17 #include <dlfcn.h>
18 #include <link.h>
19 #include <poll.h>
20 #include <pthread.h>
21 #include <pwd.h>
22 #include <sched.h>
23 #include <signal.h>
24 #include <stdarg.h>
25 #include <stdint.h>
26 #include <stdio.h>
27 #include <stdlib.h>
28 #include <string.h>
29 #include <sys/epoll.h>
30 #include <sys/resource.h>
31 #include <sys/select.h>
32 #include <sys/socket.h>
33 #include <sys/stat.h>
34 #include <sys/time.h>
35 #include <sys/types.h>
36 #include <time.h>
37 #include <unistd.h>
38 
39 #include "dfsan/dfsan.h"
40 #include "dfsan/dfsan_chained_origin_depot.h"
41 #include "dfsan/dfsan_flags.h"
42 #include "dfsan/dfsan_thread.h"
43 #include "sanitizer_common/sanitizer_common.h"
44 #include "sanitizer_common/sanitizer_internal_defs.h"
45 #include "sanitizer_common/sanitizer_linux.h"
46 #include "sanitizer_common/sanitizer_stackdepot.h"
47 
48 using namespace __dfsan;
49 
50 #define CALL_WEAK_INTERCEPTOR_HOOK(f, ...)                                     \
51   do {                                                                         \
52     if (f)                                                                     \
53       f(__VA_ARGS__);                                                          \
54   } while (false)
55 #define DECLARE_WEAK_INTERCEPTOR_HOOK(f, ...) \
56 SANITIZER_INTERFACE_ATTRIBUTE SANITIZER_WEAK_ATTRIBUTE void f(__VA_ARGS__);
57 
58 #define WRAPPER_ALIAS(fun, real)                                          \
59   SANITIZER_INTERFACE_ATTRIBUTE void __dfsw_##fun() ALIAS(__dfsw_##real); \
60   SANITIZER_INTERFACE_ATTRIBUTE void __dfso_##fun() ALIAS(__dfso_##real);
61 
62 // Async-safe, non-reentrant spin lock.
63 class SignalSpinLocker {
64  public:
65   SignalSpinLocker() {
66     sigset_t all_set;
67     sigfillset(&all_set);
68     pthread_sigmask(SIG_SETMASK, &all_set, &saved_thread_mask_);
69     sigactions_mu.Lock();
70   }
71   ~SignalSpinLocker() {
72     sigactions_mu.Unlock();
73     pthread_sigmask(SIG_SETMASK, &saved_thread_mask_, nullptr);
74   }
75 
76  private:
77   static StaticSpinMutex sigactions_mu;
78   sigset_t saved_thread_mask_;
79 
80   SignalSpinLocker(const SignalSpinLocker &) = delete;
81   SignalSpinLocker &operator=(const SignalSpinLocker &) = delete;
82 };
83 
84 StaticSpinMutex SignalSpinLocker::sigactions_mu;
85 
86 extern "C" {
87 SANITIZER_INTERFACE_ATTRIBUTE int
88 __dfsw_stat(const char *path, struct stat *buf, dfsan_label path_label,
89             dfsan_label buf_label, dfsan_label *ret_label) {
90   int ret = stat(path, buf);
91   if (ret == 0)
92     dfsan_set_label(0, buf, sizeof(struct stat));
93   *ret_label = 0;
94   return ret;
95 }
96 
97 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_stat(
98     const char *path, struct stat *buf, dfsan_label path_label,
99     dfsan_label buf_label, dfsan_label *ret_label, dfsan_origin path_origin,
100     dfsan_origin buf_origin, dfsan_origin *ret_origin) {
101   int ret = __dfsw_stat(path, buf, path_label, buf_label, ret_label);
102   return ret;
103 }
104 
105 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_fstat(int fd, struct stat *buf,
106                                                dfsan_label fd_label,
107                                                dfsan_label buf_label,
108                                                dfsan_label *ret_label) {
109   int ret = fstat(fd, buf);
110   if (ret == 0)
111     dfsan_set_label(0, buf, sizeof(struct stat));
112   *ret_label = 0;
113   return ret;
114 }
115 
116 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_fstat(
117     int fd, struct stat *buf, dfsan_label fd_label, dfsan_label buf_label,
118     dfsan_label *ret_label, dfsan_origin fd_origin, dfsan_origin buf_origin,
119     dfsan_origin *ret_origin) {
120   int ret = __dfsw_fstat(fd, buf, fd_label, buf_label, ret_label);
121   return ret;
122 }
123 
124 static char *dfsan_strchr_with_label(const char *s, int c, size_t *bytes_read,
125                                      dfsan_label s_label, dfsan_label c_label,
126                                      dfsan_label *ret_label) {
127   char *match_pos = nullptr;
128   for (size_t i = 0;; ++i) {
129     if (s[i] == c || s[i] == 0) {
130       // If s[i] is the \0 at the end of the string, and \0 is not the
131       // character we are searching for, then return null.
132       *bytes_read = i + 1;
133       match_pos = s[i] == 0 && c != 0 ? nullptr : const_cast<char *>(s + i);
134       break;
135     }
136   }
137   if (flags().strict_data_dependencies)
138     *ret_label = s_label;
139   else
140     *ret_label = dfsan_union(dfsan_read_label(s, *bytes_read),
141                              dfsan_union(s_label, c_label));
142   return match_pos;
143 }
144 
145 SANITIZER_INTERFACE_ATTRIBUTE char *__dfsw_strchr(const char *s, int c,
146                                                   dfsan_label s_label,
147                                                   dfsan_label c_label,
148                                                   dfsan_label *ret_label) {
149   size_t bytes_read;
150   return dfsan_strchr_with_label(s, c, &bytes_read, s_label, c_label,
151                                  ret_label);
152 }
153 
154 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strchr(
155     const char *s, int c, dfsan_label s_label, dfsan_label c_label,
156     dfsan_label *ret_label, dfsan_origin s_origin, dfsan_origin c_origin,
157     dfsan_origin *ret_origin) {
158   size_t bytes_read;
159   char *r =
160       dfsan_strchr_with_label(s, c, &bytes_read, s_label, c_label, ret_label);
161   if (flags().strict_data_dependencies) {
162     *ret_origin = s_origin;
163   } else if (*ret_label) {
164     dfsan_origin o = dfsan_read_origin_of_first_taint(s, bytes_read);
165     *ret_origin = o ? o : (s_label ? s_origin : c_origin);
166   }
167   return r;
168 }
169 
170 SANITIZER_INTERFACE_ATTRIBUTE char *__dfsw_strpbrk(const char *s,
171                                                    const char *accept,
172                                                    dfsan_label s_label,
173                                                    dfsan_label accept_label,
174                                                    dfsan_label *ret_label) {
175   const char *ret = strpbrk(s, accept);
176   if (flags().strict_data_dependencies) {
177     *ret_label = ret ? s_label : 0;
178   } else {
179     size_t s_bytes_read = (ret ? ret - s : strlen(s)) + 1;
180     *ret_label =
181         dfsan_union(dfsan_read_label(s, s_bytes_read),
182                     dfsan_union(dfsan_read_label(accept, strlen(accept) + 1),
183                                 dfsan_union(s_label, accept_label)));
184   }
185   return const_cast<char *>(ret);
186 }
187 
188 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strpbrk(
189     const char *s, const char *accept, dfsan_label s_label,
190     dfsan_label accept_label, dfsan_label *ret_label, dfsan_origin s_origin,
191     dfsan_origin accept_origin, dfsan_origin *ret_origin) {
192   const char *ret = __dfsw_strpbrk(s, accept, s_label, accept_label, ret_label);
193   if (flags().strict_data_dependencies) {
194     if (ret)
195       *ret_origin = s_origin;
196   } else {
197     if (*ret_label) {
198       size_t s_bytes_read = (ret ? ret - s : strlen(s)) + 1;
199       dfsan_origin o = dfsan_read_origin_of_first_taint(s, s_bytes_read);
200       if (o) {
201         *ret_origin = o;
202       } else {
203         o = dfsan_read_origin_of_first_taint(accept, strlen(accept) + 1);
204         *ret_origin = o ? o : (s_label ? s_origin : accept_origin);
205       }
206     }
207   }
208   return const_cast<char *>(ret);
209 }
210 
211 SANITIZER_INTERFACE_ATTRIBUTE char *__dfsw_strsep(char **s, const char *delim,
212                                                   dfsan_label s_label,
213                                                   dfsan_label delim_label,
214                                                   dfsan_label *ret_label) {
215   dfsan_label base_label = dfsan_read_label(s, sizeof(*s));
216   char *base = *s;
217   char *res = strsep(s, delim);
218   if (res != *s) {
219     char *token_start = res;
220     int token_length = strlen(res);
221     // the delimiter byte has been set to NULL
222     dfsan_set_label(0, token_start + token_length, 1);
223   }
224 
225   if (flags().strict_data_dependencies) {
226     *ret_label = res ? base_label : 0;
227   } else {
228     size_t s_bytes_read = (res ? strlen(res) : strlen(base)) + 1;
229     *ret_label = dfsan_union(
230         dfsan_union(base_label, dfsan_read_label(base, sizeof(s_bytes_read))),
231         dfsan_union(dfsan_read_label(delim, strlen(delim) + 1),
232                     dfsan_union(s_label, delim_label)));
233   }
234 
235   return res;
236 }
237 
238 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strsep(
239     char **s, const char *delim, dfsan_label s_label, dfsan_label delim_label,
240     dfsan_label *ret_label, dfsan_origin s_origin, dfsan_origin delim_origin,
241     dfsan_origin *ret_origin) {
242   dfsan_origin base_origin = dfsan_read_origin_of_first_taint(s, sizeof(*s));
243   char *res = __dfsw_strsep(s, delim, s_label, delim_label, ret_label);
244   if (flags().strict_data_dependencies) {
245     if (res)
246       *ret_origin = base_origin;
247   } else {
248     if (*ret_label) {
249       if (base_origin) {
250         *ret_origin = base_origin;
251       } else {
252         dfsan_origin o =
253             dfsan_read_origin_of_first_taint(delim, strlen(delim) + 1);
254         *ret_origin = o ? o : (s_label ? s_origin : delim_origin);
255       }
256     }
257   }
258 
259   return res;
260 }
261 
262 static int dfsan_memcmp_bcmp(const void *s1, const void *s2, size_t n,
263                              size_t *bytes_read) {
264   const char *cs1 = (const char *) s1, *cs2 = (const char *) s2;
265   for (size_t i = 0; i != n; ++i) {
266     if (cs1[i] != cs2[i]) {
267       *bytes_read = i + 1;
268       return cs1[i] - cs2[i];
269     }
270   }
271   *bytes_read = n;
272   return 0;
273 }
274 
275 static dfsan_label dfsan_get_memcmp_label(const void *s1, const void *s2,
276                                           size_t pos) {
277   if (flags().strict_data_dependencies)
278     return 0;
279   return dfsan_union(dfsan_read_label(s1, pos), dfsan_read_label(s2, pos));
280 }
281 
282 static void dfsan_get_memcmp_origin(const void *s1, const void *s2, size_t pos,
283                                     dfsan_label *ret_label,
284                                     dfsan_origin *ret_origin) {
285   *ret_label = dfsan_get_memcmp_label(s1, s2, pos);
286   if (*ret_label == 0)
287     return;
288   dfsan_origin o = dfsan_read_origin_of_first_taint(s1, pos);
289   *ret_origin = o ? o : dfsan_read_origin_of_first_taint(s2, pos);
290 }
291 
292 static int dfsan_memcmp_bcmp_label(const void *s1, const void *s2, size_t n,
293                                    dfsan_label *ret_label) {
294   size_t bytes_read;
295   int r = dfsan_memcmp_bcmp(s1, s2, n, &bytes_read);
296   *ret_label = dfsan_get_memcmp_label(s1, s2, bytes_read);
297   return r;
298 }
299 
300 static int dfsan_memcmp_bcmp_origin(const void *s1, const void *s2, size_t n,
301                                     dfsan_label *ret_label,
302                                     dfsan_origin *ret_origin) {
303   size_t bytes_read;
304   int r = dfsan_memcmp_bcmp(s1, s2, n, &bytes_read);
305   dfsan_get_memcmp_origin(s1, s2, bytes_read, ret_label, ret_origin);
306   return r;
307 }
308 
309 DECLARE_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_memcmp, uptr caller_pc,
310                               const void *s1, const void *s2, size_t n,
311                               dfsan_label s1_label, dfsan_label s2_label,
312                               dfsan_label n_label)
313 
314 DECLARE_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_origin_memcmp, uptr caller_pc,
315                               const void *s1, const void *s2, size_t n,
316                               dfsan_label s1_label, dfsan_label s2_label,
317                               dfsan_label n_label, dfsan_origin s1_origin,
318                               dfsan_origin s2_origin, dfsan_origin n_origin)
319 
320 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_memcmp(const void *s1, const void *s2,
321                                                 size_t n, dfsan_label s1_label,
322                                                 dfsan_label s2_label,
323                                                 dfsan_label n_label,
324                                                 dfsan_label *ret_label) {
325   CALL_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_memcmp, GET_CALLER_PC(), s1, s2, n,
326                              s1_label, s2_label, n_label);
327   return dfsan_memcmp_bcmp_label(s1, s2, n, ret_label);
328 }
329 
330 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_memcmp(
331     const void *s1, const void *s2, size_t n, dfsan_label s1_label,
332     dfsan_label s2_label, dfsan_label n_label, dfsan_label *ret_label,
333     dfsan_origin s1_origin, dfsan_origin s2_origin, dfsan_origin n_origin,
334     dfsan_origin *ret_origin) {
335   CALL_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_origin_memcmp, GET_CALLER_PC(), s1,
336                              s2, n, s1_label, s2_label, n_label, s1_origin,
337                              s2_origin, n_origin);
338   return dfsan_memcmp_bcmp_origin(s1, s2, n, ret_label, ret_origin);
339 }
340 
341 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_bcmp(const void *s1, const void *s2,
342                                               size_t n, dfsan_label s1_label,
343                                               dfsan_label s2_label,
344                                               dfsan_label n_label,
345                                               dfsan_label *ret_label) {
346   return dfsan_memcmp_bcmp_label(s1, s2, n, ret_label);
347 }
348 
349 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_bcmp(
350     const void *s1, const void *s2, size_t n, dfsan_label s1_label,
351     dfsan_label s2_label, dfsan_label n_label, dfsan_label *ret_label,
352     dfsan_origin s1_origin, dfsan_origin s2_origin, dfsan_origin n_origin,
353     dfsan_origin *ret_origin) {
354   return dfsan_memcmp_bcmp_origin(s1, s2, n, ret_label, ret_origin);
355 }
356 
357 // When n == 0, compare strings without byte limit.
358 // When n > 0, compare the first (at most) n bytes of s1 and s2.
359 static int dfsan_strncmp(const char *s1, const char *s2, size_t n,
360                          size_t *bytes_read) {
361   for (size_t i = 0;; ++i) {
362     if (s1[i] != s2[i] || s1[i] == 0 || s2[i] == 0 || (n > 0 && i == n - 1)) {
363       *bytes_read = i + 1;
364       return s1[i] - s2[i];
365     }
366   }
367 }
368 
369 DECLARE_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_strcmp, uptr caller_pc,
370                               const char *s1, const char *s2,
371                               dfsan_label s1_label, dfsan_label s2_label)
372 
373 DECLARE_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_origin_strcmp, uptr caller_pc,
374                               const char *s1, const char *s2,
375                               dfsan_label s1_label, dfsan_label s2_label,
376                               dfsan_origin s1_origin, dfsan_origin s2_origin)
377 
378 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_strcmp(const char *s1, const char *s2,
379                                                 dfsan_label s1_label,
380                                                 dfsan_label s2_label,
381                                                 dfsan_label *ret_label) {
382   CALL_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_strcmp, GET_CALLER_PC(), s1, s2,
383                              s1_label, s2_label);
384   size_t bytes_read;
385   int r = dfsan_strncmp(s1, s2, 0, &bytes_read);
386   *ret_label = dfsan_get_memcmp_label(s1, s2, bytes_read);
387   return r;
388 }
389 
390 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_strcmp(
391     const char *s1, const char *s2, dfsan_label s1_label, dfsan_label s2_label,
392     dfsan_label *ret_label, dfsan_origin s1_origin, dfsan_origin s2_origin,
393     dfsan_origin *ret_origin) {
394   CALL_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_origin_strcmp, GET_CALLER_PC(), s1,
395                              s2, s1_label, s2_label, s1_origin, s2_origin);
396   size_t bytes_read;
397   int r = dfsan_strncmp(s1, s2, 0, &bytes_read);
398   dfsan_get_memcmp_origin(s1, s2, bytes_read, ret_label, ret_origin);
399   return r;
400 }
401 
402 // When n == 0, compare strings without byte limit.
403 // When n > 0, compare the first (at most) n bytes of s1 and s2.
404 static int dfsan_strncasecmp(const char *s1, const char *s2, size_t n,
405                              size_t *bytes_read) {
406   for (size_t i = 0;; ++i) {
407     char s1_lower = tolower(s1[i]);
408     char s2_lower = tolower(s2[i]);
409 
410     if (s1_lower != s2_lower || s1[i] == 0 || s2[i] == 0 ||
411         (n > 0 && i == n - 1)) {
412       *bytes_read = i + 1;
413       return s1_lower - s2_lower;
414     }
415   }
416 }
417 
418 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_strcasecmp(const char *s1,
419                                                     const char *s2,
420                                                     dfsan_label s1_label,
421                                                     dfsan_label s2_label,
422                                                     dfsan_label *ret_label) {
423   size_t bytes_read;
424   int r = dfsan_strncasecmp(s1, s2, 0, &bytes_read);
425   *ret_label = dfsan_get_memcmp_label(s1, s2, bytes_read);
426   return r;
427 }
428 
429 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_strcasecmp(
430     const char *s1, const char *s2, dfsan_label s1_label, dfsan_label s2_label,
431     dfsan_label *ret_label, dfsan_origin s1_origin, dfsan_origin s2_origin,
432     dfsan_origin *ret_origin) {
433   size_t bytes_read;
434   int r = dfsan_strncasecmp(s1, s2, 0, &bytes_read);
435   dfsan_get_memcmp_origin(s1, s2, bytes_read, ret_label, ret_origin);
436   return r;
437 }
438 
439 DECLARE_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_strncmp, uptr caller_pc,
440                               const char *s1, const char *s2, size_t n,
441                               dfsan_label s1_label, dfsan_label s2_label,
442                               dfsan_label n_label)
443 
444 DECLARE_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_origin_strncmp, uptr caller_pc,
445                               const char *s1, const char *s2, size_t n,
446                               dfsan_label s1_label, dfsan_label s2_label,
447                               dfsan_label n_label, dfsan_origin s1_origin,
448                               dfsan_origin s2_origin, dfsan_origin n_origin)
449 
450 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_strncmp(const char *s1, const char *s2,
451                                                  size_t n, dfsan_label s1_label,
452                                                  dfsan_label s2_label,
453                                                  dfsan_label n_label,
454                                                  dfsan_label *ret_label) {
455   if (n == 0) {
456     *ret_label = 0;
457     return 0;
458   }
459 
460   CALL_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_strncmp, GET_CALLER_PC(), s1, s2,
461                              n, s1_label, s2_label, n_label);
462 
463   size_t bytes_read;
464   int r = dfsan_strncmp(s1, s2, n, &bytes_read);
465   *ret_label = dfsan_get_memcmp_label(s1, s2, bytes_read);
466   return r;
467 }
468 
469 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_strncmp(
470     const char *s1, const char *s2, size_t n, dfsan_label s1_label,
471     dfsan_label s2_label, dfsan_label n_label, dfsan_label *ret_label,
472     dfsan_origin s1_origin, dfsan_origin s2_origin, dfsan_origin n_origin,
473     dfsan_origin *ret_origin) {
474   if (n == 0) {
475     *ret_label = 0;
476     return 0;
477   }
478 
479   CALL_WEAK_INTERCEPTOR_HOOK(dfsan_weak_hook_origin_strncmp, GET_CALLER_PC(),
480                              s1, s2, n, s1_label, s2_label, n_label, s1_origin,
481                              s2_origin, n_origin);
482 
483   size_t bytes_read;
484   int r = dfsan_strncmp(s1, s2, n, &bytes_read);
485   dfsan_get_memcmp_origin(s1, s2, bytes_read, ret_label, ret_origin);
486   return r;
487 }
488 
489 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_strncasecmp(
490     const char *s1, const char *s2, size_t n, dfsan_label s1_label,
491     dfsan_label s2_label, dfsan_label n_label, dfsan_label *ret_label) {
492   if (n == 0) {
493     *ret_label = 0;
494     return 0;
495   }
496 
497   size_t bytes_read;
498   int r = dfsan_strncasecmp(s1, s2, n, &bytes_read);
499   *ret_label = dfsan_get_memcmp_label(s1, s2, bytes_read);
500   return r;
501 }
502 
503 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_strncasecmp(
504     const char *s1, const char *s2, size_t n, dfsan_label s1_label,
505     dfsan_label s2_label, dfsan_label n_label, dfsan_label *ret_label,
506     dfsan_origin s1_origin, dfsan_origin s2_origin, dfsan_origin n_origin,
507     dfsan_origin *ret_origin) {
508   if (n == 0) {
509     *ret_label = 0;
510     return 0;
511   }
512 
513   size_t bytes_read;
514   int r = dfsan_strncasecmp(s1, s2, n, &bytes_read);
515   dfsan_get_memcmp_origin(s1, s2, bytes_read, ret_label, ret_origin);
516   return r;
517 }
518 
519 
520 SANITIZER_INTERFACE_ATTRIBUTE size_t
521 __dfsw_strlen(const char *s, dfsan_label s_label, dfsan_label *ret_label) {
522   size_t ret = strlen(s);
523   if (flags().strict_data_dependencies) {
524     *ret_label = 0;
525   } else {
526     *ret_label = dfsan_read_label(s, ret + 1);
527   }
528   return ret;
529 }
530 
531 SANITIZER_INTERFACE_ATTRIBUTE size_t __dfso_strlen(const char *s,
532                                                    dfsan_label s_label,
533                                                    dfsan_label *ret_label,
534                                                    dfsan_origin s_origin,
535                                                    dfsan_origin *ret_origin) {
536   size_t ret = __dfsw_strlen(s, s_label, ret_label);
537   if (!flags().strict_data_dependencies)
538     *ret_origin = dfsan_read_origin_of_first_taint(s, ret + 1);
539   return ret;
540 }
541 
542 SANITIZER_INTERFACE_ATTRIBUTE size_t __dfsw_strnlen(const char *s,
543                                                     size_t maxlen,
544                                                     dfsan_label s_label,
545                                                     dfsan_label maxlen_label,
546                                                     dfsan_label *ret_label) {
547   size_t ret = strnlen(s, maxlen);
548   if (flags().strict_data_dependencies) {
549     *ret_label = 0;
550   } else {
551     size_t full_len = strlen(s);
552     size_t covered_len = maxlen > (full_len + 1) ? (full_len + 1) : maxlen;
553     *ret_label = dfsan_union(maxlen_label, dfsan_read_label(s, covered_len));
554   }
555   return ret;
556 }
557 
558 SANITIZER_INTERFACE_ATTRIBUTE size_t __dfso_strnlen(
559     const char *s, size_t maxlen, dfsan_label s_label, dfsan_label maxlen_label,
560     dfsan_label *ret_label, dfsan_origin s_origin, dfsan_origin maxlen_origin,
561     dfsan_origin *ret_origin) {
562   size_t ret = __dfsw_strnlen(s, maxlen, s_label, maxlen_label, ret_label);
563   if (!flags().strict_data_dependencies) {
564     size_t full_len = strlen(s);
565     size_t covered_len = maxlen > (full_len + 1) ? (full_len + 1) : maxlen;
566     dfsan_origin o = dfsan_read_origin_of_first_taint(s, covered_len);
567     *ret_origin = o ? o : maxlen_origin;
568   }
569   return ret;
570 }
571 
572 static void *dfsan_memmove(void *dest, const void *src, size_t n) {
573   dfsan_label *sdest = shadow_for(dest);
574   const dfsan_label *ssrc = shadow_for(src);
575   internal_memmove((void *)sdest, (const void *)ssrc, n * sizeof(dfsan_label));
576   return internal_memmove(dest, src, n);
577 }
578 
579 static void *dfsan_memmove_with_origin(void *dest, const void *src, size_t n) {
580   dfsan_mem_origin_transfer(dest, src, n);
581   return dfsan_memmove(dest, src, n);
582 }
583 
584 static void *dfsan_memcpy(void *dest, const void *src, size_t n) {
585   dfsan_mem_shadow_transfer(dest, src, n);
586   return internal_memcpy(dest, src, n);
587 }
588 
589 static void *dfsan_memcpy_with_origin(void *dest, const void *src, size_t n) {
590   dfsan_mem_origin_transfer(dest, src, n);
591   return dfsan_memcpy(dest, src, n);
592 }
593 
594 static void dfsan_memset(void *s, int c, dfsan_label c_label, size_t n) {
595   internal_memset(s, c, n);
596   dfsan_set_label(c_label, s, n);
597 }
598 
599 static void dfsan_memset_with_origin(void *s, int c, dfsan_label c_label,
600                                      dfsan_origin c_origin, size_t n) {
601   internal_memset(s, c, n);
602   dfsan_set_label_origin(c_label, c_origin, s, n);
603 }
604 
605 SANITIZER_INTERFACE_ATTRIBUTE
606 void *__dfsw_memcpy(void *dest, const void *src, size_t n,
607                     dfsan_label dest_label, dfsan_label src_label,
608                     dfsan_label n_label, dfsan_label *ret_label) {
609   *ret_label = dest_label;
610   return dfsan_memcpy(dest, src, n);
611 }
612 
613 SANITIZER_INTERFACE_ATTRIBUTE
614 void *__dfso_memcpy(void *dest, const void *src, size_t n,
615                     dfsan_label dest_label, dfsan_label src_label,
616                     dfsan_label n_label, dfsan_label *ret_label,
617                     dfsan_origin dest_origin, dfsan_origin src_origin,
618                     dfsan_origin n_origin, dfsan_origin *ret_origin) {
619   *ret_label = dest_label;
620   *ret_origin = dest_origin;
621   return dfsan_memcpy_with_origin(dest, src, n);
622 }
623 
624 SANITIZER_INTERFACE_ATTRIBUTE
625 void *__dfsw_memmove(void *dest, const void *src, size_t n,
626                      dfsan_label dest_label, dfsan_label src_label,
627                      dfsan_label n_label, dfsan_label *ret_label) {
628   *ret_label = dest_label;
629   return dfsan_memmove(dest, src, n);
630 }
631 
632 SANITIZER_INTERFACE_ATTRIBUTE
633 void *__dfso_memmove(void *dest, const void *src, size_t n,
634                      dfsan_label dest_label, dfsan_label src_label,
635                      dfsan_label n_label, dfsan_label *ret_label,
636                      dfsan_origin dest_origin, dfsan_origin src_origin,
637                      dfsan_origin n_origin, dfsan_origin *ret_origin) {
638   *ret_label = dest_label;
639   *ret_origin = dest_origin;
640   return dfsan_memmove_with_origin(dest, src, n);
641 }
642 
643 SANITIZER_INTERFACE_ATTRIBUTE
644 void *__dfsw_memset(void *s, int c, size_t n,
645                     dfsan_label s_label, dfsan_label c_label,
646                     dfsan_label n_label, dfsan_label *ret_label) {
647   dfsan_memset(s, c, c_label, n);
648   *ret_label = s_label;
649   return s;
650 }
651 
652 SANITIZER_INTERFACE_ATTRIBUTE
653 void *__dfso_memset(void *s, int c, size_t n, dfsan_label s_label,
654                     dfsan_label c_label, dfsan_label n_label,
655                     dfsan_label *ret_label, dfsan_origin s_origin,
656                     dfsan_origin c_origin, dfsan_origin n_origin,
657                     dfsan_origin *ret_origin) {
658   dfsan_memset_with_origin(s, c, c_label, c_origin, n);
659   *ret_label = s_label;
660   *ret_origin = s_origin;
661   return s;
662 }
663 
664 SANITIZER_INTERFACE_ATTRIBUTE char *__dfsw_strcat(char *dest, const char *src,
665                                                   dfsan_label dest_label,
666                                                   dfsan_label src_label,
667                                                   dfsan_label *ret_label) {
668   size_t dest_len = strlen(dest);
669   char *ret = strcat(dest, src);
670   dfsan_mem_shadow_transfer(dest + dest_len, src, strlen(src));
671   *ret_label = dest_label;
672   return ret;
673 }
674 
675 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strcat(
676     char *dest, const char *src, dfsan_label dest_label, dfsan_label src_label,
677     dfsan_label *ret_label, dfsan_origin dest_origin, dfsan_origin src_origin,
678     dfsan_origin *ret_origin) {
679   size_t dest_len = strlen(dest);
680   char *ret = strcat(dest, src);
681   size_t src_len = strlen(src);
682   dfsan_mem_origin_transfer(dest + dest_len, src, src_len);
683   dfsan_mem_shadow_transfer(dest + dest_len, src, src_len);
684   *ret_label = dest_label;
685   *ret_origin = dest_origin;
686   return ret;
687 }
688 
689 SANITIZER_INTERFACE_ATTRIBUTE char *__dfsw_strncat(
690     char *dest, const char *src, size_t num, dfsan_label dest_label,
691     dfsan_label src_label, dfsan_label num_label, dfsan_label *ret_label) {
692   size_t src_len = strlen(src);
693   src_len = src_len < num ? src_len : num;
694   size_t dest_len = strlen(dest);
695 
696   char *ret = strncat(dest, src, num);
697   dfsan_mem_shadow_transfer(dest + dest_len, src, src_len);
698   *ret_label = dest_label;
699   return ret;
700 }
701 
702 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strncat(
703     char *dest, const char *src, size_t num, dfsan_label dest_label,
704     dfsan_label src_label, dfsan_label num_label, dfsan_label *ret_label,
705     dfsan_origin dest_origin, dfsan_origin src_origin, dfsan_origin num_origin,
706     dfsan_origin *ret_origin) {
707   size_t src_len = strlen(src);
708   src_len = src_len < num ? src_len : num;
709   size_t dest_len = strlen(dest);
710 
711   char *ret = strncat(dest, src, num);
712 
713   dfsan_mem_origin_transfer(dest + dest_len, src, src_len);
714   dfsan_mem_shadow_transfer(dest + dest_len, src, src_len);
715   *ret_label = dest_label;
716   *ret_origin = dest_origin;
717   return ret;
718 }
719 
720 SANITIZER_INTERFACE_ATTRIBUTE char *
721 __dfsw_strdup(const char *s, dfsan_label s_label, dfsan_label *ret_label) {
722   size_t len = strlen(s);
723   void *p = malloc(len+1);
724   dfsan_memcpy(p, s, len+1);
725   *ret_label = 0;
726   return static_cast<char *>(p);
727 }
728 
729 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strdup(const char *s,
730                                                   dfsan_label s_label,
731                                                   dfsan_label *ret_label,
732                                                   dfsan_origin s_origin,
733                                                   dfsan_origin *ret_origin) {
734   size_t len = strlen(s);
735   void *p = malloc(len + 1);
736   dfsan_memcpy_with_origin(p, s, len + 1);
737   *ret_label = 0;
738   return static_cast<char *>(p);
739 }
740 
741 SANITIZER_INTERFACE_ATTRIBUTE char *
742 __dfsw_strncpy(char *s1, const char *s2, size_t n, dfsan_label s1_label,
743                dfsan_label s2_label, dfsan_label n_label,
744                dfsan_label *ret_label) {
745   size_t len = strlen(s2);
746   if (len < n) {
747     dfsan_memcpy(s1, s2, len+1);
748     dfsan_memset(s1+len+1, 0, 0, n-len-1);
749   } else {
750     dfsan_memcpy(s1, s2, n);
751   }
752 
753   *ret_label = s1_label;
754   return s1;
755 }
756 
757 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strncpy(
758     char *s1, const char *s2, size_t n, dfsan_label s1_label,
759     dfsan_label s2_label, dfsan_label n_label, dfsan_label *ret_label,
760     dfsan_origin s1_origin, dfsan_origin s2_origin, dfsan_origin n_origin,
761     dfsan_origin *ret_origin) {
762   size_t len = strlen(s2);
763   if (len < n) {
764     dfsan_memcpy_with_origin(s1, s2, len + 1);
765     dfsan_memset_with_origin(s1 + len + 1, 0, 0, 0, n - len - 1);
766   } else {
767     dfsan_memcpy_with_origin(s1, s2, n);
768   }
769 
770   *ret_label = s1_label;
771   *ret_origin = s1_origin;
772   return s1;
773 }
774 
775 SANITIZER_INTERFACE_ATTRIBUTE ssize_t
776 __dfsw_pread(int fd, void *buf, size_t count, off_t offset,
777              dfsan_label fd_label, dfsan_label buf_label,
778              dfsan_label count_label, dfsan_label offset_label,
779              dfsan_label *ret_label) {
780   ssize_t ret = pread(fd, buf, count, offset);
781   if (ret > 0)
782     dfsan_set_label(0, buf, ret);
783   *ret_label = 0;
784   return ret;
785 }
786 
787 SANITIZER_INTERFACE_ATTRIBUTE ssize_t __dfso_pread(
788     int fd, void *buf, size_t count, off_t offset, dfsan_label fd_label,
789     dfsan_label buf_label, dfsan_label count_label, dfsan_label offset_label,
790     dfsan_label *ret_label, dfsan_origin fd_origin, dfsan_origin buf_origin,
791     dfsan_origin count_origin, dfsan_label offset_origin,
792     dfsan_origin *ret_origin) {
793   return __dfsw_pread(fd, buf, count, offset, fd_label, buf_label, count_label,
794                       offset_label, ret_label);
795 }
796 
797 SANITIZER_INTERFACE_ATTRIBUTE ssize_t
798 __dfsw_read(int fd, void *buf, size_t count,
799              dfsan_label fd_label, dfsan_label buf_label,
800              dfsan_label count_label,
801              dfsan_label *ret_label) {
802   ssize_t ret = read(fd, buf, count);
803   if (ret > 0)
804     dfsan_set_label(0, buf, ret);
805   *ret_label = 0;
806   return ret;
807 }
808 
809 SANITIZER_INTERFACE_ATTRIBUTE ssize_t __dfso_read(
810     int fd, void *buf, size_t count, dfsan_label fd_label,
811     dfsan_label buf_label, dfsan_label count_label, dfsan_label *ret_label,
812     dfsan_origin fd_origin, dfsan_origin buf_origin, dfsan_origin count_origin,
813     dfsan_origin *ret_origin) {
814   return __dfsw_read(fd, buf, count, fd_label, buf_label, count_label,
815                      ret_label);
816 }
817 
818 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_clock_gettime(clockid_t clk_id,
819                                                        struct timespec *tp,
820                                                        dfsan_label clk_id_label,
821                                                        dfsan_label tp_label,
822                                                        dfsan_label *ret_label) {
823   int ret = clock_gettime(clk_id, tp);
824   if (ret == 0)
825     dfsan_set_label(0, tp, sizeof(struct timespec));
826   *ret_label = 0;
827   return ret;
828 }
829 
830 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_clock_gettime(
831     clockid_t clk_id, struct timespec *tp, dfsan_label clk_id_label,
832     dfsan_label tp_label, dfsan_label *ret_label, dfsan_origin clk_id_origin,
833     dfsan_origin tp_origin, dfsan_origin *ret_origin) {
834   return __dfsw_clock_gettime(clk_id, tp, clk_id_label, tp_label, ret_label);
835 }
836 
837 static void dfsan_set_zero_label(const void *ptr, uptr size) {
838   dfsan_set_label(0, const_cast<void *>(ptr), size);
839 }
840 
841 // dlopen() ultimately calls mmap() down inside the loader, which generally
842 // doesn't participate in dynamic symbol resolution.  Therefore we won't
843 // intercept its calls to mmap, and we have to hook it here.
844 SANITIZER_INTERFACE_ATTRIBUTE void *
845 __dfsw_dlopen(const char *filename, int flag, dfsan_label filename_label,
846               dfsan_label flag_label, dfsan_label *ret_label) {
847   void *handle = dlopen(filename, flag);
848   link_map *map = GET_LINK_MAP_BY_DLOPEN_HANDLE(handle);
849   if (filename && map)
850     ForEachMappedRegion(map, dfsan_set_zero_label);
851   *ret_label = 0;
852   return handle;
853 }
854 
855 SANITIZER_INTERFACE_ATTRIBUTE void *__dfso_dlopen(
856     const char *filename, int flag, dfsan_label filename_label,
857     dfsan_label flag_label, dfsan_label *ret_label,
858     dfsan_origin filename_origin, dfsan_origin flag_origin,
859     dfsan_origin *ret_origin) {
860   return __dfsw_dlopen(filename, flag, filename_label, flag_label, ret_label);
861 }
862 
863 static void *DFsanThreadStartFunc(void *arg) {
864   DFsanThread *t = (DFsanThread *)arg;
865   SetCurrentThread(t);
866   t->Init();
867   SetSigProcMask(&t->starting_sigset_, nullptr);
868   return t->ThreadStart();
869 }
870 
871 static int dfsan_pthread_create(pthread_t *thread, const pthread_attr_t *attr,
872                                 void *start_routine, void *arg,
873                                 dfsan_label *ret_label,
874                                 bool track_origins = false) {
875   pthread_attr_t myattr;
876   if (!attr) {
877     pthread_attr_init(&myattr);
878     attr = &myattr;
879   }
880 
881   // Ensure that the thread stack is large enough to hold all TLS data.
882   AdjustStackSize((void *)(const_cast<pthread_attr_t *>(attr)));
883 
884   DFsanThread *t =
885       DFsanThread::Create((thread_callback_t)start_routine, arg, track_origins);
886   ScopedBlockSignals block(&t->starting_sigset_);
887   int res = pthread_create(thread, attr, DFsanThreadStartFunc, t);
888 
889   if (attr == &myattr)
890     pthread_attr_destroy(&myattr);
891   *ret_label = 0;
892   return res;
893 }
894 
895 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_pthread_create(
896     pthread_t *thread, const pthread_attr_t *attr, void *start_routine,
897     void *arg, dfsan_label thread_label, dfsan_label attr_label,
898     dfsan_label start_routine_label, dfsan_label arg_label,
899     dfsan_label *ret_label) {
900   return dfsan_pthread_create(thread, attr, start_routine, arg, ret_label);
901 }
902 
903 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_pthread_create(
904     pthread_t *thread, const pthread_attr_t *attr, void *start_routine,
905     void *arg, dfsan_label thread_label, dfsan_label attr_label,
906     dfsan_label start_routine_label, dfsan_label arg_label,
907     dfsan_label *ret_label, dfsan_origin thread_origin,
908     dfsan_origin attr_origin, dfsan_origin start_routine_origin,
909     dfsan_origin arg_origin, dfsan_origin *ret_origin) {
910   return dfsan_pthread_create(thread, attr, start_routine, arg, ret_label,
911                               true);
912 }
913 
914 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_pthread_join(pthread_t thread,
915                                                       void **retval,
916                                                       dfsan_label thread_label,
917                                                       dfsan_label retval_label,
918                                                       dfsan_label *ret_label) {
919   int ret = pthread_join(thread, retval);
920   if (ret == 0 && retval)
921     dfsan_set_label(0, retval, sizeof(*retval));
922   *ret_label = 0;
923   return ret;
924 }
925 
926 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_pthread_join(
927     pthread_t thread, void **retval, dfsan_label thread_label,
928     dfsan_label retval_label, dfsan_label *ret_label,
929     dfsan_origin thread_origin, dfsan_origin retval_origin,
930     dfsan_origin *ret_origin) {
931   return __dfsw_pthread_join(thread, retval, thread_label, retval_label,
932                              ret_label);
933 }
934 
935 struct dl_iterate_phdr_info {
936   int (*callback)(struct dl_phdr_info *info, size_t size, void *data);
937   void *data;
938 };
939 
940 int dl_iterate_phdr_cb(struct dl_phdr_info *info, size_t size, void *data) {
941   dl_iterate_phdr_info *dipi = (dl_iterate_phdr_info *)data;
942   dfsan_set_label(0, *info);
943   dfsan_set_label(0, const_cast<char *>(info->dlpi_name),
944                   strlen(info->dlpi_name) + 1);
945   dfsan_set_label(
946       0, const_cast<char *>(reinterpret_cast<const char *>(info->dlpi_phdr)),
947       sizeof(*info->dlpi_phdr) * info->dlpi_phnum);
948 
949   dfsan_clear_thread_local_state();
950   return dipi->callback(info, size, dipi->data);
951 }
952 
953 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_dl_iterate_phdr(
954     int (*callback)(struct dl_phdr_info *info, size_t size, void *data),
955     void *data, dfsan_label callback_label, dfsan_label data_label,
956     dfsan_label *ret_label) {
957   dl_iterate_phdr_info dipi = {callback, data};
958   *ret_label = 0;
959   return dl_iterate_phdr(dl_iterate_phdr_cb, &dipi);
960 }
961 
962 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_dl_iterate_phdr(
963     int (*callback)(struct dl_phdr_info *info, size_t size, void *data),
964     void *data, dfsan_label callback_label, dfsan_label data_label,
965     dfsan_label *ret_label, dfsan_origin callback_origin,
966     dfsan_origin data_origin, dfsan_origin *ret_origin) {
967   dl_iterate_phdr_info dipi = {callback, data};
968   *ret_label = 0;
969   return dl_iterate_phdr(dl_iterate_phdr_cb, &dipi);
970 }
971 
972 // This function is only available for glibc 2.27 or newer.  Mark it weak so
973 // linking succeeds with older glibcs.
974 SANITIZER_WEAK_ATTRIBUTE void _dl_get_tls_static_info(size_t *sizep,
975                                                       size_t *alignp);
976 
977 SANITIZER_INTERFACE_ATTRIBUTE void __dfsw__dl_get_tls_static_info(
978     size_t *sizep, size_t *alignp, dfsan_label sizep_label,
979     dfsan_label alignp_label) {
980   assert(_dl_get_tls_static_info);
981   _dl_get_tls_static_info(sizep, alignp);
982   dfsan_set_label(0, sizep, sizeof(*sizep));
983   dfsan_set_label(0, alignp, sizeof(*alignp));
984 }
985 
986 SANITIZER_INTERFACE_ATTRIBUTE void __dfso__dl_get_tls_static_info(
987     size_t *sizep, size_t *alignp, dfsan_label sizep_label,
988     dfsan_label alignp_label, dfsan_origin sizep_origin,
989     dfsan_origin alignp_origin) {
990   __dfsw__dl_get_tls_static_info(sizep, alignp, sizep_label, alignp_label);
991 }
992 
993 SANITIZER_INTERFACE_ATTRIBUTE
994 char *__dfsw_ctime_r(const time_t *timep, char *buf, dfsan_label timep_label,
995                      dfsan_label buf_label, dfsan_label *ret_label) {
996   char *ret = ctime_r(timep, buf);
997   if (ret) {
998     dfsan_set_label(dfsan_read_label(timep, sizeof(time_t)), buf,
999                     strlen(buf) + 1);
1000     *ret_label = buf_label;
1001   } else {
1002     *ret_label = 0;
1003   }
1004   return ret;
1005 }
1006 
1007 SANITIZER_INTERFACE_ATTRIBUTE
1008 char *__dfso_ctime_r(const time_t *timep, char *buf, dfsan_label timep_label,
1009                      dfsan_label buf_label, dfsan_label *ret_label,
1010                      dfsan_origin timep_origin, dfsan_origin buf_origin,
1011                      dfsan_origin *ret_origin) {
1012   char *ret = ctime_r(timep, buf);
1013   if (ret) {
1014     dfsan_set_label_origin(
1015         dfsan_read_label(timep, sizeof(time_t)),
1016         dfsan_read_origin_of_first_taint(timep, sizeof(time_t)), buf,
1017         strlen(buf) + 1);
1018     *ret_label = buf_label;
1019     *ret_origin = buf_origin;
1020   } else {
1021     *ret_label = 0;
1022   }
1023   return ret;
1024 }
1025 
1026 SANITIZER_INTERFACE_ATTRIBUTE
1027 char *__dfsw_fgets(char *s, int size, FILE *stream, dfsan_label s_label,
1028                    dfsan_label size_label, dfsan_label stream_label,
1029                    dfsan_label *ret_label) {
1030   char *ret = fgets(s, size, stream);
1031   if (ret) {
1032     dfsan_set_label(0, ret, strlen(ret) + 1);
1033     *ret_label = s_label;
1034   } else {
1035     *ret_label = 0;
1036   }
1037   return ret;
1038 }
1039 
1040 SANITIZER_INTERFACE_ATTRIBUTE
1041 char *__dfso_fgets(char *s, int size, FILE *stream, dfsan_label s_label,
1042                    dfsan_label size_label, dfsan_label stream_label,
1043                    dfsan_label *ret_label, dfsan_origin s_origin,
1044                    dfsan_origin size_origin, dfsan_origin stream_origin,
1045                    dfsan_origin *ret_origin) {
1046   char *ret = __dfsw_fgets(s, size, stream, s_label, size_label, stream_label,
1047                            ret_label);
1048   if (ret)
1049     *ret_origin = s_origin;
1050   return ret;
1051 }
1052 
1053 SANITIZER_INTERFACE_ATTRIBUTE
1054 char *__dfsw_getcwd(char *buf, size_t size, dfsan_label buf_label,
1055                     dfsan_label size_label, dfsan_label *ret_label) {
1056   char *ret = getcwd(buf, size);
1057   if (ret) {
1058     dfsan_set_label(0, ret, strlen(ret) + 1);
1059     *ret_label = buf_label;
1060   } else {
1061     *ret_label = 0;
1062   }
1063   return ret;
1064 }
1065 
1066 SANITIZER_INTERFACE_ATTRIBUTE
1067 char *__dfso_getcwd(char *buf, size_t size, dfsan_label buf_label,
1068                     dfsan_label size_label, dfsan_label *ret_label,
1069                     dfsan_origin buf_origin, dfsan_origin size_origin,
1070                     dfsan_origin *ret_origin) {
1071   char *ret = __dfsw_getcwd(buf, size, buf_label, size_label, ret_label);
1072   if (ret)
1073     *ret_origin = buf_origin;
1074   return ret;
1075 }
1076 
1077 SANITIZER_INTERFACE_ATTRIBUTE
1078 char *__dfsw_get_current_dir_name(dfsan_label *ret_label) {
1079   char *ret = get_current_dir_name();
1080   if (ret)
1081     dfsan_set_label(0, ret, strlen(ret) + 1);
1082   *ret_label = 0;
1083   return ret;
1084 }
1085 
1086 SANITIZER_INTERFACE_ATTRIBUTE
1087 char *__dfso_get_current_dir_name(dfsan_label *ret_label,
1088                                   dfsan_origin *ret_origin) {
1089   return __dfsw_get_current_dir_name(ret_label);
1090 }
1091 
1092 // This function is only available for glibc 2.25 or newer.  Mark it weak so
1093 // linking succeeds with older glibcs.
1094 SANITIZER_WEAK_ATTRIBUTE int getentropy(void *buffer, size_t length);
1095 
1096 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_getentropy(void *buffer, size_t length,
1097                                                     dfsan_label buffer_label,
1098                                                     dfsan_label length_label,
1099                                                     dfsan_label *ret_label) {
1100   int ret = getentropy(buffer, length);
1101   if (ret == 0) {
1102     dfsan_set_label(0, buffer, length);
1103   }
1104   *ret_label = 0;
1105   return ret;
1106 }
1107 
1108 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_getentropy(void *buffer, size_t length,
1109                                                     dfsan_label buffer_label,
1110                                                     dfsan_label length_label,
1111                                                     dfsan_label *ret_label,
1112                                                     dfsan_origin buffer_origin,
1113                                                     dfsan_origin length_origin,
1114                                                     dfsan_origin *ret_origin) {
1115   return __dfsw_getentropy(buffer, length, buffer_label, length_label,
1116                            ret_label);
1117 }
1118 
1119 SANITIZER_INTERFACE_ATTRIBUTE
1120 int __dfsw_gethostname(char *name, size_t len, dfsan_label name_label,
1121                        dfsan_label len_label, dfsan_label *ret_label) {
1122   int ret = gethostname(name, len);
1123   if (ret == 0) {
1124     dfsan_set_label(0, name, strlen(name) + 1);
1125   }
1126   *ret_label = 0;
1127   return ret;
1128 }
1129 
1130 SANITIZER_INTERFACE_ATTRIBUTE
1131 int __dfso_gethostname(char *name, size_t len, dfsan_label name_label,
1132                        dfsan_label len_label, dfsan_label *ret_label,
1133                        dfsan_origin name_origin, dfsan_origin len_origin,
1134                        dfsan_label *ret_origin) {
1135   return __dfsw_gethostname(name, len, name_label, len_label, ret_label);
1136 }
1137 
1138 SANITIZER_INTERFACE_ATTRIBUTE
1139 int __dfsw_getrlimit(int resource, struct rlimit *rlim,
1140                      dfsan_label resource_label, dfsan_label rlim_label,
1141                      dfsan_label *ret_label) {
1142   int ret = getrlimit(resource, rlim);
1143   if (ret == 0) {
1144     dfsan_set_label(0, rlim, sizeof(struct rlimit));
1145   }
1146   *ret_label = 0;
1147   return ret;
1148 }
1149 
1150 SANITIZER_INTERFACE_ATTRIBUTE
1151 int __dfso_getrlimit(int resource, struct rlimit *rlim,
1152                      dfsan_label resource_label, dfsan_label rlim_label,
1153                      dfsan_label *ret_label, dfsan_origin resource_origin,
1154                      dfsan_origin rlim_origin, dfsan_origin *ret_origin) {
1155   return __dfsw_getrlimit(resource, rlim, resource_label, rlim_label,
1156                           ret_label);
1157 }
1158 
1159 SANITIZER_INTERFACE_ATTRIBUTE
1160 int __dfsw_getrusage(int who, struct rusage *usage, dfsan_label who_label,
1161                      dfsan_label usage_label, dfsan_label *ret_label) {
1162   int ret = getrusage(who, usage);
1163   if (ret == 0) {
1164     dfsan_set_label(0, usage, sizeof(struct rusage));
1165   }
1166   *ret_label = 0;
1167   return ret;
1168 }
1169 
1170 SANITIZER_INTERFACE_ATTRIBUTE
1171 int __dfso_getrusage(int who, struct rusage *usage, dfsan_label who_label,
1172                      dfsan_label usage_label, dfsan_label *ret_label,
1173                      dfsan_origin who_origin, dfsan_origin usage_origin,
1174                      dfsan_label *ret_origin) {
1175   return __dfsw_getrusage(who, usage, who_label, usage_label, ret_label);
1176 }
1177 
1178 SANITIZER_INTERFACE_ATTRIBUTE
1179 char *__dfsw_strcpy(char *dest, const char *src, dfsan_label dst_label,
1180                     dfsan_label src_label, dfsan_label *ret_label) {
1181   char *ret = strcpy(dest, src);
1182   if (ret) {
1183     dfsan_mem_shadow_transfer(dest, src, strlen(src) + 1);
1184   }
1185   *ret_label = dst_label;
1186   return ret;
1187 }
1188 
1189 SANITIZER_INTERFACE_ATTRIBUTE
1190 char *__dfso_strcpy(char *dest, const char *src, dfsan_label dst_label,
1191                     dfsan_label src_label, dfsan_label *ret_label,
1192                     dfsan_origin dst_origin, dfsan_origin src_origin,
1193                     dfsan_origin *ret_origin) {
1194   char *ret = strcpy(dest, src);
1195   if (ret) {
1196     size_t str_len = strlen(src) + 1;
1197     dfsan_mem_origin_transfer(dest, src, str_len);
1198     dfsan_mem_shadow_transfer(dest, src, str_len);
1199   }
1200   *ret_label = dst_label;
1201   *ret_origin = dst_origin;
1202   return ret;
1203 }
1204 }
1205 
1206 template <typename Fn>
1207 static ALWAYS_INLINE auto dfsan_strtol_impl(
1208     Fn real, const char *nptr, char **endptr, int base,
1209     char **tmp_endptr) -> decltype(real(nullptr, nullptr, 0)) {
1210   assert(tmp_endptr);
1211   auto ret = real(nptr, tmp_endptr, base);
1212   if (endptr)
1213     *endptr = *tmp_endptr;
1214   return ret;
1215 }
1216 
1217 extern "C" {
1218 static void dfsan_strtolong_label(const char *nptr, const char *tmp_endptr,
1219                                   dfsan_label base_label,
1220                                   dfsan_label *ret_label) {
1221   if (tmp_endptr > nptr) {
1222     // If *tmp_endptr is '\0' include its label as well.
1223     *ret_label = dfsan_union(
1224         base_label,
1225         dfsan_read_label(nptr, tmp_endptr - nptr + (*tmp_endptr ? 0 : 1)));
1226   } else {
1227     *ret_label = 0;
1228   }
1229 }
1230 
1231 static void dfsan_strtolong_origin(const char *nptr, const char *tmp_endptr,
1232                                    dfsan_label base_label,
1233                                    dfsan_label *ret_label,
1234                                    dfsan_origin base_origin,
1235                                    dfsan_origin *ret_origin) {
1236   if (tmp_endptr > nptr) {
1237     // When multiple inputs are tainted, we propagate one of its origins.
1238     // Because checking if base_label is tainted does not need additional
1239     // computation, we prefer to propagating base_origin.
1240     *ret_origin = base_label
1241                       ? base_origin
1242                       : dfsan_read_origin_of_first_taint(
1243                             nptr, tmp_endptr - nptr + (*tmp_endptr ? 0 : 1));
1244   }
1245 }
1246 
1247 static double dfsan_strtod(const char *nptr, char **endptr, char **tmp_endptr) {
1248   assert(tmp_endptr);
1249   double ret = strtod(nptr, tmp_endptr);
1250   if (endptr)
1251     *endptr = *tmp_endptr;
1252   return ret;
1253 }
1254 
1255 static void dfsan_strtod_label(const char *nptr, const char *tmp_endptr,
1256                                dfsan_label *ret_label) {
1257   if (tmp_endptr > nptr) {
1258     // If *tmp_endptr is '\0' include its label as well.
1259     *ret_label = dfsan_read_label(
1260         nptr,
1261         tmp_endptr - nptr + (*tmp_endptr ? 0 : 1));
1262   } else {
1263     *ret_label = 0;
1264   }
1265 }
1266 
1267 SANITIZER_INTERFACE_ATTRIBUTE
1268 double __dfsw_strtod(const char *nptr, char **endptr, dfsan_label nptr_label,
1269                      dfsan_label endptr_label, dfsan_label *ret_label) {
1270   char *tmp_endptr;
1271   double ret = dfsan_strtod(nptr, endptr, &tmp_endptr);
1272   dfsan_strtod_label(nptr, tmp_endptr, ret_label);
1273   return ret;
1274 }
1275 
1276 SANITIZER_INTERFACE_ATTRIBUTE
1277 double __dfso_strtod(const char *nptr, char **endptr, dfsan_label nptr_label,
1278                      dfsan_label endptr_label, dfsan_label *ret_label,
1279                      dfsan_origin nptr_origin, dfsan_origin endptr_origin,
1280                      dfsan_origin *ret_origin) {
1281   char *tmp_endptr;
1282   double ret = dfsan_strtod(nptr, endptr, &tmp_endptr);
1283   dfsan_strtod_label(nptr, tmp_endptr, ret_label);
1284   if (tmp_endptr > nptr) {
1285     // If *tmp_endptr is '\0' include its label as well.
1286     *ret_origin = dfsan_read_origin_of_first_taint(
1287         nptr, tmp_endptr - nptr + (*tmp_endptr ? 0 : 1));
1288   } else {
1289     *ret_origin = 0;
1290   }
1291   return ret;
1292 }
1293 
1294 WRAPPER_ALIAS(__isoc23_strtod, strtod)
1295 
1296 #define WRAPPER_STRTO(ret_type, fun)                                     \
1297   SANITIZER_INTERFACE_ATTRIBUTE ret_type __dfsw_##fun(                   \
1298       const char *nptr, char **endptr, int base, dfsan_label nptr_label, \
1299       dfsan_label endptr_label, dfsan_label base_label,                  \
1300       dfsan_label *ret_label) {                                          \
1301     char *tmp_endptr;                                                    \
1302     auto ret = dfsan_strtol_impl(fun, nptr, endptr, base, &tmp_endptr);  \
1303     dfsan_strtolong_label(nptr, tmp_endptr, base_label, ret_label);      \
1304     return ret;                                                          \
1305   }                                                                      \
1306   SANITIZER_INTERFACE_ATTRIBUTE ret_type __dfso_##fun(                   \
1307       const char *nptr, char **endptr, int base, dfsan_label nptr_label, \
1308       dfsan_label endptr_label, dfsan_label base_label,                  \
1309       dfsan_label *ret_label, dfsan_origin nptr_origin,                  \
1310       dfsan_origin endptr_origin, dfsan_origin base_origin,              \
1311       dfsan_origin *ret_origin) {                                        \
1312     char *tmp_endptr;                                                    \
1313     auto ret = dfsan_strtol_impl(fun, nptr, endptr, base, &tmp_endptr);  \
1314     dfsan_strtolong_label(nptr, tmp_endptr, base_label, ret_label);      \
1315     dfsan_strtolong_origin(nptr, tmp_endptr, base_label, ret_label,      \
1316                            base_origin, ret_origin);                     \
1317     return ret;                                                          \
1318   }
1319 
1320 WRAPPER_STRTO(long, strtol)
1321 WRAPPER_STRTO(long long, strtoll)
1322 WRAPPER_STRTO(unsigned long, strtoul)
1323 WRAPPER_STRTO(unsigned long long, strtoull)
1324 WRAPPER_ALIAS(__isoc23_strtol, strtol)
1325 WRAPPER_ALIAS(__isoc23_strtoll, strtoll)
1326 WRAPPER_ALIAS(__isoc23_strtoul, strtoul)
1327 WRAPPER_ALIAS(__isoc23_strtoull, strtoull)
1328 
1329 SANITIZER_INTERFACE_ATTRIBUTE
1330 time_t __dfsw_time(time_t *t, dfsan_label t_label, dfsan_label *ret_label) {
1331   time_t ret = time(t);
1332   if (ret != (time_t) -1 && t) {
1333     dfsan_set_label(0, t, sizeof(time_t));
1334   }
1335   *ret_label = 0;
1336   return ret;
1337 }
1338 
1339 SANITIZER_INTERFACE_ATTRIBUTE
1340 time_t __dfso_time(time_t *t, dfsan_label t_label, dfsan_label *ret_label,
1341                    dfsan_origin t_origin, dfsan_origin *ret_origin) {
1342   return __dfsw_time(t, t_label, ret_label);
1343 }
1344 
1345 SANITIZER_INTERFACE_ATTRIBUTE
1346 int __dfsw_inet_pton(int af, const char *src, void *dst, dfsan_label af_label,
1347                      dfsan_label src_label, dfsan_label dst_label,
1348                      dfsan_label *ret_label) {
1349   int ret = inet_pton(af, src, dst);
1350   if (ret == 1) {
1351     dfsan_set_label(dfsan_read_label(src, strlen(src) + 1), dst,
1352                     af == AF_INET ? sizeof(struct in_addr) : sizeof(in6_addr));
1353   }
1354   *ret_label = 0;
1355   return ret;
1356 }
1357 
1358 SANITIZER_INTERFACE_ATTRIBUTE
1359 int __dfso_inet_pton(int af, const char *src, void *dst, dfsan_label af_label,
1360                      dfsan_label src_label, dfsan_label dst_label,
1361                      dfsan_label *ret_label, dfsan_origin af_origin,
1362                      dfsan_origin src_origin, dfsan_origin dst_origin,
1363                      dfsan_origin *ret_origin) {
1364   int ret = inet_pton(af, src, dst);
1365   if (ret == 1) {
1366     int src_len = strlen(src) + 1;
1367     dfsan_set_label_origin(
1368         dfsan_read_label(src, src_len),
1369         dfsan_read_origin_of_first_taint(src, src_len), dst,
1370         af == AF_INET ? sizeof(struct in_addr) : sizeof(in6_addr));
1371   }
1372   *ret_label = 0;
1373   return ret;
1374 }
1375 
1376 SANITIZER_INTERFACE_ATTRIBUTE
1377 struct tm *__dfsw_localtime_r(const time_t *timep, struct tm *result,
1378                               dfsan_label timep_label, dfsan_label result_label,
1379                               dfsan_label *ret_label) {
1380   struct tm *ret = localtime_r(timep, result);
1381   if (ret) {
1382     dfsan_set_label(dfsan_read_label(timep, sizeof(time_t)), result,
1383                     sizeof(struct tm));
1384     *ret_label = result_label;
1385   } else {
1386     *ret_label = 0;
1387   }
1388   return ret;
1389 }
1390 
1391 SANITIZER_INTERFACE_ATTRIBUTE
1392 struct tm *__dfso_localtime_r(const time_t *timep, struct tm *result,
1393                               dfsan_label timep_label, dfsan_label result_label,
1394                               dfsan_label *ret_label, dfsan_origin timep_origin,
1395                               dfsan_origin result_origin,
1396                               dfsan_origin *ret_origin) {
1397   struct tm *ret = localtime_r(timep, result);
1398   if (ret) {
1399     dfsan_set_label_origin(
1400         dfsan_read_label(timep, sizeof(time_t)),
1401         dfsan_read_origin_of_first_taint(timep, sizeof(time_t)), result,
1402         sizeof(struct tm));
1403     *ret_label = result_label;
1404     *ret_origin = result_origin;
1405   } else {
1406     *ret_label = 0;
1407   }
1408   return ret;
1409 }
1410 
1411 SANITIZER_INTERFACE_ATTRIBUTE
1412 int __dfsw_getpwuid_r(id_t uid, struct passwd *pwd,
1413                       char *buf, size_t buflen, struct passwd **result,
1414                       dfsan_label uid_label, dfsan_label pwd_label,
1415                       dfsan_label buf_label, dfsan_label buflen_label,
1416                       dfsan_label result_label, dfsan_label *ret_label) {
1417   // Store the data in pwd, the strings referenced from pwd in buf, and the
1418   // address of pwd in *result.  On failure, NULL is stored in *result.
1419   int ret = getpwuid_r(uid, pwd, buf, buflen, result);
1420   if (ret == 0) {
1421     dfsan_set_label(0, pwd, sizeof(struct passwd));
1422     dfsan_set_label(0, buf, strlen(buf) + 1);
1423   }
1424   *ret_label = 0;
1425   dfsan_set_label(0, result, sizeof(struct passwd*));
1426   return ret;
1427 }
1428 
1429 SANITIZER_INTERFACE_ATTRIBUTE
1430 int __dfso_getpwuid_r(id_t uid, struct passwd *pwd, char *buf, size_t buflen,
1431                       struct passwd **result, dfsan_label uid_label,
1432                       dfsan_label pwd_label, dfsan_label buf_label,
1433                       dfsan_label buflen_label, dfsan_label result_label,
1434                       dfsan_label *ret_label, dfsan_origin uid_origin,
1435                       dfsan_origin pwd_origin, dfsan_origin buf_origin,
1436                       dfsan_origin buflen_origin, dfsan_origin result_origin,
1437                       dfsan_origin *ret_origin) {
1438   return __dfsw_getpwuid_r(uid, pwd, buf, buflen, result, uid_label, pwd_label,
1439                            buf_label, buflen_label, result_label, ret_label);
1440 }
1441 
1442 SANITIZER_INTERFACE_ATTRIBUTE
1443 int __dfsw_epoll_wait(int epfd, struct epoll_event *events, int maxevents,
1444                       int timeout, dfsan_label epfd_label,
1445                       dfsan_label events_label, dfsan_label maxevents_label,
1446                       dfsan_label timeout_label, dfsan_label *ret_label) {
1447   int ret = epoll_wait(epfd, events, maxevents, timeout);
1448   if (ret > 0)
1449     dfsan_set_label(0, events, ret * sizeof(*events));
1450   *ret_label = 0;
1451   return ret;
1452 }
1453 
1454 SANITIZER_INTERFACE_ATTRIBUTE
1455 int __dfso_epoll_wait(int epfd, struct epoll_event *events, int maxevents,
1456                       int timeout, dfsan_label epfd_label,
1457                       dfsan_label events_label, dfsan_label maxevents_label,
1458                       dfsan_label timeout_label, dfsan_label *ret_label,
1459                       dfsan_origin epfd_origin, dfsan_origin events_origin,
1460                       dfsan_origin maxevents_origin,
1461                       dfsan_origin timeout_origin, dfsan_origin *ret_origin) {
1462   return __dfsw_epoll_wait(epfd, events, maxevents, timeout, epfd_label,
1463                            events_label, maxevents_label, timeout_label,
1464                            ret_label);
1465 }
1466 
1467 SANITIZER_INTERFACE_ATTRIBUTE
1468 int __dfsw_poll(struct pollfd *fds, nfds_t nfds, int timeout,
1469                 dfsan_label dfs_label, dfsan_label nfds_label,
1470                 dfsan_label timeout_label, dfsan_label *ret_label) {
1471   int ret = poll(fds, nfds, timeout);
1472   if (ret >= 0) {
1473     for (; nfds > 0; --nfds) {
1474       dfsan_set_label(0, &fds[nfds - 1].revents, sizeof(fds[nfds - 1].revents));
1475     }
1476   }
1477   *ret_label = 0;
1478   return ret;
1479 }
1480 
1481 SANITIZER_INTERFACE_ATTRIBUTE
1482 int __dfso_poll(struct pollfd *fds, nfds_t nfds, int timeout,
1483                 dfsan_label dfs_label, dfsan_label nfds_label,
1484                 dfsan_label timeout_label, dfsan_label *ret_label,
1485                 dfsan_origin dfs_origin, dfsan_origin nfds_origin,
1486                 dfsan_origin timeout_origin, dfsan_origin *ret_origin) {
1487   return __dfsw_poll(fds, nfds, timeout, dfs_label, nfds_label, timeout_label,
1488                      ret_label);
1489 }
1490 
1491 SANITIZER_INTERFACE_ATTRIBUTE
1492 int __dfsw_select(int nfds, fd_set *readfds, fd_set *writefds,
1493                   fd_set *exceptfds, struct timeval *timeout,
1494                   dfsan_label nfds_label, dfsan_label readfds_label,
1495                   dfsan_label writefds_label, dfsan_label exceptfds_label,
1496                   dfsan_label timeout_label, dfsan_label *ret_label) {
1497   int ret = select(nfds, readfds, writefds, exceptfds, timeout);
1498   // Clear everything (also on error) since their content is either set or
1499   // undefined.
1500   if (readfds) {
1501     dfsan_set_label(0, readfds, sizeof(fd_set));
1502   }
1503   if (writefds) {
1504     dfsan_set_label(0, writefds, sizeof(fd_set));
1505   }
1506   if (exceptfds) {
1507     dfsan_set_label(0, exceptfds, sizeof(fd_set));
1508   }
1509   dfsan_set_label(0, timeout, sizeof(struct timeval));
1510   *ret_label = 0;
1511   return ret;
1512 }
1513 
1514 SANITIZER_INTERFACE_ATTRIBUTE
1515 int __dfso_select(int nfds, fd_set *readfds, fd_set *writefds,
1516                   fd_set *exceptfds, struct timeval *timeout,
1517                   dfsan_label nfds_label, dfsan_label readfds_label,
1518                   dfsan_label writefds_label, dfsan_label exceptfds_label,
1519                   dfsan_label timeout_label, dfsan_label *ret_label,
1520                   dfsan_origin nfds_origin, dfsan_origin readfds_origin,
1521                   dfsan_origin writefds_origin, dfsan_origin exceptfds_origin,
1522                   dfsan_origin timeout_origin, dfsan_origin *ret_origin) {
1523   return __dfsw_select(nfds, readfds, writefds, exceptfds, timeout, nfds_label,
1524                        readfds_label, writefds_label, exceptfds_label,
1525                        timeout_label, ret_label);
1526 }
1527 
1528 SANITIZER_INTERFACE_ATTRIBUTE
1529 int __dfsw_sched_getaffinity(pid_t pid, size_t cpusetsize, cpu_set_t *mask,
1530                              dfsan_label pid_label,
1531                              dfsan_label cpusetsize_label,
1532                              dfsan_label mask_label, dfsan_label *ret_label) {
1533   int ret = sched_getaffinity(pid, cpusetsize, mask);
1534   if (ret == 0) {
1535     dfsan_set_label(0, mask, cpusetsize);
1536   }
1537   *ret_label = 0;
1538   return ret;
1539 }
1540 
1541 SANITIZER_INTERFACE_ATTRIBUTE
1542 int __dfso_sched_getaffinity(pid_t pid, size_t cpusetsize, cpu_set_t *mask,
1543                              dfsan_label pid_label,
1544                              dfsan_label cpusetsize_label,
1545                              dfsan_label mask_label, dfsan_label *ret_label,
1546                              dfsan_origin pid_origin,
1547                              dfsan_origin cpusetsize_origin,
1548                              dfsan_origin mask_origin,
1549                              dfsan_origin *ret_origin) {
1550   return __dfsw_sched_getaffinity(pid, cpusetsize, mask, pid_label,
1551                                   cpusetsize_label, mask_label, ret_label);
1552 }
1553 
1554 SANITIZER_INTERFACE_ATTRIBUTE
1555 int __dfsw_sigemptyset(sigset_t *set, dfsan_label set_label,
1556                        dfsan_label *ret_label) {
1557   int ret = sigemptyset(set);
1558   dfsan_set_label(0, set, sizeof(sigset_t));
1559   *ret_label = 0;
1560   return ret;
1561 }
1562 
1563 SANITIZER_INTERFACE_ATTRIBUTE
1564 int __dfso_sigemptyset(sigset_t *set, dfsan_label set_label,
1565                        dfsan_label *ret_label, dfsan_origin set_origin,
1566                        dfsan_origin *ret_origin) {
1567   return __dfsw_sigemptyset(set, set_label, ret_label);
1568 }
1569 
1570 class SignalHandlerScope {
1571  public:
1572   SignalHandlerScope() {
1573     if (DFsanThread *t = GetCurrentThread())
1574       t->EnterSignalHandler();
1575   }
1576   ~SignalHandlerScope() {
1577     if (DFsanThread *t = GetCurrentThread())
1578       t->LeaveSignalHandler();
1579   }
1580 };
1581 
1582 // Clear DFSan runtime TLS state at the end of a scope.
1583 //
1584 // Implementation must be async-signal-safe and use small data size, because
1585 // instances of this class may live on the signal handler stack.
1586 //
1587 // DFSan uses TLS to pass metadata of arguments and return values. When an
1588 // instrumented function accesses the TLS, if a signal callback happens, and the
1589 // callback calls other instrumented functions with updating the same TLS, the
1590 // TLS is in an inconsistent state after the callback ends. This may cause
1591 // either under-tainting or over-tainting.
1592 //
1593 // The current implementation simply resets TLS at restore. This prevents from
1594 // over-tainting. Although under-tainting may still happen, a taint flow can be
1595 // found eventually if we run a DFSan-instrumented program multiple times. The
1596 // alternative option is saving the entire TLS. However the TLS storage takes
1597 // 2k bytes, and signal calls could be nested. So it does not seem worth.
1598 class ScopedClearThreadLocalState {
1599  public:
1600   ScopedClearThreadLocalState() {}
1601   ~ScopedClearThreadLocalState() { dfsan_clear_thread_local_state(); }
1602 };
1603 
1604 // SignalSpinLocker::sigactions_mu guarantees atomicity of sigaction() calls.
1605 const int kMaxSignals = 1024;
1606 static atomic_uintptr_t sigactions[kMaxSignals];
1607 
1608 static void SignalHandler(int signo) {
1609   SignalHandlerScope signal_handler_scope;
1610   ScopedClearThreadLocalState scoped_clear_tls;
1611 
1612   // Clear shadows for all inputs provided by system.
1613   dfsan_clear_arg_tls(0, sizeof(dfsan_label));
1614 
1615   typedef void (*signal_cb)(int x);
1616   signal_cb cb =
1617       (signal_cb)atomic_load(&sigactions[signo], memory_order_relaxed);
1618   cb(signo);
1619 }
1620 
1621 static void SignalAction(int signo, siginfo_t *si, void *uc) {
1622   SignalHandlerScope signal_handler_scope;
1623   ScopedClearThreadLocalState scoped_clear_tls;
1624 
1625   // Clear shadows for all inputs provided by system. Similar to SignalHandler.
1626   dfsan_clear_arg_tls(0, 3 * sizeof(dfsan_label));
1627   dfsan_set_label(0, si, sizeof(*si));
1628   dfsan_set_label(0, uc, sizeof(ucontext_t));
1629 
1630   typedef void (*sigaction_cb)(int, siginfo_t *, void *);
1631   sigaction_cb cb =
1632       (sigaction_cb)atomic_load(&sigactions[signo], memory_order_relaxed);
1633   cb(signo, si, uc);
1634 }
1635 
1636 SANITIZER_INTERFACE_ATTRIBUTE
1637 int __dfsw_sigaction(int signum, const struct sigaction *act,
1638                      struct sigaction *oldact, dfsan_label signum_label,
1639                      dfsan_label act_label, dfsan_label oldact_label,
1640                      dfsan_label *ret_label) {
1641   CHECK_LT(signum, kMaxSignals);
1642   SignalSpinLocker lock;
1643   uptr old_cb = atomic_load(&sigactions[signum], memory_order_relaxed);
1644   struct sigaction new_act;
1645   struct sigaction *pnew_act = act ? &new_act : nullptr;
1646   if (act) {
1647     internal_memcpy(pnew_act, act, sizeof(struct sigaction));
1648     if (pnew_act->sa_flags & SA_SIGINFO) {
1649       uptr cb = (uptr)(pnew_act->sa_sigaction);
1650       if (cb != (uptr)SIG_IGN && cb != (uptr)SIG_DFL) {
1651         atomic_store(&sigactions[signum], cb, memory_order_relaxed);
1652         pnew_act->sa_sigaction = SignalAction;
1653       }
1654     } else {
1655       uptr cb = (uptr)(pnew_act->sa_handler);
1656       if (cb != (uptr)SIG_IGN && cb != (uptr)SIG_DFL) {
1657         atomic_store(&sigactions[signum], cb, memory_order_relaxed);
1658         pnew_act->sa_handler = SignalHandler;
1659       }
1660     }
1661   }
1662 
1663   int ret = sigaction(signum, pnew_act, oldact);
1664 
1665   if (ret == 0 && oldact) {
1666     if (oldact->sa_flags & SA_SIGINFO) {
1667       if (oldact->sa_sigaction == SignalAction)
1668         oldact->sa_sigaction = (decltype(oldact->sa_sigaction))old_cb;
1669     } else {
1670       if (oldact->sa_handler == SignalHandler)
1671         oldact->sa_handler = (decltype(oldact->sa_handler))old_cb;
1672     }
1673   }
1674 
1675   if (oldact) {
1676     dfsan_set_label(0, oldact, sizeof(struct sigaction));
1677   }
1678   *ret_label = 0;
1679   return ret;
1680 }
1681 
1682 SANITIZER_INTERFACE_ATTRIBUTE
1683 int __dfso_sigaction(int signum, const struct sigaction *act,
1684                      struct sigaction *oldact, dfsan_label signum_label,
1685                      dfsan_label act_label, dfsan_label oldact_label,
1686                      dfsan_label *ret_label, dfsan_origin signum_origin,
1687                      dfsan_origin act_origin, dfsan_origin oldact_origin,
1688                      dfsan_origin *ret_origin) {
1689   return __dfsw_sigaction(signum, act, oldact, signum_label, act_label,
1690                           oldact_label, ret_label);
1691 }
1692 
1693 static sighandler_t dfsan_signal(int signum, sighandler_t handler,
1694                                  dfsan_label *ret_label) {
1695   CHECK_LT(signum, kMaxSignals);
1696   SignalSpinLocker lock;
1697   uptr old_cb = atomic_load(&sigactions[signum], memory_order_relaxed);
1698   if (handler != SIG_IGN && handler != SIG_DFL) {
1699     atomic_store(&sigactions[signum], (uptr)handler, memory_order_relaxed);
1700     handler = &SignalHandler;
1701   }
1702 
1703   sighandler_t ret = signal(signum, handler);
1704 
1705   if (ret == SignalHandler)
1706     ret = (sighandler_t)old_cb;
1707 
1708   *ret_label = 0;
1709   return ret;
1710 }
1711 
1712 SANITIZER_INTERFACE_ATTRIBUTE
1713 sighandler_t __dfsw_signal(int signum, sighandler_t handler,
1714                            dfsan_label signum_label, dfsan_label handler_label,
1715                            dfsan_label *ret_label) {
1716   return dfsan_signal(signum, handler, ret_label);
1717 }
1718 
1719 SANITIZER_INTERFACE_ATTRIBUTE
1720 sighandler_t __dfso_signal(int signum, sighandler_t handler,
1721                            dfsan_label signum_label, dfsan_label handler_label,
1722                            dfsan_label *ret_label, dfsan_origin signum_origin,
1723                            dfsan_origin handler_origin,
1724                            dfsan_origin *ret_origin) {
1725   return dfsan_signal(signum, handler, ret_label);
1726 }
1727 
1728 SANITIZER_INTERFACE_ATTRIBUTE
1729 int __dfsw_sigaltstack(const stack_t *ss, stack_t *old_ss, dfsan_label ss_label,
1730                        dfsan_label old_ss_label, dfsan_label *ret_label) {
1731   int ret = sigaltstack(ss, old_ss);
1732   if (ret != -1 && old_ss)
1733     dfsan_set_label(0, old_ss, sizeof(*old_ss));
1734   *ret_label = 0;
1735   return ret;
1736 }
1737 
1738 SANITIZER_INTERFACE_ATTRIBUTE
1739 int __dfso_sigaltstack(const stack_t *ss, stack_t *old_ss, dfsan_label ss_label,
1740                        dfsan_label old_ss_label, dfsan_label *ret_label,
1741                        dfsan_origin ss_origin, dfsan_origin old_ss_origin,
1742                        dfsan_origin *ret_origin) {
1743   return __dfsw_sigaltstack(ss, old_ss, ss_label, old_ss_label, ret_label);
1744 }
1745 
1746 SANITIZER_INTERFACE_ATTRIBUTE
1747 int __dfsw_gettimeofday(struct timeval *tv, struct timezone *tz,
1748                         dfsan_label tv_label, dfsan_label tz_label,
1749                         dfsan_label *ret_label) {
1750   int ret = gettimeofday(tv, tz);
1751   if (tv) {
1752     dfsan_set_label(0, tv, sizeof(struct timeval));
1753   }
1754   if (tz) {
1755     dfsan_set_label(0, tz, sizeof(struct timezone));
1756   }
1757   *ret_label = 0;
1758   return ret;
1759 }
1760 
1761 SANITIZER_INTERFACE_ATTRIBUTE
1762 int __dfso_gettimeofday(struct timeval *tv, struct timezone *tz,
1763                         dfsan_label tv_label, dfsan_label tz_label,
1764                         dfsan_label *ret_label, dfsan_origin tv_origin,
1765                         dfsan_origin tz_origin, dfsan_origin *ret_origin) {
1766   return __dfsw_gettimeofday(tv, tz, tv_label, tz_label, ret_label);
1767 }
1768 
1769 SANITIZER_INTERFACE_ATTRIBUTE void *__dfsw_memchr(void *s, int c, size_t n,
1770                                                   dfsan_label s_label,
1771                                                   dfsan_label c_label,
1772                                                   dfsan_label n_label,
1773                                                   dfsan_label *ret_label) {
1774   void *ret = memchr(s, c, n);
1775   if (flags().strict_data_dependencies) {
1776     *ret_label = ret ? s_label : 0;
1777   } else {
1778     size_t len =
1779         ret ? reinterpret_cast<char *>(ret) - reinterpret_cast<char *>(s) + 1
1780             : n;
1781     *ret_label =
1782         dfsan_union(dfsan_read_label(s, len), dfsan_union(s_label, c_label));
1783   }
1784   return ret;
1785 }
1786 
1787 SANITIZER_INTERFACE_ATTRIBUTE void *__dfso_memchr(
1788     void *s, int c, size_t n, dfsan_label s_label, dfsan_label c_label,
1789     dfsan_label n_label, dfsan_label *ret_label, dfsan_origin s_origin,
1790     dfsan_origin c_origin, dfsan_origin n_origin, dfsan_origin *ret_origin) {
1791   void *ret = __dfsw_memchr(s, c, n, s_label, c_label, n_label, ret_label);
1792   if (flags().strict_data_dependencies) {
1793     if (ret)
1794       *ret_origin = s_origin;
1795   } else {
1796     size_t len =
1797         ret ? reinterpret_cast<char *>(ret) - reinterpret_cast<char *>(s) + 1
1798             : n;
1799     dfsan_origin o = dfsan_read_origin_of_first_taint(s, len);
1800     *ret_origin = o ? o : (s_label ? s_origin : c_origin);
1801   }
1802   return ret;
1803 }
1804 
1805 SANITIZER_INTERFACE_ATTRIBUTE char *__dfsw_strrchr(char *s, int c,
1806                                                    dfsan_label s_label,
1807                                                    dfsan_label c_label,
1808                                                    dfsan_label *ret_label) {
1809   char *ret = strrchr(s, c);
1810   if (flags().strict_data_dependencies) {
1811     *ret_label = ret ? s_label : 0;
1812   } else {
1813     *ret_label =
1814         dfsan_union(dfsan_read_label(s, strlen(s) + 1),
1815                     dfsan_union(s_label, c_label));
1816   }
1817 
1818   return ret;
1819 }
1820 
1821 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strrchr(
1822     char *s, int c, dfsan_label s_label, dfsan_label c_label,
1823     dfsan_label *ret_label, dfsan_origin s_origin, dfsan_origin c_origin,
1824     dfsan_origin *ret_origin) {
1825   char *ret = __dfsw_strrchr(s, c, s_label, c_label, ret_label);
1826   if (flags().strict_data_dependencies) {
1827     if (ret)
1828       *ret_origin = s_origin;
1829   } else {
1830     size_t s_len = strlen(s) + 1;
1831     dfsan_origin o = dfsan_read_origin_of_first_taint(s, s_len);
1832     *ret_origin = o ? o : (s_label ? s_origin : c_origin);
1833   }
1834 
1835   return ret;
1836 }
1837 
1838 SANITIZER_INTERFACE_ATTRIBUTE char *__dfsw_strstr(char *haystack, char *needle,
1839                                                   dfsan_label haystack_label,
1840                                                   dfsan_label needle_label,
1841                                                   dfsan_label *ret_label) {
1842   char *ret = strstr(haystack, needle);
1843   if (flags().strict_data_dependencies) {
1844     *ret_label = ret ? haystack_label : 0;
1845   } else {
1846     size_t len = ret ? ret + strlen(needle) - haystack : strlen(haystack) + 1;
1847     *ret_label =
1848         dfsan_union(dfsan_read_label(haystack, len),
1849                     dfsan_union(dfsan_read_label(needle, strlen(needle) + 1),
1850                                 dfsan_union(haystack_label, needle_label)));
1851   }
1852 
1853   return ret;
1854 }
1855 
1856 SANITIZER_INTERFACE_ATTRIBUTE char *__dfso_strstr(char *haystack, char *needle,
1857                                                   dfsan_label haystack_label,
1858                                                   dfsan_label needle_label,
1859                                                   dfsan_label *ret_label,
1860                                                   dfsan_origin haystack_origin,
1861                                                   dfsan_origin needle_origin,
1862                                                   dfsan_origin *ret_origin) {
1863   char *ret =
1864       __dfsw_strstr(haystack, needle, haystack_label, needle_label, ret_label);
1865   if (flags().strict_data_dependencies) {
1866     if (ret)
1867       *ret_origin = haystack_origin;
1868   } else {
1869     size_t needle_len = strlen(needle);
1870     size_t len = ret ? ret + needle_len - haystack : strlen(haystack) + 1;
1871     dfsan_origin o = dfsan_read_origin_of_first_taint(haystack, len);
1872     if (o) {
1873       *ret_origin = o;
1874     } else {
1875       o = dfsan_read_origin_of_first_taint(needle, needle_len + 1);
1876       *ret_origin = o ? o : (haystack_label ? haystack_origin : needle_origin);
1877     }
1878   }
1879 
1880   return ret;
1881 }
1882 
1883 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_nanosleep(const struct timespec *req,
1884                                                    struct timespec *rem,
1885                                                    dfsan_label req_label,
1886                                                    dfsan_label rem_label,
1887                                                    dfsan_label *ret_label) {
1888   int ret = nanosleep(req, rem);
1889   *ret_label = 0;
1890   if (ret == -1) {
1891     // Interrupted by a signal, rem is filled with the remaining time.
1892     dfsan_set_label(0, rem, sizeof(struct timespec));
1893   }
1894   return ret;
1895 }
1896 
1897 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_nanosleep(
1898     const struct timespec *req, struct timespec *rem, dfsan_label req_label,
1899     dfsan_label rem_label, dfsan_label *ret_label, dfsan_origin req_origin,
1900     dfsan_origin rem_origin, dfsan_origin *ret_origin) {
1901   return __dfsw_nanosleep(req, rem, req_label, rem_label, ret_label);
1902 }
1903 
1904 static void clear_msghdr_labels(size_t bytes_written, struct msghdr *msg) {
1905   dfsan_set_label(0, msg, sizeof(*msg));
1906   dfsan_set_label(0, msg->msg_name, msg->msg_namelen);
1907   dfsan_set_label(0, msg->msg_control, msg->msg_controllen);
1908   for (size_t i = 0; bytes_written > 0; ++i) {
1909     assert(i < msg->msg_iovlen);
1910     struct iovec *iov = &msg->msg_iov[i];
1911     size_t iov_written =
1912         bytes_written < iov->iov_len ? bytes_written : iov->iov_len;
1913     dfsan_set_label(0, iov->iov_base, iov_written);
1914     bytes_written -= iov_written;
1915   }
1916 }
1917 
1918 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_recvmmsg(
1919     int sockfd, struct mmsghdr *msgvec, unsigned int vlen, int flags,
1920     struct timespec *timeout, dfsan_label sockfd_label,
1921     dfsan_label msgvec_label, dfsan_label vlen_label, dfsan_label flags_label,
1922     dfsan_label timeout_label, dfsan_label *ret_label) {
1923   int ret = recvmmsg(sockfd, msgvec, vlen, flags, timeout);
1924   for (int i = 0; i < ret; ++i) {
1925     dfsan_set_label(0, &msgvec[i].msg_len, sizeof(msgvec[i].msg_len));
1926     clear_msghdr_labels(msgvec[i].msg_len, &msgvec[i].msg_hdr);
1927   }
1928   *ret_label = 0;
1929   return ret;
1930 }
1931 
1932 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_recvmmsg(
1933     int sockfd, struct mmsghdr *msgvec, unsigned int vlen, int flags,
1934     struct timespec *timeout, dfsan_label sockfd_label,
1935     dfsan_label msgvec_label, dfsan_label vlen_label, dfsan_label flags_label,
1936     dfsan_label timeout_label, dfsan_label *ret_label,
1937     dfsan_origin sockfd_origin, dfsan_origin msgvec_origin,
1938     dfsan_origin vlen_origin, dfsan_origin flags_origin,
1939     dfsan_origin timeout_origin, dfsan_origin *ret_origin) {
1940   return __dfsw_recvmmsg(sockfd, msgvec, vlen, flags, timeout, sockfd_label,
1941                          msgvec_label, vlen_label, flags_label, timeout_label,
1942                          ret_label);
1943 }
1944 
1945 SANITIZER_INTERFACE_ATTRIBUTE ssize_t __dfsw_recvmsg(
1946     int sockfd, struct msghdr *msg, int flags, dfsan_label sockfd_label,
1947     dfsan_label msg_label, dfsan_label flags_label, dfsan_label *ret_label) {
1948   ssize_t ret = recvmsg(sockfd, msg, flags);
1949   if (ret >= 0)
1950     clear_msghdr_labels(ret, msg);
1951   *ret_label = 0;
1952   return ret;
1953 }
1954 
1955 SANITIZER_INTERFACE_ATTRIBUTE ssize_t __dfso_recvmsg(
1956     int sockfd, struct msghdr *msg, int flags, dfsan_label sockfd_label,
1957     dfsan_label msg_label, dfsan_label flags_label, dfsan_label *ret_label,
1958     dfsan_origin sockfd_origin, dfsan_origin msg_origin,
1959     dfsan_origin flags_origin, dfsan_origin *ret_origin) {
1960   return __dfsw_recvmsg(sockfd, msg, flags, sockfd_label, msg_label,
1961                         flags_label, ret_label);
1962 }
1963 
1964 SANITIZER_INTERFACE_ATTRIBUTE int
1965 __dfsw_socketpair(int domain, int type, int protocol, int sv[2],
1966                   dfsan_label domain_label, dfsan_label type_label,
1967                   dfsan_label protocol_label, dfsan_label sv_label,
1968                   dfsan_label *ret_label) {
1969   int ret = socketpair(domain, type, protocol, sv);
1970   *ret_label = 0;
1971   if (ret == 0) {
1972     dfsan_set_label(0, sv, sizeof(*sv) * 2);
1973   }
1974   return ret;
1975 }
1976 
1977 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_socketpair(
1978     int domain, int type, int protocol, int sv[2], dfsan_label domain_label,
1979     dfsan_label type_label, dfsan_label protocol_label, dfsan_label sv_label,
1980     dfsan_label *ret_label, dfsan_origin domain_origin,
1981     dfsan_origin type_origin, dfsan_origin protocol_origin,
1982     dfsan_origin sv_origin, dfsan_origin *ret_origin) {
1983   return __dfsw_socketpair(domain, type, protocol, sv, domain_label, type_label,
1984                            protocol_label, sv_label, ret_label);
1985 }
1986 
1987 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_getsockopt(
1988     int sockfd, int level, int optname, void *optval, socklen_t *optlen,
1989     dfsan_label sockfd_label, dfsan_label level_label,
1990     dfsan_label optname_label, dfsan_label optval_label,
1991     dfsan_label optlen_label, dfsan_label *ret_label) {
1992   int ret = getsockopt(sockfd, level, optname, optval, optlen);
1993   if (ret != -1 && optval && optlen) {
1994     dfsan_set_label(0, optlen, sizeof(*optlen));
1995     dfsan_set_label(0, optval, *optlen);
1996   }
1997   *ret_label = 0;
1998   return ret;
1999 }
2000 
2001 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_getsockopt(
2002     int sockfd, int level, int optname, void *optval, socklen_t *optlen,
2003     dfsan_label sockfd_label, dfsan_label level_label,
2004     dfsan_label optname_label, dfsan_label optval_label,
2005     dfsan_label optlen_label, dfsan_label *ret_label,
2006     dfsan_origin sockfd_origin, dfsan_origin level_origin,
2007     dfsan_origin optname_origin, dfsan_origin optval_origin,
2008     dfsan_origin optlen_origin, dfsan_origin *ret_origin) {
2009   return __dfsw_getsockopt(sockfd, level, optname, optval, optlen, sockfd_label,
2010                            level_label, optname_label, optval_label,
2011                            optlen_label, ret_label);
2012 }
2013 
2014 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_getsockname(
2015     int sockfd, struct sockaddr *addr, socklen_t *addrlen,
2016     dfsan_label sockfd_label, dfsan_label addr_label, dfsan_label addrlen_label,
2017     dfsan_label *ret_label) {
2018   socklen_t origlen = addrlen ? *addrlen : 0;
2019   int ret = getsockname(sockfd, addr, addrlen);
2020   if (ret != -1 && addr && addrlen) {
2021     socklen_t written_bytes = origlen < *addrlen ? origlen : *addrlen;
2022     dfsan_set_label(0, addrlen, sizeof(*addrlen));
2023     dfsan_set_label(0, addr, written_bytes);
2024   }
2025   *ret_label = 0;
2026   return ret;
2027 }
2028 
2029 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_getsockname(
2030     int sockfd, struct sockaddr *addr, socklen_t *addrlen,
2031     dfsan_label sockfd_label, dfsan_label addr_label, dfsan_label addrlen_label,
2032     dfsan_label *ret_label, dfsan_origin sockfd_origin,
2033     dfsan_origin addr_origin, dfsan_origin addrlen_origin,
2034     dfsan_origin *ret_origin) {
2035   return __dfsw_getsockname(sockfd, addr, addrlen, sockfd_label, addr_label,
2036                             addrlen_label, ret_label);
2037 }
2038 
2039 SANITIZER_INTERFACE_ATTRIBUTE int __dfsw_getpeername(
2040     int sockfd, struct sockaddr *addr, socklen_t *addrlen,
2041     dfsan_label sockfd_label, dfsan_label addr_label, dfsan_label addrlen_label,
2042     dfsan_label *ret_label) {
2043   socklen_t origlen = addrlen ? *addrlen : 0;
2044   int ret = getpeername(sockfd, addr, addrlen);
2045   if (ret != -1 && addr && addrlen) {
2046     socklen_t written_bytes = origlen < *addrlen ? origlen : *addrlen;
2047     dfsan_set_label(0, addrlen, sizeof(*addrlen));
2048     dfsan_set_label(0, addr, written_bytes);
2049   }
2050   *ret_label = 0;
2051   return ret;
2052 }
2053 
2054 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_getpeername(
2055     int sockfd, struct sockaddr *addr, socklen_t *addrlen,
2056     dfsan_label sockfd_label, dfsan_label addr_label, dfsan_label addrlen_label,
2057     dfsan_label *ret_label, dfsan_origin sockfd_origin,
2058     dfsan_origin addr_origin, dfsan_origin addrlen_origin,
2059     dfsan_origin *ret_origin) {
2060   return __dfsw_getpeername(sockfd, addr, addrlen, sockfd_label, addr_label,
2061                             addrlen_label, ret_label);
2062 }
2063 
2064 // Type of the function passed to dfsan_set_write_callback.
2065 typedef void (*write_dfsan_callback_t)(int fd, const void *buf, ssize_t count);
2066 
2067 // Calls to dfsan_set_write_callback() set the values in this struct.
2068 // Calls to the custom version of write() read (and invoke) them.
2069 static struct {
2070   write_dfsan_callback_t write_callback = nullptr;
2071 } write_callback_info;
2072 
2073 SANITIZER_INTERFACE_ATTRIBUTE void __dfsw_dfsan_set_write_callback(
2074     write_dfsan_callback_t write_callback, dfsan_label write_callback_label,
2075     dfsan_label *ret_label) {
2076   write_callback_info.write_callback = write_callback;
2077 }
2078 
2079 SANITIZER_INTERFACE_ATTRIBUTE void __dfso_dfsan_set_write_callback(
2080     write_dfsan_callback_t write_callback, dfsan_label write_callback_label,
2081     dfsan_label *ret_label, dfsan_origin write_callback_origin,
2082     dfsan_origin *ret_origin) {
2083   write_callback_info.write_callback = write_callback;
2084 }
2085 
2086 static inline void setup_tls_args_for_write_callback(
2087     dfsan_label fd_label, dfsan_label buf_label, dfsan_label count_label,
2088     bool origins, dfsan_origin fd_origin, dfsan_origin buf_origin,
2089     dfsan_origin count_origin) {
2090   // The callback code will expect argument shadow labels in the args TLS,
2091   // and origin labels in the origin args TLS.
2092   // Previously this was done by a trampoline, but we want to remove this:
2093   // https://github.com/llvm/llvm-project/issues/54172
2094   //
2095   // Instead, this code is manually setting up the args TLS data.
2096   //
2097   // The offsets used need to correspond with the instrumentation code,
2098   // see llvm/lib/Transforms/Instrumentation/DataFlowSanitizer.cpp
2099   // DFSanFunction::getShadowForTLSArgument.
2100   // https://github.com/llvm/llvm-project/blob/0acc9e4b5edd8b39ff3d4c6d0e17f02007671c4e/llvm/lib/Transforms/Instrumentation/DataFlowSanitizer.cpp#L1684
2101   // https://github.com/llvm/llvm-project/blob/0acc9e4b5edd8b39ff3d4c6d0e17f02007671c4e/llvm/lib/Transforms/Instrumentation/DataFlowSanitizer.cpp#L125
2102   //
2103   // Here the arguments are all primitives, but it can be more complex
2104   // to compute offsets for array/aggregate type arguments.
2105   //
2106   // TODO(browneee): Consider a builtin to improve maintainabliity.
2107   // With a builtin, we would provide the argument labels via builtin,
2108   // and the builtin would reuse parts of the instrumentation code to ensure
2109   // that this code and the instrumentation can never be out of sync.
2110   // Note: Currently DFSan instrumentation does not run on this code, so
2111   // the builtin may need to be handled outside DFSan instrumentation.
2112   dfsan_set_arg_tls(0, fd_label);
2113   dfsan_set_arg_tls(1, buf_label);
2114   dfsan_set_arg_tls(2, count_label);
2115   if (origins) {
2116     dfsan_set_arg_origin_tls(0, fd_origin);
2117     dfsan_set_arg_origin_tls(1, buf_origin);
2118     dfsan_set_arg_origin_tls(2, count_origin);
2119   }
2120 }
2121 
2122 SANITIZER_INTERFACE_ATTRIBUTE int
2123 __dfsw_write(int fd, const void *buf, size_t count,
2124              dfsan_label fd_label, dfsan_label buf_label,
2125              dfsan_label count_label, dfsan_label *ret_label) {
2126   if (write_callback_info.write_callback) {
2127     setup_tls_args_for_write_callback(fd_label, buf_label, count_label, false,
2128                                       0, 0, 0);
2129     write_callback_info.write_callback(fd, buf, count);
2130   }
2131 
2132   *ret_label = 0;
2133   return write(fd, buf, count);
2134 }
2135 
2136 SANITIZER_INTERFACE_ATTRIBUTE int __dfso_write(
2137     int fd, const void *buf, size_t count, dfsan_label fd_label,
2138     dfsan_label buf_label, dfsan_label count_label, dfsan_label *ret_label,
2139     dfsan_origin fd_origin, dfsan_origin buf_origin, dfsan_origin count_origin,
2140     dfsan_origin *ret_origin) {
2141   if (write_callback_info.write_callback) {
2142     setup_tls_args_for_write_callback(fd_label, buf_label, count_label, true,
2143                                       fd_origin, buf_origin, count_origin);
2144     write_callback_info.write_callback(fd, buf, count);
2145   }
2146 
2147   *ret_label = 0;
2148   return write(fd, buf, count);
2149 }
2150 }  // namespace __dfsan
2151 
2152 // Type used to extract a dfsan_label with va_arg()
2153 typedef int dfsan_label_va;
2154 
2155 // Formats a chunk either a constant string or a single format directive (e.g.,
2156 // '%.3f').
2157 struct Formatter {
2158   Formatter(char *str_, const char *fmt_, size_t size_)
2159       : str(str_),
2160         str_off(0),
2161         size(size_),
2162         fmt_start(fmt_),
2163         fmt_cur(fmt_),
2164         width(-1),
2165         num_scanned(-1),
2166         skip(false) {}
2167 
2168   int format() {
2169     char *tmp_fmt = build_format_string();
2170     int retval =
2171         snprintf(str + str_off, str_off < size ? size - str_off : 0, tmp_fmt,
2172                  0 /* used only to avoid warnings */);
2173     free(tmp_fmt);
2174     return retval;
2175   }
2176 
2177   template <typename T> int format(T arg) {
2178     char *tmp_fmt = build_format_string();
2179     int retval;
2180     if (width >= 0) {
2181       retval = snprintf(str + str_off, str_off < size ? size - str_off : 0,
2182                         tmp_fmt, width, arg);
2183     } else {
2184       retval = snprintf(str + str_off, str_off < size ? size - str_off : 0,
2185                         tmp_fmt, arg);
2186     }
2187     free(tmp_fmt);
2188     return retval;
2189   }
2190 
2191   int scan() {
2192     char *tmp_fmt = build_format_string(true);
2193     int read_count = 0;
2194     int retval = sscanf(str + str_off, tmp_fmt, &read_count);
2195     if (retval > 0) {
2196       if (-1 == num_scanned)
2197         num_scanned = 0;
2198       num_scanned += retval;
2199     }
2200     free(tmp_fmt);
2201     return read_count;
2202   }
2203 
2204   template <typename T>
2205   int scan(T arg) {
2206     char *tmp_fmt = build_format_string(true);
2207     int read_count = 0;
2208     int retval = sscanf(str + str_off, tmp_fmt, arg, &read_count);
2209     if (retval > 0) {
2210       if (-1 == num_scanned)
2211         num_scanned = 0;
2212       num_scanned += retval;
2213     }
2214     free(tmp_fmt);
2215     return read_count;
2216   }
2217 
2218   // with_n -> toggles adding %n on/off; off by default
2219   char *build_format_string(bool with_n = false) {
2220     size_t fmt_size = fmt_cur - fmt_start + 1;
2221     size_t add_size = 0;
2222     if (with_n)
2223       add_size = 2;
2224     char *new_fmt = (char *)malloc(fmt_size + 1 + add_size);
2225     assert(new_fmt);
2226     internal_memcpy(new_fmt, fmt_start, fmt_size);
2227     if (!with_n) {
2228       new_fmt[fmt_size] = '\0';
2229     } else {
2230       new_fmt[fmt_size] = '%';
2231       new_fmt[fmt_size + 1] = 'n';
2232       new_fmt[fmt_size + 2] = '\0';
2233     }
2234 
2235     return new_fmt;
2236   }
2237 
2238   char *str_cur() { return str + str_off; }
2239 
2240   size_t num_written_bytes(int retval) {
2241     if (retval < 0) {
2242       return 0;
2243     }
2244 
2245     size_t num_avail = str_off < size ? size - str_off : 0;
2246     if (num_avail == 0) {
2247       return 0;
2248     }
2249 
2250     size_t num_written = retval;
2251     // A return value of {v,}snprintf of size or more means that the output was
2252     // truncated.
2253     if (num_written >= num_avail) {
2254       num_written -= num_avail;
2255     }
2256 
2257     return num_written;
2258   }
2259 
2260   char *str;
2261   size_t str_off;
2262   size_t size;
2263   const char *fmt_start;
2264   const char *fmt_cur;
2265   int width;
2266   int num_scanned;
2267   bool skip;
2268 };
2269 
2270 // Formats the input and propagates the input labels to the output. The output
2271 // is stored in 'str'. 'size' bounds the number of output bytes. 'format' and
2272 // 'ap' are the format string and the list of arguments for formatting. Returns
2273 // the return value vsnprintf would return.
2274 //
2275 // The function tokenizes the format string in chunks representing either a
2276 // constant string or a single format directive (e.g., '%.3f') and formats each
2277 // chunk independently into the output string. This approach allows to figure
2278 // out which bytes of the output string depends on which argument and thus to
2279 // propagate labels more precisely.
2280 //
2281 // WARNING: This implementation does not support conversion specifiers with
2282 // positional arguments.
2283 static int format_buffer(char *str, size_t size, const char *fmt,
2284                          dfsan_label *va_labels, dfsan_label *ret_label,
2285                          dfsan_origin *va_origins, dfsan_origin *ret_origin,
2286                          va_list ap) {
2287   Formatter formatter(str, fmt, size);
2288 
2289   while (*formatter.fmt_cur) {
2290     formatter.fmt_start = formatter.fmt_cur;
2291     formatter.width = -1;
2292     int retval = 0;
2293 
2294     if (*formatter.fmt_cur != '%') {
2295       // Ordinary character. Consume all the characters until a '%' or the end
2296       // of the string.
2297       for (; *(formatter.fmt_cur + 1) && *(formatter.fmt_cur + 1) != '%';
2298            ++formatter.fmt_cur) {}
2299       retval = formatter.format();
2300       dfsan_set_label(0, formatter.str_cur(),
2301                       formatter.num_written_bytes(retval));
2302     } else {
2303       // Conversion directive. Consume all the characters until a conversion
2304       // specifier or the end of the string.
2305       bool end_fmt = false;
2306       for (; *formatter.fmt_cur && !end_fmt; ) {
2307         switch (*++formatter.fmt_cur) {
2308         case 'd':
2309         case 'i':
2310         case 'o':
2311         case 'u':
2312         case 'x':
2313         case 'X':
2314           switch (*(formatter.fmt_cur - 1)) {
2315           case 'h':
2316             // Also covers the 'hh' case (since the size of the arg is still
2317             // an int).
2318             retval = formatter.format(va_arg(ap, int));
2319             break;
2320           case 'l':
2321             if (formatter.fmt_cur - formatter.fmt_start >= 2 &&
2322                 *(formatter.fmt_cur - 2) == 'l') {
2323               retval = formatter.format(va_arg(ap, long long int));
2324             } else {
2325               retval = formatter.format(va_arg(ap, long int));
2326             }
2327             break;
2328           case 'q':
2329             retval = formatter.format(va_arg(ap, long long int));
2330             break;
2331           case 'j':
2332             retval = formatter.format(va_arg(ap, intmax_t));
2333             break;
2334           case 'z':
2335           case 't':
2336             retval = formatter.format(va_arg(ap, size_t));
2337             break;
2338           default:
2339             retval = formatter.format(va_arg(ap, int));
2340           }
2341           if (va_origins == nullptr)
2342             dfsan_set_label(*va_labels++, formatter.str_cur(),
2343                             formatter.num_written_bytes(retval));
2344           else
2345             dfsan_set_label_origin(*va_labels++, *va_origins++,
2346                                    formatter.str_cur(),
2347                                    formatter.num_written_bytes(retval));
2348           end_fmt = true;
2349           break;
2350 
2351         case 'a':
2352         case 'A':
2353         case 'e':
2354         case 'E':
2355         case 'f':
2356         case 'F':
2357         case 'g':
2358         case 'G':
2359           if (*(formatter.fmt_cur - 1) == 'L') {
2360             retval = formatter.format(va_arg(ap, long double));
2361           } else {
2362             retval = formatter.format(va_arg(ap, double));
2363           }
2364           if (va_origins == nullptr)
2365             dfsan_set_label(*va_labels++, formatter.str_cur(),
2366                             formatter.num_written_bytes(retval));
2367           else
2368             dfsan_set_label_origin(*va_labels++, *va_origins++,
2369                                    formatter.str_cur(),
2370                                    formatter.num_written_bytes(retval));
2371           end_fmt = true;
2372           break;
2373 
2374         case 'c':
2375           retval = formatter.format(va_arg(ap, int));
2376           if (va_origins == nullptr)
2377             dfsan_set_label(*va_labels++, formatter.str_cur(),
2378                             formatter.num_written_bytes(retval));
2379           else
2380             dfsan_set_label_origin(*va_labels++, *va_origins++,
2381                                    formatter.str_cur(),
2382                                    formatter.num_written_bytes(retval));
2383           end_fmt = true;
2384           break;
2385 
2386         case 's': {
2387           char *arg = va_arg(ap, char *);
2388           retval = formatter.format(arg);
2389           if (va_origins) {
2390             va_origins++;
2391             dfsan_mem_origin_transfer(formatter.str_cur(), arg,
2392                                       formatter.num_written_bytes(retval));
2393           }
2394           va_labels++;
2395           dfsan_mem_shadow_transfer(formatter.str_cur(), arg,
2396                                     formatter.num_written_bytes(retval));
2397           end_fmt = true;
2398           break;
2399         }
2400 
2401         case 'p':
2402           retval = formatter.format(va_arg(ap, void *));
2403           if (va_origins == nullptr)
2404             dfsan_set_label(*va_labels++, formatter.str_cur(),
2405                             formatter.num_written_bytes(retval));
2406           else
2407             dfsan_set_label_origin(*va_labels++, *va_origins++,
2408                                    formatter.str_cur(),
2409                                    formatter.num_written_bytes(retval));
2410           end_fmt = true;
2411           break;
2412 
2413         case 'n': {
2414           int *ptr = va_arg(ap, int *);
2415           *ptr = (int)formatter.str_off;
2416           va_labels++;
2417           if (va_origins)
2418             va_origins++;
2419           dfsan_set_label(0, ptr, sizeof(ptr));
2420           end_fmt = true;
2421           break;
2422         }
2423 
2424         case '%':
2425           retval = formatter.format();
2426           dfsan_set_label(0, formatter.str_cur(),
2427                           formatter.num_written_bytes(retval));
2428           end_fmt = true;
2429           break;
2430 
2431         case '*':
2432           formatter.width = va_arg(ap, int);
2433           va_labels++;
2434           if (va_origins)
2435             va_origins++;
2436           break;
2437 
2438         default:
2439           break;
2440         }
2441       }
2442     }
2443 
2444     if (retval < 0) {
2445       return retval;
2446     }
2447 
2448     formatter.fmt_cur++;
2449     formatter.str_off += retval;
2450   }
2451 
2452   *ret_label = 0;
2453   if (ret_origin)
2454     *ret_origin = 0;
2455 
2456   // Number of bytes written in total.
2457   return formatter.str_off;
2458 }
2459 
2460 // This function is an inverse of format_buffer: we take the input buffer,
2461 // scan it in search for format strings and store the results in the varargs.
2462 // The labels are propagated from the input buffer to the varargs.
2463 static int scan_buffer(char *str, size_t size, const char *fmt,
2464                        dfsan_label *va_labels, dfsan_label *ret_label,
2465                        dfsan_origin *str_origin, dfsan_origin *ret_origin,
2466                        va_list ap) {
2467   Formatter formatter(str, fmt, size);
2468   while (*formatter.fmt_cur) {
2469     formatter.fmt_start = formatter.fmt_cur;
2470     formatter.width = -1;
2471     formatter.skip = false;
2472     int read_count = 0;
2473     void *dst_ptr = 0;
2474     size_t write_size = 0;
2475     if (*formatter.fmt_cur != '%') {
2476       // Ordinary character. Consume all the characters until a '%' or the end
2477       // of the string.
2478       for (; *(formatter.fmt_cur + 1) && *(formatter.fmt_cur + 1) != '%';
2479            ++formatter.fmt_cur) {
2480       }
2481       read_count = formatter.scan();
2482       dfsan_set_label(0, formatter.str_cur(),
2483                       formatter.num_written_bytes(read_count));
2484     } else {
2485       // Conversion directive. Consume all the characters until a conversion
2486       // specifier or the end of the string.
2487       bool end_fmt = false;
2488       for (; *formatter.fmt_cur && !end_fmt;) {
2489         switch (*++formatter.fmt_cur) {
2490         case 'd':
2491         case 'i':
2492         case 'o':
2493         case 'u':
2494         case 'x':
2495         case 'X':
2496           if (formatter.skip) {
2497             read_count = formatter.scan();
2498           } else {
2499             switch (*(formatter.fmt_cur - 1)) {
2500             case 'h':
2501               // Also covers the 'hh' case (since the size of the arg is still
2502               // an int).
2503               dst_ptr = va_arg(ap, int *);
2504               read_count = formatter.scan((int *)dst_ptr);
2505               write_size = sizeof(int);
2506               break;
2507             case 'l':
2508               if (formatter.fmt_cur - formatter.fmt_start >= 2 &&
2509                   *(formatter.fmt_cur - 2) == 'l') {
2510                 dst_ptr = va_arg(ap, long long int *);
2511                 read_count = formatter.scan((long long int *)dst_ptr);
2512                 write_size = sizeof(long long int);
2513               } else {
2514                 dst_ptr = va_arg(ap, long int *);
2515                 read_count = formatter.scan((long int *)dst_ptr);
2516                 write_size = sizeof(long int);
2517               }
2518               break;
2519             case 'q':
2520               dst_ptr = va_arg(ap, long long int *);
2521               read_count = formatter.scan((long long int *)dst_ptr);
2522               write_size = sizeof(long long int);
2523               break;
2524             case 'j':
2525               dst_ptr = va_arg(ap, intmax_t *);
2526               read_count = formatter.scan((intmax_t *)dst_ptr);
2527               write_size = sizeof(intmax_t);
2528               break;
2529             case 'z':
2530             case 't':
2531               dst_ptr = va_arg(ap, size_t *);
2532               read_count = formatter.scan((size_t *)dst_ptr);
2533               write_size = sizeof(size_t);
2534               break;
2535             default:
2536               dst_ptr = va_arg(ap, int *);
2537               read_count = formatter.scan((int *)dst_ptr);
2538               write_size = sizeof(int);
2539             }
2540             // get the label associated with the string at the corresponding
2541             // place
2542             dfsan_label l = dfsan_read_label(
2543                 formatter.str_cur(), formatter.num_written_bytes(read_count));
2544             dfsan_set_label(l, dst_ptr, write_size);
2545             if (str_origin != nullptr) {
2546             dfsan_set_label(l, dst_ptr, write_size);
2547             size_t scan_count = formatter.num_written_bytes(read_count);
2548             size_t size = scan_count > write_size ? write_size : scan_count;
2549             dfsan_mem_origin_transfer(dst_ptr, formatter.str_cur(), size);
2550             }
2551           }
2552           end_fmt = true;
2553 
2554           break;
2555 
2556         case 'a':
2557         case 'A':
2558         case 'e':
2559         case 'E':
2560         case 'f':
2561         case 'F':
2562         case 'g':
2563         case 'G':
2564           if (formatter.skip) {
2565             read_count = formatter.scan();
2566           } else {
2567             if (*(formatter.fmt_cur - 1) == 'L') {
2568             dst_ptr = va_arg(ap, long double *);
2569             read_count = formatter.scan((long double *)dst_ptr);
2570             write_size = sizeof(long double);
2571             } else if (*(formatter.fmt_cur - 1) == 'l') {
2572             dst_ptr = va_arg(ap, double *);
2573             read_count = formatter.scan((double *)dst_ptr);
2574             write_size = sizeof(double);
2575             } else {
2576             dst_ptr = va_arg(ap, float *);
2577             read_count = formatter.scan((float *)dst_ptr);
2578             write_size = sizeof(float);
2579             }
2580             dfsan_label l = dfsan_read_label(
2581                 formatter.str_cur(), formatter.num_written_bytes(read_count));
2582             dfsan_set_label(l, dst_ptr, write_size);
2583             if (str_origin != nullptr) {
2584             dfsan_set_label(l, dst_ptr, write_size);
2585             size_t scan_count = formatter.num_written_bytes(read_count);
2586             size_t size = scan_count > write_size ? write_size : scan_count;
2587             dfsan_mem_origin_transfer(dst_ptr, formatter.str_cur(), size);
2588             }
2589           }
2590           end_fmt = true;
2591           break;
2592 
2593         case 'c':
2594           if (formatter.skip) {
2595             read_count = formatter.scan();
2596           } else {
2597             dst_ptr = va_arg(ap, char *);
2598             read_count = formatter.scan((char *)dst_ptr);
2599             write_size = sizeof(char);
2600             dfsan_label l = dfsan_read_label(
2601                 formatter.str_cur(), formatter.num_written_bytes(read_count));
2602             dfsan_set_label(l, dst_ptr, write_size);
2603             if (str_origin != nullptr) {
2604             size_t scan_count = formatter.num_written_bytes(read_count);
2605             size_t size = scan_count > write_size ? write_size : scan_count;
2606             dfsan_mem_origin_transfer(dst_ptr, formatter.str_cur(), size);
2607             }
2608           }
2609           end_fmt = true;
2610           break;
2611 
2612         case 's': {
2613           if (formatter.skip) {
2614             read_count = formatter.scan();
2615           } else {
2616             dst_ptr = va_arg(ap, char *);
2617             read_count = formatter.scan((char *)dst_ptr);
2618             if (1 == read_count) {
2619             // special case: we have parsed a single string and we need to
2620             // update read_count with the string size
2621             read_count = strlen((char *)dst_ptr);
2622             }
2623             if (str_origin)
2624             dfsan_mem_origin_transfer(dst_ptr, formatter.str_cur(),
2625                                       formatter.num_written_bytes(read_count));
2626             va_labels++;
2627             dfsan_mem_shadow_transfer(dst_ptr, formatter.str_cur(),
2628                                       formatter.num_written_bytes(read_count));
2629           }
2630           end_fmt = true;
2631           break;
2632         }
2633 
2634         case 'p':
2635           if (formatter.skip) {
2636             read_count = formatter.scan();
2637           } else {
2638             dst_ptr = va_arg(ap, void *);
2639             read_count =
2640                 formatter.scan((int *)dst_ptr);  // note: changing void* to int*
2641                                                  // since we need to call sizeof
2642             write_size = sizeof(int);
2643 
2644             dfsan_label l = dfsan_read_label(
2645                 formatter.str_cur(), formatter.num_written_bytes(read_count));
2646             dfsan_set_label(l, dst_ptr, write_size);
2647             if (str_origin != nullptr) {
2648             dfsan_set_label(l, dst_ptr, write_size);
2649             size_t scan_count = formatter.num_written_bytes(read_count);
2650             size_t size = scan_count > write_size ? write_size : scan_count;
2651             dfsan_mem_origin_transfer(dst_ptr, formatter.str_cur(), size);
2652             }
2653           }
2654           end_fmt = true;
2655           break;
2656 
2657         case 'n': {
2658           if (!formatter.skip) {
2659             int *ptr = va_arg(ap, int *);
2660             *ptr = (int)formatter.str_off;
2661             *va_labels++ = 0;
2662             dfsan_set_label(0, ptr, sizeof(*ptr));
2663             if (str_origin != nullptr)
2664             *str_origin++ = 0;
2665           }
2666           end_fmt = true;
2667           break;
2668         }
2669 
2670         case '%':
2671           read_count = formatter.scan();
2672           end_fmt = true;
2673           break;
2674 
2675         case '*':
2676           formatter.skip = true;
2677           break;
2678 
2679         default:
2680           break;
2681         }
2682       }
2683     }
2684 
2685     if (read_count < 0) {
2686       // There was an error.
2687       return read_count;
2688     }
2689 
2690     formatter.fmt_cur++;
2691     formatter.str_off += read_count;
2692   }
2693 
2694   (void)va_labels; // Silence unused-but-set-parameter warning
2695   *ret_label = 0;
2696   if (ret_origin)
2697     *ret_origin = 0;
2698 
2699   // Number of items scanned in total.
2700   return formatter.num_scanned;
2701 }
2702 
2703 extern "C" {
2704 SANITIZER_INTERFACE_ATTRIBUTE
2705 int __dfsw_sprintf(char *str, const char *format, dfsan_label str_label,
2706                    dfsan_label format_label, dfsan_label *va_labels,
2707                    dfsan_label *ret_label, ...) {
2708   va_list ap;
2709   va_start(ap, ret_label);
2710 
2711   int ret = format_buffer(str, INT32_MAX, format, va_labels, ret_label, nullptr,
2712                           nullptr, ap);
2713   va_end(ap);
2714   return ret;
2715 }
2716 
2717 SANITIZER_INTERFACE_ATTRIBUTE
2718 int __dfso_sprintf(char *str, const char *format, dfsan_label str_label,
2719                    dfsan_label format_label, dfsan_label *va_labels,
2720                    dfsan_label *ret_label, dfsan_origin str_origin,
2721                    dfsan_origin format_origin, dfsan_origin *va_origins,
2722                    dfsan_origin *ret_origin, ...) {
2723   va_list ap;
2724   va_start(ap, ret_origin);
2725   int ret = format_buffer(str, INT32_MAX, format, va_labels, ret_label,
2726                           va_origins, ret_origin, ap);
2727   va_end(ap);
2728   return ret;
2729 }
2730 
2731 SANITIZER_INTERFACE_ATTRIBUTE
2732 int __dfsw_snprintf(char *str, size_t size, const char *format,
2733                     dfsan_label str_label, dfsan_label size_label,
2734                     dfsan_label format_label, dfsan_label *va_labels,
2735                     dfsan_label *ret_label, ...) {
2736   va_list ap;
2737   va_start(ap, ret_label);
2738   int ret = format_buffer(str, size, format, va_labels, ret_label, nullptr,
2739                           nullptr, ap);
2740   va_end(ap);
2741   return ret;
2742 }
2743 
2744 SANITIZER_INTERFACE_ATTRIBUTE
2745 int __dfso_snprintf(char *str, size_t size, const char *format,
2746                     dfsan_label str_label, dfsan_label size_label,
2747                     dfsan_label format_label, dfsan_label *va_labels,
2748                     dfsan_label *ret_label, dfsan_origin str_origin,
2749                     dfsan_origin size_origin, dfsan_origin format_origin,
2750                     dfsan_origin *va_origins, dfsan_origin *ret_origin, ...) {
2751   va_list ap;
2752   va_start(ap, ret_origin);
2753   int ret = format_buffer(str, size, format, va_labels, ret_label, va_origins,
2754                           ret_origin, ap);
2755   va_end(ap);
2756   return ret;
2757 }
2758 
2759 SANITIZER_INTERFACE_ATTRIBUTE
2760 int __dfsw_sscanf(char *str, const char *format, dfsan_label str_label,
2761                   dfsan_label format_label, dfsan_label *va_labels,
2762                   dfsan_label *ret_label, ...) {
2763   va_list ap;
2764   va_start(ap, ret_label);
2765   int ret = scan_buffer(str, ~0ul, format, va_labels, ret_label, nullptr,
2766                         nullptr, ap);
2767   va_end(ap);
2768   return ret;
2769 }
2770 
2771 SANITIZER_INTERFACE_ATTRIBUTE
2772 int __dfso_sscanf(char *str, const char *format, dfsan_label str_label,
2773                   dfsan_label format_label, dfsan_label *va_labels,
2774                   dfsan_label *ret_label, dfsan_origin str_origin,
2775                   dfsan_origin format_origin, dfsan_origin *va_origins,
2776                   dfsan_origin *ret_origin, ...) {
2777   va_list ap;
2778   va_start(ap, ret_origin);
2779   int ret = scan_buffer(str, ~0ul, format, va_labels, ret_label, &str_origin,
2780                         ret_origin, ap);
2781   va_end(ap);
2782   return ret;
2783 }
2784 
2785 WRAPPER_ALIAS(__isoc99_sscanf, sscanf)
2786 WRAPPER_ALIAS(__isoc23_sscanf, sscanf)
2787 
2788 static void BeforeFork() {
2789   StackDepotLockBeforeFork();
2790   ChainedOriginDepotLockBeforeFork();
2791 }
2792 
2793 static void AfterFork(bool fork_child) {
2794   ChainedOriginDepotUnlockAfterFork(fork_child);
2795   StackDepotUnlockAfterFork(fork_child);
2796 }
2797 
2798 SANITIZER_INTERFACE_ATTRIBUTE
2799 pid_t __dfsw_fork(dfsan_label *ret_label) {
2800   pid_t pid = fork();
2801   *ret_label = 0;
2802   return pid;
2803 }
2804 
2805 SANITIZER_INTERFACE_ATTRIBUTE
2806 pid_t __dfso_fork(dfsan_label *ret_label, dfsan_origin *ret_origin) {
2807   BeforeFork();
2808   pid_t pid = __dfsw_fork(ret_label);
2809   AfterFork(/* fork_child= */ pid == 0);
2810   return pid;
2811 }
2812 
2813 // Default empty implementations (weak). Users should redefine them.
2814 SANITIZER_INTERFACE_WEAK_DEF(void, __sanitizer_cov_trace_pc_guard, u32 *) {}
2815 SANITIZER_INTERFACE_WEAK_DEF(void, __sanitizer_cov_trace_pc_guard_init, u32 *,
2816                              u32 *) {}
2817 SANITIZER_INTERFACE_WEAK_DEF(void, __sanitizer_cov_pcs_init, const uptr *beg,
2818                              const uptr *end) {}
2819 SANITIZER_INTERFACE_WEAK_DEF(void, __sanitizer_cov_trace_pc_indir, void) {}
2820 
2821 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_cmp, void) {}
2822 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_cmp1, void) {}
2823 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_cmp2, void) {}
2824 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_cmp4, void) {}
2825 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_cmp8, void) {}
2826 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_const_cmp1,
2827                              void) {}
2828 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_const_cmp2,
2829                              void) {}
2830 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_const_cmp4,
2831                              void) {}
2832 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_const_cmp8,
2833                              void) {}
2834 SANITIZER_INTERFACE_WEAK_DEF(void, __dfsw___sanitizer_cov_trace_switch, void) {}
2835 }  // extern "C"
2836