Home | History | Annotate | Download | only in Linux
      1 // RUN: %clangxx_msan -std=c++11 -O0 %s -o %t && %run %t
      2 // RUN: %clangxx_msan -std=c++11 -O0 %s -o %t -DPOSITIVE && not %run %t |& FileCheck %s
      3 
      4 #include <assert.h>
      5 #include <dlfcn.h>
      6 #include <sanitizer/msan_interface.h>
      7 #include <stdio.h>
      8 #include <string.h>
      9 #include <sys/types.h>
     10 #include <sys/uio.h>
     11 #include <unistd.h>
     12 #include <errno.h>
     13 
     14 typedef ssize_t (*process_vm_readwritev_fn)(pid_t, const iovec *, unsigned long,
     15                                             const iovec *, unsigned long,
     16                                             unsigned long);
     17 
     18 // Exit with success, emulating the expected output.
     19 int exit_dummy()
     20 {
     21 #ifdef POSITIVE
     22     printf("process_vm_readv not found or not implemented!\n");
     23     printf(
     24         "WARNING: MemorySanitizer: use-of-uninitialized-value (not really)\n");
     25     return 1;
     26 #else
     27     return 0;
     28 #endif
     29 }
     30 
     31 int main(void) {
     32   // This requires glibc 2.15.
     33   process_vm_readwritev_fn libc_process_vm_readv =
     34       (process_vm_readwritev_fn)dlsym(RTLD_NEXT, "process_vm_readv");
     35   if (!libc_process_vm_readv)
     36     return exit_dummy();
     37 
     38   process_vm_readwritev_fn process_vm_readv =
     39       (process_vm_readwritev_fn)dlsym(RTLD_DEFAULT, "process_vm_readv");
     40   process_vm_readwritev_fn process_vm_writev =
     41       (process_vm_readwritev_fn)dlsym(RTLD_DEFAULT, "process_vm_writev");
     42 
     43   char a[100];
     44   memset(a, 0xab, 100);
     45 
     46   char b[100];
     47   iovec iov_a[] = {{(void *)a, 20}, (void *)(a + 50), 10};
     48   iovec iov_b[] = {{(void *)(b + 10), 10}, (void *)(b + 30), 20};
     49 
     50   __msan_poison(&b, sizeof(b));
     51   ssize_t res = process_vm_readv(getpid(), iov_b, 2, iov_a, 2, 0);
     52   if (errno == ENOSYS) // Function not implemented
     53     return exit_dummy();
     54 
     55   assert(res == 30);
     56   __msan_check_mem_is_initialized(b + 10, 10);
     57   __msan_check_mem_is_initialized(b + 30, 20);
     58   assert(__msan_test_shadow(b + 9, 1) == 0);
     59   assert(__msan_test_shadow(b + 20, 1) == 0);
     60   assert(__msan_test_shadow(b + 29, 1) == 0);
     61   assert(__msan_test_shadow(b + 50, 1) == 0);
     62 
     63 #ifdef POSITIVE
     64   __msan_unpoison(&b, sizeof(b));
     65   __msan_poison(b + 32, 1);
     66   res = process_vm_writev(getpid(), iov_b, 2, iov_a, 2, 0);
     67 // CHECK: WARNING: MemorySanitizer: use-of-uninitialized-value
     68 #else
     69   __msan_unpoison(&b, sizeof(b));
     70   res = process_vm_writev(getpid(), iov_b, 2, iov_a, 2, 0);
     71   assert(res == 30);
     72 #endif
     73 
     74   return 0;
     75 }
     76