1 /*
2 * Copyright (C) 2016 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 "wifi_system/interface_tool.h"
18
19 #include <net/if.h>
20 #include <net/if_arp.h>
21 #include <netinet/in.h>
22 #include <sys/socket.h>
23
24 #include <linux/if_bridge.h>
25 #include <string.h>
26
27 #include <linux/ethtool.h>
28 /* We need linux/if.h for flags like IFF_UP. Sadly, it forward declares
29 struct sockaddr and must be included after sys/socket.h. */
30 #include <linux/if.h>
31
32 #include <android-base/logging.h>
33 #include <android-base/unique_fd.h>
34
35 #define IFNAMSIZ 16
36
37 namespace android {
38 namespace wifi_system {
39 namespace {
40
41 const char kWlan0InterfaceName[] = "wlan0";
42
GetIfState(const char * if_name,int sock,struct ifreq * ifr)43 bool GetIfState(const char* if_name, int sock, struct ifreq* ifr) {
44 memset(ifr, 0, sizeof(*ifr));
45 if (strlcpy(ifr->ifr_name, if_name, sizeof(ifr->ifr_name)) >=
46 sizeof(ifr->ifr_name)) {
47 LOG(ERROR) << "Interface name is too long: " << if_name;
48 return false;
49 }
50
51 if (TEMP_FAILURE_RETRY(ioctl(sock, SIOCGIFFLAGS, ifr)) != 0) {
52 LOG(ERROR) << "Could not read interface state for " << if_name
53 << " (" << strerror(errno) << ")";
54 return false;
55 }
56
57 return true;
58 }
59
60 } // namespace
61
GetUpState(const char * if_name)62 bool InterfaceTool::GetUpState(const char* if_name) {
63 base::unique_fd sock(socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0));
64 if (sock.get() < 0) {
65 LOG(ERROR) << "Failed to open socket to set up/down state ("
66 << strerror(errno) << ")";
67 return false;
68 }
69
70 struct ifreq ifr;
71 if (!GetIfState(if_name, sock.get(), &ifr)) {
72 return false; // logging done internally
73 }
74
75 return ifr.ifr_flags & IFF_UP;
76 }
77
SetUpState(const char * if_name,bool request_up)78 bool InterfaceTool::SetUpState(const char* if_name, bool request_up) {
79 base::unique_fd sock(socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0));
80 if (sock.get() < 0) {
81 LOG(ERROR) << "Failed to open socket to set up/down state ("
82 << strerror(errno) << ")";
83 return false;
84 }
85
86 struct ifreq ifr;
87 if (!GetIfState(if_name, sock.get(), &ifr)) {
88 return false; // logging done internally
89 }
90
91 const bool currently_up = ifr.ifr_flags & IFF_UP;
92 if (currently_up == request_up) {
93 return true;
94 }
95
96 if (request_up) {
97 ifr.ifr_flags |= IFF_UP;
98 } else {
99 ifr.ifr_flags &= ~IFF_UP;
100 }
101
102 if (TEMP_FAILURE_RETRY(ioctl(sock.get(), SIOCSIFFLAGS, &ifr)) != 0) {
103 LOG(ERROR) << "Could not set interface flags for " << if_name
104 << " (" << strerror(errno) << ")";
105 return false;
106 }
107
108 return true;
109 }
110
SetWifiUpState(bool request_up)111 bool InterfaceTool::SetWifiUpState(bool request_up) {
112 return SetUpState(kWlan0InterfaceName, request_up);
113 }
114
SetMacAddress(const char * if_name,const std::array<uint8_t,ETH_ALEN> & new_address)115 bool InterfaceTool::SetMacAddress(const char* if_name,
116 const std::array<uint8_t, ETH_ALEN>& new_address) {
117 struct ifreq ifr;
118 static_assert(ETH_ALEN <= sizeof(ifr.ifr_hwaddr.sa_data),
119 "new address is too long");
120
121 base::unique_fd sock(socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0));
122 if (sock.get() < 0) {
123 LOG(ERROR) << "Failed to open socket to set MAC address ("
124 << strerror(errno) << ")";
125 return false;
126 }
127
128 if (!GetIfState(if_name, sock.get(), &ifr)) {
129 return false; // logging done internally
130 }
131
132 memset(&ifr.ifr_hwaddr, 0, sizeof(ifr.ifr_hwaddr));
133 ifr.ifr_hwaddr.sa_family = ARPHRD_ETHER;
134 memcpy(ifr.ifr_hwaddr.sa_data, new_address.data(), new_address.size());
135 if (TEMP_FAILURE_RETRY(ioctl(sock.get(), SIOCSIFHWADDR, &ifr)) != 0) {
136 LOG(ERROR) << "Could not set interface MAC address for " << if_name
137 << " (" << strerror(errno) << ")";
138 return false;
139 }
140
141 return true;
142 }
143
GetFactoryMacAddress(const char * if_name)144 std::array<uint8_t, ETH_ALEN> InterfaceTool::GetFactoryMacAddress(const char* if_name) {
145 std::array<uint8_t, ETH_ALEN> paddr = {};
146 struct ifreq ifr;
147 union {
148 struct ethtool_perm_addr epaddr;
149 // Add ETH_ALEN bytes to size of structure.
150 uint8_t raw[sizeof(ethtool_perm_addr) + ETH_ALEN];
151 } epaddr_buf;
152 struct ethtool_perm_addr* epaddr = &epaddr_buf.epaddr;
153
154 base::unique_fd sock(socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0));
155 if (sock.get() < 0) {
156 LOG(ERROR) << "Failed to open socket to get factory MAC address ("
157 << strerror(errno) << ")";
158 return paddr;
159 }
160
161 if (!GetIfState(if_name, sock.get(), &ifr)) {
162 return paddr; // logging done internally
163 }
164
165 epaddr->cmd = ETHTOOL_GPERMADDR;
166 epaddr->size = ETH_ALEN;
167 ifr.ifr_data = epaddr;
168
169 if (TEMP_FAILURE_RETRY(ioctl(sock.get(), SIOCETHTOOL, &ifr)) != 0) {
170 LOG(ERROR) << "Could not get factory address MAC for " << if_name
171 << " (" << strerror(errno) << ")";
172 } else if (epaddr->size == ETH_ALEN) {
173 memcpy(paddr.data(), epaddr->data, ETH_ALEN);
174 }
175 return paddr;
176 }
177
createBridge(const std::string & br_name)178 bool InterfaceTool::createBridge(const std::string& br_name) {
179 base::unique_fd sock(socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0));
180
181 if (TEMP_FAILURE_RETRY(ioctl(sock, SIOCBRADDBR, br_name.c_str())) != 0) {
182 LOG(ERROR) << "Could not add bridge " << br_name.c_str()
183 << " (" << strerror(errno) << ")";
184 return false;
185 }
186
187 return true;
188 }
189
deleteBridge(const std::string & br_name)190 bool InterfaceTool::deleteBridge(const std::string& br_name) {
191 base::unique_fd sock(socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0));
192
193 if (TEMP_FAILURE_RETRY(ioctl(sock, SIOCBRDELBR, br_name.c_str())) != 0) {
194 LOG(ERROR) << "Could not remove bridge " << br_name.c_str()
195 << " (" << strerror(errno) << ")";
196 return false;
197 }
198 return true;
199 }
200
addIfaceToBridge(const std::string & br_name,const std::string & if_name)201 bool InterfaceTool::addIfaceToBridge(const std::string& br_name, const std::string& if_name) {
202 struct ifreq ifr;
203 memset(&ifr, 0, sizeof(ifr));
204
205 ifr.ifr_ifindex = if_nametoindex(if_name.c_str());
206 if (ifr.ifr_ifindex == 0) {
207 LOG(ERROR) << "Interface is not exist: " << if_name.c_str();
208 return false;
209 }
210 strlcpy(ifr.ifr_name, br_name.c_str(), IFNAMSIZ);
211
212 base::unique_fd sock(socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0));
213 if (TEMP_FAILURE_RETRY(ioctl(sock, SIOCBRADDIF, &ifr)) != 0) {
214 LOG(ERROR) << "Could not add interface " << if_name.c_str()
215 << " into bridge " << ifr.ifr_name
216 << " (" << strerror(errno) << ")";
217 return false;
218 }
219 return true;
220 }
221
removeIfaceFromBridge(const std::string & br_name,const std::string & if_name)222 bool InterfaceTool::removeIfaceFromBridge(const std::string& br_name, const std::string& if_name) {
223 struct ifreq ifr;
224 memset(&ifr, 0, sizeof(ifr));
225
226 ifr.ifr_ifindex = if_nametoindex(if_name.c_str());
227 if (ifr.ifr_ifindex == 0) {
228 LOG(ERROR) << "Interface is not exist: " << if_name.c_str();
229 return false;
230 }
231 strlcpy(ifr.ifr_name, br_name.c_str(), IFNAMSIZ);
232
233 base::unique_fd sock(socket(PF_INET, SOCK_DGRAM | SOCK_CLOEXEC, 0));
234 if (TEMP_FAILURE_RETRY(ioctl(sock, SIOCBRDELIF, &ifr)) != 0) {
235 LOG(ERROR) << "Could not remove interface " << if_name.c_str()
236 << " from bridge " << ifr.ifr_name
237 << " (" << strerror(errno) << ")";
238 return false;
239 }
240
241 return true;
242 }
243 } // namespace wifi_system
244 } // namespace android
245