Subject: pkg/34961: sysutils/gkrellm support for acpibat
To: None <pkg-manager@netbsd.org, gnats-admin@netbsd.org,>
From: \"\" <anthony.mallet@useless-ficus.net>
List: pkgsrc-bugs
Date: 11/01/2006 13:45:00
>Number:         34961
>Category:       pkg
>Synopsis:       sysutils/gkrellm support for acpibat
>Confidential:   no
>Severity:       non-critical
>Priority:       low
>Responsible:    pkg-manager
>State:          open
>Class:          change-request
>Submitter-Id:   net
>Arrival-Date:   Wed Nov 01 13:45:00 +0000 2006
>Originator:     "\"\" <Anthony Mallet" <anthony.mallet@useless-ficus.net>
>Release:        NetBSD 4.99.3 i386
>Organization:
>Environment:


System: NetBSD 4.99.3 (CITRUS) #4: Sat Oct 21 03:38:26 CEST 2006
	troot@citrus:/usr/obj/sys/arch/i386/compile/CITRUS



>Description:


I added support for acpibat in gkrellm battery monitor, for those laptops that don't have
apm support. This is implemented via envsys(4) API.

The gkrellm_sys_battery_init() function first look if apm support is present, and falls back to
 envsys(4) if /dev/apm cannot be opened.

Patches are provided and diffed against 2.2.10 *patched* version that I provided in
PR#34957. If you name my patch files patch-b{a,b,c} this should work :p.

Cheers.



>How-To-Repeat:





>Fix:


--- patch-ba begins here ---
$NetBSD$

--- src/sysdeps/openbsd.c~	2006-03-30 00:23:37.000000000 +0200
+++ src/sysdeps/openbsd.c	2006-10-31 22:12:03.000000000 +0100
@@ -199,3 +199,55 @@
 	return FALSE;
 	}
 
+
+/* ===================================================================== */
+/* Battery monitor interface */
+#include <sys/ioctl.h>
+
+#if defined(__i386__) || defined(__powerpc__)
+
+#include <machine/apmvar.h>
+#define	APMDEV		"/dev/apm"
+
+void
+gkrellm_sys_battery_read_data(void)
+	{
+	int			f, r;
+	struct apm_power_info info;
+	gboolean    available, on_line, charging;
+	gint        percent, time_left;
+
+	if ((f = open(APMDEV, O_RDONLY)) == -1) return;
+	memset(&info, 0, sizeof(info));
+	r = ioctl(f, APM_IOC_GETPOWER, &info);
+	close(f);
+	if (r == -1) return;
+
+	available = (info.battery_state != APM_BATT_UNKNOWN);
+	on_line = (info.ac_state == APM_AC_ON) ? TRUE : FALSE;
+	charging = (info.battery_state == APM_BATT_CHARGING) ? TRUE : FALSE;
+	percent = info.battery_life;
+	time_left = info.minutes_left;
+	gkrellm_battery_assign_data(0, available, on_line, charging,
+				percent, time_left);
+	}
+
+gboolean
+gkrellm_sys_battery_init()
+	{
+	return TRUE;
+	}
+
+#else
+
+void
+gkrellm_sys_battery_read_data(void)
+	{
+	}
+
+gboolean
+gkrellm_sys_battery_init()
+	{
+	return FALSE;
+	}
+#endif
--- patch-ba ends here ---
--- patch-bb begins here ---
$NetBSD$

--- src/sysdeps/bsd-net-open.c~	2006-11-01 12:06:53.000000000 +0100
+++ src/sysdeps/bsd-net-open.c	2006-11-01 12:07:01.000000000 +0100
@@ -98,57 +98,5 @@
 gboolean
 gkrellm_sys_inet_init(void)
 	{
 	return TRUE;
 	}
-
-/* ===================================================================== */
-/* Battery monitor interface */
-#include <sys/ioctl.h>
-
-#if defined(__i386__) || defined(__powerpc__)
-
-#include <machine/apmvar.h>
-#define	APMDEV		"/dev/apm"
-
-void
-gkrellm_sys_battery_read_data(void)
-	{
-	int			f, r;
-	struct apm_power_info info;
-	gboolean    available, on_line, charging;
-	gint        percent, time_left;
-
-	if ((f = open(APMDEV, O_RDONLY)) == -1) return;
-	memset(&info, 0, sizeof(info));
-	r = ioctl(f, APM_IOC_GETPOWER, &info);
-	close(f);
-	if (r == -1) return;
-
-	available = (info.battery_state != APM_BATT_UNKNOWN);
-	on_line = (info.ac_state == APM_AC_ON) ? TRUE : FALSE;
-	charging = (info.battery_state == APM_BATT_CHARGING) ? TRUE : FALSE;
-	percent = info.battery_life;
-	time_left = info.minutes_left;
-	gkrellm_battery_assign_data(0, available, on_line, charging,
-				percent, time_left);
-	}
-
-gboolean
-gkrellm_sys_battery_init()
-	{
-	return TRUE;
-	}
-
-#else
-
-void
-gkrellm_sys_battery_read_data(void)
-	{
-	}
-
-gboolean
-gkrellm_sys_battery_init()
-	{
-	return FALSE;
-	}
-#endif
--- patch-bb ends here ---
--- patch-bc begins here ---
$NetBSD$

