iwictl.c revision 1.3 1 1.3 christos /* $NetBSD: iwictl.c,v 1.3 2005/04/03 17:21:29 christos Exp $ */
2 1.1 skrll
3 1.1 skrll /*-
4 1.1 skrll * Copyright (c) 2004, 2005
5 1.1 skrll * Damien Bergamini <damien.bergamini (at) free.fr>. All rights reserved.
6 1.1 skrll *
7 1.1 skrll * Redistribution and use in source and binary forms, with or without
8 1.1 skrll * modification, are permitted provided that the following conditions
9 1.1 skrll * are met:
10 1.1 skrll * 1. Redistributions of source code must retain the above copyright
11 1.1 skrll * notice unmodified, this list of conditions, and the following
12 1.1 skrll * disclaimer.
13 1.1 skrll * 2. Redistributions in binary form must reproduce the above copyright
14 1.1 skrll * notice, this list of conditions and the following disclaimer in the
15 1.1 skrll * documentation and/or other materials provided with the distribution.
16 1.1 skrll *
17 1.1 skrll * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18 1.1 skrll * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 1.1 skrll * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 1.1 skrll * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21 1.1 skrll * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 1.1 skrll * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 1.1 skrll * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 1.1 skrll * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 1.1 skrll * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 1.1 skrll * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 1.1 skrll * SUCH DAMAGE.
28 1.1 skrll */
29 1.1 skrll
30 1.1 skrll #include <sys/cdefs.h>
31 1.3 christos __RCSID("$NetBSD: iwictl.c,v 1.3 2005/04/03 17:21:29 christos Exp $");
32 1.1 skrll
33 1.1 skrll #include <sys/types.h>
34 1.1 skrll #include <sys/ioctl.h>
35 1.1 skrll #include <sys/mman.h>
36 1.1 skrll #include <sys/socket.h>
37 1.1 skrll #include <sys/stat.h>
38 1.1 skrll
39 1.1 skrll #include <net/if.h>
40 1.1 skrll
41 1.1 skrll #include <err.h>
42 1.1 skrll #include <errno.h>
43 1.1 skrll #include <fcntl.h>
44 1.1 skrll #include <stdio.h>
45 1.1 skrll #include <stdlib.h>
46 1.1 skrll #include <string.h>
47 1.1 skrll #include <sysexits.h>
48 1.1 skrll #include <unistd.h>
49 1.1 skrll
50 1.1 skrll #define SIOCSLOADFW _IOW('i', 137, struct ifreq)
51 1.1 skrll #define SIOCSKILLFW _IOW('i', 138, struct ifreq)
52 1.1 skrll #define SIOCGRADIO _IOWR('i', 139, struct ifreq)
53 1.1 skrll #define SIOCGTABLE0 _IOWR('i', 140, struct ifreq)
54 1.1 skrll
55 1.1 skrll struct firmware {
56 1.1 skrll void *boot;
57 1.1 skrll int boot_size;
58 1.1 skrll void *ucode;
59 1.1 skrll int ucode_size;
60 1.1 skrll void *main;
61 1.1 skrll int main_size;
62 1.1 skrll };
63 1.1 skrll
64 1.1 skrll struct header {
65 1.1 skrll u_int32_t version;
66 1.1 skrll u_int32_t mode;
67 1.1 skrll } __attribute__((__packed__));
68 1.1 skrll
69 1.3 christos static void usage(void) __attribute__((__noreturn__));
70 1.3 christos static int do_req(const char *, unsigned long, void *);
71 1.3 christos static void mmap_file(const char *, void **, size_t *);
72 1.3 christos static void load_firmware(const char *, const char *, const char *);
73 1.3 christos static void kill_firmware(const char *);
74 1.3 christos static void get_radio_state(const char *);
75 1.3 christos static void get_statistics(const char *);
76 1.1 skrll
77 1.1 skrll int
78 1.1 skrll main(int argc, char **argv)
79 1.1 skrll {
80 1.1 skrll int ch;
81 1.3 christos char *iface = NULL, *path = NULL;
82 1.3 christos const char *mode = "bss";
83 1.1 skrll int noflag = 1, kflag = 0, rflag = 0;
84 1.1 skrll
85 1.3 christos setprogname(argv[0]);
86 1.1 skrll if (argc > 1 && argv[1][0] != '-') {
87 1.1 skrll iface = argv[1];
88 1.1 skrll optind++;
89 1.1 skrll }
90 1.1 skrll
91 1.1 skrll while ((ch = getopt(argc, argv, "d:i:km:r")) != -1) {
92 1.1 skrll if (ch != 'i')
93 1.1 skrll noflag = 0;
94 1.1 skrll
95 1.1 skrll switch (ch) {
96 1.1 skrll case 'd':
97 1.1 skrll path = optarg;
98 1.1 skrll break;
99 1.1 skrll
100 1.1 skrll case 'i':
101 1.1 skrll iface = optarg;
102 1.1 skrll break;
103 1.1 skrll
104 1.1 skrll case 'k':
105 1.1 skrll kflag = 1;
106 1.1 skrll break;
107 1.1 skrll
108 1.1 skrll case 'm':
109 1.1 skrll mode = optarg;
110 1.1 skrll break;
111 1.1 skrll
112 1.1 skrll case 'r':
113 1.1 skrll rflag = 1;
114 1.1 skrll break;
115 1.1 skrll
116 1.1 skrll default:
117 1.1 skrll usage();
118 1.1 skrll }
119 1.1 skrll }
120 1.1 skrll
121 1.1 skrll if (iface == NULL)
122 1.1 skrll usage();
123 1.1 skrll
124 1.1 skrll if (kflag && (path != NULL || rflag))
125 1.1 skrll usage();
126 1.1 skrll
127 1.1 skrll if (kflag)
128 1.1 skrll kill_firmware(iface);
129 1.1 skrll
130 1.1 skrll if (path != NULL)
131 1.1 skrll load_firmware(iface, path, mode);
132 1.1 skrll
133 1.1 skrll if (rflag)
134 1.1 skrll get_radio_state(iface);
135 1.1 skrll
136 1.1 skrll if (noflag)
137 1.1 skrll get_statistics(iface);
138 1.1 skrll
139 1.1 skrll return EX_OK;
140 1.1 skrll }
141 1.1 skrll
142 1.1 skrll static void
143 1.1 skrll usage(void)
144 1.1 skrll {
145 1.3 christos (void)fprintf(stderr, "Usage: %s iface\n"
146 1.1 skrll "\t%s iface -d path [-m bss|ibss]\n"
147 1.1 skrll "\t%s iface -k\n"
148 1.3 christos "\t%s iface -r\n", getprogname(), getprogname(), getprogname(),
149 1.3 christos getprogname());
150 1.1 skrll
151 1.1 skrll exit(EX_USAGE);
152 1.1 skrll }
153 1.1 skrll
154 1.1 skrll static int
155 1.3 christos do_req(const char *iface, unsigned long req, void *data)
156 1.1 skrll {
157 1.1 skrll int s;
158 1.1 skrll struct ifreq ifr;
159 1.3 christos int error, serrno;
160 1.1 skrll
161 1.1 skrll if ((s = socket(AF_INET, SOCK_DGRAM, 0)) == -1)
162 1.1 skrll err(EX_OSERR, "Can't create socket");
163 1.1 skrll
164 1.3 christos (void)memset(&ifr, 0, sizeof(ifr));
165 1.3 christos (void)strncpy(ifr.ifr_name, iface, sizeof(ifr.ifr_name));
166 1.1 skrll ifr.ifr_data = data;
167 1.1 skrll error = ioctl(s, req, &ifr);
168 1.3 christos serrno = errno;
169 1.1 skrll (void)close(s);
170 1.3 christos errno = serrno;
171 1.1 skrll
172 1.1 skrll return error;
173 1.1 skrll }
174 1.1 skrll
175 1.1 skrll static void
176 1.3 christos mmap_file(const char *filename, void **addr, size_t *len)
177 1.1 skrll {
178 1.1 skrll int fd;
179 1.1 skrll struct stat st;
180 1.1 skrll
181 1.1 skrll if ((fd = open(filename, O_RDONLY)) == -1)
182 1.1 skrll err(EX_OSERR, "%s", filename);
183 1.1 skrll
184 1.1 skrll if (fstat(fd, &st) == -1)
185 1.1 skrll err(EX_OSERR, "Unable to stat %s", filename);
186 1.1 skrll
187 1.3 christos *len = (size_t)st.st_size;
188 1.1 skrll
189 1.3 christos *addr = mmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, (off_t)0);
190 1.3 christos if (*addr == MAP_FAILED)
191 1.1 skrll err(EX_OSERR, "Can't map %s into memory", filename);
192 1.1 skrll
193 1.3 christos *(char **)addr += sizeof(struct header);
194 1.3 christos *len -= sizeof(struct header);
195 1.1 skrll
196 1.1 skrll (void)close(fd);
197 1.1 skrll }
198 1.1 skrll
199 1.1 skrll static void
200 1.3 christos load_firmware(const char *iface, const char *path, const char *mode)
201 1.1 skrll {
202 1.1 skrll char filename[FILENAME_MAX];
203 1.1 skrll struct firmware fw;
204 1.1 skrll
205 1.3 christos (void)snprintf(filename, sizeof(filename), "%s/iwi-boot.fw", path);
206 1.1 skrll mmap_file(filename, &fw.boot, &fw.boot_size);
207 1.1 skrll
208 1.3 christos (void)snprintf(filename, sizeof(filename), "%s/iwi-ucode-%s.fw", path,
209 1.1 skrll mode);
210 1.1 skrll mmap_file(filename, &fw.ucode, &fw.ucode_size);
211 1.1 skrll
212 1.3 christos (void)snprintf(filename, sizeof(filename), "%s/iwi-%s.fw", path, mode);
213 1.1 skrll mmap_file(filename, &fw.main, &fw.main_size);
214 1.1 skrll
215 1.1 skrll if (do_req(iface, SIOCSLOADFW, &fw) == -1)
216 1.1 skrll err(EX_OSERR, "Can't load firmware to driver");
217 1.1 skrll }
218 1.1 skrll
219 1.1 skrll static void
220 1.3 christos kill_firmware(const char *iface)
221 1.1 skrll {
222 1.1 skrll if (do_req(iface, SIOCSKILLFW, NULL) == -1)
223 1.1 skrll err(EX_OSERR, "Can't kill firmware");
224 1.1 skrll }
225 1.1 skrll
226 1.1 skrll static void
227 1.3 christos get_radio_state(const char *iface)
228 1.1 skrll {
229 1.1 skrll int radio;
230 1.1 skrll
231 1.1 skrll if (do_req(iface, SIOCGRADIO, &radio) == -1)
232 1.1 skrll err(EX_OSERR, "Can't read radio");
233 1.1 skrll
234 1.1 skrll (void)printf("Radio is %s\n", radio ? "ON" : "OFF");
235 1.1 skrll }
236 1.1 skrll
237 1.1 skrll struct statistic {
238 1.1 skrll int index;
239 1.1 skrll const char *desc;
240 1.1 skrll };
241 1.1 skrll
242 1.1 skrll static const struct statistic tbl[] = {
243 1.1 skrll { 1, "Current transmission rate" },
244 1.1 skrll { 2, "Fragmentation threshold" },
245 1.1 skrll { 3, "RTS threshold" },
246 1.1 skrll { 4, "Number of frames submitted for transfer" },
247 1.1 skrll { 5, "Number of frames transmitted" },
248 1.1 skrll { 6, "Number of unicast frames transmitted" },
249 1.1 skrll { 7, "Number of unicast 802.11b frames transmitted at 1Mb/s" },
250 1.1 skrll { 8, "Number of unicast 802.11b frames transmitted at 2Mb/s" },
251 1.1 skrll { 9, "Number of unicast 802.11b frames transmitted at 5.5Mb/s" },
252 1.1 skrll { 10, "Number of unicast 802.11b frames transmitted at 11Mb/s" },
253 1.1 skrll
254 1.1 skrll { 19, "Number of unicast 802.11g frames transmitted at 1Mb/s" },
255 1.1 skrll { 20, "Number of unicast 802.11g frames transmitted at 2Mb/s" },
256 1.1 skrll { 21, "Number of unicast 802.11g frames transmitted at 5.5Mb/s" },
257 1.1 skrll { 22, "Number of unicast 802.11g frames transmitted at 6Mb/s" },
258 1.1 skrll { 23, "Number of unicast 802.11g frames transmitted at 9Mb/s" },
259 1.1 skrll { 24, "Number of unicast 802.11g frames transmitted at 11Mb/s" },
260 1.1 skrll { 25, "Number of unicast 802.11g frames transmitted at 12Mb/s" },
261 1.1 skrll { 26, "Number of unicast 802.11g frames transmitted at 18Mb/s" },
262 1.1 skrll { 27, "Number of unicast 802.11g frames transmitted at 24Mb/s" },
263 1.1 skrll { 28, "Number of unicast 802.11g frames transmitted at 36Mb/s" },
264 1.1 skrll { 29, "Number of unicast 802.11g frames transmitted at 48Mb/s" },
265 1.1 skrll { 30, "Number of unicast 802.11g frames transmitted at 54Mb/s" },
266 1.1 skrll { 31, "Number of multicast frames transmitted" },
267 1.1 skrll { 32, "Number of multicast 802.11b frames transmitted at 1Mb/s" },
268 1.1 skrll { 33, "Number of multicast 802.11b frames transmitted at 2Mb/s" },
269 1.1 skrll { 34, "Number of multicast 802.11b frames transmitted at 5.5Mb/s" },
270 1.1 skrll { 35, "Number of multicast 802.11b frames transmitted at 11Mb/s" },
271 1.1 skrll
272 1.1 skrll { 44, "Number of multicast 802.11g frames transmitted at 1Mb/s" },
273 1.1 skrll { 45, "Number of multicast 802.11g frames transmitted at 2Mb/s" },
274 1.1 skrll { 46, "Number of multicast 802.11g frames transmitted at 5.5Mb/s" },
275 1.1 skrll { 47, "Number of multicast 802.11g frames transmitted at 6Mb/s" },
276 1.1 skrll { 48, "Number of multicast 802.11g frames transmitted at 9Mb/s" },
277 1.1 skrll { 49, "Number of multicast 802.11g frames transmitted at 11Mb/s" },
278 1.1 skrll { 50, "Number of multicast 802.11g frames transmitted at 12Mb/s" },
279 1.1 skrll { 51, "Number of multicast 802.11g frames transmitted at 18Mb/s" },
280 1.1 skrll { 52, "Number of multicast 802.11g frames transmitted at 24Mb/s" },
281 1.1 skrll { 53, "Number of multicast 802.11g frames transmitted at 36Mb/s" },
282 1.1 skrll { 54, "Number of multicast 802.11g frames transmitted at 48Mb/s" },
283 1.1 skrll { 55, "Number of multicast 802.11g frames transmitted at 54Mb/s" },
284 1.1 skrll { 56, "Number of transmission retries" },
285 1.1 skrll { 57, "Number of transmission failures" },
286 1.1 skrll { 58, "Number of frames with a bad CRC received" },
287 1.1 skrll
288 1.1 skrll { 61, "Number of full scans" },
289 1.1 skrll { 62, "Number of partial scans" },
290 1.1 skrll
291 1.1 skrll { 64, "Number of bytes transmitted" },
292 1.1 skrll { 65, "Current RSSI" },
293 1.1 skrll { 66, "Number of beacons received" },
294 1.1 skrll { 67, "Number of beacons missed" },
295 1.1 skrll
296 1.1 skrll { 0, NULL }
297 1.1 skrll };
298 1.1 skrll
299 1.1 skrll static void
300 1.3 christos get_statistics(const char *iface)
301 1.1 skrll {
302 1.1 skrll static u_int32_t stats[256];
303 1.3 christos const struct statistic *st;
304 1.1 skrll
305 1.1 skrll if (do_req(iface, SIOCGTABLE0, stats) == -1)
306 1.1 skrll err(EX_OSERR, "Can't read statistics");
307 1.1 skrll
308 1.3 christos for (st = tbl; st->index != 0; st++)
309 1.3 christos (void)printf("%-60s[%u]\n", st->desc, stats[st->index]);
310 1.1 skrll }
311