2 * Copyright 1997, 1998, 1999
3 * Bill Paul <wpaul@ee.columbia.edu>. All rights reserved.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. All advertising materials mentioning features or use of this software
14 * must display the following acknowledgement:
15 * This product includes software developed by Bill Paul.
16 * 4. Neither the name of the author nor the names of any co-contributors
17 * may be used to endorse or promote products derived from this software
18 * without specific prior written permission.
20 * THIS SOFTWARE IS PROVIDED BY Bill Paul AND CONTRIBUTORS ``AS IS'' AND
21 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED. IN NO EVENT SHALL Bill Paul OR THE VOICES IN HIS HEAD
24 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
25 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
26 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
27 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
28 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
29 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
30 * THE POSSIBILITY OF SUCH DAMAGE.
35 static const char copyright
[] = "@(#) Copyright (c) 1997, 1998, 1999\
36 Bill Paul. All rights reserved.";
39 #include <sys/cdefs.h>
40 __FBSDID("$FreeBSD$");
42 #include <sys/types.h>
43 #include <sys/socket.h>
44 #include <sys/ioctl.h>
46 #include <arpa/inet.h>
49 #include <net/ethernet.h>
51 #include <dev/an/if_aironet_ieee.h>
62 static int an_getval(const char *, struct an_req
*);
63 static void an_setval(const char *, struct an_req
*);
64 static void an_printwords(const u_int16_t
*, int);
65 static void an_printspeeds(const u_int8_t
*, int);
66 static void an_printbool(int);
67 static void an_printhex(const char *, int);
68 static void an_printstr(char *, int);
69 static void an_dumpstatus(const char *);
70 static void an_dumpstats(const char *);
71 static void an_dumpconfig(const char *);
72 static void an_dumpcaps(const char *);
73 static void an_dumpssid(const char *);
74 static void an_dumpap(const char *);
75 static void an_setconfig(const char *, int, void *);
76 static void an_setssid(const char *, int, void *);
77 static void an_setap(const char *, int, void *);
78 static void an_setspeed(const char *, int, void *);
79 static void an_readkeyinfo(const char *);
81 static void an_zerocache(const char *);
82 static void an_readcache(const char *);
84 static int an_hex2int(char);
85 static void an_str2key(const char *, struct an_ltv_key
*);
86 static void an_setkeys(const char *, const char *, int);
87 static void an_enable_tx_key(const char *, const char *);
88 static void an_enable_leap_mode(const char *, const char *);
89 static void an_dumprssimap(const char *);
90 static void usage(const char *);
92 #define ACT_DUMPSTATS 1
93 #define ACT_DUMPCONFIG 2
94 #define ACT_DUMPSTATUS 3
95 #define ACT_DUMPCAPS 4
96 #define ACT_DUMPSSID 5
99 #define ACT_SET_OPMODE 7
100 #define ACT_SET_SSID 8
101 #define ACT_SET_FREQ 11
102 #define ACT_SET_AP1 12
103 #define ACT_SET_AP2 13
104 #define ACT_SET_AP3 14
105 #define ACT_SET_AP4 15
106 #define ACT_SET_DRIVERNAME 16
107 #define ACT_SET_SCANMODE 17
108 #define ACT_SET_TXRATE 18
109 #define ACT_SET_RTS_THRESH 19
110 #define ACT_SET_PWRSAVE 20
111 #define ACT_SET_DIVERSITY_RX 21
112 #define ACT_SET_DIVERSITY_TX 22
113 #define ACT_SET_RTS_RETRYLIM 23
114 #define ACT_SET_WAKE_DURATION 24
115 #define ACT_SET_BEACON_PERIOD 25
116 #define ACT_SET_TXPWR 26
117 #define ACT_SET_FRAG_THRESH 27
118 #define ACT_SET_NETJOIN 28
119 #define ACT_SET_MYNAME 29
120 #define ACT_SET_MAC 30
122 #define ACT_DUMPCACHE 31
123 #define ACT_ZEROCACHE 32
125 #define ACT_ENABLE_WEP 33
126 #define ACT_SET_KEY_TYPE 34
127 #define ACT_SET_KEYS 35
128 #define ACT_ENABLE_TX_KEY 36
129 #define ACT_SET_MONITOR_MODE 37
130 #define ACT_SET_LEAP_MODE 38
132 #define ACT_DUMPRSSIMAP 39
135 an_getval(const char *iface
, struct an_req
*areq
)
140 bzero(&ifr
, sizeof(ifr
));
142 strlcpy(ifr
.ifr_name
, iface
, sizeof(ifr
.ifr_name
));
143 ifr
.ifr_data
= (caddr_t
)areq
;
145 s
= socket(AF_INET
, SOCK_DGRAM
, 0);
150 if (ioctl(s
, SIOCGAIRONET
, &ifr
) == -1) {
152 err(1, "SIOCGAIRONET");
161 an_setval(const char *iface
, struct an_req
*areq
)
166 bzero(&ifr
, sizeof(ifr
));
168 strlcpy(ifr
.ifr_name
, iface
, sizeof(ifr
.ifr_name
));
169 ifr
.ifr_data
= (caddr_t
)areq
;
171 s
= socket(AF_INET
, SOCK_DGRAM
, 0);
176 if (ioctl(s
, SIOCSAIRONET
, &ifr
) == -1)
177 err(1, "SIOCSAIRONET");
185 an_printstr(char *str
, int len
)
189 for (i
= 0; i
< len
- 1; i
++) {
194 printf("[ %.*s ]", len
, str
);
198 an_printwords(const u_int16_t
*w
, int len
)
203 for (i
= 0; i
< len
; i
++)
209 an_printspeeds(const u_int8_t
*w
, int len
)
214 for (i
= 0; i
< len
&& w
[i
]; i
++)
215 printf("%2.1fMbps ", w
[i
] * 0.500);
220 an_printbool(int val
)
229 an_printhex(const char *ptr
, int len
)
234 for (i
= 0; i
< len
; i
++) {
235 printf("%02x", ptr
[i
] & 0xFF);
246 an_dumpstatus(const char *iface
)
248 struct an_ltv_status
*sts
;
250 struct an_ltv_rssi_map an_rssimap
;
251 int rssimap_valid
= 0;
254 * Try to get RSSI to percent and dBM table
257 an_rssimap
.an_len
= sizeof(an_rssimap
);
258 an_rssimap
.an_type
= AN_RID_RSSI_MAP
;
259 rssimap_valid
= an_getval(iface
, (struct an_req
*)&an_rssimap
);
262 printf("RSSI table:\t\t[ present ]\n");
264 printf("RSSI table:\t\t[ not available ]\n");
266 areq
.an_len
= sizeof(areq
);
267 areq
.an_type
= AN_RID_STATUS
;
269 an_getval(iface
, &areq
);
271 sts
= (struct an_ltv_status
*)&areq
;
273 printf("MAC address:\t\t");
274 an_printhex((char *)&sts
->an_macaddr
, ETHER_ADDR_LEN
);
275 printf("\nOperating mode:\t\t[ ");
276 if (sts
->an_opmode
& AN_STATUS_OPMODE_CONFIGURED
)
277 printf("configured ");
278 if (sts
->an_opmode
& AN_STATUS_OPMODE_MAC_ENABLED
)
280 if (sts
->an_opmode
& AN_STATUS_OPMODE_RX_ENABLED
)
282 if (sts
->an_opmode
& AN_STATUS_OPMODE_IN_SYNC
)
284 if (sts
->an_opmode
& AN_STATUS_OPMODE_ASSOCIATED
)
285 printf("associated ");
286 if (sts
->an_opmode
& AN_STATUS_OPMODE_LEAP
)
288 if (sts
->an_opmode
& AN_STATUS_OPMODE_ERROR
)
291 printf("Error code:\t\t");
292 an_printhex((char *)&sts
->an_errcode
, 1);
294 printf("\nSignal strength:\t[ %u%% ]",
295 an_rssimap
.an_entries
[
296 sts
->an_normalized_strength
].an_rss_pct
);
298 printf("\nSignal strength:\t[ %u%% ]",
299 sts
->an_normalized_strength
);
300 printf("\nAverage Noise:\t\t[ %u%% ]", sts
->an_avg_noise_prev_min_pc
);
302 printf("\nSignal quality:\t\t[ %u%% ]",
303 an_rssimap
.an_entries
[
304 sts
->an_cur_signal_quality
].an_rss_pct
);
306 printf("\nSignal quality:\t\t[ %u ]",
307 sts
->an_cur_signal_quality
);
308 printf("\nMax Noise:\t\t[ %u%% ]", sts
->an_max_noise_prev_min_pc
);
310 * XXX: This uses the old definition of the rate field (units of
311 * 500kbps). Technically the new definition is that this field
312 * contains arbitrary values, but no devices which need this
313 * support exist and the IEEE seems to intend to use the old
314 * definition until they get something big so we'll keep using
315 * it as well because this will work with new cards with
318 printf("\nCurrent TX rate:\t[ %u%s ]", sts
->an_current_tx_rate
/ 2,
319 (sts
->an_current_tx_rate
% 2) ? ".5" : "");
320 printf("\nCurrent SSID:\t\t");
321 an_printstr((char *)&sts
->an_ssid
, sts
->an_ssidlen
);
322 printf("\nCurrent AP name:\t");
323 an_printstr((char *)&sts
->an_ap_name
, 16);
324 printf("\nCurrent BSSID:\t\t");
325 an_printhex((char *)&sts
->an_cur_bssid
, ETHER_ADDR_LEN
);
326 printf("\nBeacon period:\t\t");
327 an_printwords(&sts
->an_beacon_period
, 1);
328 printf("\nDTIM period:\t\t");
329 an_printwords(&sts
->an_dtim_period
, 1);
330 printf("\nATIM duration:\t\t");
331 an_printwords(&sts
->an_atim_duration
, 1);
332 printf("\nHOP period:\t\t");
333 an_printwords(&sts
->an_hop_period
, 1);
334 printf("\nChannel set:\t\t");
335 an_printwords(&sts
->an_channel_set
, 1);
336 printf("\nCurrent channel:\t");
337 an_printwords(&sts
->an_cur_channel
, 1);
338 printf("\nHops to backbone:\t");
339 an_printwords(&sts
->an_hops_to_backbone
, 1);
340 printf("\nTotal AP load:\t\t");
341 an_printwords(&sts
->an_ap_total_load
, 1);
342 printf("\nOur generated load:\t");
343 an_printwords(&sts
->an_our_generated_load
, 1);
344 printf("\nAccumulated ARL:\t");
345 an_printwords(&sts
->an_accumulated_arl
, 1);
351 an_dumpcaps(const char *iface
)
353 struct an_ltv_caps
*caps
;
357 areq
.an_len
= sizeof(areq
);
358 areq
.an_type
= AN_RID_CAPABILITIES
;
360 an_getval(iface
, &areq
);
362 caps
= (struct an_ltv_caps
*)&areq
;
364 printf("OUI:\t\t\t");
365 an_printhex((char *)&caps
->an_oui
, 3);
366 printf("\nProduct number:\t\t");
367 an_printwords(&caps
->an_prodnum
, 1);
368 printf("\nManufacturer name:\t");
369 an_printstr((char *)&caps
->an_manufname
, 32);
370 printf("\nProduce name:\t\t");
371 an_printstr((char *)&caps
->an_prodname
, 16);
372 printf("\nFirmware version:\t");
373 an_printstr((char *)&caps
->an_prodvers
, 1);
374 printf("\nOEM MAC address:\t");
375 an_printhex((char *)&caps
->an_oemaddr
, ETHER_ADDR_LEN
);
376 printf("\nAironet MAC address:\t");
377 an_printhex((char *)&caps
->an_aironetaddr
, ETHER_ADDR_LEN
);
378 printf("\nRadio type:\t\t[ ");
379 if (caps
->an_radiotype
& AN_RADIOTYPE_80211_FH
)
381 else if (caps
->an_radiotype
& AN_RADIOTYPE_80211_DS
)
383 else if (caps
->an_radiotype
& AN_RADIOTYPE_LM2000_DS
)
386 printf("unknown (%x)", caps
->an_radiotype
);
388 printf("\nRegulatory domain:\t");
389 an_printwords(&caps
->an_regdomain
, 1);
390 printf("\nAssigned CallID:\t");
391 an_printhex((char *)&caps
->an_callid
, 6);
392 printf("\nSupported speeds:\t");
393 an_printspeeds(caps
->an_rates
, 8);
394 printf("\nRX Diversity:\t\t[ ");
395 if (caps
->an_rx_diversity
== AN_DIVERSITY_FACTORY_DEFAULT
)
396 printf("factory default");
397 else if (caps
->an_rx_diversity
== AN_DIVERSITY_ANTENNA_1_ONLY
)
398 printf("antenna 1 only");
399 else if (caps
->an_rx_diversity
== AN_DIVERSITY_ANTENNA_2_ONLY
)
400 printf("antenna 2 only");
401 else if (caps
->an_rx_diversity
== AN_DIVERSITY_ANTENNA_1_AND_2
)
402 printf("antenna 1 and 2");
404 printf("\nTX Diversity:\t\t[ ");
405 if (caps
->an_tx_diversity
== AN_DIVERSITY_FACTORY_DEFAULT
)
406 printf("factory default");
407 else if (caps
->an_tx_diversity
== AN_DIVERSITY_ANTENNA_1_ONLY
)
408 printf("antenna 1 only");
409 else if (caps
->an_tx_diversity
== AN_DIVERSITY_ANTENNA_2_ONLY
)
410 printf("antenna 2 only");
411 else if (caps
->an_tx_diversity
== AN_DIVERSITY_ANTENNA_1_AND_2
)
412 printf("antenna 1 and 2");
414 printf("\nSupported power levels:\t");
415 an_printwords(caps
->an_tx_powerlevels
, 8);
416 printf("\nHardware revision:\t");
417 tmp
= ntohs(caps
->an_hwrev
);
418 an_printhex((char *)&tmp
, 2);
419 printf("\nSoftware revision:\t");
420 tmp
= ntohs(caps
->an_fwrev
);
421 an_printhex((char *)&tmp
, 2);
422 printf("\nSoftware subrevision:\t");
423 tmp
= ntohs(caps
->an_fwsubrev
);
424 an_printhex((char *)&tmp
, 2);
425 printf("\nInterface revision:\t");
426 tmp
= ntohs(caps
->an_ifacerev
);
427 an_printhex((char *)&tmp
, 2);
428 printf("\nBootblock revision:\t");
429 tmp
= ntohs(caps
->an_bootblockrev
);
430 an_printhex((char *)&tmp
, 2);
436 an_dumpstats(const char *iface
)
438 struct an_ltv_stats
*stats
;
441 areq
.an_len
= sizeof(areq
);
442 areq
.an_type
= AN_RID_32BITS_CUM
;
444 an_getval(iface
, &areq
);
446 stats
= (struct an_ltv_stats
*)((uint16_t *)&areq
- 1);
448 printf("RX overruns:\t\t\t\t\t[ %u ]\n", stats
->an_rx_overruns
);
449 printf("RX PLCP CSUM errors:\t\t\t\t[ %u ]\n",
450 stats
->an_rx_plcp_csum_errs
);
451 printf("RX PLCP format errors:\t\t\t\t[ %u ]\n",
452 stats
->an_rx_plcp_format_errs
);
453 printf("RX PLCP length errors:\t\t\t\t[ %u ]\n",
454 stats
->an_rx_plcp_len_errs
);
455 printf("RX MAC CRC errors:\t\t\t\t[ %u ]\n",
456 stats
->an_rx_mac_crc_errs
);
457 printf("RX MAC CRC OK:\t\t\t\t\t[ %u ]\n",
458 stats
->an_rx_mac_crc_ok
);
459 printf("RX WEP errors:\t\t\t\t\t[ %u ]\n",
460 stats
->an_rx_wep_errs
);
461 printf("RX WEP OK:\t\t\t\t\t[ %u ]\n",
462 stats
->an_rx_wep_ok
);
463 printf("Long retries:\t\t\t\t\t[ %u ]\n",
464 stats
->an_retry_long
);
465 printf("Short retries:\t\t\t\t\t[ %u ]\n",
466 stats
->an_retry_short
);
467 printf("Retries exhausted:\t\t\t\t[ %u ]\n",
468 stats
->an_retry_max
);
469 printf("Bad ACK:\t\t\t\t\t[ %u ]\n",
471 printf("Bad CTS:\t\t\t\t\t[ %u ]\n",
473 printf("RX good ACKs:\t\t\t\t\t[ %u ]\n",
474 stats
->an_rx_ack_ok
);
475 printf("RX good CTSs:\t\t\t\t\t[ %u ]\n",
476 stats
->an_rx_cts_ok
);
477 printf("TX good ACKs:\t\t\t\t\t[ %u ]\n",
478 stats
->an_tx_ack_ok
);
479 printf("TX good RTSs:\t\t\t\t\t[ %u ]\n",
480 stats
->an_tx_rts_ok
);
481 printf("TX good CTSs:\t\t\t\t\t[ %u ]\n",
482 stats
->an_tx_cts_ok
);
483 printf("LMAC multicasts transmitted:\t\t\t[ %u ]\n",
484 stats
->an_tx_lmac_mcasts
);
485 printf("LMAC broadcasts transmitted:\t\t\t[ %u ]\n",
486 stats
->an_tx_lmac_bcasts
);
487 printf("LMAC unicast frags transmitted:\t\t\t[ %u ]\n",
488 stats
->an_tx_lmac_ucast_frags
);
489 printf("LMAC unicasts transmitted:\t\t\t[ %u ]\n",
490 stats
->an_tx_lmac_ucasts
);
491 printf("Beacons transmitted:\t\t\t\t[ %u ]\n",
492 stats
->an_tx_beacons
);
493 printf("Beacons received:\t\t\t\t[ %u ]\n",
494 stats
->an_rx_beacons
);
495 printf("Single transmit collisions:\t\t\t[ %u ]\n",
496 stats
->an_tx_single_cols
);
497 printf("Multiple transmit collisions:\t\t\t[ %u ]\n",
498 stats
->an_tx_multi_cols
);
499 printf("Transmits without deferrals:\t\t\t[ %u ]\n",
500 stats
->an_tx_defers_no
);
501 printf("Transmits deferred due to protocol:\t\t[ %u ]\n",
502 stats
->an_tx_defers_prot
);
503 printf("Transmits deferred due to energy detect:\t\t[ %u ]\n",
504 stats
->an_tx_defers_energy
);
505 printf("RX duplicate frames/frags:\t\t\t[ %u ]\n",
507 printf("RX partial frames:\t\t\t\t[ %u ]\n",
508 stats
->an_rx_partial
);
509 printf("TX max lifetime exceeded:\t\t\t[ %u ]\n",
510 stats
->an_tx_too_old
);
511 printf("RX max lifetime exceeded:\t\t\t[ %u ]\n",
512 stats
->an_tx_too_old
);
513 printf("Sync lost due to too many missed beacons:\t[ %u ]\n",
514 stats
->an_lostsync_missed_beacons
);
515 printf("Sync lost due to ARL exceeded:\t\t\t[ %u ]\n",
516 stats
->an_lostsync_arl_exceeded
);
517 printf("Sync lost due to deauthentication:\t\t[ %u ]\n",
518 stats
->an_lostsync_deauthed
);
519 printf("Sync lost due to disassociation:\t\t[ %u ]\n",
520 stats
->an_lostsync_disassociated
);
521 printf("Sync lost due to excess change in TSF timing:\t[ %u ]\n",
522 stats
->an_lostsync_tsf_timing
);
523 printf("Host transmitted multicasts:\t\t\t[ %u ]\n",
524 stats
->an_tx_host_mcasts
);
525 printf("Host transmitted broadcasts:\t\t\t[ %u ]\n",
526 stats
->an_tx_host_bcasts
);
527 printf("Host transmitted unicasts:\t\t\t[ %u ]\n",
528 stats
->an_tx_host_ucasts
);
529 printf("Host transmission failures:\t\t\t[ %u ]\n",
530 stats
->an_tx_host_failed
);
531 printf("Host received multicasts:\t\t\t[ %u ]\n",
532 stats
->an_rx_host_mcasts
);
533 printf("Host received broadcasts:\t\t\t[ %u ]\n",
534 stats
->an_rx_host_bcasts
);
535 printf("Host received unicasts:\t\t\t\t[ %u ]\n",
536 stats
->an_rx_host_ucasts
);
537 printf("Host receive discards:\t\t\t\t[ %u ]\n",
538 stats
->an_rx_host_discarded
);
539 printf("HMAC transmitted multicasts:\t\t\t[ %u ]\n",
540 stats
->an_tx_hmac_mcasts
);
541 printf("HMAC transmitted broadcasts:\t\t\t[ %u ]\n",
542 stats
->an_tx_hmac_bcasts
);
543 printf("HMAC transmitted unicasts:\t\t\t[ %u ]\n",
544 stats
->an_tx_hmac_ucasts
);
545 printf("HMAC transmissions failed:\t\t\t[ %u ]\n",
546 stats
->an_tx_hmac_failed
);
547 printf("HMAC received multicasts:\t\t\t[ %u ]\n",
548 stats
->an_rx_hmac_mcasts
);
549 printf("HMAC received broadcasts:\t\t\t[ %u ]\n",
550 stats
->an_rx_hmac_bcasts
);
551 printf("HMAC received unicasts:\t\t\t\t[ %u ]\n",
552 stats
->an_rx_hmac_ucasts
);
553 printf("HMAC receive discards:\t\t\t\t[ %u ]\n",
554 stats
->an_rx_hmac_discarded
);
555 printf("HMAC transmits accepted:\t\t\t[ %u ]\n",
556 stats
->an_tx_hmac_accepted
);
557 printf("SSID mismatches:\t\t\t\t[ %u ]\n",
558 stats
->an_ssid_mismatches
);
559 printf("Access point mismatches:\t\t\t[ %u ]\n",
560 stats
->an_ap_mismatches
);
561 printf("Speed mismatches:\t\t\t\t[ %u ]\n",
562 stats
->an_rates_mismatches
);
563 printf("Authentication rejects:\t\t\t\t[ %u ]\n",
564 stats
->an_auth_rejects
);
565 printf("Authentication timeouts:\t\t\t[ %u ]\n",
566 stats
->an_auth_timeouts
);
567 printf("Association rejects:\t\t\t\t[ %u ]\n",
568 stats
->an_assoc_rejects
);
569 printf("Association timeouts:\t\t\t\t[ %u ]\n",
570 stats
->an_assoc_timeouts
);
571 printf("Management frames received:\t\t\t[ %u ]\n",
572 stats
->an_rx_mgmt_pkts
);
573 printf("Management frames transmitted:\t\t\t[ %u ]\n",
574 stats
->an_tx_mgmt_pkts
);
575 printf("Refresh frames received:\t\t\t[ %u ]\n",
576 stats
->an_rx_refresh_pkts
),
577 printf("Refresh frames transmitted:\t\t\t[ %u ]\n",
578 stats
->an_tx_refresh_pkts
),
579 printf("Poll frames received:\t\t\t\t[ %u ]\n",
580 stats
->an_rx_poll_pkts
);
581 printf("Poll frames transmitted:\t\t\t[ %u ]\n",
582 stats
->an_tx_poll_pkts
);
583 printf("Host requested sync losses:\t\t\t[ %u ]\n",
584 stats
->an_lostsync_hostreq
);
585 printf("Host transmitted bytes:\t\t\t\t[ %u ]\n",
586 stats
->an_host_tx_bytes
);
587 printf("Host received bytes:\t\t\t\t[ %u ]\n",
588 stats
->an_host_rx_bytes
);
589 printf("Uptime in microseconds:\t\t\t\t[ %u ]\n",
590 stats
->an_uptime_usecs
);
591 printf("Uptime in seconds:\t\t\t\t[ %u ]\n",
592 stats
->an_uptime_secs
);
593 printf("Sync lost due to better AP:\t\t\t[ %u ]\n",
594 stats
->an_lostsync_better_ap
);
598 an_dumpap(const char *iface
)
600 struct an_ltv_aplist
*ap
;
603 areq
.an_len
= sizeof(areq
);
604 areq
.an_type
= AN_RID_APLIST
;
606 an_getval(iface
, &areq
);
608 ap
= (struct an_ltv_aplist
*)&areq
;
609 printf("Access point 1:\t\t\t");
610 an_printhex((char *)&ap
->an_ap1
, ETHER_ADDR_LEN
);
611 printf("\nAccess point 2:\t\t\t");
612 an_printhex((char *)&ap
->an_ap2
, ETHER_ADDR_LEN
);
613 printf("\nAccess point 3:\t\t\t");
614 an_printhex((char *)&ap
->an_ap3
, ETHER_ADDR_LEN
);
615 printf("\nAccess point 4:\t\t\t");
616 an_printhex((char *)&ap
->an_ap4
, ETHER_ADDR_LEN
);
623 an_dumpssid(const char *iface
)
625 struct an_ltv_ssidlist_new
*ssid
;
629 areq
.an_len
= sizeof(areq
);
630 areq
.an_type
= AN_RID_SSIDLIST
;
632 an_getval(iface
, &areq
);
634 max
= (areq
.an_len
- 4) / sizeof(struct an_ltv_ssid_entry
);
635 if ( max
> MAX_SSIDS
) {
636 printf("Too many SSIDs only printing %d of %d\n",
640 ssid
= (struct an_ltv_ssidlist_new
*)&areq
;
641 for (i
= 0; i
< max
; i
++)
642 printf("SSID %2d:\t\t\t[ %.*s ]\n", i
+ 1,
643 ssid
->an_entry
[i
].an_len
,
644 ssid
->an_entry
[i
].an_ssid
);
650 an_dumpconfig(const char *iface
)
652 struct an_ltv_genconfig
*cfg
;
654 unsigned char diversity
;
656 areq
.an_len
= sizeof(areq
);
657 areq
.an_type
= AN_RID_ACTUALCFG
;
659 an_getval(iface
, &areq
);
661 cfg
= (struct an_ltv_genconfig
*)&areq
;
663 printf("Operating mode:\t\t\t\t[ ");
664 if ((cfg
->an_opmode
& 0x7) == AN_OPMODE_IBSS_ADHOC
)
666 if ((cfg
->an_opmode
& 0x7) == AN_OPMODE_INFRASTRUCTURE_STATION
)
667 printf("infrastructure");
668 if ((cfg
->an_opmode
& 0x7) == AN_OPMODE_AP
)
669 printf("access point");
670 if ((cfg
->an_opmode
& 0x7) == AN_OPMODE_AP_REPEATER
)
671 printf("access point repeater");
673 printf("\nReceive mode:\t\t\t\t[ ");
674 if ((cfg
->an_rxmode
& 0x7) == AN_RXMODE_BC_MC_ADDR
)
675 printf("broadcast/multicast/unicast");
676 if ((cfg
->an_rxmode
& 0x7) == AN_RXMODE_BC_ADDR
)
677 printf("broadcast/unicast");
678 if ((cfg
->an_rxmode
& 0x7) == AN_RXMODE_ADDR
)
680 if ((cfg
->an_rxmode
& 0x7) == AN_RXMODE_80211_MONITOR_CURBSS
)
681 printf("802.11 monitor, current BSSID");
682 if ((cfg
->an_rxmode
& 0x7) == AN_RXMODE_80211_MONITOR_ANYBSS
)
683 printf("802.11 monitor, any BSSID");
684 if ((cfg
->an_rxmode
& 0x7) == AN_RXMODE_LAN_MONITOR_CURBSS
)
685 printf("LAN monitor, current BSSID");
687 printf("\nFragment threshold:\t\t\t");
688 an_printwords(&cfg
->an_fragthresh
, 1);
689 printf("\nRTS threshold:\t\t\t\t");
690 an_printwords(&cfg
->an_rtsthresh
, 1);
691 printf("\nMAC address:\t\t\t\t");
692 an_printhex((char *)&cfg
->an_macaddr
, ETHER_ADDR_LEN
);
693 printf("\nSupported rates:\t\t\t");
694 an_printspeeds(cfg
->an_rates
, 8);
695 printf("\nShort retry limit:\t\t\t");
696 an_printwords(&cfg
->an_shortretry_limit
, 1);
697 printf("\nLong retry limit:\t\t\t");
698 an_printwords(&cfg
->an_longretry_limit
, 1);
699 printf("\nTX MSDU lifetime:\t\t\t");
700 an_printwords(&cfg
->an_tx_msdu_lifetime
, 1);
701 printf("\nRX MSDU lifetime:\t\t\t");
702 an_printwords(&cfg
->an_rx_msdu_lifetime
, 1);
703 printf("\nStationary:\t\t\t\t");
704 an_printbool(cfg
->an_stationary
);
705 printf("\nOrdering:\t\t\t\t");
706 an_printbool(cfg
->an_ordering
);
707 printf("\nDevice type:\t\t\t\t[ ");
708 if (cfg
->an_devtype
== AN_DEVTYPE_PC4500
)
710 else if (cfg
->an_devtype
== AN_DEVTYPE_PC4800
)
713 printf("unknown (%x)", cfg
->an_devtype
);
715 printf("\nScanning mode:\t\t\t\t[ ");
716 if (cfg
->an_scanmode
== AN_SCANMODE_ACTIVE
)
718 if (cfg
->an_scanmode
== AN_SCANMODE_PASSIVE
)
720 if (cfg
->an_scanmode
== AN_SCANMODE_AIRONET_ACTIVE
)
721 printf("Aironet active");
723 printf("\nProbe delay:\t\t\t\t");
724 an_printwords(&cfg
->an_probedelay
, 1);
725 printf("\nProbe energy timeout:\t\t\t");
726 an_printwords(&cfg
->an_probe_energy_timeout
, 1);
727 printf("\nProbe response timeout:\t\t\t");
728 an_printwords(&cfg
->an_probe_response_timeout
, 1);
729 printf("\nBeacon listen timeout:\t\t\t");
730 an_printwords(&cfg
->an_beacon_listen_timeout
, 1);
731 printf("\nIBSS join network timeout:\t\t");
732 an_printwords(&cfg
->an_ibss_join_net_timeout
, 1);
733 printf("\nAuthentication timeout:\t\t\t");
734 an_printwords(&cfg
->an_auth_timeout
, 1);
735 printf("\nWEP enabled:\t\t\t\t[ ");
736 if (cfg
->an_authtype
& AN_AUTHTYPE_PRIVACY_IN_USE
)
738 if (cfg
->an_authtype
& AN_AUTHTYPE_LEAP
)
740 else if (cfg
->an_authtype
& AN_AUTHTYPE_ALLOW_UNENCRYPTED
)
741 printf("mixed cell");
748 printf("\nAuthentication type:\t\t\t[ ");
749 if ((cfg
->an_authtype
& AN_AUTHTYPE_MASK
) == AN_AUTHTYPE_NONE
)
751 if ((cfg
->an_authtype
& AN_AUTHTYPE_MASK
) == AN_AUTHTYPE_OPEN
)
753 if ((cfg
->an_authtype
& AN_AUTHTYPE_MASK
) == AN_AUTHTYPE_SHAREDKEY
)
754 printf("shared key");
756 printf("\nAssociation timeout:\t\t\t");
757 an_printwords(&cfg
->an_assoc_timeout
, 1);
758 printf("\nSpecified AP association timeout:\t");
759 an_printwords(&cfg
->an_specified_ap_timeout
, 1);
760 printf("\nOffline scan interval:\t\t\t");
761 an_printwords(&cfg
->an_offline_scan_interval
, 1);
762 printf("\nOffline scan duration:\t\t\t");
763 an_printwords(&cfg
->an_offline_scan_duration
, 1);
764 printf("\nLink loss delay:\t\t\t");
765 an_printwords(&cfg
->an_link_loss_delay
, 1);
766 printf("\nMax beacon loss time:\t\t\t");
767 an_printwords(&cfg
->an_max_beacon_lost_time
, 1);
768 printf("\nRefresh interval:\t\t\t");
769 an_printwords(&cfg
->an_refresh_interval
, 1);
770 printf("\nPower save mode:\t\t\t[ ");
771 if (cfg
->an_psave_mode
== AN_PSAVE_NONE
)
773 if (cfg
->an_psave_mode
== AN_PSAVE_CAM
)
774 printf("constantly awake mode");
775 if (cfg
->an_psave_mode
== AN_PSAVE_PSP
)
777 if (cfg
->an_psave_mode
== AN_PSAVE_PSP_CAM
)
778 printf("PSP-CAM (fast PSP)");
780 printf("\nSleep through DTIMs:\t\t\t");
781 an_printbool(cfg
->an_sleep_for_dtims
);
782 printf("\nPower save listen interval:\t\t");
783 an_printwords(&cfg
->an_listen_interval
, 1);
784 printf("\nPower save fast listen interval:\t");
785 an_printwords(&cfg
->an_fast_listen_interval
, 1);
786 printf("\nPower save listen decay:\t\t");
787 an_printwords(&cfg
->an_listen_decay
, 1);
788 printf("\nPower save fast listen decay:\t\t");
789 an_printwords(&cfg
->an_fast_listen_decay
, 1);
790 printf("\nAP/ad-hoc Beacon period:\t\t");
791 an_printwords(&cfg
->an_beacon_period
, 1);
792 printf("\nAP/ad-hoc ATIM duration:\t\t");
793 an_printwords(&cfg
->an_atim_duration
, 1);
794 printf("\nAP/ad-hoc current channel:\t\t");
795 an_printwords(&cfg
->an_ds_channel
, 1);
796 printf("\nAP/ad-hoc DTIM period:\t\t\t");
797 an_printwords(&cfg
->an_dtim_period
, 1);
798 printf("\nRadio type:\t\t\t\t[ ");
799 if (cfg
->an_radiotype
& AN_RADIOTYPE_80211_FH
)
801 else if (cfg
->an_radiotype
& AN_RADIOTYPE_80211_DS
)
803 else if (cfg
->an_radiotype
& AN_RADIOTYPE_LM2000_DS
)
806 printf("unknown (%x)", cfg
->an_radiotype
);
808 printf("\nRX Diversity:\t\t\t\t[ ");
809 diversity
= cfg
->an_diversity
& 0xFF;
810 if (diversity
== AN_DIVERSITY_FACTORY_DEFAULT
)
811 printf("factory default");
812 else if (diversity
== AN_DIVERSITY_ANTENNA_1_ONLY
)
813 printf("antenna 1 only");
814 else if (diversity
== AN_DIVERSITY_ANTENNA_2_ONLY
)
815 printf("antenna 2 only");
816 else if (diversity
== AN_DIVERSITY_ANTENNA_1_AND_2
)
817 printf("antenna 1 and 2");
819 printf("\nTX Diversity:\t\t\t\t[ ");
820 diversity
= (cfg
->an_diversity
>> 8) & 0xFF;
821 if (diversity
== AN_DIVERSITY_FACTORY_DEFAULT
)
822 printf("factory default");
823 else if (diversity
== AN_DIVERSITY_ANTENNA_1_ONLY
)
824 printf("antenna 1 only");
825 else if (diversity
== AN_DIVERSITY_ANTENNA_2_ONLY
)
826 printf("antenna 2 only");
827 else if (diversity
== AN_DIVERSITY_ANTENNA_1_AND_2
)
828 printf("antenna 1 and 2");
830 printf("\nTransmit power level:\t\t\t");
831 an_printwords(&cfg
->an_tx_power
, 1);
832 printf("\nRSS threshold:\t\t\t\t");
833 an_printwords(&cfg
->an_rss_thresh
, 1);
834 printf("\nNode name:\t\t\t\t");
835 an_printstr((char *)&cfg
->an_nodename
, 16);
836 printf("\nARL threshold:\t\t\t\t");
837 an_printwords(&cfg
->an_arl_thresh
, 1);
838 printf("\nARL decay:\t\t\t\t");
839 an_printwords(&cfg
->an_arl_decay
, 1);
840 printf("\nARL delay:\t\t\t\t");
841 an_printwords(&cfg
->an_arl_delay
, 1);
842 printf("\nConfiguration:\t\t\t\t[ ");
843 if (cfg
->an_home_product
& AN_HOME_NETWORK
)
844 printf("Home Configuration");
846 printf("Enterprise Configuration");
851 an_readkeyinfo(iface
);
855 an_dumprssimap(const char *iface
)
857 struct an_ltv_rssi_map
*rssi
;
861 areq
.an_len
= sizeof(areq
);
862 areq
.an_type
= AN_RID_RSSI_MAP
;
864 an_getval(iface
, &areq
);
866 rssi
= (struct an_ltv_rssi_map
*)&areq
;
868 printf("idx\tpct\t dBm\n");
870 for (i
= 0; i
< 0xFF; i
++) {
872 * negate the dBm value: it's the only way the power
875 printf("%3d\t%3d\t%4d\n", i
,
876 rssi
->an_entries
[i
].an_rss_pct
,
877 - rssi
->an_entries
[i
].an_rss_dbm
);
884 fprintf(stderr
, "usage: %s -i iface -A (show specified APs)\n", p
);
885 fprintf(stderr
, "\t%s -i iface -N (show specified SSIDss)\n", p
);
886 fprintf(stderr
, "\t%s -i iface -S (show NIC status)\n", p
);
887 fprintf(stderr
, "\t%s -i iface -I (show NIC capabilities)\n", p
);
888 fprintf(stderr
, "\t%s -i iface -T (show stats counters)\n", p
);
889 fprintf(stderr
, "\t%s -i iface -C (show current config)\n", p
);
890 fprintf(stderr
, "\t%s -i iface -R (show RSSI map)\n", p
);
891 fprintf(stderr
, "\t%s -i iface -t 0-4 (set TX speed)\n", p
);
892 fprintf(stderr
, "\t%s -i iface -s 0-3 (set power save mode)\n", p
);
893 fprintf(stderr
, "\t%s -i iface [-v 1-4] -a AP (specify AP)\n", p
);
894 fprintf(stderr
, "\t%s -i iface -b val (set beacon period)\n", p
);
895 fprintf(stderr
, "\t%s -i iface [-v 0|1] -d val (set diversity)\n", p
);
896 fprintf(stderr
, "\t%s -i iface -j val (set netjoin timeout)\n", p
);
897 fprintf(stderr
, "\t%s -i iface -e 0-4 (enable transmit key)\n", p
);
898 fprintf(stderr
, "\t%s -i iface [-v 0-8] -k key (set key)\n", p
);
899 fprintf(stderr
, "\t%s -i iface -K 0-2 (no auth/open/shared secret)\n", p
);
900 fprintf(stderr
, "\t%s -i iface -W 0-2 (no WEP/full WEP/mixed cell)\n", p
);
901 fprintf(stderr
, "\t%s -i iface -l val (set station name)\n", p
);
902 fprintf(stderr
, "\t%s -i iface -m val (set MAC address)\n", p
);
903 fprintf(stderr
, "\t%s -i iface [-v 1-3] -n SSID "
904 "(specify SSID)\n", p
);
905 fprintf(stderr
, "\t%s -i iface -o 0|1 (set operating mode)\n", p
);
906 fprintf(stderr
, "\t%s -i iface -c val (set ad-hoc channel)\n", p
);
907 fprintf(stderr
, "\t%s -i iface -f val (set frag threshold)\n", p
);
908 fprintf(stderr
, "\t%s -i iface -r val (set RTS threshold)\n", p
);
909 fprintf(stderr
, "\t%s -i iface -M 0-15 (set monitor mode)\n", p
);
910 fprintf(stderr
, "\t%s -i iface -L user (enter LEAP authentication mode)\n", p
);
912 fprintf(stderr
, "\t%s -i iface -Q print signal quality cache\n", p
);
913 fprintf(stderr
, "\t%s -i iface -Z zero out signal cache\n", p
);
916 fprintf(stderr
, "\t%s -h (display this message)\n", p
);
922 an_setconfig(const char *iface
, int act
, void *arg
)
924 struct an_ltv_genconfig
*cfg
;
925 struct an_ltv_caps
*caps
;
927 struct an_req areq_caps
;
928 u_int16_t diversity
= 0;
929 struct ether_addr
*addr
;
932 areq
.an_len
= sizeof(areq
);
933 areq
.an_type
= AN_RID_GENCONFIG
;
934 an_getval(iface
, &areq
);
935 cfg
= (struct an_ltv_genconfig
*)&areq
;
937 areq_caps
.an_len
= sizeof(areq
);
938 areq_caps
.an_type
= AN_RID_CAPABILITIES
;
939 an_getval(iface
, &areq_caps
);
940 caps
= (struct an_ltv_caps
*)&areq_caps
;
944 cfg
->an_opmode
= atoi(arg
);
947 cfg
->an_ds_channel
= atoi(arg
);
949 case ACT_SET_PWRSAVE
:
950 cfg
->an_psave_mode
= atoi(arg
);
952 case ACT_SET_SCANMODE
:
953 cfg
->an_scanmode
= atoi(arg
);
955 case ACT_SET_DIVERSITY_RX
:
956 case ACT_SET_DIVERSITY_TX
:
959 diversity
= AN_DIVERSITY_FACTORY_DEFAULT
;
962 diversity
= AN_DIVERSITY_ANTENNA_1_ONLY
;
965 diversity
= AN_DIVERSITY_ANTENNA_2_ONLY
;
968 diversity
= AN_DIVERSITY_ANTENNA_1_AND_2
;
971 errx(1, "bad diversity setting: %u", diversity
);
974 if (act
== ACT_SET_DIVERSITY_RX
) {
975 cfg
->an_diversity
&= 0xFF00;
976 cfg
->an_diversity
|= diversity
;
978 cfg
->an_diversity
&= 0x00FF;
979 cfg
->an_diversity
|= (diversity
<< 8);
983 for (i
= 0; i
< 8; i
++) {
984 if (caps
->an_tx_powerlevels
[i
] == atoi(arg
))
988 errx(1, "unsupported power level: %dmW", atoi(arg
));
990 cfg
->an_tx_power
= atoi(arg
);
992 case ACT_SET_RTS_THRESH
:
993 cfg
->an_rtsthresh
= atoi(arg
);
995 case ACT_SET_RTS_RETRYLIM
:
996 cfg
->an_shortretry_limit
=
997 cfg
->an_longretry_limit
= atoi(arg
);
999 case ACT_SET_BEACON_PERIOD
:
1000 cfg
->an_beacon_period
= atoi(arg
);
1002 case ACT_SET_WAKE_DURATION
:
1003 cfg
->an_atim_duration
= atoi(arg
);
1005 case ACT_SET_FRAG_THRESH
:
1006 cfg
->an_fragthresh
= atoi(arg
);
1008 case ACT_SET_NETJOIN
:
1009 cfg
->an_ibss_join_net_timeout
= atoi(arg
);
1011 case ACT_SET_MYNAME
:
1012 bzero(cfg
->an_nodename
, 16);
1013 strncpy((char *)&cfg
->an_nodename
, optarg
, 16);
1016 addr
= ether_aton((char *)arg
);
1019 errx(1, "badly formatted address");
1020 bzero(cfg
->an_macaddr
, ETHER_ADDR_LEN
);
1021 bcopy(addr
, &cfg
->an_macaddr
, ETHER_ADDR_LEN
);
1023 case ACT_ENABLE_WEP
:
1024 switch (atoi (arg
)) {
1027 cfg
->an_authtype
&= ~(AN_AUTHTYPE_PRIVACY_IN_USE
1028 | AN_AUTHTYPE_ALLOW_UNENCRYPTED
1029 | AN_AUTHTYPE_LEAP
);
1033 cfg
->an_authtype
|= AN_AUTHTYPE_PRIVACY_IN_USE
;
1034 cfg
->an_authtype
&= ~AN_AUTHTYPE_ALLOW_UNENCRYPTED
;
1035 cfg
->an_authtype
&= ~AN_AUTHTYPE_LEAP
;
1039 cfg
->an_authtype
= AN_AUTHTYPE_PRIVACY_IN_USE
1040 | AN_AUTHTYPE_ALLOW_UNENCRYPTED
;
1044 case ACT_SET_KEY_TYPE
:
1045 cfg
->an_authtype
= (cfg
->an_authtype
& ~AN_AUTHTYPE_MASK
)
1048 case ACT_SET_MONITOR_MODE
:
1049 areq
.an_type
= AN_RID_MONITOR_MODE
;
1050 cfg
->an_len
= atoi(arg
); /* mode is put in length */
1053 errx(1, "unknown action");
1057 an_setval(iface
, &areq
);
1062 an_setspeed(const char *iface
, int act __unused
, void *arg
)
1065 struct an_ltv_caps
*caps
;
1068 areq
.an_len
= sizeof(areq
);
1069 areq
.an_type
= AN_RID_CAPABILITIES
;
1071 an_getval(iface
, &areq
);
1072 caps
= (struct an_ltv_caps
*)&areq
;
1079 speed
= AN_RATE_1MBPS
;
1082 speed
= AN_RATE_2MBPS
;
1085 if (caps
->an_rates
[2] != AN_RATE_5_5MBPS
)
1086 errx(1, "5.5Mbps not supported on this card");
1087 speed
= AN_RATE_5_5MBPS
;
1090 if (caps
->an_rates
[3] != AN_RATE_11MBPS
)
1091 errx(1, "11Mbps not supported on this card");
1092 speed
= AN_RATE_11MBPS
;
1095 errx(1, "unsupported speed");
1100 areq
.an_type
= AN_RID_TX_SPEED
;
1101 areq
.an_val
[0] = speed
;
1103 an_setval(iface
, &areq
);
1108 an_setap(const char *iface
, int act
, void *arg
)
1110 struct an_ltv_aplist
*ap
;
1112 struct ether_addr
*addr
;
1114 areq
.an_len
= sizeof(areq
);
1115 areq
.an_type
= AN_RID_APLIST
;
1117 an_getval(iface
, &areq
);
1118 ap
= (struct an_ltv_aplist
*)&areq
;
1120 addr
= ether_aton((char *)arg
);
1123 errx(1, "badly formatted address");
1127 bzero(ap
->an_ap1
, ETHER_ADDR_LEN
);
1128 bcopy(addr
, &ap
->an_ap1
, ETHER_ADDR_LEN
);
1131 bzero(ap
->an_ap2
, ETHER_ADDR_LEN
);
1132 bcopy(addr
, &ap
->an_ap2
, ETHER_ADDR_LEN
);
1135 bzero(ap
->an_ap3
, ETHER_ADDR_LEN
);
1136 bcopy(addr
, &ap
->an_ap3
, ETHER_ADDR_LEN
);
1139 bzero(ap
->an_ap4
, ETHER_ADDR_LEN
);
1140 bcopy(addr
, &ap
->an_ap4
, ETHER_ADDR_LEN
);
1143 errx(1, "unknown action");
1147 an_setval(iface
, &areq
);
1152 an_setssid(const char *iface
, int act
, void *arg
)
1154 struct an_ltv_ssidlist_new
*ssid
;
1158 areq
.an_len
= sizeof(areq
);
1159 areq
.an_type
= AN_RID_SSIDLIST
;
1161 an_getval(iface
, &areq
);
1162 ssid
= (struct an_ltv_ssidlist_new
*)&areq
;
1164 max
= (areq
.an_len
- 4) / sizeof(struct an_ltv_ssid_entry
);
1165 if ( max
> MAX_SSIDS
) {
1166 printf("Too many SSIDs only printing %d of %d\n",
1172 errx(1, "bad modifier %d: there "
1173 "are only %d SSID settings", act
, max
);
1177 bzero(ssid
->an_entry
[act
-1].an_ssid
,
1178 sizeof(ssid
->an_entry
[act
-1].an_ssid
));
1179 strlcpy(ssid
->an_entry
[act
-1].an_ssid
, (char *)arg
,
1180 sizeof(ssid
->an_entry
[act
-1].an_ssid
));
1181 ssid
->an_entry
[act
-1].an_len
1182 = strlen(ssid
->an_entry
[act
-1].an_ssid
);
1184 an_setval(iface
, &areq
);
1191 an_zerocache(const char *iface
)
1195 bzero(&areq
, sizeof(areq
));
1197 areq
.an_type
= AN_RID_ZERO_CACHE
;
1199 an_getval(iface
, &areq
);
1203 an_readcache(const char *iface
)
1206 uint16_t *an_sigitems
;
1207 struct an_sigcache
*sc
;
1211 errx(1, "must specify interface name");
1213 bzero(&areq
, sizeof(areq
));
1214 areq
.an_len
= AN_MAX_DATALEN
;
1215 areq
.an_type
= AN_RID_READ_CACHE
;
1217 an_getval(iface
, &areq
);
1219 an_sigitems
= areq
.an_val
;
1220 sc
= (struct an_sigcache
*)((int32_t *)areq
.an_val
+ 1);
1222 for (i
= 0; i
< *an_sigitems
; i
++) {
1223 printf("[%d/%d]:", i
+1, *an_sigitems
);
1224 printf(" %02x:%02x:%02x:%02x:%02x:%02x,",
1230 sc
->macsrc
[5]&0xff);
1231 printf(" %d.%d.%d.%d,",((sc
->ipsrc
>> 0) & 0xff),
1232 ((sc
->ipsrc
>> 8) & 0xff),
1233 ((sc
->ipsrc
>> 16) & 0xff),
1234 ((sc
->ipsrc
>> 24) & 0xff));
1235 printf(" sig: %d, noise: %d, qual: %d\n",
1247 if (c
>= '0' && c
<= '9')
1249 if (c
>= 'A' && c
<= 'F')
1250 return (c
- 'A' + 10);
1251 if (c
>= 'a' && c
<= 'f')
1252 return (c
- 'a' + 10);
1258 an_str2key(const char *s
, struct an_ltv_key
*k
)
1263 /* Is this a hex string? */
1264 if (s
[0] == '0' && (s
[1] == 'x' || s
[1] == 'X')) {
1265 /* Yes, convert to int. */
1267 p
= (char *)&k
->key
[0];
1268 for (i
= 2; s
[i
] != '\0' && s
[i
+ 1] != '\0'; i
+= 2) {
1269 *p
++ = (an_hex2int(s
[i
]) << 4) + an_hex2int(s
[i
+ 1]);
1273 errx(1, "hex strings must be of even length");
1276 /* No, just copy it in. */
1277 bcopy(s
, k
->key
, strlen(s
));
1278 k
->klen
= strlen(s
);
1285 an_setkeys(const char *iface
, const char *key
, int keytype
)
1288 struct an_ltv_key
*k
;
1290 bzero(&areq
, sizeof(areq
));
1291 k
= (struct an_ltv_key
*)&areq
;
1293 if (strlen(key
) > 28) {
1294 err(1, "encryption key must be no "
1295 "more than 18 characters long");
1300 k
->kindex
=keytype
/2;
1302 if (!(k
->klen
==0 || k
->klen
==5 || k
->klen
==13)) {
1303 err(1, "encryption key must be 0, 5 or 13 bytes long");
1306 /* default mac and only valid one (from manual) 1.0.0.0.0.0 */
1314 switch(keytype
& 1) {
1316 areq
.an_len
= sizeof(struct an_ltv_key
);
1317 areq
.an_type
= AN_RID_WEP_PERM
;
1318 an_setval(iface
, &areq
);
1321 areq
.an_len
= sizeof(struct an_ltv_key
);
1322 areq
.an_type
= AN_RID_WEP_TEMP
;
1323 an_setval(iface
, &areq
);
1329 an_readkeyinfo(const char *iface
)
1332 struct an_ltv_genconfig
*cfg
;
1333 struct an_ltv_key
*k
;
1337 areq
.an_len
= sizeof(areq
);
1338 areq
.an_type
= AN_RID_ACTUALCFG
;
1339 an_getval(iface
, &areq
);
1340 cfg
= (struct an_ltv_genconfig
*)&areq
;
1341 if (cfg
->an_home_product
& AN_HOME_NETWORK
)
1346 bzero(&areq
, sizeof(areq
));
1347 k
= (struct an_ltv_key
*)&areq
;
1349 printf("WEP Key status:\n");
1350 areq
.an_type
= AN_RID_WEP_TEMP
; /* read first key */
1351 for(i
=0; i
<5; i
++) {
1352 areq
.an_len
= sizeof(struct an_ltv_key
);
1353 an_getval(iface
, &areq
);
1354 if (k
->kindex
== 0xffff)
1358 printf("\tKey %u is unset\n", k
->kindex
);
1361 printf("\tKey %u is set 40 bits\n", k
->kindex
);
1364 printf("\tKey %u is set 128 bits\n", k
->kindex
);
1367 printf("\tWEP Key %d has an unknown size %u\n",
1371 areq
.an_type
= AN_RID_WEP_PERM
; /* read next key */
1374 areq
.an_len
= sizeof(struct an_ltv_key
);
1375 an_getval(iface
, &areq
);
1376 printf("\tThe active transmit key is %d\n", 4 * home
+ k
->mac
[0]);
1382 an_enable_tx_key(const char *iface
, const char *arg
)
1385 struct an_ltv_key
*k
;
1386 struct an_ltv_genconfig
*config
;
1388 bzero(&areq
, sizeof(areq
));
1390 /* set home or not home mode */
1391 areq
.an_len
= sizeof(struct an_ltv_genconfig
);
1392 areq
.an_type
= AN_RID_GENCONFIG
;
1393 an_getval(iface
, &areq
);
1394 config
= (struct an_ltv_genconfig
*)&areq
;
1395 if (atoi(arg
) == 4) {
1396 config
->an_home_product
|= AN_HOME_NETWORK
;
1398 config
->an_home_product
&= ~AN_HOME_NETWORK
;
1400 an_setval(iface
, &areq
);
1402 bzero(&areq
, sizeof(areq
));
1404 k
= (struct an_ltv_key
*)&areq
;
1406 /* From a Cisco engineer write the transmit key to use in the
1407 first MAC, index is FFFF*/
1411 k
->mac
[0]=atoi(arg
);
1418 areq
.an_len
= sizeof(struct an_ltv_key
);
1419 areq
.an_type
= AN_RID_WEP_PERM
;
1420 an_setval(iface
, &areq
);
1424 an_enable_leap_mode(const char *iface
, const char *username
)
1427 struct an_ltv_status
*sts
;
1428 struct an_ltv_genconfig
*cfg
;
1429 struct an_ltv_caps
*caps
;
1430 struct an_ltv_leap_username an_username
;
1431 struct an_ltv_leap_password an_password
;
1436 char unicode_password
[LEAP_PASSWORD_MAX
* 2];
1438 areq
.an_len
= sizeof(areq
);
1439 areq
.an_type
= AN_RID_CAPABILITIES
;
1441 an_getval(iface
, &areq
);
1443 caps
= (struct an_ltv_caps
*)&areq
;
1445 if (!(caps
->an_softcaps
& AN_AUTHTYPE_LEAP
)) {
1446 fprintf(stderr
, "Firmware does not support LEAP\n");
1450 bzero(&an_username
, sizeof(an_username
));
1451 bzero(&an_password
, sizeof(an_password
));
1453 len
= strlen(username
);
1454 if (len
> LEAP_USERNAME_MAX
) {
1455 printf("Username too long (max %d)\n", LEAP_USERNAME_MAX
);
1458 strncpy(an_username
.an_username
, username
, len
);
1459 an_username
.an_username_len
= len
;
1460 an_username
.an_len
= sizeof(an_username
);
1461 an_username
.an_type
= AN_RID_LEAPUSERNAME
;
1463 password
= getpass("Enter LEAP password:");
1465 len
= strlen(password
);
1466 if (len
> LEAP_PASSWORD_MAX
) {
1467 printf("Password too long (max %d)\n", LEAP_PASSWORD_MAX
);
1471 bzero(&unicode_password
, sizeof(unicode_password
));
1472 for(i
= 0; i
< len
; i
++) {
1473 unicode_password
[i
* 2] = *password
++;
1478 MD4Update(&context
, unicode_password
, len
* 2);
1479 MD4Final(&an_password
.an_password
[0], &context
);
1483 MD4Update (&context
, &an_password
.an_password
[0], 16);
1484 MD4Final (&an_password
.an_password
[16], &context
);
1486 an_password
.an_password_len
= 32;
1487 an_password
.an_len
= sizeof(an_password
);
1488 an_password
.an_type
= AN_RID_LEAPPASSWORD
;
1490 an_setval(iface
, (struct an_req
*)&an_username
);
1491 an_setval(iface
, (struct an_req
*)&an_password
);
1493 areq
.an_len
= sizeof(areq
);
1494 areq
.an_type
= AN_RID_GENCONFIG
;
1495 an_getval(iface
, &areq
);
1496 cfg
= (struct an_ltv_genconfig
*)&areq
;
1497 cfg
->an_authtype
= (AN_AUTHTYPE_PRIVACY_IN_USE
| AN_AUTHTYPE_LEAP
);
1498 an_setval(iface
, &areq
);
1500 sts
= (struct an_ltv_status
*)&areq
;
1501 areq
.an_type
= AN_RID_STATUS
;
1503 for (i
= 60; i
> 0; i
--) {
1504 an_getval(iface
, &areq
);
1505 if (sts
->an_opmode
& AN_STATUS_OPMODE_LEAP
) {
1506 printf("Authenticated\n");
1513 fprintf(stderr
, "Failed LEAP authentication\n");
1519 main(int argc
, char *argv
[])
1523 const char *iface
= NULL
;
1529 /* Get the interface name */
1531 ch
= getopt(argc
, argv
, "i:");
1535 if (argc
> 1 && *argv
[1] != '-') {
1546 while ((ch
= getopt(argc
, argv
,
1547 "ANISCTRht:a:e:o:s:n:v:d:j:b:c:f:r:p:w:m:l:k:K:W:QZM:L:")) != -1) {
1551 act
= ACT_ZEROCACHE
;
1553 errx(1, "ANCACHE not available");
1558 act
= ACT_DUMPCACHE
;
1560 errx(1, "ANCACHE not available");
1570 act
= ACT_DUMPSTATUS
;
1576 act
= ACT_DUMPSTATS
;
1579 act
= ACT_DUMPCONFIG
;
1582 act
= ACT_DUMPRSSIMAP
;
1585 act
= ACT_SET_TXRATE
;
1589 act
= ACT_SET_PWRSAVE
;
1593 act
= ACT_SET_TXPWR
;
1597 modifier
= atoi(optarg
);
1615 errx(1, "bad modifier %d: there "
1616 "are only 4 access point settings",
1624 act
= ACT_SET_BEACON_PERIOD
;
1630 act
= ACT_SET_DIVERSITY_RX
;
1633 act
= ACT_SET_DIVERSITY_TX
;
1636 errx(1, "must specify RX or TX diversity");
1639 if (!isdigit(*optarg
)) {
1640 errx(1, "%s is not numeric", optarg
);
1646 act
= ACT_SET_NETJOIN
;
1650 act
= ACT_SET_MYNAME
;
1664 act
= ACT_SET_OPMODE
;
1672 act
= ACT_SET_FRAG_THRESH
;
1676 act
= ACT_ENABLE_WEP
;
1680 act
= ACT_SET_KEY_TYPE
;
1688 act
= ACT_ENABLE_TX_KEY
;
1692 act
= ACT_SET_RTS_RETRYLIM
;
1696 act
= ACT_SET_RTS_THRESH
;
1700 act
= ACT_SET_WAKE_DURATION
;
1704 act
= ACT_SET_MONITOR_MODE
;
1708 act
= ACT_SET_LEAP_MODE
;
1717 if (iface
== NULL
|| (!act
&& !key
))
1721 case ACT_DUMPSTATUS
:
1722 an_dumpstatus(iface
);
1728 an_dumpstats(iface
);
1730 case ACT_DUMPCONFIG
:
1731 an_dumpconfig(iface
);
1739 case ACT_DUMPRSSIMAP
:
1740 an_dumprssimap(iface
);
1743 an_setssid(iface
, modifier
, arg
);
1749 an_setap(iface
, act
, arg
);
1751 case ACT_SET_TXRATE
:
1752 an_setspeed(iface
, act
, arg
);
1756 an_zerocache(iface
);
1759 an_readcache(iface
);
1764 an_setkeys(iface
, key
, modifier
);
1766 case ACT_ENABLE_TX_KEY
:
1767 an_enable_tx_key(iface
, arg
);
1769 case ACT_SET_LEAP_MODE
:
1770 an_enable_leap_mode(iface
, arg
);
1773 an_setconfig(iface
, act
, arg
);