--- src/sysdeps/netbsd.c~	2006-03-30 00:23:37.000000000 +0200
+++ src/sysdeps/netbsd.c	2006-10-31 22:17:41.000000000 +0100
@@ -336,3 +336,226 @@
    return found_sensors;
 }
 
+
+
+/* ===================================================================== */
+/* Battery monitor interface */
+
+#if defined(__i386__) || defined(__powerpc__)
+# include <sys/ioctl.h>
+# include <machine/apmvar.h>
+
+# define	APMDEV		"/dev/apm"
+#endif
+
+static int battery_use_apm = 1;
+
+/* battery data, index by [battery number] */
+static struct battery_acpi_data {
+  gboolean available;	int available_index;
+  gboolean on_line;	int on_line_index;
+  gboolean charging;	int charging_index;
+  gint full_cap;	int full_cap_index;
+  gint cap;		int cap_index;
+  gint discharge_rate;	int discharge_rate_index;
+  gint charge_rate;	int charge_rate_index;
+} *battery_acpi_data;
+static int battery_acpi_data_items;
+
+static int
+gkrellm_battery_data_alloc(int bat)
+{
+  if (bat + 1 > battery_acpi_data_items)
+    battery_acpi_data_items = bat + 1;
+
+  battery_acpi_data = realloc(battery_acpi_data,
+			      battery_acpi_data_items*
+			      sizeof(*battery_acpi_data));
+  return battery_acpi_data?1:0;
+}
+
+void
+gkrellm_sys_battery_read_data(void)
+{
+  int i;
+  int fd;			/* file desc. for /dev/sysmon or /dev/apm */
+  envsys_tre_data_t data;	/* sensor data */
+  int time_left;
+
+  if (battery_use_apm) {
+#if defined(__i386__) || defined(__powerpc__)
+    int			r;
+    struct apm_power_info apminfo;
+    gboolean available, on_line, charging;
+    gint percent, time_left;
+
+    if ((fd = open(APMDEV, O_RDONLY)) == -1) return;
+    memset(&apminfo, 0, sizeof(apminfo));
+    r = ioctl(fd, APM_IOC_GETPOWER, &apminfo);
+    close(fd);
+    if (r == -1) return;
+
+    available = (apminfo.battery_state != APM_BATT_UNKNOWN);
+    on_line = (apminfo.ac_state == APM_AC_ON) ? TRUE : FALSE;
+    charging = (apminfo.battery_state == APM_BATT_CHARGING) ? TRUE : FALSE;
+    percent = apminfo.battery_life;
+    time_left = apminfo.minutes_left;
+    gkrellm_battery_assign_data(0, available, on_line, charging,
+				percent, time_left);
+#else
+    return;
+#endif
+  }
+
+   fd = open(SENSORS_DIR, O_RDONLY);
+   if (fd < 0) return;
+
+   data.sensor = battery_acpi_data[0].on_line_index;
+   if (ioctl(fd, ENVSYS_GTREDATA, &data) < 0) return;
+   if (!(data.validflags & ENVSYS_FVALID)) return;
+   battery_acpi_data[0].on_line = data.cur.data_us ? TRUE:FALSE;
+
+   /* iterate through available batteries */
+   for(i=0; i<battery_acpi_data_items; i++) {
+#define read_sensor(x)							\
+     do {								\
+       data.sensor = battery_acpi_data[i].x ## _index;			\
+       if (ioctl(fd, ENVSYS_GTREDATA, &data) < 0) continue;		\
+       if (!(data.validflags & ENVSYS_FCURVALID))			\
+	 battery_acpi_data[i].x = -1;					\
+       else								\
+	 battery_acpi_data[i].x = data.cur.data_s;			\
+     } while(0)
+
+     read_sensor(available);
+     read_sensor(charging);
+     read_sensor(full_cap);
+     read_sensor(cap);
+     read_sensor(discharge_rate);
+     read_sensor(charge_rate);
+#undef read_sensor
+
+     if (battery_acpi_data[i].discharge_rate > 0)
+       time_left =
+	 battery_acpi_data[i].cap * 60 / battery_acpi_data[i].discharge_rate;
+     else if (battery_acpi_data[i].charge_rate > 0)
+       time_left =
+	 (battery_acpi_data[i].full_cap - battery_acpi_data[i].cap) * 60 /
+	 battery_acpi_data[i].charge_rate;
+     else
+       time_left = -1;
+
+     if (battery_acpi_data[i].available) {
+       gkrellm_battery_assign_data(i,
+				   battery_acpi_data[i].available,
+				   battery_acpi_data[0].on_line,
+				   battery_acpi_data[i].charging,
+
+				   /* percent */
+				   battery_acpi_data[i].cap * 100 /
+				   battery_acpi_data[i].full_cap,
+
+				   /* time left (minutes) */
+				   time_left);
+     } else
+       gkrellm_battery_assign_data(i, 0, 0, 0, -1, -1);
+   }
+
+   close(fd);
+}
+
+gboolean
+gkrellm_sys_battery_init()
+{
+   int fd;			/* file desc. for /dev/sysmon or /dev/apm */
+   envsys_basic_info_t info;	/* sensor misc. info */
+   gboolean found_sensors = FALSE;
+   char fake[2];
+   int r, bat;
+
+   /* --- check APM first --- */
+
+#if defined(__i386__) || defined(__powerpc__)
+   do {
+     struct apm_power_info info;
+
+     if ((fd = open(APMDEV, O_RDONLY)) == -1) break;
+     r = ioctl(fd, APM_IOC_GETPOWER, &info);
+     close(fd);
+     if (r != -1) {
+       battery_use_apm = 1;
+       return TRUE;
+     }
+   } while(0);
+#endif
+
+   /* --- check for some envsys(4) acpi battery --- */
+
+   battery_use_apm = 0;
+   battery_acpi_data_items = 0;
+   battery_acpi_data = NULL; /* this is never freed */
+
+   fd = open(SENSORS_DIR, O_RDONLY);
+   if (fd < 0) return FALSE;
+
+   /* iterate through available sensors */
+   for(info.sensor=0; ; info.sensor++) {
+      /* stop if we can't ioctl() */
+      if (ioctl(fd, ENVSYS_GTREINFO, &info) < 0) break;
+      /* stop if that sensor is not valid */
+      if (!(info.validflags & ENVSYS_FVALID)) break;
+
+      do {
+	if (info.units == ENVSYS_INDICATOR &&
+	    sscanf(info.desc, "acpibat%d presen%1[t]", &bat, fake) == 2) {
+	  if (!gkrellm_battery_data_alloc(bat)) return FALSE;
+	  battery_acpi_data[bat].available_index = info.sensor;
+	  found_sensors = TRUE;
+
+	} else if (info.units == ENVSYS_INDICATOR &&
+		   sscanf(info.desc, "acpiacad%*d connecte%1[d]", fake) == 1) {
+	  if (!gkrellm_battery_data_alloc(0)) return FALSE;
+	  battery_acpi_data[0].on_line_index = info.sensor;
+
+	} else if (info.units == ENVSYS_INDICATOR &&
+		   sscanf(info.desc,
+			  "acpibat%d chargin%1[g]", &bat, fake) == 2) {
+	  if (!gkrellm_battery_data_alloc(bat)) return FALSE;
+	  battery_acpi_data[bat].charging_index = info.sensor;
+	  found_sensors = TRUE;
+
+	} else if (info.units == ENVSYS_SAMPHOUR &&
+		   sscanf(info.desc,
+			  "acpibat%d design ca%1[p]", &bat, fake) == 2) {
+	  if (!gkrellm_battery_data_alloc(bat)) return FALSE;
+	  battery_acpi_data[bat].full_cap_index = info.sensor;
+	  found_sensors = TRUE;
+
+	} else if (info.units == ENVSYS_SAMPHOUR &&
+		   sscanf(info.desc,
+			  "acpibat%d charg%1[e]", &bat, fake) == 2) {
+	  if (!gkrellm_battery_data_alloc(bat)) return FALSE;
+	  battery_acpi_data[bat].cap_index = info.sensor;
+	  found_sensors = TRUE;
+
+	} else if (info.units == ENVSYS_SAMPS &&
+		   sscanf(info.desc,
+			  "acpibat%d discharge rat%1[e]", &bat, fake) == 2) {
+	  if (!gkrellm_battery_data_alloc(bat)) return FALSE;
+	  battery_acpi_data[bat].discharge_rate_index = info.sensor;
+	  found_sensors = TRUE;
+
+	} else if (info.units == ENVSYS_SAMPS &&
+		   sscanf(info.desc,
+			  "acpibat%d charge rat%1[e]", &bat, fake) == 2) {
+	  if (!gkrellm_battery_data_alloc(bat)) return FALSE;
+	  battery_acpi_data[bat].charge_rate_index = info.sensor;
+	  found_sensors = TRUE;
+	}
+      } while(0);
+   }
+
+   close(fd);
+
+   return found_sensors;
+}
--- patch-bc ends here ---