blob: 29abc0c59e8383ec17240b57a6852869982238ba [file] [log] [blame]
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07001/*
2 * Linux ioctl helper functions for driver wrappers
3 * Copyright (c) 2002-2010, Jouni Malinen <j@w1.fi>
4 *
Dmitry Shmidtc5ec7f52012-03-06 16:33:24 -08005 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -07007 */
8
9#include "utils/includes.h"
10#include <sys/ioctl.h>
11#include <net/if.h>
12#include <net/if_arp.h>
13
14#include "utils/common.h"
Hai Shalom74f70d42019-02-11 14:42:39 -080015#include "common/linux_bridge.h"
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070016#include "linux_ioctl.h"
17
18
19int linux_set_iface_flags(int sock, const char *ifname, int dev_up)
20{
21 struct ifreq ifr;
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080022 int ret;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070023
24 if (sock < 0)
25 return -1;
26
27 os_memset(&ifr, 0, sizeof(ifr));
28 os_strlcpy(ifr.ifr_name, ifname, IFNAMSIZ);
29
30 if (ioctl(sock, SIOCGIFFLAGS, &ifr) != 0) {
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080031 ret = errno ? -errno : -999;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070032 wpa_printf(MSG_ERROR, "Could not read interface %s flags: %s",
33 ifname, strerror(errno));
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080034 return ret;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070035 }
36
37 if (dev_up) {
38 if (ifr.ifr_flags & IFF_UP)
39 return 0;
40 ifr.ifr_flags |= IFF_UP;
41 } else {
42 if (!(ifr.ifr_flags & IFF_UP))
43 return 0;
44 ifr.ifr_flags &= ~IFF_UP;
45 }
46
47 if (ioctl(sock, SIOCSIFFLAGS, &ifr) != 0) {
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080048 ret = errno ? -errno : -999;
Dmitry Shmidt61d9df32012-08-29 16:22:06 -070049 wpa_printf(MSG_ERROR, "Could not set interface %s flags (%s): "
50 "%s",
51 ifname, dev_up ? "UP" : "DOWN", strerror(errno));
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080052 return ret;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070053 }
54
55 return 0;
56}
57
58
Dmitry Shmidt1f69aa52012-01-24 16:10:04 -080059int linux_iface_up(int sock, const char *ifname)
60{
61 struct ifreq ifr;
62 int ret;
63
64 if (sock < 0)
65 return -1;
66
67 os_memset(&ifr, 0, sizeof(ifr));
68 os_strlcpy(ifr.ifr_name, ifname, IFNAMSIZ);
69
70 if (ioctl(sock, SIOCGIFFLAGS, &ifr) != 0) {
71 ret = errno ? -errno : -999;
72 wpa_printf(MSG_ERROR, "Could not read interface %s flags: %s",
73 ifname, strerror(errno));
74 return ret;
75 }
76
77 return !!(ifr.ifr_flags & IFF_UP);
78}
79
80
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -070081int linux_get_ifhwaddr(int sock, const char *ifname, u8 *addr)
82{
83 struct ifreq ifr;
84
85 os_memset(&ifr, 0, sizeof(ifr));
86 os_strlcpy(ifr.ifr_name, ifname, IFNAMSIZ);
87 if (ioctl(sock, SIOCGIFHWADDR, &ifr)) {
88 wpa_printf(MSG_ERROR, "Could not get interface %s hwaddr: %s",
89 ifname, strerror(errno));
90 return -1;
91 }
92
93 if (ifr.ifr_hwaddr.sa_family != ARPHRD_ETHER) {
94 wpa_printf(MSG_ERROR, "%s: Invalid HW-addr family 0x%04x",
95 ifname, ifr.ifr_hwaddr.sa_family);
96 return -1;
97 }
98 os_memcpy(addr, ifr.ifr_hwaddr.sa_data, ETH_ALEN);
99
100 return 0;
101}
102
103
104int linux_set_ifhwaddr(int sock, const char *ifname, const u8 *addr)
105{
106 struct ifreq ifr;
107
108 os_memset(&ifr, 0, sizeof(ifr));
109 os_strlcpy(ifr.ifr_name, ifname, IFNAMSIZ);
110 os_memcpy(ifr.ifr_hwaddr.sa_data, addr, ETH_ALEN);
111 ifr.ifr_hwaddr.sa_family = ARPHRD_ETHER;
112
113 if (ioctl(sock, SIOCSIFHWADDR, &ifr)) {
114 wpa_printf(MSG_DEBUG, "Could not set interface %s hwaddr: %s",
115 ifname, strerror(errno));
116 return -1;
117 }
118
119 return 0;
120}
121
122
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700123int linux_br_add(int sock, const char *brname)
124{
125 if (ioctl(sock, SIOCBRADDBR, brname) < 0) {
Hai Shalom74f70d42019-02-11 14:42:39 -0800126 int saved_errno = errno;
127
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700128 wpa_printf(MSG_DEBUG, "Could not add bridge %s: %s",
129 brname, strerror(errno));
Hai Shalom74f70d42019-02-11 14:42:39 -0800130 errno = saved_errno;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700131 return -1;
132 }
133
134 return 0;
135}
136
137
138int linux_br_del(int sock, const char *brname)
139{
140 if (ioctl(sock, SIOCBRDELBR, brname) < 0) {
141 wpa_printf(MSG_DEBUG, "Could not remove bridge %s: %s",
142 brname, strerror(errno));
143 return -1;
144 }
145
146 return 0;
147}
148
149
150int linux_br_add_if(int sock, const char *brname, const char *ifname)
151{
152 struct ifreq ifr;
153 int ifindex;
154
155 ifindex = if_nametoindex(ifname);
156 if (ifindex == 0)
157 return -1;
158
159 os_memset(&ifr, 0, sizeof(ifr));
160 os_strlcpy(ifr.ifr_name, brname, IFNAMSIZ);
161 ifr.ifr_ifindex = ifindex;
162 if (ioctl(sock, SIOCBRADDIF, &ifr) < 0) {
Hai Shalom74f70d42019-02-11 14:42:39 -0800163 int saved_errno = errno;
Sunil Ravib0ac25f2024-07-12 01:42:03 +0000164 char in_br[IFNAMSIZ];
Hai Shalom74f70d42019-02-11 14:42:39 -0800165
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700166 wpa_printf(MSG_DEBUG, "Could not add interface %s into bridge "
167 "%s: %s", ifname, brname, strerror(errno));
Hai Shalom74f70d42019-02-11 14:42:39 -0800168 errno = saved_errno;
Sunil Ravib0ac25f2024-07-12 01:42:03 +0000169
170 /* If ioctl() returns EBUSY when adding an interface into the
171 * bridge, the interface might have already been added by an
172 * external operation, so check whether the interface is
173 * currently on the right bridge and ignore the error if it is.
174 */
175 if (errno != EBUSY || linux_br_get(in_br, ifname) != 0 ||
176 os_strcmp(in_br, brname) != 0)
177 return -1;
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700178 }
179
180 return 0;
181}
182
183
184int linux_br_del_if(int sock, const char *brname, const char *ifname)
185{
186 struct ifreq ifr;
187 int ifindex;
188
189 ifindex = if_nametoindex(ifname);
190 if (ifindex == 0)
191 return -1;
192
193 os_memset(&ifr, 0, sizeof(ifr));
194 os_strlcpy(ifr.ifr_name, brname, IFNAMSIZ);
195 ifr.ifr_ifindex = ifindex;
196 if (ioctl(sock, SIOCBRDELIF, &ifr) < 0) {
197 wpa_printf(MSG_DEBUG, "Could not remove interface %s from "
198 "bridge %s: %s", ifname, brname, strerror(errno));
199 return -1;
200 }
201
202 return 0;
203}
204
205
206int linux_br_get(char *brname, const char *ifname)
207{
208 char path[128], brlink[128], *pos;
Dmitry Shmidt68d0e3e2013-10-28 17:59:21 -0700209 ssize_t res;
210
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700211 os_snprintf(path, sizeof(path), "/sys/class/net/%s/brport/bridge",
212 ifname);
Dmitry Shmidt68d0e3e2013-10-28 17:59:21 -0700213 res = readlink(path, brlink, sizeof(brlink));
214 if (res < 0 || (size_t) res >= sizeof(brlink))
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700215 return -1;
Dmitry Shmidt68d0e3e2013-10-28 17:59:21 -0700216 brlink[res] = '\0';
Dmitry Shmidt8d520ff2011-05-09 14:06:53 -0700217 pos = os_strrchr(brlink, '/');
218 if (pos == NULL)
219 return -1;
220 pos++;
221 os_strlcpy(brname, pos, IFNAMSIZ);
222 return 0;
223}
Dmitry Shmidtd80a4012015-11-05 16:35:40 -0800224
225
226int linux_master_get(char *master_ifname, const char *ifname)
227{
228 char buf[128], masterlink[128], *pos;
229 ssize_t res;
230
231 /* check whether there is a master */
232 os_snprintf(buf, sizeof(buf), "/sys/class/net/%s/master", ifname);
233
234 res = readlink(buf, masterlink, sizeof(masterlink));
235 if (res < 0 || (size_t) res >= sizeof(masterlink))
236 return -1;
237
238 masterlink[res] = '\0';
239
240 pos = os_strrchr(masterlink, '/');
241 if (pos == NULL)
242 return -1;
243 pos++;
244 os_strlcpy(master_ifname, pos, IFNAMSIZ);
245 return 0;
246}