1*7c3d14c8STreehugger Robot // RUN: %clangxx_msan -O0 %s -o %t && %run %t %p 2>&1
2*7c3d14c8STreehugger Robot // RUN: %clangxx_msan -O0 -D_FILE_OFFSET_BITS=64 %s -o %t && %run %t %p 2>&1
3*7c3d14c8STreehugger Robot // RUN: %clangxx_msan -O3 %s -o %t && %run %t %p 2>&1
4*7c3d14c8STreehugger Robot
5*7c3d14c8STreehugger Robot #include <assert.h>
6*7c3d14c8STreehugger Robot #include <errno.h>
7*7c3d14c8STreehugger Robot #include <ifaddrs.h>
8*7c3d14c8STreehugger Robot #include <stdio.h>
9*7c3d14c8STreehugger Robot #include <string.h>
10*7c3d14c8STreehugger Robot
11*7c3d14c8STreehugger Robot #include <vector>
12*7c3d14c8STreehugger Robot
13*7c3d14c8STreehugger Robot #if defined(__FreeBSD__)
14*7c3d14c8STreehugger Robot #include <sys/socket.h> // To define 'struct sockaddr'.
15*7c3d14c8STreehugger Robot #endif
16*7c3d14c8STreehugger Robot
17*7c3d14c8STreehugger Robot #include <sanitizer/msan_interface.h>
18*7c3d14c8STreehugger Robot
19*7c3d14c8STreehugger Robot #define CHECK_AND_PUSH(addr, size) \
20*7c3d14c8STreehugger Robot if (addr) { \
21*7c3d14c8STreehugger Robot assert(-1 == __msan_test_shadow(addr, sizeof(size))); \
22*7c3d14c8STreehugger Robot ranges.push_back(std::make_pair((void *)addr, (size_t)size)); \
23*7c3d14c8STreehugger Robot }
24*7c3d14c8STreehugger Robot
main(int argc,char * argv[])25*7c3d14c8STreehugger Robot int main(int argc, char *argv[]) {
26*7c3d14c8STreehugger Robot struct ifaddrs *ifas;
27*7c3d14c8STreehugger Robot
28*7c3d14c8STreehugger Robot assert(0 == __msan_test_shadow(&ifas, sizeof(ifaddrs *)));
29*7c3d14c8STreehugger Robot int res = getifaddrs(&ifas);
30*7c3d14c8STreehugger Robot if (res == -1) {
31*7c3d14c8STreehugger Robot assert(errno == ENOSYS);
32*7c3d14c8STreehugger Robot printf("getifaddrs() is not implemented\n");
33*7c3d14c8STreehugger Robot return 0;
34*7c3d14c8STreehugger Robot }
35*7c3d14c8STreehugger Robot assert(res == 0);
36*7c3d14c8STreehugger Robot assert(-1 == __msan_test_shadow(&ifas, sizeof(ifaddrs *)));
37*7c3d14c8STreehugger Robot
38*7c3d14c8STreehugger Robot std::vector<std::pair<void *, size_t> > ranges;
39*7c3d14c8STreehugger Robot ifaddrs *p = ifas;
40*7c3d14c8STreehugger Robot while (p) {
41*7c3d14c8STreehugger Robot CHECK_AND_PUSH(p, sizeof(ifaddrs));
42*7c3d14c8STreehugger Robot CHECK_AND_PUSH(p->ifa_name, strlen(p->ifa_name) + 1);
43*7c3d14c8STreehugger Robot CHECK_AND_PUSH(p->ifa_addr, sizeof(*p->ifa_addr));
44*7c3d14c8STreehugger Robot CHECK_AND_PUSH(p->ifa_netmask, sizeof(*p->ifa_netmask));
45*7c3d14c8STreehugger Robot CHECK_AND_PUSH(p->ifa_broadaddr, sizeof(*p->ifa_broadaddr));
46*7c3d14c8STreehugger Robot CHECK_AND_PUSH(p->ifa_dstaddr, sizeof(*p->ifa_dstaddr));
47*7c3d14c8STreehugger Robot p = p->ifa_next;
48*7c3d14c8STreehugger Robot }
49*7c3d14c8STreehugger Robot
50*7c3d14c8STreehugger Robot freeifaddrs(ifas);
51*7c3d14c8STreehugger Robot for (int i = 0; i < ranges.size(); i++)
52*7c3d14c8STreehugger Robot assert(0 == __msan_test_shadow(ranges[i].first, ranges[i].second));
53*7c3d14c8STreehugger Robot return 0;
54*7c3d14c8STreehugger Robot }
55