1 /* acpi.c - show power state
2 *
3 * Written by Isaac Dunham, 2013
4 *
5 * No standard.
6
7 USE_ACPI(NEWTOY(acpi, "abctV", TOYFLAG_USR|TOYFLAG_BIN))
8
9 config ACPI
10 bool "acpi"
11 default y
12 help
13 usage: acpi [-abctV]
14
15 Show status of power sources and thermal devices.
16
17 -a show power adapters
18 -b show batteries
19 -c show cooling device state
20 -t show temperatures
21 -V show everything
22 */
23
24 #define FOR_acpi
25 #include "toys.h"
26
GLOBALS(int ac,bat,therm,cool;char * cpath;)27 GLOBALS(
28 int ac, bat, therm, cool;
29 char *cpath;
30 )
31
32 static int read_int_at(int dirfd, char *name)
33 {
34 int fd, ret=0;
35 FILE *fil;
36
37 if ((fd = openat(dirfd, name, O_RDONLY)) < 0) return -1;
38 if (!fscanf(fil = xfdopen(fd, "r"), "%d", &ret)) perror_exit_raw(name);
39 fclose(fil);
40
41 return ret;
42 }
43
acpi_callback(struct dirtree * tree)44 static int acpi_callback(struct dirtree *tree)
45 {
46 int dfd, fd, len, on;
47
48 errno = 0;
49
50 if (tree->name[0]=='.') return 0;
51
52 if (!tree->parent)
53 return DIRTREE_RECURSE|DIRTREE_SYMFOLLOW;
54
55 if (0 <= (dfd = open((TT.cpath=dirtree_path(tree, NULL)), O_RDONLY))) {
56 if ((fd = openat(dfd, "type", O_RDONLY)) < 0) goto done;
57 len = readall(fd, toybuf, sizeof(toybuf));
58 close(fd);
59 if (len < 1) goto done;
60
61 if (!strncmp(toybuf, "Battery", 7)) {
62 if ((toys.optflags & FLAG_b) || (!toys.optflags)) {
63 int cap = 0, curr = 0, max = 0;
64
65 if ((cap = read_int_at(dfd, "capacity")) < 0) {
66 if ((max = read_int_at(dfd, "charge_full")) > 0)
67 curr = read_int_at(dfd, "charge_now");
68 else if ((max = read_int_at(dfd, "energy_full")) > 0)
69 curr = read_int_at(dfd, "energy_now");
70 if (max > 0 && curr >= 0) cap = 100 * curr / max;
71 }
72 if (cap >= 0) printf("Battery %d: %d%%\n", TT.bat++, cap);
73 }
74 } else if (toys.optflags & FLAG_a) {
75 if ((on = read_int_at(dfd, "online")) >= 0)
76 printf("Adapter %d: %s-line\n", TT.ac++, (on ? "on" : "off"));
77 }
78 done:
79 close(dfd);
80 }
81 free(TT.cpath);
82 return 0;
83 }
84
temp_callback(struct dirtree * tree)85 static int temp_callback(struct dirtree *tree)
86 {
87 int dfd, temp;
88
89 if (*tree->name=='.') return 0;
90 if (!tree->parent || !tree->parent->parent)
91 return DIRTREE_RECURSE|DIRTREE_SYMFOLLOW;
92 errno = 0;
93
94 if (0 <= (dfd = open((TT.cpath=dirtree_path(tree, NULL)), O_RDONLY))) {
95 if ((0 < (temp = read_int_at(dfd, "temp"))) || !errno) {
96 //some tempertures are in milli-C, some in deci-C
97 //reputedly some are in deci-K, but I have not seen them
98 if (((temp >= 1000) || (temp <= -1000)) && (temp%100 == 0)) temp /= 100;
99 printf("Thermal %d: %d.%d degrees C\n", TT.therm++, temp/10, temp%10);
100 }
101 close(dfd);
102 }
103 free(TT.cpath);
104
105 return 0;
106 }
107
cool_callback(struct dirtree * tree)108 static int cool_callback(struct dirtree *tree)
109 {
110 int dfd=5, cur, max;
111
112 errno = 0;
113 memset(toybuf, 0, sizeof(toybuf));
114
115 if (*tree->name == '.') return 0;
116 if (!tree->parent) return DIRTREE_RECURSE|DIRTREE_SYMFOLLOW;
117
118
119 if (0 <= (dfd = open((TT.cpath=dirtree_path(tree, &dfd)), O_RDONLY))) {
120 TT.cpath = strcat(TT.cpath, "/type");
121 if (readfile(TT.cpath, toybuf, 256) && !errno) {
122 toybuf[strlen(toybuf) -1] = 0;
123 cur=read_int_at(dfd, "cur_state");
124 max=read_int_at(dfd, "max_state");
125 if (errno)
126 printf("Cooling %d: %s no state information\n", TT.cool++, toybuf);
127 else printf("Cooling %d: %s %d of %d\n", TT.cool++, toybuf, cur, max);
128 }
129 close(dfd);
130 }
131 free(TT.cpath);
132 return 0;
133 }
134
acpi_main(void)135 void acpi_main(void)
136 {
137 if (toys.optflags & FLAG_V) toys.optflags = FLAG_a|FLAG_b|FLAG_c|FLAG_t;
138 if (!toys.optflags) toys.optflags = FLAG_b;
139 if (toys.optflags & (FLAG_a|FLAG_b))
140 dirtree_read("/sys/class/power_supply", acpi_callback);
141 if (toys.optflags & FLAG_t) dirtree_read("/sys/class", temp_callback);
142 if (toys.optflags & FLAG_c) dirtree_read("/sys/class/thermal", cool_callback);
143 }
144