My slstatus configuration
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

366 lines
8.2 KiB

  1. /* See LICENSE file for copyright and license details. */
  2. /* global libraries */
  3. #include <alsa/asoundlib.h>
  4. #include <stdarg.h>
  5. #include <stdio.h>
  6. #include <stdlib.h>
  7. #include <string.h>
  8. #include <time.h>
  9. #include <unistd.h>
  10. #include <X11/Xlib.h>
  11. /* local libraries */
  12. #include "config.h"
  13. /* functions */
  14. void setstatus(char *str);
  15. char *get_battery();
  16. char *get_cpu_temperature();
  17. char *get_cpu_usage();
  18. char *get_datetime();
  19. char *get_ram_usage();
  20. char *smprintf(char *fmt, ...);
  21. char *get_volume();
  22. char *get_wifi_signal();
  23. /* global variables */
  24. static Display *dpy;
  25. /* set statusbar (WM_NAME) */
  26. void
  27. setstatus(char *str)
  28. {
  29. XStoreName(dpy, DefaultRootWindow(dpy), str);
  30. XSync(dpy, False);
  31. }
  32. /* battery percentage */
  33. char *
  34. get_battery()
  35. {
  36. int battery_now, battery_full, battery_perc;
  37. FILE *fp;
  38. /* open battery now file */
  39. if (!(fp = fopen(batterynowfile, "r"))) {
  40. fprintf(stderr, "Error opening battery file.");
  41. exit(1);
  42. }
  43. /* read value */
  44. fscanf(fp, "%i", &battery_now);
  45. /* close battery now file */
  46. fclose(fp);
  47. /* open battery full file */
  48. if (!(fp = fopen(batteryfullfile, "r"))) {
  49. fprintf(stderr, "Error opening battery file.");
  50. exit(1);
  51. }
  52. /* read value */
  53. fscanf(fp, "%i", &battery_full);
  54. /* close battery full file */
  55. fclose(fp);
  56. /* calculate percent */
  57. battery_perc = battery_now / (battery_full / 100);
  58. /* return batt_perc as string */
  59. return smprintf("%d%%", battery_perc);
  60. }
  61. /* cpu temperature */
  62. char *
  63. get_cpu_temperature()
  64. {
  65. int temperature;
  66. FILE *fp;
  67. /* open temperature file */
  68. if (!(fp = fopen(tempfile, "r"))) {
  69. fprintf(stderr, "Could not open temperature file.\n");
  70. exit(1);
  71. }
  72. /* extract temperature */
  73. fscanf(fp, "%d", &temperature);
  74. /* close temperature file */
  75. fclose(fp);
  76. /* return temperature in degrees */
  77. return smprintf("%d°C", temperature / 1000);
  78. }
  79. /* cpu percentage */
  80. char *
  81. get_cpu_usage()
  82. {
  83. int cpu_perc;
  84. long double a[4], b[4];
  85. FILE *fp;
  86. /* open stat file */
  87. if (!(fp = fopen("/proc/stat","r"))) {
  88. fprintf(stderr, "Error opening stat file.");
  89. exit(1);
  90. }
  91. /* read values */
  92. fscanf(fp, "%*s %Lf %Lf %Lf %Lf", &a[0], &a[1], &a[2], &a[3]);
  93. /* close stat file */
  94. fclose(fp);
  95. /* wait a second (for avg values) */
  96. sleep(1);
  97. /* open stat file */
  98. if (!(fp = fopen("/proc/stat","r"))) {
  99. fprintf(stderr, "Error opening stat file.");
  100. exit(1);
  101. }
  102. /* read values */
  103. fscanf(fp, "%*s %Lf %Lf %Lf %Lf", &b[0], &b[1], &b[2], &b[3]);
  104. /* close stat file */
  105. fclose(fp);
  106. /* calculate avg in this second */
  107. cpu_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]));
  108. /* return cpu_perc as string */
  109. return smprintf("%d%%", cpu_perc);
  110. }
  111. /* date and time */
  112. char *
  113. get_datetime()
  114. {
  115. time_t tm;
  116. size_t bufsize = 64;
  117. char *buf = malloc(bufsize);
  118. /* get time in format */
  119. time(&tm);
  120. if(!strftime(buf, bufsize, timeformat, localtime(&tm))) {
  121. fprintf(stderr, "Strftime failed.\n");
  122. exit(1);
  123. }
  124. /* return time */
  125. return buf;
  126. }
  127. /* ram percentage */
  128. char *
  129. get_ram_usage()
  130. {
  131. int ram_perc;
  132. long total, free, buffers, cached;
  133. FILE *fp;
  134. /* open meminfo file */
  135. if (!(fp = fopen("/proc/meminfo", "r"))) {
  136. fprintf(stderr, "Error opening meminfo file.");
  137. exit(1);
  138. }
  139. /* read the values */
  140. fscanf(fp, "MemTotal: %ld kB\n", &total);
  141. fscanf(fp, "MemFree: %ld kB\n", &free);
  142. fscanf(fp, "MemAvailable: %ld kB\nBuffers: %ld kB\n", &buffers, &buffers);
  143. fscanf(fp, "Cached: %ld kB\n", &cached);
  144. /* close meminfo file */
  145. fclose(fp);
  146. /* calculate percentage */
  147. ram_perc = 100 * ((total - free) - (buffers + cached)) / total;
  148. /* return ram_perc as string */
  149. return smprintf("%d%%",ram_perc);
  150. }
  151. /* smprintf function */
  152. char *
  153. smprintf(char *fmt, ...)
  154. {
  155. va_list fmtargs;
  156. char *ret;
  157. int len;
  158. va_start(fmtargs, fmt);
  159. len = vsnprintf(NULL, 0, fmt, fmtargs);
  160. va_end(fmtargs);
  161. ret = malloc(++len);
  162. if (ret == NULL) {
  163. fprintf(stderr, "Malloc error.");
  164. exit(1);
  165. }
  166. va_start(fmtargs, fmt);
  167. vsnprintf(ret, len, fmt, fmtargs);
  168. va_end(fmtargs);
  169. return ret;
  170. }
  171. /* alsa volume percentage */
  172. char *
  173. get_volume()
  174. {
  175. int mute = 0;
  176. long vol = 0, max = 0, min = 0;
  177. /* get volume from alsa */
  178. snd_mixer_t *handle;
  179. snd_mixer_elem_t *pcm_mixer, *mas_mixer;
  180. snd_mixer_selem_id_t *vol_info, *mute_info;
  181. snd_mixer_open(&handle, 0);
  182. snd_mixer_attach(handle, soundcard);
  183. snd_mixer_selem_register(handle, NULL, NULL);
  184. snd_mixer_load(handle);
  185. snd_mixer_selem_id_malloc(&vol_info);
  186. snd_mixer_selem_id_malloc(&mute_info);
  187. snd_mixer_selem_id_set_name(vol_info, channel);
  188. snd_mixer_selem_id_set_name(mute_info, channel);
  189. pcm_mixer = snd_mixer_find_selem(handle, vol_info);
  190. mas_mixer = snd_mixer_find_selem(handle, mute_info);
  191. snd_mixer_selem_get_playback_volume_range((snd_mixer_elem_t *)pcm_mixer, &min, &max);
  192. snd_mixer_selem_get_playback_volume((snd_mixer_elem_t *)pcm_mixer, SND_MIXER_SCHN_MONO, &vol);
  193. snd_mixer_selem_get_playback_switch(mas_mixer, SND_MIXER_SCHN_MONO, &mute);
  194. if (vol_info)
  195. snd_mixer_selem_id_free(vol_info);
  196. if (mute_info)
  197. snd_mixer_selem_id_free(mute_info);
  198. if (handle)
  199. snd_mixer_close(handle);
  200. /* return the string (mute) */
  201. if (!mute)
  202. return smprintf("mute");
  203. else
  204. return smprintf("%d%%", (vol * 100) / max);
  205. }
  206. /* wifi percentage */
  207. char *
  208. get_wifi_signal()
  209. {
  210. int bufsize = 255;
  211. int strength;
  212. char buf[bufsize];
  213. char *datastart;
  214. char path_start[16] = "/sys/class/net/";
  215. char path_end[11] = "/operstate";
  216. char path[32];
  217. char status[5];
  218. char needle[sizeof wificard + 1];
  219. FILE *fp;
  220. /* generate the path name */
  221. memset(path, 0, sizeof path);
  222. strcat(path, path_start);
  223. strcat(path, wificard);
  224. strcat(path, path_end);
  225. /* open wifi file */
  226. if(!(fp = fopen(path, "r"))) {
  227. fprintf(stderr, "Error opening wifi operstate file.");
  228. exit(1);
  229. }
  230. /* read the status */
  231. fgets(status, 5, fp);
  232. /* close wifi file */
  233. fclose(fp);
  234. /* check if interface down */
  235. if(strcmp(status, "up\n") != 0){
  236. return "n/a";
  237. }
  238. /* open wifi file */
  239. if (!(fp = fopen("/proc/net/wireless", "r"))) {
  240. fprintf(stderr, "Error opening wireless file.");
  241. exit(1);
  242. }
  243. /* extract the signal strength */
  244. strcpy(needle, wificard);
  245. strcat(needle, ":");
  246. fgets(buf, bufsize, fp);
  247. fgets(buf, bufsize, fp);
  248. fgets(buf, bufsize, fp);
  249. if ((datastart = strstr(buf, needle)) != NULL) {
  250. datastart = strstr(buf, ":");
  251. sscanf(datastart + 1, " %*d %d %*d %*d %*d %*d %*d %*d %*d %*d", &strength);
  252. }
  253. /* close wifi file */
  254. fclose(fp);
  255. /* return strength in percent */
  256. return smprintf("%d%%", strength);
  257. }
  258. /* main function */
  259. int
  260. main()
  261. {
  262. char status[1024];
  263. char *wifi_signal = NULL;
  264. char *battery = NULL;
  265. char *cpu_usage = NULL;
  266. char *cpu_temperature = NULL;
  267. char *ram_usage = NULL;
  268. char *volume = NULL;
  269. char *datetime = NULL;
  270. /* open display */
  271. if (!(dpy = XOpenDisplay(0x0))) {
  272. fprintf(stderr, "Cannot open display!\n");
  273. exit(1);
  274. }
  275. /* return status every second */
  276. for (;;) {
  277. /* assign the values */
  278. wifi_signal = get_wifi_signal();
  279. battery = get_battery();
  280. cpu_usage = get_cpu_usage();
  281. cpu_temperature = get_cpu_temperature();
  282. ram_usage = get_ram_usage();
  283. volume = get_volume();
  284. datetime = get_datetime();
  285. /* return the status */
  286. sprintf(status, FORMATSTRING, ARGUMENTS);
  287. setstatus(status);
  288. /* free the values */
  289. free(wifi_signal);
  290. free(battery);
  291. free(cpu_usage);
  292. free(cpu_temperature);
  293. free(ram_usage);
  294. free(volume);
  295. free(datetime);
  296. }
  297. /* close display */
  298. XCloseDisplay(dpy);
  299. /* exit successfully */
  300. return 0;
  301. }