My slstatus configuration
選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。
 
 
 
 

361 行
7.9 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 *battery();
  16. char *cpu_temperature();
  17. char *cpu_usage();
  18. char *datetime();
  19. char *ram_usage();
  20. char *smprintf(char *fmt, ...);
  21. char *volume();
  22. char *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. 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. 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. 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. datetime()
  114. {
  115. time_t tm;
  116. size_t bufsize = 19;
  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. ram_usage()
  130. {
  131. int ram_perc;
  132. long total, free, available;
  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\n", &available);
  143. /* close meminfo file */
  144. fclose(fp);
  145. /* calculate percentage */
  146. ram_perc = 100 * (total - available) / total;
  147. /* return ram_perc as string */
  148. return smprintf("%d%%",ram_perc);
  149. }
  150. /* smprintf function */
  151. char *
  152. smprintf(char *fmt, ...)
  153. {
  154. va_list fmtargs;
  155. char *ret;
  156. int len;
  157. va_start(fmtargs, fmt);
  158. len = vsnprintf(NULL, 0, fmt, fmtargs);
  159. va_end(fmtargs);
  160. ret = malloc(++len);
  161. if (ret == NULL) {
  162. fprintf(stderr, "Malloc error.");
  163. exit(1);
  164. }
  165. va_start(fmtargs, fmt);
  166. vsnprintf(ret, len, fmt, fmtargs);
  167. va_end(fmtargs);
  168. return ret;
  169. }
  170. /* alsa volume percentage */
  171. char *
  172. volume()
  173. {
  174. int mute = 0;
  175. long vol = 0, max = 0, min = 0;
  176. /* get volume from alsa */
  177. snd_mixer_t *handle;
  178. snd_mixer_elem_t *pcm_mixer, *mas_mixer;
  179. snd_mixer_selem_id_t *vol_info, *mute_info;
  180. snd_mixer_open(&handle, 0);
  181. snd_mixer_attach(handle, soundcard);
  182. snd_mixer_selem_register(handle, NULL, NULL);
  183. snd_mixer_load(handle);
  184. snd_mixer_selem_id_malloc(&vol_info);
  185. snd_mixer_selem_id_malloc(&mute_info);
  186. snd_mixer_selem_id_set_name(vol_info, channel);
  187. snd_mixer_selem_id_set_name(mute_info, channel);
  188. pcm_mixer = snd_mixer_find_selem(handle, vol_info);
  189. mas_mixer = snd_mixer_find_selem(handle, mute_info);
  190. snd_mixer_selem_get_playback_volume_range((snd_mixer_elem_t *)pcm_mixer, &min, &max);
  191. snd_mixer_selem_get_playback_volume((snd_mixer_elem_t *)pcm_mixer, SND_MIXER_SCHN_MONO, &vol);
  192. snd_mixer_selem_get_playback_switch(mas_mixer, SND_MIXER_SCHN_MONO, &mute);
  193. if (vol_info)
  194. snd_mixer_selem_id_free(vol_info);
  195. if (mute_info)
  196. snd_mixer_selem_id_free(mute_info);
  197. if (handle)
  198. snd_mixer_close(handle);
  199. /* return the string (mute) */
  200. if (!mute)
  201. return smprintf("mute");
  202. else
  203. return smprintf("%d%%", (vol * 100) / max);
  204. }
  205. /* wifi percentage */
  206. char *
  207. wifi_signal()
  208. {
  209. int bufsize = 255;
  210. int strength;
  211. char buf[bufsize];
  212. char *datastart;
  213. char path_start[16] = "/sys/class/net/";
  214. char path_end[11] = "/operstate";
  215. char path[32];
  216. char status[5];
  217. char needle[sizeof wificard + 1];
  218. FILE *fp;
  219. /* generate the path name */
  220. memset(path, 0, sizeof path);
  221. strcat(path, path_start);
  222. strcat(path, wificard);
  223. strcat(path, path_end);
  224. /* open wifi file */
  225. if(!(fp = fopen(path, "r"))) {
  226. fprintf(stderr, "Error opening wifi operstate file.");
  227. exit(1);
  228. }
  229. /* read the status */
  230. fgets(status, 5, fp);
  231. /* close wifi file */
  232. fclose(fp);
  233. /* check if interface down */
  234. if(strcmp(status, "up\n") != 0){
  235. return "n/a";
  236. }
  237. /* open wifi file */
  238. if (!(fp = fopen("/proc/net/wireless", "r"))) {
  239. fprintf(stderr, "Error opening wireless file.");
  240. exit(1);
  241. }
  242. /* extract the signal strength */
  243. strcpy(needle, wificard);
  244. strcat(needle, ":");
  245. fgets(buf, bufsize, fp);
  246. fgets(buf, bufsize, fp);
  247. fgets(buf, bufsize, fp);
  248. if ((datastart = strstr(buf, needle)) != NULL) {
  249. datastart = strstr(buf, ":");
  250. sscanf(datastart + 1, " %*d %d %*d %*d %*d %*d %*d %*d %*d %*d", &strength);
  251. }
  252. /* close wifi file */
  253. fclose(fp);
  254. /* return strength in percent */
  255. return smprintf("%d%%", strength);
  256. }
  257. /* main function */
  258. int
  259. main()
  260. {
  261. char status[1024];
  262. /* open display */
  263. if (!(dpy = XOpenDisplay(0x0))) {
  264. fprintf(stderr, "Cannot open display!\n");
  265. exit(1);
  266. }
  267. char *pWifi_signal = NULL;
  268. char *pBattery = NULL;
  269. char *pCpu_usage = NULL;
  270. char *pCpu_temperature = NULL;
  271. char *pRam_usage = NULL;
  272. char *pVolume = NULL;
  273. char *pDatetime = NULL;
  274. /* return status every second */
  275. for (;;) {
  276. pWifi_signal = wifi_signal();
  277. pBattery = battery();
  278. pCpu_usage = cpu_usage();
  279. pCpu_temperature = cpu_temperature();
  280. pRam_usage = ram_usage();
  281. pVolume = volume();
  282. pDatetime = datetime();
  283. sprintf(status, FORMATSTRING, ARGUMENTS);
  284. setstatus(status);
  285. free(pWifi_signal);
  286. free(pBattery);
  287. free(pCpu_usage);
  288. free(pCpu_temperature);
  289. free(pRam_usage);
  290. free(pVolume);
  291. free(pDatetime);
  292. }
  293. /* close display */
  294. XCloseDisplay(dpy);
  295. /* exit successfully */
  296. return 0;
  297. }