Xinqi Bao's Git

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