• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2019 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  *
16  */
17 
18 #include "resolv_test_utils.h"
19 
20 #include <arpa/inet.h>
21 
22 #include <android-base/chrono_utils.h>
23 #include <android-base/logging.h>
24 
25 using android::netdutils::ScopedAddrinfo;
26 
ToString(const hostent * he)27 std::string ToString(const hostent* he) {
28     if (he == nullptr) return "<null>";
29     char buffer[INET6_ADDRSTRLEN];
30     if (!inet_ntop(he->h_addrtype, he->h_addr_list[0], buffer, sizeof(buffer))) {
31         return "<invalid>";
32     }
33     return buffer;
34 }
35 
ToString(const addrinfo * ai)36 std::string ToString(const addrinfo* ai) {
37     if (!ai) return "<null>";
38 
39     char host[NI_MAXHOST];
40     int rv = getnameinfo(ai->ai_addr, ai->ai_addrlen, host, sizeof(host), nullptr, 0,
41                          NI_NUMERICHOST);
42     if (rv != 0) return gai_strerror(rv);
43     return host;
44 }
45 
ToString(const ScopedAddrinfo & ai)46 std::string ToString(const ScopedAddrinfo& ai) {
47     return ToString(ai.get());
48 }
49 
ToString(const sockaddr_storage * addr)50 std::string ToString(const sockaddr_storage* addr) {
51     if (!addr) return "<null>";
52     char host[NI_MAXHOST];
53     int rv = getnameinfo((const sockaddr*)addr, sizeof(sockaddr_storage), host, sizeof(host),
54                          nullptr, 0, NI_NUMERICHOST);
55     if (rv != 0) return gai_strerror(rv);
56     return host;
57 }
58 
ToStrings(const hostent * he)59 std::vector<std::string> ToStrings(const hostent* he) {
60     std::vector<std::string> hosts;
61     if (he == nullptr) {
62         hosts.push_back("<null>");
63         return hosts;
64     }
65     uint32_t i = 0;
66     while (he->h_addr_list[i] != nullptr) {
67         char host[INET6_ADDRSTRLEN];
68         if (!inet_ntop(he->h_addrtype, he->h_addr_list[i], host, sizeof(host))) {
69             hosts.push_back("<invalid>");
70             return hosts;
71         } else {
72             hosts.push_back(host);
73         }
74         i++;
75     }
76     if (hosts.empty()) hosts.push_back("<invalid>");
77     return hosts;
78 }
79 
ToStrings(const addrinfo * ai)80 std::vector<std::string> ToStrings(const addrinfo* ai) {
81     std::vector<std::string> hosts;
82     if (!ai) {
83         hosts.push_back("<null>");
84         return hosts;
85     }
86     for (const auto* aip = ai; aip != nullptr; aip = aip->ai_next) {
87         char host[NI_MAXHOST];
88         int rv = getnameinfo(aip->ai_addr, aip->ai_addrlen, host, sizeof(host), nullptr, 0,
89                              NI_NUMERICHOST);
90         if (rv != 0) {
91             hosts.clear();
92             hosts.push_back(gai_strerror(rv));
93             return hosts;
94         } else {
95             hosts.push_back(host);
96         }
97     }
98     if (hosts.empty()) hosts.push_back("<invalid>");
99     return hosts;
100 }
101 
ToStrings(const ScopedAddrinfo & ai)102 std::vector<std::string> ToStrings(const ScopedAddrinfo& ai) {
103     return ToStrings(ai.get());
104 }
105 
GetNumQueries(const test::DNSResponder & dns,const char * name)106 size_t GetNumQueries(const test::DNSResponder& dns, const char* name) {
107     std::vector<test::DNSResponder::QueryInfo> queries = dns.queries();
108     size_t found = 0;
109     for (const auto& p : queries) {
110         if (p.name == name) {
111             ++found;
112         }
113     }
114     return found;
115 }
116 
GetNumQueriesForProtocol(const test::DNSResponder & dns,const int protocol,const char * name)117 size_t GetNumQueriesForProtocol(const test::DNSResponder& dns, const int protocol,
118                                 const char* name) {
119     std::vector<test::DNSResponder::QueryInfo> queries = dns.queries();
120     size_t found = 0;
121     for (const auto& p : queries) {
122         if (p.protocol == protocol && p.name == name) {
123             ++found;
124         }
125     }
126     return found;
127 }
128 
GetNumQueriesForType(const test::DNSResponder & dns,ns_type type,const char * name)129 size_t GetNumQueriesForType(const test::DNSResponder& dns, ns_type type, const char* name) {
130     std::vector<test::DNSResponder::QueryInfo> queries = dns.queries();
131     size_t found = 0;
132     for (const auto& p : queries) {
133         if (p.type == type && p.name == name) {
134             ++found;
135         }
136     }
137     return found;
138 }
139 
PollForCondition(const std::function<bool ()> & condition,std::chrono::milliseconds timeout)140 bool PollForCondition(const std::function<bool()>& condition, std::chrono::milliseconds timeout) {
141     constexpr std::chrono::milliseconds retryIntervalMs{5};
142     android::base::Timer t;
143     while (t.duration() < timeout) {
144         if (condition()) return true;
145         std::this_thread::sleep_for(retryIntervalMs);
146     }
147     return false;
148 }
149 
safe_getaddrinfo(const char * node,const char * service,const struct addrinfo * hints)150 ScopedAddrinfo safe_getaddrinfo(const char* node, const char* service,
151                                 const struct addrinfo* hints) {
152     addrinfo* result = nullptr;
153     if (getaddrinfo(node, service, hints, &result) != 0) {
154         result = nullptr;  // Should already be the case, but...
155     }
156     return ScopedAddrinfo(result);
157 }
158 
WaitChild(pid_t pid)159 int WaitChild(pid_t pid) {
160     int status;
161     const pid_t got_pid = TEMP_FAILURE_RETRY(waitpid(pid, &status, 0));
162 
163     if (got_pid != pid) {
164         PLOG(WARNING) << __func__ << ": waitpid failed: wanted " << pid << ", got " << got_pid;
165         return 1;
166     }
167 
168     if (WIFEXITED(status) && WEXITSTATUS(status) == 0) {
169         return 0;
170     } else {
171         return status;
172     }
173 }
174 
ForkAndRun(const std::vector<std::string> & args)175 int ForkAndRun(const std::vector<std::string>& args) {
176     std::vector<const char*> argv;
177     argv.resize(args.size() + 1, nullptr);
178     std::transform(args.begin(), args.end(), argv.begin(),
179                    [](const std::string& in) { return in.c_str(); });
180 
181     pid_t pid = fork();
182     if (pid == -1) {
183         // Fork failed.
184         PLOG(ERROR) << __func__ << ": Unable to fork";
185         return -1;
186     }
187 
188     if (pid == 0) {
189         execv(argv[0], const_cast<char**>(argv.data()));
190         PLOG(ERROR) << __func__ << ": execv failed";
191         _exit(1);
192     }
193 
194     int rc = WaitChild(pid);
195     if (rc != 0) {
196         PLOG(ERROR) << __func__ << ": Failed run: status=" << rc;
197     }
198     return rc;
199 }
200 
201 // Add routing rules for MDNS packets, or MDNS packets won't know the destination is MDNS
202 // muticast address "224.0.0.251".
SetMdnsRoute()203 void SetMdnsRoute() {
204     const std::vector<std::string> args = {
205             "system/bin/ip", "route",  "add",   "local", "224.0.0.251", "dev",       "lo",
206             "proto",         "static", "scope", "host",  "src",         "127.0.0.1",
207     };
208     EXPECT_EQ(0, ForkAndRun(args));
209 }
210 
RemoveMdnsRoute()211 void RemoveMdnsRoute() {
212     const std::vector<std::string> args = {
213             "system/bin/ip", "route",  "del",   "local", "224.0.0.251", "dev",       "lo",
214             "proto",         "static", "scope", "host",  "src",         "127.0.0.1",
215     };
216     EXPECT_EQ(0, ForkAndRun(args));
217 }