1 /* rfkill.c - Enable/disable wireless devices.
2  *
3  * Copyright 2014 Ranjan Kumar <ranjankumar.bth@gmail.com>
4  * Copyright 2014 Kyungwan Han <asura321@gmail.com>
5  *
6  * No Standard
7 
8 USE_RFKILL(NEWTOY(rfkill, "<1>2", TOYFLAG_USR|TOYFLAG_SBIN))
9 
10 config RFKILL
11   bool "rfkill"
12   default y
13   help
14     usage: rfkill COMMAND [DEVICE]
15 
16     Enable/disable wireless devices.
17 
18     Commands:
19     list [DEVICE]   List current state
20     block DEVICE    Disable device
21     unblock DEVICE  Enable device
22 
23     DEVICE is an index number, or one of:
24     all, wlan(wifi), bluetooth, uwb(ultrawideband), wimax, wwan, gps, fm.
25 */
26 
27 #define FOR_rfkill
28 #include "toys.h"
29 #include <linux/rfkill.h>
30 
rfkill_main(void)31 void rfkill_main(void)
32 {
33   struct rfkill_event rfevent;
34   int fd, tvar, idx = -1, tid = RFKILL_TYPE_ALL;
35   char **optargs = toys.optargs;
36 
37   // Parse command line options
38   for (tvar = 0; tvar < 3; tvar++)
39     if (!strcmp((char *[]){"list", "block", "unblock"}[tvar], *optargs)) break;
40   if (tvar == 3) error_exit("unknown cmd '%s'", *optargs);
41   if (tvar) {
42     int i;
43     struct arglist {
44       char *name;
45       int idx;
46     } rftypes[] = {{"all", RFKILL_TYPE_ALL}, {"wifi", RFKILL_TYPE_WLAN},
47       {"wlan", RFKILL_TYPE_WLAN}, {"bluetooth", RFKILL_TYPE_BLUETOOTH},
48       {"uwb", RFKILL_TYPE_UWB}, {"ultrawideband", RFKILL_TYPE_UWB},
49       {"wimax", RFKILL_TYPE_WIMAX}, {"wwan", RFKILL_TYPE_WWAN},
50       {"gps", RFKILL_TYPE_GPS}, {"fm", 7}}; // RFKILL_TYPE_FM = 7
51 
52     if (!*++optargs) error_exit("'%s' needs IDENTIFIER", optargs[-1]);
53     for (i = 0; i < ARRAY_LEN(rftypes); i++)
54       if (!strcmp(rftypes[i].name, *optargs)) break;
55     if (i == ARRAY_LEN(rftypes)) idx = atolx_range(*optargs, 0, INT_MAX);
56     else tid = rftypes[i].idx;
57   }
58 
59   // Perform requested action
60   fd = xopen("/dev/rfkill", (tvar ? O_RDWR : O_RDONLY)|O_NONBLOCK);
61   if (tvar) {
62     // block/unblock
63     memset(&rfevent, 0, sizeof(rfevent));
64     rfevent.soft = tvar == 1;
65     if (idx >= 0) {
66       rfevent.idx = idx;
67       rfevent.op = RFKILL_OP_CHANGE;
68     } else {
69       rfevent.type = tid;
70       rfevent.op = RFKILL_OP_CHANGE_ALL;
71     }
72     xwrite(fd, &rfevent, sizeof(rfevent));
73   } else {
74     // show list.
75     while (sizeof(rfevent) == readall(fd, &rfevent, sizeof(rfevent))) {
76       char *line = 0, *name = 0, *type = 0;
77       FILE *fp;
78       size_t l = 0;
79       ssize_t len;
80 
81       // filter list items
82       if ((tid > 0 && tid != rfevent.type) || (idx != -1 && idx != rfevent.idx))
83         continue;
84 
85       sprintf(toybuf, "/sys/class/rfkill/rfkill%u/uevent", rfevent.idx);
86       fp = xfopen(toybuf, "r");
87       while ((len = getline(&line, &l, fp)) > 0) {
88         char *s = line;
89 
90         line[len-1] = 0;
91         if (strstart(&s, "RFKILL_NAME=")) name = xstrdup(s);
92         else if (strstart(&s, "RFKILL_TYPE=")) type = xstrdup(s);
93       }
94       free(line);
95       fclose(fp);
96 
97       xprintf("%u: %s: %s\n", rfevent.idx, name, type);
98       xprintf("\tSoft blocked: %s\n", rfevent.soft ? "yes" : "no");
99       xprintf("\tHard blocked: %s\n", rfevent.hard ? "yes" : "no");
100       free(name);
101       free(type);
102     }
103   }
104   xclose(fd);
105 }
106