15 char *sasprintf(const char *fmt
, ...) {
19 if (vasprintf(&result
, fmt
, args
) == -1) {
20 fprintf(stderr
, "Can't allocate memory!?\n");
28 char *voltage_filename
;
29 char *current_filename
;
34 static float fatof(const char *filename
) {
35 FILE *f
= fopen(filename
, "r");
40 const int bufsize
= 1024;
42 const int read
= fread(buf
, sizeof(char), bufsize
, f
);
43 if (read
== 0 || read
== bufsize
|| ferror(f
)) {
51 float val
= strtof(buf
, &end
);
52 int parsed
= end
- buf
;
53 gboolean parsed_all
= parsed
== read
;
54 gboolean parsed_all_but_space
= parsed
== read
- 1 && isspace(*end
);
55 gboolean parse_ok
= parsed_all
|| parsed_all_but_space
;
56 if (errno
!= 0 || parsed
== 0 || !parse_ok
) {
62 static gboolean
collect_data(struct State
*state
) {
63 float now
= g_get_monotonic_time() / 1e6
;
64 float voltage
= fatof(state
->voltage_filename
);
65 float current
= fatof(state
->current_filename
);
66 bv_chart_add_point(state
->voltage
, now
, voltage
);
67 bv_chart_add_point(state
->current
, now
, current
);
68 gtk_widget_queue_draw(GTK_WIDGET(state
->voltage
));
69 gtk_widget_queue_draw(GTK_WIDGET(state
->current
));
73 static void activate(GtkApplication
*app
, gpointer user_data
) {
75 GtkWidget
*window
= gtk_application_window_new(app
);
76 gtk_window_set_title(GTK_WINDOW(window
), "BatteryViewer");
77 gtk_window_set_default_size(GTK_WINDOW(window
), 200, 200);
79 GtkWidget
*box
= gtk_box_new(GTK_ORIENTATION_VERTICAL
, 1);
80 gtk_container_add(GTK_CONTAINER(window
), box
);
82 struct State
*state
= (struct State
*)user_data
;
83 state
->voltage
= BV_CHART(bv_chart_new());
84 gboolean expand
= TRUE
;
87 gtk_box_pack_start(GTK_BOX(box
), GTK_WIDGET(state
->voltage
), expand
, fill
,
90 state
->current
= BV_CHART(bv_chart_new());
91 gtk_box_pack_end(GTK_BOX(box
), GTK_WIDGET(state
->current
), expand
, fill
,
94 gtk_widget_show_all(window
);
96 g_timeout_add_seconds(1, (GSourceFunc
)collect_data
, user_data
);
99 char *find_battery_dir() {
100 glob_t globbuf
= {.gl_offs
= 0};
101 char pattern
[] = "/sys/class/power_supply/*[Bb][Aa][Tt]*";
102 if (glob(pattern
, GLOB_ERR
, NULL
, &globbuf
) != 0 || globbuf
.gl_pathc
== 0) {
103 fprintf(stderr
, "Could not find battery dir %s\n", pattern
);
106 if (globbuf
.gl_pathc
!= 1) {
107 // TODO: Filter for directories that have {voltage,current}_now files.
108 // TODO: Support multiple batteries
109 fprintf(stderr
, "Multiple batteries found. Arbitrarily choosing %s\n",
110 globbuf
.gl_pathv
[0]);
112 char *dir
= strdup(globbuf
.gl_pathv
[0]);
117 int main(int argc
, char **argv
) {
118 char *battery_dir
= find_battery_dir();
119 struct State state
= {
120 .voltage_filename
= sasprintf("%s/voltage_now", battery_dir
),
121 .current_filename
= sasprintf("%s/current_now", battery_dir
),
124 GtkApplication
*app
= gtk_application_new("com.scottworley.batteryviewer",
125 G_APPLICATION_DEFAULT_FLAGS
);
126 g_signal_connect(app
, "activate", G_CALLBACK(activate
), &state
);
127 int status
= g_application_run(G_APPLICATION(app
), argc
, argv
);