[package] iwinfo: replace internal constant mode strings with enums
[openwrt.git] / package / iwinfo / src / iwinfo_cli.c
1 /*
2  * iwinfo - Wireless Information Library - Command line frontend
3  *
4  *   Copyright (C) 2011 Jo-Philipp Wich <xm@subsignal.org>
5  *
6  * The iwinfo library is free software: you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License version 2
8  * as published by the Free Software Foundation.
9  *
10  * The iwinfo library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13  * See the GNU General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License along
16  * with the iwinfo library. If not, see http://www.gnu.org/licenses/.
17  */
18
19 #include <stdio.h>
20
21 #include "iwinfo.h"
22
23
24 static char * format_bssid(unsigned char *mac)
25 {
26         static char buf[18];
27
28         snprintf(buf, sizeof(buf), "%02X:%02X:%02X:%02X:%02X:%02X",
29                 mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
30
31         return buf;
32 }
33
34 static char * format_ssid(char *ssid)
35 {
36         static char buf[IWINFO_ESSID_MAX_SIZE+3];
37
38         if (ssid && ssid[0])
39                 snprintf(buf, sizeof(buf), "\"%s\"", ssid);
40         else
41                 snprintf(buf, sizeof(buf), "unknown");
42
43         return buf;
44 }
45
46 static char * format_channel(int ch)
47 {
48         static char buf[8];
49
50         if (ch <= 0)
51                 snprintf(buf, sizeof(buf), "unknown");
52         else
53                 snprintf(buf, sizeof(buf), "%d", ch);
54
55         return buf;
56 }
57
58 static char * format_frequency(int freq)
59 {
60         static char buf[10];
61
62         if (freq <= 0)
63                 snprintf(buf, sizeof(buf), "unknown");
64         else
65                 snprintf(buf, sizeof(buf), "%.3f GHz", ((float)freq / 1000.0));
66
67         return buf;
68 }
69
70 static char * format_txpower(int pwr)
71 {
72         static char buf[10];
73
74         if (pwr < 0)
75                 snprintf(buf, sizeof(buf), "unknown");
76         else
77                 snprintf(buf, sizeof(buf), "%d dBm", pwr);
78
79         return buf;
80 }
81
82 static char * format_quality(int qual)
83 {
84         static char buf[8];
85
86         if (qual < 0)
87                 snprintf(buf, sizeof(buf), "unknown");
88         else
89                 snprintf(buf, sizeof(buf), "%d", qual);
90
91         return buf;
92 }
93
94 static char * format_quality_max(int qmax)
95 {
96         static char buf[8];
97
98         if (qmax < 0)
99                 snprintf(buf, sizeof(buf), "unknown");
100         else
101                 snprintf(buf, sizeof(buf), "%d", qmax);
102
103         return buf;
104 }
105
106 static char * format_signal(int sig)
107 {
108         static char buf[10];
109
110         if (!sig)
111                 snprintf(buf, sizeof(buf), "unknown");
112         else
113                 snprintf(buf, sizeof(buf), "%d dBm", sig);
114
115         return buf;
116 }
117
118 static char * format_noise(int noise)
119 {
120         static char buf[10];
121
122         if (!noise)
123                 snprintf(buf, sizeof(buf), "unknown");
124         else
125                 snprintf(buf, sizeof(buf), "%d dBm", noise);
126
127         return buf;
128 }
129
130 static char * format_rate(int rate)
131 {
132         static char buf[14];
133
134         if (rate <= 0)
135                 snprintf(buf, sizeof(buf), "unknown");
136         else
137                 snprintf(buf, sizeof(buf), "%.1f MBit/s", ((float)rate / 1000.0));
138
139         return buf;
140 }
141
142 static char * format_enc_ciphers(int ciphers)
143 {
144         static char str[128] = { 0 };
145         char *pos = str;
146
147         if (ciphers & IWINFO_CIPHER_WEP40)
148                 pos += sprintf(pos, "WEP-40, ");
149
150         if (ciphers & IWINFO_CIPHER_WEP104)
151                 pos += sprintf(pos, "WEP-104, ");
152
153         if (ciphers & IWINFO_CIPHER_TKIP)
154                 pos += sprintf(pos, "TKIP, ");
155
156         if (ciphers & IWINFO_CIPHER_CCMP)
157                 pos += sprintf(pos, "CCMP, ");
158
159         if (ciphers & IWINFO_CIPHER_WRAP)
160                 pos += sprintf(pos, "WRAP, ");
161
162         if (ciphers & IWINFO_CIPHER_AESOCB)
163                 pos += sprintf(pos, "AES-OCB, ");
164
165         if (ciphers & IWINFO_CIPHER_CKIP)
166                 pos += sprintf(pos, "CKIP, ");
167
168         if (!ciphers || (ciphers & IWINFO_CIPHER_NONE))
169                 pos += sprintf(pos, "NONE, ");
170
171         *(pos - 2) = 0;
172
173         return str;
174 }
175
176 static char * format_enc_suites(int suites)
177 {
178         static char str[64] = { 0 };
179         char *pos = str;
180
181         if (suites & IWINFO_KMGMT_PSK)
182                 pos += sprintf(pos, "PSK/");
183
184         if (suites & IWINFO_KMGMT_8021x)
185                 pos += sprintf(pos, "802.1X/");
186
187         if (!suites || (suites & IWINFO_KMGMT_NONE))
188                 pos += sprintf(pos, "NONE/");
189
190         *(pos - 1) = 0;
191
192         return str;
193 }
194
195 static char * format_encryption(struct iwinfo_crypto_entry *c)
196 {
197         static char buf[512];
198
199         if (!c)
200         {
201                 snprintf(buf, sizeof(buf), "unknown");
202         }
203         else if (c->enabled)
204         {
205                 /* WEP */
206                 if (c->auth_algs && !c->wpa_version)
207                 {
208                         if ((c->auth_algs & IWINFO_AUTH_OPEN) &&
209                                 (c->auth_algs & IWINFO_AUTH_SHARED))
210                         {
211                                 snprintf(buf, sizeof(buf), "WEP Open/Shared (%s)",
212                                         format_enc_ciphers(c->pair_ciphers));
213                         }
214                         else if (c->auth_algs & IWINFO_AUTH_OPEN)
215                         {
216                                 snprintf(buf, sizeof(buf), "WEP Open System (%s)",
217                                         format_enc_ciphers(c->pair_ciphers));
218                         }
219                         else if (c->auth_algs & IWINFO_AUTH_SHARED)
220                         {
221                                 snprintf(buf, sizeof(buf), "WEP Shared Auth (%s)",
222                                         format_enc_ciphers(c->pair_ciphers));
223                         }
224                 }
225
226                 /* WPA */
227                 else if (c->wpa_version)
228                 {
229                         switch (c->wpa_version) {
230                                 case 3:
231                                         snprintf(buf, sizeof(buf), "mixed WPA/WPA2 %s (%s)",
232                                                 format_enc_suites(c->auth_suites),
233                                                 format_enc_ciphers(c->pair_ciphers | c->group_ciphers));
234                                         break;
235
236                                 case 2:
237                                         snprintf(buf, sizeof(buf), "WPA2 %s (%s)",
238                                                 format_enc_suites(c->auth_suites),
239                                                 format_enc_ciphers(c->pair_ciphers | c->group_ciphers));
240                                         break;
241
242                                 case 1:
243                                         snprintf(buf, sizeof(buf), "WPA %s (%s)",
244                                                 format_enc_suites(c->auth_suites),
245                                                 format_enc_ciphers(c->pair_ciphers | c->group_ciphers));
246                                         break;
247                         }
248                 }
249                 else
250                 {
251                         snprintf(buf, sizeof(buf), "none");
252                 }
253         }
254         else
255         {
256                 snprintf(buf, sizeof(buf), "none");
257         }
258
259         return buf;
260 }
261
262 static char * format_hwmodes(int modes)
263 {
264         static char buf[12];
265
266         if (modes <= 0)
267                 snprintf(buf, sizeof(buf), "unknown");
268         else
269                 snprintf(buf, sizeof(buf), "802.11%s%s%s%s",
270                         (modes & IWINFO_80211_A) ? "a" : "",
271                         (modes & IWINFO_80211_B) ? "b" : "",
272                         (modes & IWINFO_80211_G) ? "g" : "",
273                         (modes & IWINFO_80211_N) ? "n" : "");
274
275         return buf;
276 }
277
278 static char * format_assocrate(struct iwinfo_rate_entry *r)
279 {
280         static char buf[40];
281         char *p = buf;
282         int l = sizeof(buf);
283
284         if (r->rate <= 0)
285         {
286                 snprintf(buf, sizeof(buf), "unknown");
287         }
288         else
289         {
290                 p += snprintf(p, l, "%s", format_rate(r->rate));
291                 l = sizeof(buf) - (p - buf);
292
293                 if (r->mcs >= 0)
294                 {
295                         p += snprintf(p, l, ", MCS %d, %dMHz", r->mcs, 20 + r->is_40mhz*20);
296                         l = sizeof(buf) - (p - buf);
297
298                         if (r->is_short_gi)
299                                 p += snprintf(p, l, ", short GI");
300                 }
301         }
302
303         return buf;
304 }
305
306
307 static const char * print_type(const struct iwinfo_ops *iw, const char *ifname)
308 {
309         const char *type = iwinfo_type(ifname);
310         return type ? type : "unknown";
311 }
312
313 static char * print_hardware_id(const struct iwinfo_ops *iw, const char *ifname)
314 {
315         static char buf[20];
316         struct iwinfo_hardware_id ids;
317
318         if (!iw->hardware_id(ifname, (char *)&ids))
319         {
320                 snprintf(buf, sizeof(buf), "%04X:%04X %04X:%04X",
321                         ids.vendor_id, ids.device_id,
322                         ids.subsystem_vendor_id, ids.subsystem_device_id);
323         }
324         else
325         {
326                 snprintf(buf, sizeof(buf), "unknown");
327         }
328
329         return buf;
330 }
331
332 static char * print_hardware_name(const struct iwinfo_ops *iw, const char *ifname)
333 {
334         static char buf[128];
335
336         if (iw->hardware_name(ifname, buf))
337                 snprintf(buf, sizeof(buf), "unknown");
338
339         return buf;
340 }
341
342 static char * print_txpower_offset(const struct iwinfo_ops *iw, const char *ifname)
343 {
344         int off;
345         static char buf[12];
346
347         if (iw->txpower_offset(ifname, &off))
348                 snprintf(buf, sizeof(buf), "unknown");
349         else if (off != 0)
350                 snprintf(buf, sizeof(buf), "%d dB", off);
351         else
352                 snprintf(buf, sizeof(buf), "none");
353
354         return buf;
355 }
356
357 static char * print_frequency_offset(const struct iwinfo_ops *iw, const char *ifname)
358 {
359         int off;
360         static char buf[12];
361
362         if (iw->frequency_offset(ifname, &off))
363                 snprintf(buf, sizeof(buf), "unknown");
364         else if (off != 0)
365                 snprintf(buf, sizeof(buf), "%.3f GHz", ((float)off / 1000.0));
366         else
367                 snprintf(buf, sizeof(buf), "none");
368
369         return buf;
370 }
371
372 static char * print_ssid(const struct iwinfo_ops *iw, const char *ifname)
373 {
374         char buf[IWINFO_ESSID_MAX_SIZE+1] = { 0 };
375
376         if (iw->ssid(ifname, buf))
377                 memset(buf, 0, sizeof(buf));
378
379         return format_ssid(buf);
380 }
381
382 static char * print_bssid(const struct iwinfo_ops *iw, const char *ifname)
383 {
384         static char buf[18] = { 0 };
385
386         if (iw->bssid(ifname, buf))
387                 snprintf(buf, sizeof(buf), "00:00:00:00:00:00");
388
389         return buf;
390 }
391
392 static char * print_mode(const struct iwinfo_ops *iw, const char *ifname)
393 {
394         int mode;
395         static char buf[128];
396
397         if (iw->mode(ifname, &mode))
398                 mode = IWINFO_OPMODE_UNKNOWN;
399
400         snprintf(buf, sizeof(buf), "%s", IWINFO_OPMODE_NAMES[mode]);
401
402         return buf;
403 }
404
405 static char * print_channel(const struct iwinfo_ops *iw, const char *ifname)
406 {
407         int ch;
408         if (iw->channel(ifname, &ch))
409                 ch = -1;
410
411         return format_channel(ch);
412 }
413
414 static char * print_frequency(const struct iwinfo_ops *iw, const char *ifname)
415 {
416         int freq;
417         if (iw->frequency(ifname, &freq))
418                 freq = -1;
419
420         return format_frequency(freq);
421 }
422
423 static char * print_txpower(const struct iwinfo_ops *iw, const char *ifname)
424 {
425         int pwr, off;
426         if (iw->txpower_offset(ifname, &off))
427                 off = 0;
428
429         if (iw->txpower(ifname, &pwr))
430                 pwr = -1;
431         else
432                 pwr += off;
433
434         return format_txpower(pwr);
435 }
436
437 static char * print_quality(const struct iwinfo_ops *iw, const char *ifname)
438 {
439         int qual;
440         if (iw->quality(ifname, &qual))
441                 qual = -1;
442
443         return format_quality(qual);
444 }
445
446 static char * print_quality_max(const struct iwinfo_ops *iw, const char *ifname)
447 {
448         int qmax;
449         if (iw->quality_max(ifname, &qmax))
450                 qmax = -1;
451
452         return format_quality_max(qmax);
453 }
454
455 static char * print_signal(const struct iwinfo_ops *iw, const char *ifname)
456 {
457         int sig;
458         if (iw->signal(ifname, &sig))
459                 sig = 0;
460
461         return format_signal(sig);
462 }
463
464 static char * print_noise(const struct iwinfo_ops *iw, const char *ifname)
465 {
466         int noise;
467         if (iw->noise(ifname, &noise))
468                 noise = 0;
469
470         return format_noise(noise);
471 }
472
473 static char * print_rate(const struct iwinfo_ops *iw, const char *ifname)
474 {
475         int rate;
476         if (iw->bitrate(ifname, &rate))
477                 rate = -1;
478
479         return format_rate(rate);
480 }
481
482 static char * print_encryption(const struct iwinfo_ops *iw, const char *ifname)
483 {
484         struct iwinfo_crypto_entry c = { 0 };
485         if (iw->encryption(ifname, (char *)&c))
486                 return format_encryption(NULL);
487
488         return format_encryption(&c);
489 }
490
491 static char * print_hwmodes(const struct iwinfo_ops *iw, const char *ifname)
492 {
493         int modes;
494         if (iw->hwmodelist(ifname, &modes))
495                 modes = -1;
496
497         return format_hwmodes(modes);
498 }
499
500 static char * print_mbssid_supp(const struct iwinfo_ops *iw, const char *ifname)
501 {
502         int supp;
503         static char buf[4];
504
505         if (iw->mbssid_support(ifname, &supp))
506                 snprintf(buf, sizeof(buf), "no");
507         else
508                 snprintf(buf, sizeof(buf), "%s", supp ? "yes" : "no");
509
510         return buf;
511 }
512
513
514 static void print_info(const struct iwinfo_ops *iw, const char *ifname)
515 {
516         printf("%-9s ESSID: %s\n",
517                 ifname,
518                 print_ssid(iw, ifname));
519         printf("          Access Point: %s\n",
520                 print_bssid(iw, ifname));
521         printf("          Mode: %s  Channel: %s (%s)\n",
522                 print_mode(iw, ifname),
523                 print_channel(iw, ifname),
524                 print_frequency(iw, ifname));
525         printf("          Tx-Power: %s  Link Quality: %s/%s\n",
526                 print_txpower(iw, ifname),
527                 print_quality(iw, ifname),
528                 print_quality_max(iw, ifname));
529         printf("          Signal: %s  Noise: %s\n",
530                 print_signal(iw, ifname),
531                 print_noise(iw, ifname));
532         printf("          Bit Rate: %s\n",
533                 print_rate(iw, ifname));
534         printf("          Encryption: %s\n",
535                 print_encryption(iw, ifname));
536         printf("          Type: %s  HW Mode(s): %s\n",
537                 print_type(iw, ifname),
538                 print_hwmodes(iw, ifname));
539         printf("          Hardware: %s [%s]\n",
540                 print_hardware_id(iw, ifname),
541                 print_hardware_name(iw, ifname));
542         printf("          TX power offset: %s\n",
543                 print_txpower_offset(iw, ifname));
544         printf("          Frequency offset: %s\n",
545                 print_frequency_offset(iw, ifname));
546         printf("          Supports VAPs: %s\n",
547                 print_mbssid_supp(iw, ifname));
548 }
549
550
551 static void print_scanlist(const struct iwinfo_ops *iw, const char *ifname)
552 {
553         int i, x, len;
554         char buf[IWINFO_BUFSIZE];
555         struct iwinfo_scanlist_entry *e;
556
557         if (iw->scanlist(ifname, buf, &len))
558         {
559                 printf("Scanning not possible\n\n");
560                 return;
561         }
562         else if (len <= 0)
563         {
564                 printf("No scan results\n\n");
565                 return;
566         }
567
568         for (i = 0, x = 1; i < len; i += sizeof(struct iwinfo_scanlist_entry), x++)
569         {
570                 e = (struct iwinfo_scanlist_entry *) &buf[i];
571
572                 printf("Cell %02d - Address: %s\n",
573                         x,
574                         format_bssid(e->mac));
575                 printf("          ESSID: %s\n",
576                         format_ssid(e->ssid));
577                 printf("          Mode: %s  Channel: %s\n",
578                         IWINFO_OPMODE_NAMES[e->mode],
579                         format_channel(e->channel));
580                 printf("          Signal: %s  Quality: %s/%s\n",
581                         format_signal(e->signal - 0x100),
582                         format_quality(e->quality),
583                         format_quality_max(e->quality_max));
584                 printf("          Encryption: %s\n\n",
585                         format_encryption(&e->crypto));
586         }
587 }
588
589
590 static void print_txpwrlist(const struct iwinfo_ops *iw, const char *ifname)
591 {
592         int len, pwr, off, i;
593         char buf[IWINFO_BUFSIZE];
594         struct iwinfo_txpwrlist_entry *e;
595
596         if (iw->txpwrlist(ifname, buf, &len) || len <= 0)
597         {
598                 printf("No TX power information available\n");
599                 return;
600         }
601
602         if (iw->txpower(ifname, &pwr))
603                 pwr = -1;
604
605         if (iw->txpower_offset(ifname, &off))
606                 off = 0;
607
608         for (i = 0; i < len; i += sizeof(struct iwinfo_txpwrlist_entry))
609         {
610                 e = (struct iwinfo_txpwrlist_entry *) &buf[i];
611
612                 printf("%s%3d dBm (%4d mW)\n",
613                         (pwr == e->dbm) ? "*" : " ",
614                         e->dbm + off,
615                         iwinfo_dbm2mw(e->dbm + off));
616         }
617 }
618
619
620 static void print_freqlist(const struct iwinfo_ops *iw, const char *ifname)
621 {
622         int i, len, ch;
623         char buf[IWINFO_BUFSIZE];
624         struct iwinfo_freqlist_entry *e;
625
626         if (iw->freqlist(ifname, buf, &len) || len <= 0)
627         {
628                 printf("No frequency information available\n");
629                 return;
630         }
631
632         if (iw->channel(ifname, &ch))
633                 ch = -1;
634
635         for (i = 0; i < len; i += sizeof(struct iwinfo_freqlist_entry))
636         {
637                 e = (struct iwinfo_freqlist_entry *) &buf[i];
638
639                 printf("%s %s (Channel %s)%s\n",
640                         (ch == e->channel) ? "*" : " ",
641                         format_frequency(e->mhz),
642                         format_channel(e->channel),
643                         e->restricted ? " [restricted]" : "");
644         }
645 }
646
647
648 static void print_assoclist(const struct iwinfo_ops *iw, const char *ifname)
649 {
650         int i, len;
651         char buf[IWINFO_BUFSIZE];
652         struct iwinfo_assoclist_entry *e;
653
654         if (iw->assoclist(ifname, buf, &len))
655         {
656                 printf("No information available\n");
657                 return;
658         }
659         else if (len <= 0)
660         {
661                 printf("No station connected\n");
662                 return;
663         }
664
665         for (i = 0; i < len; i += sizeof(struct iwinfo_assoclist_entry))
666         {
667                 e = (struct iwinfo_assoclist_entry *) &buf[i];
668
669                 printf("%s  %s / %s (SNR %d)  %d ms ago\n",
670                         format_bssid(e->mac),
671                         format_signal(e->signal),
672                         format_noise(e->noise),
673                         (e->signal - e->noise),
674                         e->inactive);
675
676                 printf("        RX: %-38s  %8d Pkts.\n",
677                         format_assocrate(&e->rx_rate),
678                         e->rx_packets
679                 );
680
681                 printf("        TX: %-38s  %8d Pkts.\n\n",
682                         format_assocrate(&e->tx_rate),
683                         e->tx_packets
684                 );
685         }
686 }
687
688
689 static char * lookup_country(char *buf, int len, int iso3166)
690 {
691         int i;
692         struct iwinfo_country_entry *c;
693
694         for (i = 0; i < len; i += sizeof(struct iwinfo_country_entry))
695         {
696                 c = (struct iwinfo_country_entry *) &buf[i];
697
698                 if (c->iso3166 == iso3166)
699                         return c->ccode;
700         }
701
702         return NULL;
703 }
704
705 static void print_countrylist(const struct iwinfo_ops *iw, const char *ifname)
706 {
707         int len;
708         char buf[IWINFO_BUFSIZE];
709         char *ccode;
710         char curcode[3];
711         const struct iwinfo_iso3166_label *l;
712
713         if (iw->countrylist(ifname, buf, &len))
714         {
715                 printf("No country code information available\n");
716                 return;
717         }
718
719         if (iw->country(ifname, curcode))
720                 memset(curcode, 0, sizeof(curcode));
721
722         for (l = IWINFO_ISO3166_NAMES; l->iso3166; l++)
723         {
724                 if ((ccode = lookup_country(buf, len, l->iso3166)) != NULL)
725                 {
726                         printf("%s %4s  %c%c\n",
727                                 strncmp(ccode, curcode, 2) ? " " : "*",
728                                 ccode, (l->iso3166 / 256), (l->iso3166 % 256));
729                 }
730         }
731 }
732
733
734 int main(int argc, char **argv)
735 {
736         int i;
737         const struct iwinfo_ops *iw;
738
739         if (argc < 3)
740         {
741                 fprintf(stderr,
742                         "Usage:\n"
743                         "       iwinfo <device> info\n"
744                         "       iwinfo <device> scan\n"
745                         "       iwinfo <device> txpowerlist\n"
746                         "       iwinfo <device> freqlist\n"
747                         "       iwinfo <device> assoclist\n"
748                         "       iwinfo <device> countrylist\n"
749                 );
750
751                 return 1;
752         }
753
754         iw = iwinfo_backend(argv[1]);
755
756         if (!iw)
757         {
758                 fprintf(stderr, "No such wireless device: %s\n", argv[1]);
759                 return 1;
760         }
761
762         for (i = 2; i < argc; i++)
763         {
764                 switch(argv[i][0])
765                 {
766                 case 'i':
767                         print_info(iw, argv[1]);
768                         break;
769
770                 case 's':
771                         print_scanlist(iw, argv[1]);
772                         break;
773
774                 case 't':
775                         print_txpwrlist(iw, argv[1]);
776                         break;
777
778                 case 'f':
779                         print_freqlist(iw, argv[1]);
780                         break;
781
782                 case 'a':
783                         print_assoclist(iw, argv[1]);
784                         break;
785
786                 case 'c':
787                         print_countrylist(iw, argv[1]);
788                         break;
789
790                 default:
791                         fprintf(stderr, "Unknown command: %s\n", argv[i]);
792                         return 1;
793                 }
794         }
795
796         iwinfo_finish();
797
798         return 0;
799 }