aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--bat_tray.c76
1 files changed, 52 insertions, 24 deletions
diff --git a/bat_tray.c b/bat_tray.c
index 3424037..e044630 100644
--- a/bat_tray.c
+++ b/bat_tray.c
@@ -31,28 +31,59 @@ int get_bat_percent()
return get_int_value_from_file(CHARGE_VALUE_PATH);
}
+
/***********************************************************************
* Return number of seconds until battery is fully charged/discharged
**********************************************************************/
-int get_bat_seconds_left()
+long get_bat_seconds_left()
{
- int charge_now, charge_full, current_now;
- char* file;
-
- // Read current current value
- asprintf(&file, POWERDIR"BAT%d/current_now",bat_num);
- current_now = get_int_value_from_file(file);
- free(file);
-
- // Return -1 if zero current (infinite charge time)
- if (current_now == 0)
- return -1;
+ int now = 0;
+ int full = 0;
+ int rate = 0;
+
+
+ /* FIXME this now/full/rate logic has probably earned its own function.
+ * Not because of DRY, but just for readability's sake perhaps */
+
+ /* Kernel battery info is reported in one of two formats:
+ * - charge units
+ * - energy units
+ * Charge Units | Energy Units
+ * ------------------+---------------
+ * charge_now (µAh) | energy_now (µWh)
+ * charge_full (µAh) | energy_full (µWh)
+ * current_now (µA) | power_now (µW)
+ *
+ * Simple arithmetic produces the desired time value (time to fill, time
+ * to empty).
+ *
+ * If a valid (non-negative) value from energy_now is received, we assume
+ * all values are in energy units and proceed from there. If an invalid
+ * value is received, we assume all values are in charge units etc.
+ */
+
+ /* Attempt to read current ENERGY level */
+ now = get_int_value_from_filef(POWERDIR"BAT%d/energy_now", bat_num);
+
+ /* FIXME battery's full charge/energy is only required when charging */
+ /* Energy units or charge units */
+ if (now >= 0)
+ {
+ debug("Energy units");
+ full = get_int_value_from_filef(POWERDIR"BAT%d/energy_full", bat_num);
+ rate = get_int_value_from_filef(POWERDIR"BAT%d/power_now", bat_num);
+ } else {
+ debug("Charge units");
+ full = get_int_value_from_filef(POWERDIR"BAT%d/charge_full", bat_num);
+ now = get_int_value_from_filef(POWERDIR"BAT%d/charge_now", bat_num);
+ rate = get_int_value_from_filef(POWERDIR"BAT%d/current_now", bat_num);
+ }
- asprintf(&file, POWERDIR"BAT%d/charge_now",bat_num);
- charge_now = get_int_value_from_file(file);
- free(file);
- if (charge_now == -1 || current_now == -1)
+ /* Note the '<=' for rate (we divide by rate) */
+ if (full < 0 ||
+ now < 0 ||
+ rate <= 0)
{
return -1;
}
@@ -60,15 +91,10 @@ int get_bat_seconds_left()
switch(get_battery_state())
{
case STATE_CHARGING:
- // We need to know full charge to calculate this one
- asprintf(&file, POWERDIR"BAT%d/charge_full",bat_num);
- charge_full = get_int_value_from_file(file);
- free(file);
-
- return (3600*(charge_full - charge_now))/current_now;
+ return (3600*(long long)(full - now))/rate;
break;
case STATE_DISCHARGING:
- return (3600*charge_now)/current_now;
+ return (3600*(long long)now)/rate;
break;
default:
return -1;
@@ -86,11 +112,13 @@ static gboolean update_tooltip(GtkStatusIcon* status_icon,gint x,gint y,gboolean
char* time_left;
+ fprintf(stderr, "%d seconds left on bat\n", seconds_left);
+
if (seconds_left == -1)
{
asprintf(&time_left, _("Unknown time left"));
} else {
- asprintf(&time_left, _("%02d:%02d:%02d left"), (int)(seconds_left/3600), (int)((seconds_left%3600)/60), seconds_left%60);
+ asprintf(&time_left, _("%02d:%02d left"), (int)(seconds_left/3600), (int)((seconds_left%3600)/60));
}
switch(get_battery_state())