Xinqi Bao's Git
1 /* See LICENSE file for copyright and license details. */
7 #include <linux/wireless.h>
16 #include <sys/ioctl.h>
18 #include <sys/statvfs.h>
19 #include <sys/socket.h>
20 #include <sys/soundcard.h>
21 #include <sys/sysinfo.h>
22 #include <sys/types.h>
23 #include <sys/utsname.h>
30 #define LEN(x) (sizeof (x) / sizeof *(x))
33 const char *(*func
)();
38 static const char *battery_perc(const char *bat
);
39 static const char *battery_power(const char *bat
);
40 static const char *battery_state(const char *bat
);
41 static const char *cpu_freq(void);
42 static const char *cpu_perc(void);
43 static const char *datetime(const char *fmt
);
44 static const char *disk_free(const char *mnt
);
45 static const char *disk_perc(const char *mnt
);
46 static const char *disk_total(const char *mnt
);
47 static const char *disk_used(const char *mnt
);
48 static const char *entropy(void);
49 static const char *gid(void);
50 static const char *hostname(void);
51 static const char *ip(const char *iface
);
52 static const char *kernel_release(void);
53 static const char *keyboard_indicators(void);
54 static const char *load_avg(void);
55 static const char *num_files(const char *dir
);
56 static const char *ram_free(void);
57 static const char *ram_perc(void);
58 static const char *ram_used(void);
59 static const char *ram_total(void);
60 static const char *run_command(const char *cmd
);
61 static const char *swap_free(void);
62 static const char *swap_perc(void);
63 static const char *swap_used(void);
64 static const char *swap_total(void);
65 static const char *temp(const char *file
);
66 static const char *uid(void);
67 static const char *uptime(void);
68 static const char *username(void);
69 static const char *vol_perc(const char *card
);
70 static const char *wifi_perc(const char *iface
);
71 static const char *wifi_essid(const char *iface
);
74 static unsigned short int delay
= 0;
75 static unsigned short int done
;
80 static char buf
[MAXLEN
];
83 bprintf(const char *fmt
, ...)
89 len
= vsnprintf(buf
, sizeof(buf
) - 1, fmt
, ap
);
92 if (len
>= sizeof(buf
))
93 buf
[sizeof(buf
)-1] = '\0';
99 battery_perc(const char *bat
)
105 snprintf(path
, sizeof(path
), "%s%s%s", "/sys/class/power_supply/", bat
, "/capacity");
106 fp
= fopen(path
, "r");
108 warn("Failed to open file %s", path
);
111 n
= fscanf(fp
, "%i", &perc
);
116 return bprintf("%d", perc
);
120 battery_power(const char *bat
)
126 snprintf(path
, sizeof(path
), "%s%s%s", "/sys/class/power_supply/", bat
, "/power_now");
127 fp
= fopen(path
, "r");
129 warn("Failed to open file %s", path
);
132 n
= fscanf(fp
, "%i", &watts
);
137 return bprintf("%d", (watts
+ 500000) / 1000000);
141 battery_state(const char *bat
)
149 { "Discharging", "-" },
155 char path
[PATH_MAX
], state
[12];
157 snprintf(path
, sizeof(path
), "%s%s%s", "/sys/class/power_supply/", bat
, "/status");
158 fp
= fopen(path
, "r");
160 warn("Failed to open file %s", path
);
163 n
= fscanf(fp
, "%12s", state
);
168 for (i
= 0; i
< LEN(map
); i
++) {
169 if (!strcmp(map
[i
].state
, state
)) {
174 return (i
== LEN(map
)) ? "?" : map
[i
].symbol
;
183 fp
= fopen("/sys/devices/system/cpu/cpu0/cpufreq/scaling_cur_freq", "r");
185 warn("Failed to open file /sys/devices/system/cpu/cpu0/cpufreq/scaling_cur_freq");
188 n
= fscanf(fp
, "%i", &freq
);
193 return bprintf("%d", (freq
+ 500) / 1000);
200 long double a
[4], b
[4];
203 fp
= fopen("/proc/stat", "r");
205 warn("Failed to open file /proc/stat");
208 n
= fscanf(fp
, "%*s %Lf %Lf %Lf %Lf", &a
[0], &a
[1], &a
[2], &a
[3]);
216 fp
= fopen("/proc/stat", "r");
218 warn("Failed to open file /proc/stat");
221 n
= fscanf(fp
, "%*s %Lf %Lf %Lf %Lf", &b
[0], &b
[1], &b
[2], &b
[3]);
226 perc
= 100 * ((b
[0]+b
[1]+b
[2]) - (a
[0]+a
[1]+a
[2])) / ((b
[0]+b
[1]+b
[2]+b
[3]) - (a
[0]+a
[1]+a
[2]+a
[3]));
227 return bprintf("%d", perc
);
231 datetime(const char *fmt
)
236 if (strftime(buf
, sizeof(buf
), fmt
, localtime(&t
)) == 0)
243 disk_free(const char *mnt
)
247 if (statvfs(mnt
, &fs
) < 0) {
248 warn("Failed to get filesystem info");
252 return bprintf("%f", (float)fs
.f_bsize
* (float)fs
.f_bfree
/ 1024 / 1024 / 1024);
256 disk_perc(const char *mnt
)
261 if (statvfs(mnt
, &fs
) < 0) {
262 warn("Failed to get filesystem info");
266 perc
= 100 * (1.0f
- ((float)fs
.f_bfree
/ (float)fs
.f_blocks
));
268 return bprintf("%d", perc
);
272 disk_total(const char *mnt
)
276 if (statvfs(mnt
, &fs
) < 0) {
277 warn("Failed to get filesystem info");
281 return bprintf("%f", (float)fs
.f_bsize
* (float)fs
.f_blocks
/ 1024 / 1024 / 1024);
285 disk_used(const char *mnt
)
289 if (statvfs(mnt
, &fs
) < 0) {
290 warn("Failed to get filesystem info");
294 return bprintf("%f", (float)fs
.f_bsize
* ((float)fs
.f_blocks
- (float)fs
.f_bfree
) / 1024 / 1024 / 1024);
303 fp
= fopen("/proc/sys/kernel/random/entropy_avail", "r");
305 warn("Failed to open file /proc/sys/kernel/random/entropy_avail");
308 n
= fscanf(fp
, "%d", &num
);
313 return bprintf("%d", num
);
319 return bprintf("%d", getgid());
325 if (gethostname(buf
, sizeof(buf
)) == -1) {
334 ip(const char *iface
)
336 struct ifaddrs
*ifaddr
, *ifa
;
338 char host
[NI_MAXHOST
];
340 if (getifaddrs(&ifaddr
) == -1) {
341 warn("Failed to get IP address for interface %s", iface
);
345 for (ifa
= ifaddr
; ifa
!= NULL
; ifa
= ifa
->ifa_next
) {
346 if (ifa
->ifa_addr
== NULL
) {
349 s
= getnameinfo(ifa
->ifa_addr
, sizeof(struct sockaddr_in
), host
, NI_MAXHOST
, NULL
, 0, NI_NUMERICHOST
);
350 if ((strcmp(ifa
->ifa_name
, iface
) == 0) && (ifa
->ifa_addr
->sa_family
== AF_INET
)) {
352 warnx("Failed to get IP address for interface %s", iface
);
355 return bprintf("%s", host
);
367 struct utsname udata
;
369 if (uname(&udata
) < 0) {
373 return bprintf("%s", udata
.release
);
377 keyboard_indicators(void)
379 Display
*dpy
= XOpenDisplay(NULL
);
380 XKeyboardState state
;
383 warnx("XOpenDisplay failed");
386 XGetKeyboardControl(dpy
, &state
);
389 switch (state
.led_mask
) {
406 if (getloadavg(avgs
, 3) < 0) {
407 warnx("Failed to get the load avg");
411 return bprintf("%.2f %.2f %.2f", avgs
[0], avgs
[1], avgs
[2]);
415 num_files(const char *dir
)
421 if ((fd
= opendir(dir
)) == NULL
) {
422 warn("Failed to get number of files in directory %s", dir
);
426 while ((dp
= readdir(fd
)) != NULL
) {
427 if (!strcmp(dp
->d_name
, ".") || !strcmp(dp
->d_name
, ".."))
428 continue; /* skip self and parent */
434 return bprintf("%d", num
);
444 fp
= fopen("/proc/meminfo", "r");
446 warn("Failed to open file /proc/meminfo");
449 n
= fscanf(fp
, "MemFree: %ld kB\n", &free
);
454 return bprintf("%f", (float)free
/ 1024 / 1024);
460 long total
, free
, buffers
, cached
;
463 fp
= fopen("/proc/meminfo", "r");
465 warn("Failed to open file /proc/meminfo");
468 if (fscanf(fp
, "MemTotal: %ld kB\n", &total
) != 1 ||
469 fscanf(fp
, "MemFree: %ld kB\n", &free
) != 1 ||
470 fscanf(fp
, "MemAvailable: %ld kB\nBuffers: %ld kB\n",
471 &buffers
, &buffers
) != 2 ||
472 fscanf(fp
, "Cached: %ld kB\n", &cached
) != 1)
476 return bprintf("%d", 100 * ((total
- free
) - (buffers
+ cached
)) / total
);
490 fp
= fopen("/proc/meminfo", "r");
492 warn("Failed to open file /proc/meminfo");
495 n
= fscanf(fp
, "MemTotal: %ld kB\n", &total
);
500 return bprintf("%f", (float)total
/ 1024 / 1024);
506 long free
, total
, buffers
, cached
;
509 fp
= fopen("/proc/meminfo", "r");
511 warn("Failed to open file /proc/meminfo");
514 if (fscanf(fp
, "MemTotal: %ld kB\n", &total
) != 1 ||
515 fscanf(fp
, "MemFree: %ld kB\n", &free
) != 1 ||
516 fscanf(fp
, "MemAvailable: %ld kB\nBuffers: %ld kB\n",
517 &buffers
, &buffers
) != 2 ||
518 fscanf(fp
, "Cached: %ld kB\n", &cached
) != 1)
522 return bprintf("%f", (float)(total
- free
- buffers
- cached
) / 1024 / 1024);
530 run_command(const char *cmd
)
535 fp
= popen(cmd
, "r");
537 warn("Failed to get command output for %s", cmd
);
540 p
= fgets(buf
, sizeof(buf
) - 1, fp
);
544 if ((p
= strrchr(buf
, '\n')) != NULL
)
547 return buf
[0] ? buf
: unknown_str
;
558 fp
= fopen("/proc/meminfo", "r");
560 warn("Failed to open file /proc/meminfo");
564 if ((bytes_read
= fread(buf
, sizeof(char), sizeof(buf
) - 1, fp
)) == 0) {
565 warn("swap_free: read error");
571 if ((match
= strstr(buf
, "SwapTotal")) == NULL
)
573 sscanf(match
, "SwapTotal: %ld kB\n", &total
);
575 if ((match
= strstr(buf
, "SwapFree")) == NULL
)
577 sscanf(match
, "SwapFree: %ld kB\n", &free
);
579 return bprintf("%f", (float)free
/ 1024 / 1024);
585 long total
, free
, cached
;
590 fp
= fopen("/proc/meminfo", "r");
592 warn("Failed to open file /proc/meminfo");
596 if ((bytes_read
= fread(buf
, sizeof(char), sizeof(buf
) - 1, fp
)) == 0) {
597 warn("swap_perc: read error");
603 if ((match
= strstr(buf
, "SwapTotal")) == NULL
)
605 sscanf(match
, "SwapTotal: %ld kB\n", &total
);
607 if ((match
= strstr(buf
, "SwapCached")) == NULL
)
609 sscanf(match
, "SwapCached: %ld kB\n", &cached
);
611 if ((match
= strstr(buf
, "SwapFree")) == NULL
)
613 sscanf(match
, "SwapFree: %ld kB\n", &free
);
615 return bprintf("%d", 100 * (total
- free
- cached
) / total
);
626 fp
= fopen("/proc/meminfo", "r");
628 warn("Failed to open file /proc/meminfo");
631 if ((bytes_read
= fread(buf
, sizeof(char), sizeof(buf
) - 1, fp
)) == 0) {
632 warn("swap_total: read error");
638 if ((match
= strstr(buf
, "SwapTotal")) == NULL
)
640 sscanf(match
, "SwapTotal: %ld kB\n", &total
);
642 return bprintf("%f", (float)total
/ 1024 / 1024);
648 long total
, free
, cached
;
653 fp
= fopen("/proc/meminfo", "r");
655 warn("Failed to open file /proc/meminfo");
658 if ((bytes_read
= fread(buf
, sizeof(char), sizeof(buf
) - 1, fp
)) == 0) {
659 warn("swap_used: read error");
665 if ((match
= strstr(buf
, "SwapTotal")) == NULL
)
667 sscanf(match
, "SwapTotal: %ld kB\n", &total
);
669 if ((match
= strstr(buf
, "SwapCached")) == NULL
)
671 sscanf(match
, "SwapCached: %ld kB\n", &cached
);
673 if ((match
= strstr(buf
, "SwapFree")) == NULL
)
675 sscanf(match
, "SwapFree: %ld kB\n", &free
);
677 return bprintf("%f", (float)(total
- free
- cached
) / 1024 / 1024);
681 temp(const char *file
)
686 fp
= fopen(file
, "r");
688 warn("Failed to open file %s", file
);
691 n
= fscanf(fp
, "%d", &temp
);
696 return bprintf("%d", temp
/ 1000);
707 h
= info
.uptime
/ 3600;
708 m
= (info
.uptime
- h
* 3600 ) / 60;
710 return bprintf("%dh %dm", h
, m
);
716 struct passwd
*pw
= getpwuid(geteuid());
719 warn("Failed to get username");
723 return bprintf("%s", pw
->pw_name
);
729 return bprintf("%d", geteuid());
734 vol_perc(const char *card
)
738 char *vnames
[] = SOUND_DEVICE_NAMES
;
740 afd
= open(card
, O_RDONLY
| O_NONBLOCK
);
742 warn("Cannot open %s", card
);
746 if (ioctl(afd
, SOUND_MIXER_READ_DEVMASK
, &devmask
) == -1) {
747 warn("Cannot get volume for %s", card
);
751 for (i
= 0; i
< LEN(vnames
); i
++) {
752 if (devmask
& (1 << i
) && !strcmp("vol", vnames
[i
])) {
753 if (ioctl(afd
, MIXER_READ(i
), &v
) == -1) {
754 warn("vol_perc: ioctl");
763 return bprintf("%d", v
& 0xff);
767 wifi_perc(const char *iface
)
775 snprintf(path
, sizeof(path
), "%s%s%s", "/sys/class/net/", iface
, "/operstate");
776 fp
= fopen(path
, "r");
778 warn("Failed to open file %s", path
);
781 p
= fgets(status
, 5, fp
);
783 if(!p
|| strcmp(status
, "up\n") != 0) {
787 fp
= fopen("/proc/net/wireless", "r");
789 warn("Failed to open file /proc/net/wireless");
793 for (i
= 0; i
< 3; i
++) {
794 if (!(p
= fgets(buf
, sizeof(buf
) - 1, fp
)))
801 if ((datastart
= strstr(buf
, iface
)) == NULL
)
804 datastart
= (datastart
+(strlen(iface
)+1));
805 sscanf(datastart
+ 1, " %*d %d %*d %*d %*d %*d %*d %*d %*d %*d", &perc
);
807 return bprintf("%d", perc
);
811 wifi_essid(const char *iface
)
813 static char id
[IW_ESSID_MAX_SIZE
+1];
814 int sockfd
= socket(AF_INET
, SOCK_DGRAM
, 0);
817 memset(&wreq
, 0, sizeof(struct iwreq
));
818 wreq
.u
.essid
.length
= IW_ESSID_MAX_SIZE
+1;
819 snprintf(wreq
.ifr_name
, sizeof(wreq
.ifr_name
), "%s", iface
);
822 warn("Failed to get ESSID for interface %s", iface
);
825 wreq
.u
.essid
.pointer
= id
;
826 if (ioctl(sockfd
,SIOCGIWESSID
, &wreq
) == -1) {
827 warn("Failed to get ESSID for interface %s", iface
);
833 if (strcmp(id
, "") == 0)
840 sighandler(const int signo
)
842 if (signo
== SIGTERM
|| signo
== SIGINT
) {
850 fprintf(stderr
, "usage: %s [-s]\n", argv0
);
855 main(int argc
, char *argv
[])
858 struct sigaction act
;
861 char status_string
[MAXLEN
];
876 memset(&act
, 0, sizeof(act
));
877 act
.sa_handler
= sighandler
;
878 sigaction(SIGINT
, &act
, 0);
879 sigaction(SIGTERM
, &act
, 0);
882 dpy
= XOpenDisplay(NULL
);
884 fprintf(stderr
, "slstatus: cannot open display");
889 setlocale(LC_ALL
, "");
892 status_string
[0] = '\0';
894 for (element
= status_string
, i
= len
= 0; i
< LEN(args
);
895 ++i
, element
+= len
) {
897 len
= snprintf(element
, sizeof(status_string
)-1 - len
,
899 argument
.func(argument
.args
));
900 if (len
>= sizeof(status_string
)) {
901 status_string
[sizeof(status_string
)-1] = '\0';
907 printf("%s\n", status_string
);
909 XStoreName(dpy
, DefaultRootWindow(dpy
), status_string
);
913 if ((update_interval
- delay
) <= 0) {
917 sleep(update_interval
- delay
);
923 XStoreName(dpy
, DefaultRootWindow(dpy
), NULL
);