]> git.scottworley.com Git - batteryviewer/blobdiff - batteryviewer.c
License
[batteryviewer] / batteryviewer.c
index 7be007c3ff54aad376199c1e38bd32fbb0dd04d8..a0fae4cd361a0f5917a203ecb781a0aa50962410 100644 (file)
+/* batteryviewer: Display battery metrics
+ * Copyright (C) 2023 Scott Worley <scottworley@scottworley.com>
+
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+
+ * You should have received a copy of the GNU General Public License
+ * along with this program.  If not, see <https://www.gnu.org/licenses/>.
+*/
+
+// For vasprintf
+#define _GNU_SOURCE
+
+#include "chart.h"
+#include <ctype.h>
+#include <errno.h>
+#include <glob.h>
 #include <gtk/gtk.h>
+#include <math.h>
+#include <stdarg.h>
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
 
-static void print_hello(GtkWidget *widget __attribute__((unused)),
-                        gpointer data __attribute__((unused))) {
-  g_print("Hello World\n");
+char *sasprintf(const char *fmt, ...) {
+  char *result;
+  va_list args;
+  va_start(args, fmt);
+  if (vasprintf(&result, fmt, args) == -1) {
+    fprintf(stderr, "Can't allocate memory!?\n");
+    abort();
+  }
+  va_end(args);
+  return result;
 }
 
-static void activate(GtkApplication *app,
-                     gpointer user_data __attribute__((unused))) {
-  GtkWidget *window;
-  GtkWidget *button;
+struct State {
+  char *voltage_filename;
+  char *current_filename;
+  BVChart *voltage;
+  BVChart *current;
+};
+
+static float fatof(const char *filename) {
+  FILE *f = fopen(filename, "r");
+  if (f == NULL) {
+    return NAN;
+  }
+
+  const int bufsize = 1024;
+  char buf[bufsize];
+  const int read = fread(buf, sizeof(char), bufsize, f);
+  if (read == 0 || read == bufsize || ferror(f)) {
+    fclose(f);
+    return NAN;
+  }
+  fclose(f);
+
+  char *end;
+  errno = 0;
+  float val = strtof(buf, &end);
+  int parsed = end - buf;
+  gboolean parsed_all = parsed == read;
+  gboolean parsed_all_but_space = parsed == read - 1 && isspace(*end);
+  gboolean parse_ok = parsed_all || parsed_all_but_space;
+  if (errno != 0 || parsed == 0 || !parse_ok) {
+    return NAN;
+  }
+  return val;
+}
 
-  window = gtk_application_window_new(app);
-  gtk_window_set_title(GTK_WINDOW(window), "Window");
+static gboolean collect_data(struct State *state) {
+  float now = g_get_monotonic_time() / 1e6;
+  float voltage = fatof(state->voltage_filename);
+  float current = fatof(state->current_filename);
+  if (!isnan(voltage)) {
+    bv_chart_add_point(state->voltage, now, voltage);
+    gtk_widget_queue_draw(GTK_WIDGET(state->voltage));
+  }
+  if (!isnan(current)) {
+    bv_chart_add_point(state->current, now, current);
+    gtk_widget_queue_draw(GTK_WIDGET(state->current));
+  }
+  return TRUE;
+}
+
+static void activate(GtkApplication *app, gpointer user_data) {
+
+  GtkWidget *window = gtk_application_window_new(app);
+  gtk_window_set_title(GTK_WINDOW(window), "BatteryViewer");
   gtk_window_set_default_size(GTK_WINDOW(window), 200, 200);
 
-  button = gtk_button_new_with_label("Hello World");
-  g_signal_connect(button, "clicked", G_CALLBACK(print_hello), NULL);
-  gtk_window_set_child(GTK_WINDOW(window), button);
+  GtkWidget *box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 1);
+  gtk_container_add(GTK_CONTAINER(window), box);
+
+  struct State *state = (struct State *)user_data;
+  state->voltage = BV_CHART(bv_chart_new());
+  gboolean expand = TRUE;
+  gboolean fill = TRUE;
+  guint padding = 0;
+  gtk_box_pack_start(GTK_BOX(box), GTK_WIDGET(state->voltage), expand, fill,
+                     padding);
+
+  state->current = BV_CHART(bv_chart_new());
+  gtk_box_pack_end(GTK_BOX(box), GTK_WIDGET(state->current), expand, fill,
+                   padding);
+
+  gtk_widget_show_all(window);
+
+  g_timeout_add_seconds(1, (GSourceFunc)collect_data, user_data);
+}
 
-  gtk_window_present(GTK_WINDOW(window));
+char *find_battery_dir() {
+  glob_t globbuf = {.gl_offs = 0};
+  char pattern[] = "/sys/class/power_supply/*[Bb][Aa][Tt]*";
+  if (glob(pattern, GLOB_ERR, NULL, &globbuf) != 0 || globbuf.gl_pathc == 0) {
+    fprintf(stderr, "Could not find battery dir %s\n", pattern);
+    exit(1);
+  }
+  if (globbuf.gl_pathc != 1) {
+    // TODO: Filter for directories that have {voltage,current}_now files.
+    // TODO: Support multiple batteries
+    fprintf(stderr, "Multiple batteries found. Arbitrarily choosing %s\n",
+            globbuf.gl_pathv[0]);
+  }
+  char *dir = strdup(globbuf.gl_pathv[0]);
+  globfree(&globbuf);
+  return dir;
 }
 
 int main(int argc, char **argv) {
-  GtkApplication *app;
-  int status;
+  char *battery_dir = find_battery_dir();
+  struct State state = {
+      .voltage_filename = sasprintf("%s/voltage_now", battery_dir),
+      .current_filename = sasprintf("%s/current_now", battery_dir),
+  };
 
-  app = gtk_application_new("org.gtk.example", G_APPLICATION_DEFAULT_FLAGS);
-  g_signal_connect(app, "activate", G_CALLBACK(activate), NULL);
-  status = g_application_run(G_APPLICATION(app), argc, argv);
+  GtkApplication *app = gtk_application_new(
+      "com.scottworley.batteryviewer",
+      // G_APPLICATION_FLAGS_NONE is deprecated, but
+      // G_APPLICATION_DEFAULT_FLAGS isn't available on Stable Debian yet.
+      G_APPLICATION_FLAGS_NONE);
+  g_signal_connect(app, "activate", G_CALLBACK(activate), &state);
+  int status = g_application_run(G_APPLICATION(app), argc, argv);
   g_object_unref(app);
 
   return status;