blob: 0d6cf5416beec1144b2a3d8aaba1bb987d7c87ed [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 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation.
8 *
9 * Alternatively, this software may be distributed under the terms of BSD
10 * license.
11 *
12 * See README and COPYING for more details.
13 */
14
15#include "utils/includes.h"
16#include <sys/ioctl.h>
17#include <net/if.h>
18#include <net/if_arp.h>
19
20#include "utils/common.h"
21#include "linux_ioctl.h"
22
23
24int linux_set_iface_flags(int sock, const char *ifname, int dev_up)
25{
26 struct ifreq ifr;
27
28 if (sock < 0)
29 return -1;
30
31 os_memset(&ifr, 0, sizeof(ifr));
32 os_strlcpy(ifr.ifr_name, ifname, IFNAMSIZ);
33
34 if (ioctl(sock, SIOCGIFFLAGS, &ifr) != 0) {
35 wpa_printf(MSG_ERROR, "Could not read interface %s flags: %s",
36 ifname, strerror(errno));
37 return -1;
38 }
39
40 if (dev_up) {
41 if (ifr.ifr_flags & IFF_UP)
42 return 0;
43 ifr.ifr_flags |= IFF_UP;
44 } else {
45 if (!(ifr.ifr_flags & IFF_UP))
46 return 0;
47 ifr.ifr_flags &= ~IFF_UP;
48 }
49
50 if (ioctl(sock, SIOCSIFFLAGS, &ifr) != 0) {
51 wpa_printf(MSG_ERROR, "Could not set interface %s flags: %s",
52 ifname, strerror(errno));
53 return -1;
54 }
55
56 return 0;
57}
58
59
60int linux_get_ifhwaddr(int sock, const char *ifname, u8 *addr)
61{
62 struct ifreq ifr;
63
64 os_memset(&ifr, 0, sizeof(ifr));
65 os_strlcpy(ifr.ifr_name, ifname, IFNAMSIZ);
66 if (ioctl(sock, SIOCGIFHWADDR, &ifr)) {
67 wpa_printf(MSG_ERROR, "Could not get interface %s hwaddr: %s",
68 ifname, strerror(errno));
69 return -1;
70 }
71
72 if (ifr.ifr_hwaddr.sa_family != ARPHRD_ETHER) {
73 wpa_printf(MSG_ERROR, "%s: Invalid HW-addr family 0x%04x",
74 ifname, ifr.ifr_hwaddr.sa_family);
75 return -1;
76 }
77 os_memcpy(addr, ifr.ifr_hwaddr.sa_data, ETH_ALEN);
78
79 return 0;
80}
81
82
83int linux_set_ifhwaddr(int sock, const char *ifname, const u8 *addr)
84{
85 struct ifreq ifr;
86
87 os_memset(&ifr, 0, sizeof(ifr));
88 os_strlcpy(ifr.ifr_name, ifname, IFNAMSIZ);
89 os_memcpy(ifr.ifr_hwaddr.sa_data, addr, ETH_ALEN);
90 ifr.ifr_hwaddr.sa_family = ARPHRD_ETHER;
91
92 if (ioctl(sock, SIOCSIFHWADDR, &ifr)) {
93 wpa_printf(MSG_DEBUG, "Could not set interface %s hwaddr: %s",
94 ifname, strerror(errno));
95 return -1;
96 }
97
98 return 0;
99}
100
101
102#ifndef SIOCBRADDBR
103#define SIOCBRADDBR 0x89a0
104#endif
105#ifndef SIOCBRDELBR
106#define SIOCBRDELBR 0x89a1
107#endif
108#ifndef SIOCBRADDIF
109#define SIOCBRADDIF 0x89a2
110#endif
111#ifndef SIOCBRDELIF
112#define SIOCBRDELIF 0x89a3
113#endif
114
115
116int linux_br_add(int sock, const char *brname)
117{
118 if (ioctl(sock, SIOCBRADDBR, brname) < 0) {
119 wpa_printf(MSG_DEBUG, "Could not add bridge %s: %s",
120 brname, strerror(errno));
121 return -1;
122 }
123
124 return 0;
125}
126
127
128int linux_br_del(int sock, const char *brname)
129{
130 if (ioctl(sock, SIOCBRDELBR, brname) < 0) {
131 wpa_printf(MSG_DEBUG, "Could not remove bridge %s: %s",
132 brname, strerror(errno));
133 return -1;
134 }
135
136 return 0;
137}
138
139
140int linux_br_add_if(int sock, const char *brname, const char *ifname)
141{
142 struct ifreq ifr;
143 int ifindex;
144
145 ifindex = if_nametoindex(ifname);
146 if (ifindex == 0)
147 return -1;
148
149 os_memset(&ifr, 0, sizeof(ifr));
150 os_strlcpy(ifr.ifr_name, brname, IFNAMSIZ);
151 ifr.ifr_ifindex = ifindex;
152 if (ioctl(sock, SIOCBRADDIF, &ifr) < 0) {
153 wpa_printf(MSG_DEBUG, "Could not add interface %s into bridge "
154 "%s: %s", ifname, brname, strerror(errno));
155 return -1;
156 }
157
158 return 0;
159}
160
161
162int linux_br_del_if(int sock, const char *brname, const char *ifname)
163{
164 struct ifreq ifr;
165 int ifindex;
166
167 ifindex = if_nametoindex(ifname);
168 if (ifindex == 0)
169 return -1;
170
171 os_memset(&ifr, 0, sizeof(ifr));
172 os_strlcpy(ifr.ifr_name, brname, IFNAMSIZ);
173 ifr.ifr_ifindex = ifindex;
174 if (ioctl(sock, SIOCBRDELIF, &ifr) < 0) {
175 wpa_printf(MSG_DEBUG, "Could not remove interface %s from "
176 "bridge %s: %s", ifname, brname, strerror(errno));
177 return -1;
178 }
179
180 return 0;
181}
182
183
184int linux_br_get(char *brname, const char *ifname)
185{
186 char path[128], brlink[128], *pos;
187 os_snprintf(path, sizeof(path), "/sys/class/net/%s/brport/bridge",
188 ifname);
189 os_memset(brlink, 0, sizeof(brlink));
190 if (readlink(path, brlink, sizeof(brlink) - 1) < 0)
191 return -1;
192 pos = os_strrchr(brlink, '/');
193 if (pos == NULL)
194 return -1;
195 pos++;
196 os_strlcpy(brname, pos, IFNAMSIZ);
197 return 0;
198}