blob: 4c5b30928fee1f935361225544ce347895231d97 [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
26#include <linux/can.h>
Tomasz Wasilczyk390c7112020-07-30 11:08:29 -070027#include <linux/rtnetlink.h>
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070028#include <net/if.h>
29
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -070030#include <sstream>
31
Tomasz Wasilczyk55f21932019-12-20 09:20:24 -080032namespace android::netdevice {
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070033
Tomasz Wasilczyk2aa1a122020-07-20 14:03:05 -070034void useSocketDomain(int domain) {
35 ifreqs::socketDomain = domain;
Tomasz Wasilczykad107412020-05-29 16:32:43 -070036}
37
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070038bool exists(std::string ifname) {
39 return nametoindex(ifname) != 0;
40}
41
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070042bool up(std::string ifname) {
Tomasz Wasilczyka5c83a52020-06-16 15:58:19 -070043 auto ifr = ifreqs::fromName(ifname);
44 if (!ifreqs::send(SIOCGIFFLAGS, ifr)) return false;
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070045 ifr.ifr_flags |= IFF_UP;
Tomasz Wasilczyka5c83a52020-06-16 15:58:19 -070046 return ifreqs::send(SIOCSIFFLAGS, ifr);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070047}
48
49bool down(std::string ifname) {
Tomasz Wasilczyka5c83a52020-06-16 15:58:19 -070050 auto ifr = ifreqs::fromName(ifname);
51 if (!ifreqs::send(SIOCGIFFLAGS, ifr)) return false;
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
56bool add(std::string dev, std::string type) {
Tomasz Wasilczyk3ab105b2020-08-05 14:07:40 -070057 nl::MessageFactory<ifinfomsg> req(RTM_NEWLINK,
58 NLM_F_REQUEST | NLM_F_CREATE | NLM_F_EXCL | NLM_F_ACK);
59 req.add(IFLA_IFNAME, dev);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070060
61 {
Tomasz Wasilczyk3ab105b2020-08-05 14:07:40 -070062 auto linkinfo = req.addNested(IFLA_LINKINFO);
63 req.add(IFLA_INFO_KIND, type);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070064 }
65
Tomasz Wasilczyk66fc9392020-08-03 10:22:52 -070066 nl::Socket sock(NETLINK_ROUTE);
Tomasz Wasilczyk71396052020-08-04 16:21:39 -070067 return sock.send(req) && sock.receiveAck(req);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070068}
69
70bool del(std::string dev) {
Tomasz Wasilczyk3ab105b2020-08-05 14:07:40 -070071 nl::MessageFactory<ifinfomsg> req(RTM_DELLINK, NLM_F_REQUEST | NLM_F_ACK);
72 req.add(IFLA_IFNAME, dev);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070073
Tomasz Wasilczyk66fc9392020-08-03 10:22:52 -070074 nl::Socket sock(NETLINK_ROUTE);
Tomasz Wasilczyk71396052020-08-04 16:21:39 -070075 return sock.send(req) && sock.receiveAck(req);
Tomasz Wasilczyk87329672019-07-12 11:43:00 -070076}
77
Tomasz Wasilczyk91c3a042020-06-24 15:30:10 -070078std::optional<hwaddr_t> getHwAddr(const std::string& ifname) {
79 auto ifr = ifreqs::fromName(ifname);
80 if (!ifreqs::send(SIOCGIFHWADDR, ifr)) return std::nullopt;
81
82 hwaddr_t hwaddr;
83 memcpy(hwaddr.data(), ifr.ifr_hwaddr.sa_data, hwaddr.size());
84 return hwaddr;
85}
86
Tomasz Wasilczykcaeae192020-06-25 13:36:48 -070087bool setHwAddr(const std::string& ifname, hwaddr_t hwaddr) {
88 auto ifr = ifreqs::fromName(ifname);
89
90 // fetch sa_family
91 if (!ifreqs::send(SIOCGIFHWADDR, ifr)) return false;
92
93 memcpy(ifr.ifr_hwaddr.sa_data, hwaddr.data(), hwaddr.size());
94 return ifreqs::send(SIOCSIFHWADDR, ifr);
95}
96
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -070097std::optional<bool> isUp(std::string ifname) {
98 auto ifr = ifreqs::fromName(ifname);
99 if (!ifreqs::send(SIOCGIFFLAGS, ifr)) return std::nullopt;
100 return ifr.ifr_flags & IFF_UP;
101}
102
103struct WaitState {
104 bool present;
105 bool up;
106
107 bool satisfied(WaitCondition cnd) const {
108 switch (cnd) {
109 case WaitCondition::PRESENT:
110 if (present) return true;
111 break;
112 case WaitCondition::PRESENT_AND_UP:
113 if (present && up) return true;
114 break;
115 case WaitCondition::DOWN_OR_GONE:
116 if (!present || !up) return true;
117 break;
118 }
119 return false;
120 }
121};
122
123static std::string toString(WaitCondition cnd) {
124 switch (cnd) {
125 case WaitCondition::PRESENT:
126 return "become present";
127 case WaitCondition::PRESENT_AND_UP:
128 return "come up";
129 case WaitCondition::DOWN_OR_GONE:
130 return "go down";
131 }
132}
133
134static std::string toString(const std::set<std::string>& ifnames) {
135 std::stringstream ss;
136 std::copy(ifnames.begin(), ifnames.end(), std::ostream_iterator<std::string>(ss, ","));
137 auto str = ss.str();
138 str.pop_back();
139 return str;
140}
141
142void waitFor(std::set<std::string> ifnames, WaitCondition cnd, bool allOf) {
143 nl::Socket sock(NETLINK_ROUTE, 0, RTMGRP_LINK);
144
145 using StatesMap = std::map<std::string, WaitState>;
146 StatesMap states = {};
147 for (const auto ifname : ifnames) {
148 const auto present = exists(ifname);
149 const auto up = present && isUp(ifname).value_or(false);
150 states[ifname] = {present, up};
151 }
152
153 const auto mapConditionChecker = [cnd](const StatesMap::iterator::value_type& it) {
154 return it.second.satisfied(cnd);
155 };
156 const auto isFullySatisfied = [&states, allOf, mapConditionChecker]() {
157 if (allOf) {
158 return std::all_of(states.begin(), states.end(), mapConditionChecker);
159 } else {
160 return std::any_of(states.begin(), states.end(), mapConditionChecker);
161 }
162 };
163
164 if (isFullySatisfied()) return;
165
166 LOG(DEBUG) << "Waiting for " << (allOf ? "" : "any of ") << toString(ifnames) << " to "
167 << toString(cnd);
Tomasz Wasilczyk2424b712020-08-07 11:04:26 -0700168 for (const auto rawMsg : sock) {
169 const auto msg = nl::Message<ifinfomsg>::parse(rawMsg, {RTM_NEWLINK, RTM_DELLINK});
170 if (!msg.has_value()) continue;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700171
Tomasz Wasilczyk2424b712020-08-07 11:04:26 -0700172 const auto ifname = msg->attributes.get<std::string>(IFLA_IFNAME);
173 if (ifnames.count(ifname) == 0) continue;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700174
Tomasz Wasilczyk2424b712020-08-07 11:04:26 -0700175 const bool present = (msg->header.nlmsg_type != RTM_DELLINK);
176 const bool up = present && (msg->data.ifi_flags & IFF_UP) != 0;
177 states[ifname] = {present, up};
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700178
Tomasz Wasilczyk2424b712020-08-07 11:04:26 -0700179 if (isFullySatisfied()) {
180 LOG(DEBUG) << "Finished waiting for " << (allOf ? "" : "some of ") << toString(ifnames)
181 << " to " << toString(cnd);
182 return;
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700183 }
184 }
Tomasz Wasilczyk2424b712020-08-07 11:04:26 -0700185 LOG(FATAL) << "Can't read Netlink socket";
Tomasz Wasilczykb759eca2020-08-06 16:12:40 -0700186}
187
Tomasz Wasilczyk55f21932019-12-20 09:20:24 -0800188} // namespace android::netdevice
Tomasz Wasilczyk91c3a042020-06-24 15:30:10 -0700189
190bool operator==(const android::netdevice::hwaddr_t lhs, const unsigned char rhs[ETH_ALEN]) {
191 static_assert(lhs.size() == ETH_ALEN);
192 return 0 == memcmp(lhs.data(), rhs, lhs.size());
193}