blob: 4c4b77b5c41231eee017d2f7217b04f2cccd9a03 [file] [log] [blame]
Tomasz Wasilczyk87329672019-07-12 11:43:00 -07001/*
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#include <libnetdevice/libnetdevice.h>
18
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070019#include "common.h"
Tomasz Wasilczyka5c83a52020-06-16 15:58:19 -070020#include "ifreqs.h"
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070021
22#include <android-base/logging.h>
Tomasz Wasilczyk66fc9392020-08-03 10:22:52 -070023#include <libnl++/MessageFactory.h>
24#include <libnl++/Socket.h>
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070025
Tomasz Wasilczykcd635b92024-10-10 14:08:11 -070026#include <arpa/inet.h>
27#include <ifaddrs.h>
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070028#include <linux/can.h>
Tomasz Wasilczyk390c7112020-07-30 11:08:29 -070029#include <linux/rtnetlink.h>
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070030#include <net/if.h>
Tomasz Wasilczykcd635b92024-10-10 14:08:11 -070031#include <netdb.h>
Tomasz Wasilczyk426708b2024-10-10 11:34:36 -070032#include <sys/ioctl.h>
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070033
Tomasz Wasilczykb87cdae2024-01-09 11:56:43 -080034#include <algorithm>
35#include <iterator>
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -070036#include <sstream>
37
Tomasz Wasilczyk55f21932019-12-20 09:20:24 -080038namespace android::netdevice {
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070039
Tomasz Wasilczyk2aa1a122020-07-20 14:03:05 -070040void useSocketDomain(int domain) {
41 ifreqs::socketDomain = domain;
Tomasz Wasilczykad107412020-05-29 16:32:43 -070042}
43
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -070044bool exists(std::string_view ifname) {
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070045 return nametoindex(ifname) != 0;
46}
47
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -070048bool up(std::string_view ifname) {
Tomasz Wasilczyka5c83a52020-06-16 15:58:19 -070049 auto ifr = ifreqs::fromName(ifname);
50 if (!ifreqs::send(SIOCGIFFLAGS, ifr)) return false;
Tomasz Wasilczykcd635b92024-10-10 14:08:11 -070051 if (ifr.ifr_flags & IFF_UP) return true;
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070052 ifr.ifr_flags |= IFF_UP;
Tomasz Wasilczyka5c83a52020-06-16 15:58:19 -070053 return ifreqs::send(SIOCSIFFLAGS, ifr);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070054}
55
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -070056bool down(std::string_view ifname) {
Tomasz Wasilczyka5c83a52020-06-16 15:58:19 -070057 auto ifr = ifreqs::fromName(ifname);
58 if (!ifreqs::send(SIOCGIFFLAGS, ifr)) return false;
Tomasz Wasilczykcd635b92024-10-10 14:08:11 -070059 if (!(ifr.ifr_flags & IFF_UP)) return true;
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070060 ifr.ifr_flags &= ~IFF_UP;
Tomasz Wasilczyka5c83a52020-06-16 15:58:19 -070061 return ifreqs::send(SIOCSIFFLAGS, ifr);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070062}
63
Tomasz Wasilczykcd635b92024-10-10 14:08:11 -070064static std::string toString(const sockaddr* addr) {
65 char host[NI_MAXHOST];
66 socklen_t addrlen = (addr->sa_family == AF_INET) ? sizeof(sockaddr_in) : sizeof(sockaddr_in6);
67 auto res = getnameinfo(addr, addrlen, host, sizeof(host), nullptr, 0, NI_NUMERICHOST);
68 CHECK(res == 0) << "getnameinfo failed: " << gai_strerror(res);
69 return host;
70}
71
72static std::unique_ptr<ifaddrs, decltype(&freeifaddrs)> getifaddrs() {
73 ifaddrs* addrs = nullptr;
74 CHECK(getifaddrs(&addrs) == 0) << "getifaddrs failed: " << strerror(errno);
75 return {addrs, freeifaddrs};
76}
77
78std::set<std::string> getAllAddr4(std::string_view ifname) {
79 std::set<std::string> addresses;
80 auto addrs = getifaddrs();
81 for (ifaddrs* addr = addrs.get(); addr != nullptr; addr = addr->ifa_next) {
82 if (ifname != addr->ifa_name) continue;
83 if (addr->ifa_addr == nullptr) continue;
84 if (addr->ifa_addr->sa_family != AF_INET) continue;
85 addresses.insert(toString(addr->ifa_addr));
86 }
87 return addresses;
88}
89
90static in_addr_t inetAddr(std::string_view addr) {
91 auto addrn = inet_addr(std::string(addr).c_str());
92 CHECK(addrn != INADDR_NONE) << "Invalid address " << addr;
93 return addrn;
94}
95
96bool setAddr4(std::string_view ifname, std::string_view addr) {
97 auto ifr = ifreqs::fromName(ifname);
98
99 struct sockaddr_in* ifrAddr = reinterpret_cast<sockaddr_in*>(&ifr.ifr_addr);
100 ifrAddr->sin_family = AF_INET;
101 ifrAddr->sin_addr.s_addr = inetAddr(addr);
102
103 return ifreqs::send(SIOCSIFADDR, ifr);
104}
105
106bool addAddr4(std::string_view ifname, std::string_view addr, uint8_t prefixlen) {
107 android::nl::MessageFactory<ifaddrmsg> req(
108 RTM_NEWADDR, NLM_F_REQUEST | NLM_F_CREATE | NLM_F_EXCL | NLM_F_ACK);
109 req->ifa_family = AF_INET;
110 req->ifa_prefixlen = prefixlen;
111 req->ifa_flags = IFA_F_SECONDARY;
112 req->ifa_index = nametoindex(ifname);
113
114 auto addrn = inetAddr(addr);
115 req.add(IFLA_ADDRESS, addrn);
116 req.add(IFLA_BROADCAST, addrn);
117
118 nl::Socket sock(NETLINK_ROUTE);
119 return sock.send(req) && sock.receiveAck(req);
120}
121
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -0700122bool add(std::string_view dev, std::string_view type) {
Tomasz Wasilczyk3ab105b2020-08-05 14:07:40 -0700123 nl::MessageFactory<ifinfomsg> req(RTM_NEWLINK,
124 NLM_F_REQUEST | NLM_F_CREATE | NLM_F_EXCL | NLM_F_ACK);
125 req.add(IFLA_IFNAME, dev);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -0700126
127 {
Tomasz Wasilczyk3ab105b2020-08-05 14:07:40 -0700128 auto linkinfo = req.addNested(IFLA_LINKINFO);
Keith Mok6355aa92022-07-12 21:03:18 +0000129 req.addBuffer(IFLA_INFO_KIND, type);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -0700130 }
131
Tomasz Wasilczyk66fc9392020-08-03 10:22:52 -0700132 nl::Socket sock(NETLINK_ROUTE);
Tomasz Wasilczyk71396052020-08-04 16:21:39 -0700133 return sock.send(req) && sock.receiveAck(req);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -0700134}
135
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -0700136bool del(std::string_view dev) {
Tomasz Wasilczyk3ab105b2020-08-05 14:07:40 -0700137 nl::MessageFactory<ifinfomsg> req(RTM_DELLINK, NLM_F_REQUEST | NLM_F_ACK);
138 req.add(IFLA_IFNAME, dev);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -0700139
Tomasz Wasilczyk66fc9392020-08-03 10:22:52 -0700140 nl::Socket sock(NETLINK_ROUTE);
Tomasz Wasilczyk71396052020-08-04 16:21:39 -0700141 return sock.send(req) && sock.receiveAck(req);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -0700142}
143
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -0700144std::optional<hwaddr_t> getHwAddr(std::string_view ifname) {
Tomasz Wasilczyk91c3a042020-06-24 15:30:10 -0700145 auto ifr = ifreqs::fromName(ifname);
146 if (!ifreqs::send(SIOCGIFHWADDR, ifr)) return std::nullopt;
147
148 hwaddr_t hwaddr;
149 memcpy(hwaddr.data(), ifr.ifr_hwaddr.sa_data, hwaddr.size());
150 return hwaddr;
151}
152
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -0700153bool setHwAddr(std::string_view ifname, hwaddr_t hwaddr) {
Tomasz Wasilczykcaeae192020-06-25 13:36:48 -0700154 auto ifr = ifreqs::fromName(ifname);
155
156 // fetch sa_family
157 if (!ifreqs::send(SIOCGIFHWADDR, ifr)) return false;
158
159 memcpy(ifr.ifr_hwaddr.sa_data, hwaddr.data(), hwaddr.size());
160 return ifreqs::send(SIOCSIFHWADDR, ifr);
161}
162
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -0700163std::optional<bool> isUp(std::string_view ifname) {
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700164 auto ifr = ifreqs::fromName(ifname);
165 if (!ifreqs::send(SIOCGIFFLAGS, ifr)) return std::nullopt;
166 return ifr.ifr_flags & IFF_UP;
167}
168
Tomasz Wasilczykf2dcb642024-10-10 12:13:29 -0700169static bool hasIpv4(std::string_view ifname) {
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000170 auto ifr = ifreqs::fromName(ifname);
Yi Kong810d41a2023-12-06 14:03:58 +0900171 switch (ifreqs::trySend(SIOCGIFADDR, ifr)) {
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000172 case 0:
173 return true;
174 case EADDRNOTAVAIL:
175 case ENODEV:
176 return false;
177 default:
178 PLOG(WARNING) << "Failed checking IPv4 address";
179 return false;
180 }
181}
182
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700183struct WaitState {
184 bool present;
185 bool up;
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000186 bool hasIpv4Addr;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700187
188 bool satisfied(WaitCondition cnd) const {
189 switch (cnd) {
190 case WaitCondition::PRESENT:
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000191 return present;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700192 case WaitCondition::PRESENT_AND_UP:
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000193 return present && up;
194 case WaitCondition::PRESENT_AND_IPV4:
195 return present && up && hasIpv4Addr;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700196 case WaitCondition::DOWN_OR_GONE:
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000197 return !present || !up;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700198 }
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700199 }
200};
201
202static std::string toString(WaitCondition cnd) {
203 switch (cnd) {
204 case WaitCondition::PRESENT:
205 return "become present";
206 case WaitCondition::PRESENT_AND_UP:
207 return "come up";
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000208 case WaitCondition::PRESENT_AND_IPV4:
209 return "get IPv4 address";
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700210 case WaitCondition::DOWN_OR_GONE:
211 return "go down";
212 }
213}
214
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000215static std::string toString(Quantifier quant) {
216 switch (quant) {
217 case Quantifier::ALL_OF:
218 return "all of";
219 case Quantifier::ANY_OF:
220 return "any of";
221 }
222}
223
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700224static std::string toString(const std::set<std::string>& ifnames) {
225 std::stringstream ss;
226 std::copy(ifnames.begin(), ifnames.end(), std::ostream_iterator<std::string>(ss, ","));
227 auto str = ss.str();
228 str.pop_back();
229 return str;
230}
231
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000232std::optional<std::string> waitFor(std::set<std::string> ifnames, WaitCondition cnd,
233 Quantifier quant) {
234 nl::Socket sock(NETLINK_ROUTE, 0, RTMGRP_LINK | RTMGRP_IPV4_IFADDR);
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700235
236 using StatesMap = std::map<std::string, WaitState>;
237 StatesMap states = {};
238 for (const auto ifname : ifnames) {
239 const auto present = exists(ifname);
240 const auto up = present && isUp(ifname).value_or(false);
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000241 const auto hasIpv4Addr = present && hasIpv4(ifname);
242 states[ifname] = {present, up, hasIpv4Addr};
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700243 }
244
245 const auto mapConditionChecker = [cnd](const StatesMap::iterator::value_type& it) {
246 return it.second.satisfied(cnd);
247 };
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000248 const auto isFullySatisfied = [&states, quant,
249 mapConditionChecker]() -> std::optional<std::string> {
250 if (quant == Quantifier::ALL_OF) {
251 if (!std::all_of(states.begin(), states.end(), mapConditionChecker)) return {};
252 return states.begin()->first;
253 } else { // Quantifier::ANY_OF
254 const auto it = std::find_if(states.begin(), states.end(), mapConditionChecker);
255 if (it == states.end()) return {};
256 return it->first;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700257 }
258 };
259
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000260 if (const auto iface = isFullySatisfied()) return iface;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700261
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000262 LOG(DEBUG) << "Waiting for " << toString(quant) << " " << toString(ifnames) << " to "
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700263 << toString(cnd);
Tomasz Wasilczyk2424b712020-08-07 11:04:26 -0700264 for (const auto rawMsg : sock) {
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000265 if (const auto msg = nl::Message<ifinfomsg>::parse(rawMsg, {RTM_NEWLINK, RTM_DELLINK});
266 msg.has_value()) {
267 // Interface added / removed
268 const auto ifname = msg->attributes.get<std::string>(IFLA_IFNAME);
269 if (ifnames.count(ifname) == 0) continue;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700270
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000271 auto& state = states[ifname];
272 state.present = (msg->header.nlmsg_type != RTM_DELLINK);
273 state.up = state.present && (msg->data.ifi_flags & IFF_UP) != 0;
274 if (!state.present) state.hasIpv4Addr = false;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700275
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000276 } else if (const auto msg =
277 nl::Message<ifaddrmsg>::parse(rawMsg, {RTM_NEWADDR, RTM_DELADDR});
278 msg.has_value()) {
279 // Address added / removed
280 const auto ifname = msg->attributes.get<std::string>(IFLA_IFNAME);
281 if (ifnames.count(ifname) == 0) continue;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700282
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000283 if (msg->header.nlmsg_type == RTM_NEWADDR) {
284 states[ifname].hasIpv4Addr = true;
285 } else {
286 // instead of tracking which one got deleted, let's just ask
287 states[ifname].hasIpv4Addr = hasIpv4(ifname);
288 }
289 }
290
291 if (const auto iface = isFullySatisfied()) {
292 LOG(DEBUG) << "Finished waiting for " << toString(quant) << " " << toString(ifnames)
Tomasz Wasilczyk2424b712020-08-07 11:04:26 -0700293 << " to " << toString(cnd);
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000294 return iface;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700295 }
296 }
Tomasz Wasilczyk2424b712020-08-07 11:04:26 -0700297 LOG(FATAL) << "Can't read Netlink socket";
Tomasz Wasilczyk2b162da2022-07-21 18:11:21 +0000298 return {};
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700299}
300
Tomasz Wasilczyk55f21932019-12-20 09:20:24 -0800301} // namespace android::netdevice
Tomasz Wasilczyk91c3a042020-06-24 15:30:10 -0700302
303bool operator==(const android::netdevice::hwaddr_t lhs, const unsigned char rhs[ETH_ALEN]) {
304 static_assert(lhs.size() == ETH_ALEN);
305 return 0 == memcmp(lhs.data(), rhs, lhs.size());
306}