1148 lines
32 KiB
C
1148 lines
32 KiB
C
/*
|
|
* wmbatteries - A dockapp to monitor ACPI status of two batteries
|
|
* Copyright (C) 2003 Florian Krohs <krohs@uni.de>
|
|
|
|
* Based on work by Thomas Nemeth <tnemeth@free.fr>
|
|
* Copyright (C) 2002 Thomas Nemeth <tnemeth@free.fr>
|
|
* and on work by Seiichi SATO <ssato@sh.rim.or.jp>
|
|
* Copyright (C) 2001,2002 Seiichi SATO <ssato@sh.rim.or.jp>
|
|
* and on work by Mark Staggs <me@markstaggs.net>
|
|
* Copyright (C) 2002 Mark Staggs <me@markstaggs.net>
|
|
|
|
* 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; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include "files.h"
|
|
#include <signal.h>
|
|
#include "dockapp.h"
|
|
#include "backlight_on.xpm"
|
|
#include "backlight_off.xpm"
|
|
#include "parts.xpm"
|
|
#include <stdlib.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
|
|
#ifdef linux
|
|
#include <sys/stat.h>
|
|
#endif
|
|
|
|
#define DEBUG
|
|
|
|
#define WMBATTERIES_VERSION "0.1.3"
|
|
|
|
#define FREE(data) {if (data) free (data); data = NULL;}
|
|
|
|
#define RATE_HISTORY 10
|
|
#define BLINK_ONLOADING_TIMEOUT 500
|
|
#define SIZE 58
|
|
#define MAXSTRLEN 512
|
|
#define WINDOWED_BG ". c #AEAAAE"
|
|
#define MAX_HISTORY 16
|
|
#define CPUNUM_NONE -1
|
|
|
|
#define CHARGING 3
|
|
#define DISCHARGING 1
|
|
#define UNKNOWN 0
|
|
|
|
#define TOGGLEMODE 1
|
|
#define TOGGLESPEED 2
|
|
|
|
#define DEFAULT_UPDATE_INTERVAL 5
|
|
|
|
#define RATE 0
|
|
#define TEMP 1
|
|
|
|
#define NONE 0
|
|
#define STATE_OK 1
|
|
#define INFO_OK 2
|
|
#define BAT_OK 3
|
|
|
|
#ifdef DEBUG
|
|
#define DEBUGSTRING(STRING) printf("DEBUG: %s\n",STRING);
|
|
#endif
|
|
|
|
#ifndef DEBUG
|
|
#define DEBUGSTRING(STRING)
|
|
#endif
|
|
|
|
# ifdef linux
|
|
# define ACPIDEV "/proc/acpi/info"
|
|
# endif
|
|
|
|
typedef struct AcpiInfos {
|
|
const char driver_version[10];
|
|
int ac_line_status;
|
|
int battery_status[2];
|
|
int battery_percentage[2];
|
|
long rate[2];
|
|
long remain[2];
|
|
long currcap[2];
|
|
int thermal_temp;
|
|
int thermal_state;
|
|
int hours_left;
|
|
int minutes_left;
|
|
int low;
|
|
} AcpiInfos;
|
|
|
|
typedef struct RateListElem {
|
|
long rate[2];
|
|
struct RateListElem *next;
|
|
} RateListElem;
|
|
|
|
typedef enum { LIGHTOFF, LIGHTON } light;
|
|
|
|
|
|
Pixmap pixmap;
|
|
Pixmap backdrop_on;
|
|
Pixmap backdrop_off;
|
|
Pixmap parts;
|
|
Pixmap mask;
|
|
static char *display_name = "";
|
|
static char light_color[256] = ""; /* back-light color */
|
|
char tmp_string[256];
|
|
static char *config_file = NULL; /* name of configfile */
|
|
static unsigned update_interval = DEFAULT_UPDATE_INTERVAL;
|
|
static light backlight = LIGHTOFF;
|
|
static unsigned switch_authorized = True;
|
|
static unsigned alarm_level = 20;
|
|
static unsigned alarm_level_temp = 70;
|
|
static char *notif_cmd = NULL;
|
|
static char *suspend_cmd = NULL;
|
|
static char *standby_cmd = NULL;
|
|
static int mode = TEMP;
|
|
static int togglemode = TOGGLEMODE;
|
|
static int togglespeed = TOGGLESPEED;
|
|
static int animationspeed = 500;
|
|
static AcpiInfos cur_acpi_infos;
|
|
static number_of_batteries = 2;
|
|
static char state_files[2][256]={BAT0_STATE_FILE,BAT1_STATE_FILE};
|
|
static char info_files[2][256]={BAT0_INFO_FILE,BAT1_INFO_FILE};
|
|
static char thermal[256]=THERMAL_FILE;
|
|
static char ac_state[256]=AC_STATE_FILE;
|
|
static int history_size = RATE_HISTORY;
|
|
|
|
static RateListElem *rateElements;
|
|
static RateListElem *firstRateElem;
|
|
|
|
#ifdef linux
|
|
# ifndef ACPI_32_BIT_SUPPORT
|
|
# define ACPI_32_BIT_SUPPORT 0x0002
|
|
# endif
|
|
#endif
|
|
|
|
|
|
/* prototypes */
|
|
static void parse_config_file(char *config);
|
|
static void update();
|
|
static void switch_light();
|
|
static void draw_remaining_time(AcpiInfos infos);
|
|
static void draw_batt(AcpiInfos infos);
|
|
static void draw_low();
|
|
static void draw_rate(AcpiInfos infos);
|
|
static void draw_temp(AcpiInfos infos);
|
|
static void draw_statusdigit(AcpiInfos infos);
|
|
static void draw_pcgraph(AcpiInfos infos);
|
|
static void parse_arguments(int argc, char **argv);
|
|
static void print_help(char *prog);
|
|
static void acpi_getinfos(AcpiInfos *infos);
|
|
static int acpi_exists();
|
|
static int my_system (char *cmd);
|
|
static void blink_batt();
|
|
static void draw_all();
|
|
|
|
static void debug(char *debug_string);
|
|
#ifdef linux
|
|
int acpi_read(AcpiInfos *i);
|
|
int init_stats(AcpiInfos *k);
|
|
#endif
|
|
|
|
int count;
|
|
int blink_pos=0;
|
|
|
|
static void debug(char *debug_string){
|
|
printf("DEBUG: %s\n",debug_string);
|
|
}
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
XEvent event;
|
|
XpmColorSymbol colors[2] = { {"Back0", NULL, 0}, {"Back1", NULL, 0} };
|
|
int ncolor = 0;
|
|
struct sigaction sa;
|
|
long counter=0;
|
|
long timeout;
|
|
int charging;
|
|
long togglecounter=0;
|
|
long animationcounter=0;
|
|
|
|
sa.sa_handler = SIG_IGN;
|
|
#ifdef SA_NOCLDWAIT
|
|
sa.sa_flags = SA_NOCLDWAIT;
|
|
#else
|
|
sa.sa_flags = 0;
|
|
#endif
|
|
|
|
|
|
printf("wmbatteries %s (c) Florian Krohs\n"
|
|
"<florian.krohs@informatik.uni-oldenburg.de>\n\n"
|
|
"This Software comes with absolut no warranty.\n"
|
|
"Use at your own risk!\n\n",WMBATTERIES_VERSION);
|
|
|
|
sigemptyset(&sa.sa_mask);
|
|
sigaction(SIGCHLD, &sa, NULL);
|
|
|
|
/* Parse CommandLine */
|
|
parse_arguments(argc, argv);
|
|
|
|
/* Check for ACPI support */
|
|
if (!acpi_exists()) {
|
|
#ifdef linux
|
|
fprintf(stderr, "No ACPI support in kernel\n");
|
|
#else
|
|
fprintf(stderr, "Unable to access ACPI info\n");
|
|
#endif
|
|
exit(1);
|
|
}
|
|
|
|
/* Initialize Application */
|
|
|
|
init_stats(&cur_acpi_infos);
|
|
//acpi_getinfos(&cur_acpi_infos);
|
|
//update();
|
|
dockapp_open_window(display_name, PACKAGE, SIZE, SIZE, argc, argv);
|
|
dockapp_set_eventmask(ButtonPressMask);
|
|
|
|
if (strcmp(light_color,"")) {
|
|
colors[0].pixel = dockapp_getcolor(light_color);
|
|
colors[1].pixel = dockapp_blendedcolor(light_color, -24, -24, -24, 1.0);
|
|
ncolor = 2;
|
|
}
|
|
|
|
/* change raw xpm data to pixmap */
|
|
if (dockapp_iswindowed)
|
|
backlight_on_xpm[1] = backlight_off_xpm[1] = WINDOWED_BG;
|
|
|
|
if (!dockapp_xpm2pixmap(backlight_on_xpm, &backdrop_on, &mask, colors, ncolor)) {
|
|
fprintf(stderr, "Error initializing backlit background image.\n");
|
|
exit(1);
|
|
}
|
|
if (!dockapp_xpm2pixmap(backlight_off_xpm, &backdrop_off, NULL, NULL, 0)) {
|
|
fprintf(stderr, "Error initializing background image.\n");
|
|
exit(1);
|
|
}
|
|
if (!dockapp_xpm2pixmap(parts_xpm, &parts, NULL, colors, ncolor)) {
|
|
fprintf(stderr, "Error initializing parts image.\n");
|
|
exit(1);
|
|
}
|
|
|
|
/* shape window */
|
|
if (!dockapp_iswindowed) dockapp_setshape(mask, 0, 0);
|
|
if (mask) XFreePixmap(display, mask);
|
|
|
|
/* pixmap : draw area */
|
|
pixmap = dockapp_XCreatePixmap(SIZE, SIZE);
|
|
|
|
/* Initialize pixmap */
|
|
if (backlight == LIGHTON)
|
|
dockapp_copyarea(backdrop_on, pixmap, 0, 0, SIZE, SIZE, 0, 0);
|
|
else
|
|
dockapp_copyarea(backdrop_off, pixmap, 0, 0, SIZE, SIZE, 0, 0);
|
|
|
|
dockapp_set_background(pixmap);
|
|
update();
|
|
dockapp_show();
|
|
long update_timeout = update_interval*1000;
|
|
long animation_timeout = animationspeed;
|
|
long toggle_timeout = togglespeed*1000;
|
|
int show = 0;
|
|
/* Main loop */
|
|
while (1) {
|
|
if (cur_acpi_infos.battery_status[0]==CHARGING || cur_acpi_infos.battery_status[1]==CHARGING)
|
|
charging = 1;
|
|
else
|
|
charging = 0;
|
|
timeout = update_timeout;
|
|
if( charging && animation_timeout<update_timeout){
|
|
if(animation_timeout<toggle_timeout)
|
|
timeout = animation_timeout;
|
|
else if(togglemode) timeout = toggle_timeout;
|
|
} else if(update_timeout<toggle_timeout)
|
|
timeout = update_timeout;
|
|
else if(togglemode) timeout = toggle_timeout;
|
|
if (dockapp_nextevent_or_timeout(&event, timeout)) {
|
|
/* Next Event */
|
|
switch (event.type) {
|
|
case ButtonPress:
|
|
switch (event.xbutton.button) {
|
|
case 1: switch_light(); break;
|
|
case 3: mode=!mode; draw_all();dockapp_copy2window(pixmap);break;
|
|
default: break;
|
|
}
|
|
break;
|
|
default: break;
|
|
}
|
|
} else {
|
|
/* Time Out */
|
|
update_timeout -= timeout;
|
|
animation_timeout -= timeout;
|
|
toggle_timeout -= timeout;
|
|
if(toggle_timeout<=0){
|
|
toggle_timeout = togglespeed*1000;
|
|
if(togglemode){
|
|
mode=!mode;
|
|
show = 1;
|
|
}
|
|
}
|
|
if(animation_timeout<=0){
|
|
animation_timeout = animationspeed;
|
|
if(charging){
|
|
blink_batt();
|
|
show = 1;
|
|
}
|
|
}
|
|
if(update_timeout<=0){
|
|
update();
|
|
show = 1;
|
|
update_timeout = update_interval*1000;
|
|
}
|
|
if(show) {
|
|
/* show */
|
|
draw_all();
|
|
if(charging) {
|
|
blink_pos--;
|
|
blink_batt();
|
|
}
|
|
dockapp_copy2window(pixmap);
|
|
show = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
int init_stats(AcpiInfos *k) {
|
|
int bat_status[2]={NONE,NONE};
|
|
FILE *fd;
|
|
char *buf;
|
|
char *ptr;
|
|
char present;
|
|
int bat;
|
|
int hist;
|
|
int i;
|
|
|
|
buf=(char *)malloc(sizeof(char)*512);
|
|
if(buf == NULL)
|
|
exit(-1);
|
|
/* get info about existing batteries */
|
|
number_of_batteries=0;
|
|
for(i=0;i<2;i++){
|
|
if((fd = fopen(state_files[i], "r"))){
|
|
fread(buf,512,1,fd);
|
|
fclose(fd);
|
|
if(ptr = strstr(buf,"present:")) {
|
|
present=*(ptr+25);
|
|
if(present == 'y'){
|
|
bat_status[i]|=STATE_OK;
|
|
}
|
|
}
|
|
if(ptr = strstr(buf,"present rate:")) {
|
|
present=*(ptr+25);
|
|
sscanf(ptr,"%d",&k->rate[bat]);
|
|
}
|
|
}
|
|
if((fd = fopen(info_files[i], "r"))){
|
|
fread(buf,512,1,fd);
|
|
fclose(fd);
|
|
if(ptr = strstr(buf,"present:")) {
|
|
present=*(ptr+25);
|
|
if(present == 'y'){
|
|
bat_status[i]|=INFO_OK;
|
|
}
|
|
}
|
|
if(ptr = strstr(buf,"last full capacity:")) {
|
|
present=*(ptr+25);
|
|
sscanf(ptr,"%d",&k->currcap[bat]);
|
|
}
|
|
}
|
|
|
|
|
|
}
|
|
if(bat_status[0]==BAT_OK && bat_status[1]==BAT_OK){
|
|
printf("BAT0 and BAT1 ok\n");
|
|
number_of_batteries=2;
|
|
} else if(bat_status[0]==BAT_OK) {
|
|
printf("BAT0 ok\n");
|
|
number_of_batteries=1;
|
|
} else if(bat_status[1]==BAT_OK) {
|
|
printf("BAT1 ok\n");
|
|
number_of_batteries=1;
|
|
strcpy(state_files[0],state_files[1]);
|
|
strcpy(info_files[0],info_files[1]);
|
|
k->currcap[0] = k->currcap[1];
|
|
k->rate[0] = k->rate[1];
|
|
}
|
|
|
|
printf("%i batter%s found in system\n",number_of_batteries,number_of_batteries==1 ? "y" : "ies");
|
|
|
|
// initialize buffer
|
|
if ((rateElements = (RateListElem *) malloc(history_size * sizeof(RateListElem))) == NULL)
|
|
exit(-1);
|
|
|
|
firstRateElem = rateElements;
|
|
|
|
|
|
/* get info about full battery charge */
|
|
|
|
for(bat=0;bat<number_of_batteries;bat++){
|
|
if ((fd = fopen(info_files[bat], "r"))) {
|
|
fread(buf,512,1,fd);
|
|
fclose(fd);
|
|
if(ptr = strstr(buf,"last full capacity:")) {
|
|
ptr += 25;
|
|
sscanf(ptr,"%d",&k->currcap[bat]);
|
|
}
|
|
}
|
|
if ((fd = fopen(state_files[bat], "r"))) {
|
|
fread(buf,512,1,fd);
|
|
fclose(fd);
|
|
if(ptr = strstr(buf,"present rate:")) {
|
|
ptr += 25;
|
|
sscanf(ptr,"%d",&k->rate[bat]);
|
|
}
|
|
}
|
|
|
|
}
|
|
for(i=0;i<2;i++){
|
|
/* link rateElements */
|
|
for(hist=0;hist<(history_size-1);hist++){
|
|
(*(rateElements+hist)).next = rateElements+hist+1;
|
|
(*(rateElements+hist)).rate[i] = k->rate[i];
|
|
}
|
|
(*(rateElements+history_size-1)).next = rateElements;
|
|
(*(rateElements+history_size-1)).rate[i] = k->rate[i];
|
|
}
|
|
free(buf);
|
|
k->ac_line_status = 0;
|
|
k->battery_status[0] = 0;
|
|
k->battery_percentage[0] = 0;
|
|
k->remain[0] = 0;
|
|
k->battery_status[1] = 0;
|
|
k->battery_percentage[1] = 0;
|
|
k->remain[1] = 0;
|
|
k->thermal_temp = 0;
|
|
k->thermal_state = 0;
|
|
DEBUGSTRING("end of init_stats()");
|
|
}
|
|
|
|
/* called by timer */
|
|
static void update() {
|
|
static light pre_backlight;
|
|
static Bool in_alarm_mode = False;
|
|
|
|
/* get current battery usage in percent */
|
|
acpi_getinfos(&cur_acpi_infos);
|
|
|
|
/* alarm mode */
|
|
if (cur_acpi_infos.low || (cur_acpi_infos.thermal_temp > alarm_level_temp)) {
|
|
if (!in_alarm_mode) {
|
|
in_alarm_mode = True;
|
|
pre_backlight = backlight;
|
|
my_system(notif_cmd);
|
|
}
|
|
if ( (switch_authorized) ||
|
|
( (! switch_authorized) && (backlight != pre_backlight) ) ) {
|
|
switch_light();
|
|
return;
|
|
}
|
|
}
|
|
else {
|
|
if (in_alarm_mode) {
|
|
in_alarm_mode = False;
|
|
if (backlight != pre_backlight) {
|
|
switch_light();
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
draw_all();
|
|
}
|
|
|
|
static void parse_config_file(char *config){
|
|
|
|
FILE *fd=NULL;
|
|
char *buf;
|
|
char stringbuffer[256];
|
|
char *ptr;
|
|
char line[256] ;
|
|
char *item;
|
|
char *value;
|
|
extern int errno;
|
|
int linenr=0;
|
|
int tmp;
|
|
char *test;
|
|
buf=(char *)malloc(sizeof(char)*512);
|
|
if(buf == NULL)
|
|
exit(-1);
|
|
if(config != NULL) { //config file by command line
|
|
DEBUGSTRING("using command line given config file name");
|
|
DEBUGSTRING(config);
|
|
if((fd = fopen(config, "r"))){
|
|
DEBUGSTRING("config file found\n");
|
|
} else {
|
|
DEBUGSTRING("config file NOT found\n");
|
|
DEBUGSTRING("falling back to default config file\n");
|
|
}
|
|
}
|
|
if(fd==NULL) { // no config file found yet
|
|
// stringbuffer=strcat(getenv("HOME"),"/.wmbatteriesrc");
|
|
strcpy(stringbuffer,getenv("HOME"));
|
|
strcat(stringbuffer,"/.wmbatteriesrc");
|
|
|
|
|
|
DEBUGSTRING("trying config file in your $HOME dir\n");
|
|
DEBUGSTRING(getenv("HOME"));
|
|
DEBUGSTRING(stringbuffer);
|
|
if((fd = fopen(stringbuffer, "r"))){
|
|
DEBUGSTRING("config file found\n");
|
|
} else {
|
|
DEBUGSTRING("config file in $HOME dir nonexistant\n");
|
|
DEBUGSTRING("trying global one in /etc\n");
|
|
if((fd = fopen("/etc/wmbatteries", "r"))){
|
|
DEBUGSTRING("config file found\n");
|
|
}
|
|
else {
|
|
DEBUGSTRING("no config file found. ignoring\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
if(fd!=NULL){ // some config file was found, try parsing
|
|
DEBUGSTRING("begin parsing\n");
|
|
while( fgets( line, 255, fd ) != NULL )
|
|
{
|
|
|
|
item = strtok( line, "\t =\n\r" ) ;
|
|
if( item != NULL && item[0] != '#' )
|
|
{
|
|
value = strtok( NULL, "\t =\n\r" ) ;
|
|
if(!strcmp(item,"backlight")){
|
|
if(strcasecmp(value,"yes") && strcasecmp(value,"true") && strcasecmp(value,"false") && strcasecmp(value,"no")) {
|
|
printf("backlight option wrong in line %i,use yes/no or true/false\n",linenr);
|
|
} else {
|
|
if(!strcasecmp(value,"true") || !strcasecmp(value,"yes")){
|
|
backlight = LIGHTON;
|
|
} else {
|
|
backlight = LIGHTOFF;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!strcmp(item,"lightcolor")){
|
|
strcpy(light_color,value);
|
|
}
|
|
|
|
if(!strcmp(item,"temperature")){
|
|
strcpy(thermal,value);
|
|
}
|
|
|
|
if(!strcmp(item,"bat0_state")){
|
|
strcpy(state_files[0],value);
|
|
}
|
|
|
|
if(!strcmp(item,"bat1_state")){
|
|
strcpy(state_files[1],value);
|
|
}
|
|
|
|
if(!strcmp(item,"bat0_info")){
|
|
strcpy(info_files[0],value);
|
|
}
|
|
|
|
if(!strcmp(item,"bat1_info")){
|
|
strcpy(info_files[1],value);
|
|
}
|
|
|
|
if(!strcmp(item,"ac_state")){
|
|
strcpy(ac_state,value);
|
|
}
|
|
|
|
|
|
if(!strcmp(item,"updateinterval")){
|
|
tmp=atoi(value);
|
|
if(tmp<1) {
|
|
printf("update interval is out of range in line %i,must be > 0\n",linenr);
|
|
} else {
|
|
update_interval=tmp;
|
|
}
|
|
}
|
|
|
|
if(!strcmp(item,"alarm")){
|
|
tmp=atoi(value);
|
|
if(tmp<1 || tmp>100) {
|
|
printf("alarm is out of range in line %i,must be > 0 and <= 100\n",linenr);
|
|
} else {
|
|
alarm_level=tmp;
|
|
}
|
|
}
|
|
|
|
if(!strcmp(item,"togglespeed")){
|
|
tmp=atoi(value);
|
|
if(tmp<1) {
|
|
printf("togglespeed variable is out of range in line %i,must be > 0\n",linenr);
|
|
} else {
|
|
togglespeed=tmp;
|
|
}
|
|
}
|
|
|
|
if(!strcmp(item,"animationspeed")){
|
|
tmp=atoi(value);
|
|
if(tmp<100) {
|
|
printf("animationspeed variable is out of range in line %i,must be >= 100\n",linenr);
|
|
} else {
|
|
animationspeed=tmp;
|
|
}
|
|
}
|
|
|
|
if(!strcmp(item,"historysize")){
|
|
tmp=atoi(value);
|
|
if(tmp<1 || tmp>1000) {
|
|
printf("historysize variable is out of range in line %i,must be >=1 and <=1000\n",linenr);
|
|
} else {
|
|
history_size=tmp;
|
|
}
|
|
}
|
|
|
|
if(!strcmp(item,"mode")){
|
|
if(strcmp(value,"rate") && strcmp(value,"toggle") && strcmp(value,"toggle")) {
|
|
printf("mode must be one of rate,temp,toggle in line %i\n",linenr);
|
|
} else {
|
|
if(strcmp(value,"rate")) mode=RATE;
|
|
if(strcmp(value,"temp")) mode=TEMP;
|
|
if(strcmp(value,"toggle")) togglemode=1;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
}
|
|
linenr++;
|
|
}
|
|
fclose(fd);
|
|
DEBUGSTRING("end parsing\n");
|
|
}
|
|
}
|
|
|
|
|
|
static void draw_all(){
|
|
int bat;
|
|
long allremain=0;
|
|
long allcapacity=0;
|
|
/* all clear */
|
|
if (backlight == LIGHTON)
|
|
dockapp_copyarea(backdrop_on, pixmap, 0, 0, 58, 58, 0, 0);
|
|
else
|
|
dockapp_copyarea(backdrop_off, pixmap, 0, 0, 58, 58, 0, 0);
|
|
/* draw digit */
|
|
draw_remaining_time(cur_acpi_infos);
|
|
if(mode==RATE) draw_rate(cur_acpi_infos);
|
|
else if(mode==TEMP) draw_temp(cur_acpi_infos);
|
|
draw_statusdigit(cur_acpi_infos);
|
|
draw_pcgraph(cur_acpi_infos);
|
|
|
|
if(cur_acpi_infos.low) draw_low();
|
|
|
|
draw_batt(cur_acpi_infos);
|
|
}
|
|
|
|
|
|
/* called when mouse button pressed */
|
|
|
|
static void switch_light() {
|
|
switch (backlight) {
|
|
case LIGHTOFF:
|
|
backlight = LIGHTON;
|
|
dockapp_copyarea(backdrop_on, pixmap, 0, 0, 58, 58, 0, 0);
|
|
break;
|
|
case LIGHTON:
|
|
backlight = LIGHTOFF;
|
|
dockapp_copyarea(backdrop_off, pixmap, 0, 0, 58, 58, 0, 0);
|
|
break;
|
|
}
|
|
|
|
draw_remaining_time(cur_acpi_infos);
|
|
if(mode==RATE) draw_rate(cur_acpi_infos);
|
|
else if(mode==TEMP) draw_temp(cur_acpi_infos);
|
|
draw_statusdigit(cur_acpi_infos);
|
|
draw_pcgraph(cur_acpi_infos);
|
|
if(cur_acpi_infos.battery_status[0]==CHARGING || cur_acpi_infos.battery_status[1]==CHARGING){
|
|
blink_batt();
|
|
} else draw_batt(cur_acpi_infos);
|
|
if(cur_acpi_infos.low){
|
|
draw_low();
|
|
}
|
|
/* show */
|
|
dockapp_copy2window(pixmap);
|
|
}
|
|
|
|
static void draw_batt(AcpiInfos infos){
|
|
int y = 0;
|
|
int i=0;
|
|
if (backlight == LIGHTON) y = 28;
|
|
for(i=0;i<number_of_batteries;i++){
|
|
if(infos.battery_status[i]==DISCHARGING){
|
|
dockapp_copyarea(parts, pixmap,33+y , 63, 9, 5, 16+i*11, 39);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void draw_remaining_time(AcpiInfos infos) {
|
|
int y = 0;
|
|
if (backlight == LIGHTON) y = 20;
|
|
if (infos.ac_line_status == 1 && !(cur_acpi_infos.battery_status[0]==CHARGING || cur_acpi_infos.battery_status[1]==CHARGING)){
|
|
dockapp_copyarea(parts, pixmap, 0, 68+68+y, 10, 20, 17, 5);
|
|
dockapp_copyarea(parts, pixmap, 10, 68+68+y, 10, 20, 32, 5);
|
|
} else {
|
|
|
|
dockapp_copyarea(parts, pixmap, (infos.hours_left / 10) * 10, 68+y, 10, 20, 5, 5);
|
|
dockapp_copyarea(parts, pixmap, (infos.hours_left % 10) * 10, 68+y, 10, 20, 17, 5);
|
|
dockapp_copyarea(parts, pixmap, (infos.minutes_left / 10) * 10, 68+y, 10, 20, 32, 5);
|
|
dockapp_copyarea(parts, pixmap, (infos.minutes_left % 10) * 10, 68+y, 10, 20, 44, 5);
|
|
}
|
|
|
|
}
|
|
|
|
static void draw_low() {
|
|
int y = 0;
|
|
if (backlight == LIGHTON) y = 28;
|
|
dockapp_copyarea(parts, pixmap,42+y , 58, 17, 7, 38, 38);
|
|
|
|
}
|
|
|
|
static void draw_temp(AcpiInfos infos) {
|
|
int temp = infos.thermal_temp;
|
|
int light_offset=0;
|
|
if (backlight == LIGHTON) {
|
|
light_offset=50;
|
|
}
|
|
|
|
if (temp < 0 || temp>99) temp = 0;
|
|
dockapp_copyarea(parts, pixmap, (temp/10)*5 + light_offset, 40, 5, 9, 23, 46);
|
|
dockapp_copyarea(parts, pixmap, (temp%10)*5 + light_offset, 40, 5, 9, 29, 46);
|
|
|
|
dockapp_copyarea(parts, pixmap, 10 + light_offset, 49, 5, 9, 36, 46); //o
|
|
dockapp_copyarea(parts, pixmap, 15 + light_offset, 49, 5, 9, 42, 46); //C
|
|
|
|
}
|
|
|
|
static void blink_batt(){
|
|
int light_offset=0;
|
|
int bat=0;
|
|
if (backlight == LIGHTON) {
|
|
light_offset=50;
|
|
}
|
|
blink_pos=(blink_pos+1)%5;
|
|
for(bat=0;bat<number_of_batteries;bat++){
|
|
if(cur_acpi_infos.battery_status[bat]==CHARGING){
|
|
dockapp_copyarea(parts, pixmap, blink_pos*9+light_offset , 117, 9, 5, 16+bat*11, 39);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void draw_statusdigit(AcpiInfos infos) {
|
|
int light_offset=0;
|
|
if (backlight == LIGHTON) {
|
|
light_offset=28;
|
|
}
|
|
if (infos.ac_line_status == 1){
|
|
dockapp_copyarea(parts, pixmap,33+light_offset , 58, 9, 5, 5, 39);
|
|
}
|
|
}
|
|
|
|
static void draw_rate(AcpiInfos infos) {
|
|
int light_offset=0;
|
|
long rate = infos.rate[0]+infos.rate[1];
|
|
if (backlight == LIGHTON) {
|
|
light_offset=50;
|
|
}
|
|
|
|
dockapp_copyarea(parts, pixmap, (rate/10000)*5 + light_offset, 40, 5, 9, 5, 46);
|
|
dockapp_copyarea(parts, pixmap, ((rate/1000)%10)*5 + light_offset, 40, 5, 9, 11, 46);
|
|
dockapp_copyarea(parts, pixmap, ((rate/100)%10)*5 + light_offset, 40, 5, 9, 17, 46);
|
|
dockapp_copyarea(parts, pixmap, ((rate/10)%10)*5 + light_offset, 40, 5, 9, 23, 46);
|
|
dockapp_copyarea(parts, pixmap, (rate%10)*5 + light_offset, 40, 5, 9, 29, 46);
|
|
|
|
dockapp_copyarea(parts, pixmap, 0 + light_offset, 49, 5, 9, 36, 46); //m
|
|
dockapp_copyarea(parts, pixmap, 5 + light_offset, 49, 5, 9, 42, 46); //W
|
|
|
|
}
|
|
|
|
static void draw_pcgraph(AcpiInfos infos) {
|
|
int num[2];
|
|
int bat;
|
|
int width;
|
|
int light_offset=0;
|
|
if (backlight == LIGHTON) {
|
|
light_offset=5;
|
|
}
|
|
for(bat=0;bat<number_of_batteries;bat++){
|
|
width = (infos.battery_percentage[bat]*32)/100;
|
|
dockapp_copyarea(parts, pixmap, 0, 58+light_offset, width, 5, 5, 26+6*bat);
|
|
if(infos.battery_percentage[bat] == 100){ // don't display leading 0
|
|
dockapp_copyarea(parts, pixmap, 4*(infos.battery_percentage[bat]/100), 126+light_offset, 3, 5, 38, 26+6*bat);
|
|
}
|
|
if(infos.battery_percentage[bat] > 9){ //don't display leading 0
|
|
dockapp_copyarea(parts, pixmap, 4*((infos.battery_percentage[bat]%100)/10), 126+light_offset, 3, 5, 42, 26+6*bat);
|
|
}
|
|
dockapp_copyarea(parts, pixmap, 4*(infos.battery_percentage[bat]%10), 126+light_offset, 3, 5, 46, 26+6*bat);
|
|
}
|
|
|
|
}
|
|
|
|
|
|
static void parse_arguments(int argc, char **argv) {
|
|
int i;
|
|
int integer;
|
|
char character;
|
|
|
|
for (i = 1; i < argc; i++) { // first search for config file option
|
|
if (!strcmp(argv[i], "--config") || !strcmp(argv[i], "-c")) {
|
|
config_file = argv[i + 1];
|
|
i++;
|
|
}
|
|
}
|
|
// parse config file before other command line options, to allow overriding
|
|
parse_config_file(config_file);
|
|
for (i = 1; i < argc; i++) {
|
|
if (!strcmp(argv[i], "--help") || !strcmp(argv[i], "-h")) {
|
|
print_help(argv[0]), exit(0);
|
|
} else if (!strcmp(argv[i], "--version") || !strcmp(argv[i], "-v")) {
|
|
printf("%s version %s\n", PACKAGE, VERSION), exit(0);
|
|
} else if (!strcmp(argv[i], "--display") || !strcmp(argv[i], "-d")) {
|
|
display_name = argv[i + 1];
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--backlight") || !strcmp(argv[i], "-bl")) {
|
|
backlight = LIGHTON;
|
|
} else if (!strcmp(argv[i], "--light-color") || !strcmp(argv[i], "-lc")) {
|
|
strcpy(light_color,argv[i + 1]);
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--config") || !strcmp(argv[i], "-c")) {
|
|
config_file = argv[i + 1];
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--interval") || !strcmp(argv[i], "-i")) {
|
|
if (argc == i + 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (sscanf(argv[i + 1], "%i", &integer) != 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (integer < 1)
|
|
fprintf(stderr, "%s: argument %s must be >=1\n",
|
|
argv[0], argv[i]), exit(1);
|
|
update_interval = integer;
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--alarm") || !strcmp(argv[i], "-a")) {
|
|
if (argc == i + 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (sscanf(argv[i + 1], "%i", &integer) != 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if ( (integer < 0) || (integer > 100) )
|
|
fprintf(stderr, "%s: argument %s must be >=0 and <=100\n",
|
|
argv[0], argv[i]), exit(1);
|
|
alarm_level = integer;
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--windowed") || !strcmp(argv[i], "-w")) {
|
|
dockapp_iswindowed = True;
|
|
} else if (!strcmp(argv[i], "--broken-wm") || !strcmp(argv[i], "-bw")) {
|
|
dockapp_isbrokenwm = True;
|
|
} else if (!strcmp(argv[i], "--notify") || !strcmp(argv[i], "-n")) {
|
|
notif_cmd = argv[i + 1];
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--suspend") || !strcmp(argv[i], "-s")) {
|
|
suspend_cmd = argv[i + 1];
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--togglespeed") || !strcmp(argv[i], "-ts")) {
|
|
if (argc == i + 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (sscanf(argv[i + 1], "%i", &integer) != 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if ( integer < 1)
|
|
fprintf(stderr, "%s: argument %s must be positive integer\n",
|
|
argv[0], argv[i],update_interval), exit(1);
|
|
togglespeed=integer;
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--animationspeed") || !strcmp(argv[i], "-as")) {
|
|
if (argc == i + 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (sscanf(argv[i + 1], "%i", &integer) != 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (integer < 100)
|
|
fprintf(stderr, "%s: argument %s must be >=100\n",
|
|
argv[0], argv[i]), exit(1);
|
|
animationspeed=integer;
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--historysize") || !strcmp(argv[i], "-hs")) {
|
|
if (argc == i + 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (sscanf(argv[i + 1], "%i", &integer) != 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (integer < 1 || integer > 1000)
|
|
fprintf(stderr, "%s: argument %s must be >=1 && <=1000\n",
|
|
argv[0], argv[i]), exit(1);
|
|
history_size=integer;
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--mode") || !strcmp(argv[i], "-m")) {
|
|
if (argc == i + 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (sscanf(argv[i + 1], "%c", &character) != 1)
|
|
fprintf(stderr, "%s: error parsing argument for option %s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if (!(character=='t' || character=='r' || character=='s'))
|
|
fprintf(stderr, "%s: argument %s must be t,r or s\n",
|
|
argv[0], argv[i]), exit(1);
|
|
if(character=='s') togglemode=1;
|
|
else if(character=='t') mode=TEMP;
|
|
else if(character=='r') mode=RATE;
|
|
i++;
|
|
} else if (!strcmp(argv[i], "--standby") || !strcmp(argv[i], "-S")) {
|
|
standby_cmd = argv[i + 1];
|
|
i++;
|
|
} else {
|
|
fprintf(stderr, "%s: unrecognized option '%s'\n", argv[0], argv[i]);
|
|
print_help(argv[0]), exit(1);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void print_help(char *prog)
|
|
{
|
|
printf("Usage : %s [OPTIONS]\n"
|
|
"%s - Window Maker mails monitor dockapp\n"
|
|
" -d, --display <string> display to use\n"
|
|
" -bl, --backlight turn on back-light\n"
|
|
" -lc, --light-color <string> back-light color(rgb:6E/C6/3B is default)\n"
|
|
" -c, --config <string> set filename of config file\n"
|
|
" -i, --interval <number> number of secs between updates (1 is default)\n"
|
|
" -a, --alarm <number> low battery level when to raise alarm\n"
|
|
" (20 is default)\n"
|
|
" -h, --help show this help text and exit\n"
|
|
" -v, --version show program version and exit\n"
|
|
" -w, --windowed run the application in windowed mode\n"
|
|
" -bw, --broken-wm activate broken window manager fix\n"
|
|
" -n, --notify <string> command to launch when alarm is on\n"
|
|
" -s, --suspend <string> set command for acpi suspend\n"
|
|
" -S, --standby <string> set command for acpi standby\n"
|
|
" -m, --mode [t|r|s] set mode for the lower row , \n"
|
|
" t=temperature,r=current rate,s=toggle\n"
|
|
" -ts --togglespeed <int> set toggle speed in seconds\n"
|
|
" -as --animationspeed <int> set speed for charging animation in msec\n"
|
|
" -hs --historysize <int> set size of history for calculating\n"
|
|
" average power consumption rate\n",
|
|
prog, prog);
|
|
/* OPTIONS SUPP :
|
|
* ? -f, --file : configuration file
|
|
*/
|
|
}
|
|
|
|
|
|
static void acpi_getinfos(AcpiInfos *infos) {
|
|
DEBUGSTRING("acpi_getinfos\n")
|
|
if (
|
|
#if defined(linux) || defined(solaris)
|
|
(acpi_read(infos))
|
|
#else
|
|
# ifdef freebsd
|
|
(acpi_read(&temp_info))
|
|
# endif
|
|
#endif
|
|
) {
|
|
fprintf(stderr, "Cannot read ACPI information: %i\n");
|
|
exit(1);
|
|
}
|
|
}
|
|
|
|
|
|
int acpi_exists() {
|
|
if (access(ACPIDEV, R_OK))
|
|
return 0;
|
|
else
|
|
return 1;
|
|
}
|
|
|
|
|
|
static int my_system (char *cmd) {
|
|
int pid;
|
|
extern char **environ;
|
|
|
|
if (cmd == 0) return 1;
|
|
pid = fork ();
|
|
if (pid == -1) return -1;
|
|
if (pid == 0) {
|
|
pid = fork ();
|
|
if (pid == 0) {
|
|
char *argv[4];
|
|
argv[0] = "sh";
|
|
argv[1] = "-c";
|
|
argv[2] = cmd;
|
|
argv[3] = 0;
|
|
execve ("/bin/sh", argv, environ);
|
|
exit (0);
|
|
}
|
|
exit (0);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
#ifdef linux
|
|
|
|
int acpi_read(AcpiInfos *i) {
|
|
FILE *fd;
|
|
int retcode = 0;
|
|
int capacity[2],remain[2];
|
|
int bat;
|
|
char *buf;
|
|
char *ptr;
|
|
char stat;
|
|
buf=(char *)malloc(sizeof(char)*512);
|
|
RateListElem currRateElement;
|
|
int hist;
|
|
long rate;
|
|
float time;
|
|
long allcapacity=0;
|
|
long allremain=0;
|
|
|
|
rate = 0;
|
|
|
|
|
|
DEBUGSTRING("acpi_read()\n")
|
|
|
|
/* get acpi thermal cpu info */
|
|
if ((fd = fopen(thermal, "r"))) {
|
|
fscanf(fd, "temperature: %d", &i->thermal_temp);
|
|
fclose(fd);
|
|
}
|
|
if ((fd = fopen(ac_state, "r"))) {
|
|
bzero(buf, 512);
|
|
fscanf(fd, "state: %s", buf);
|
|
fclose(fd);
|
|
if(strstr(buf, "on-line") != NULL) i->ac_line_status=1;
|
|
if(strstr(buf, "off-line") != NULL) i->ac_line_status=0;
|
|
}
|
|
for(bat=0;bat<number_of_batteries;bat++){
|
|
|
|
if ((fd = fopen(state_files[bat], "r"))) {
|
|
bzero(buf, 512);
|
|
fread(buf,512,1,fd);
|
|
fclose(fd);
|
|
if(( ptr = strstr(buf,"charging state:"))) {
|
|
stat = *(ptr + 25);
|
|
switch (stat)
|
|
{
|
|
case 'd':
|
|
i->battery_status[bat]=1;
|
|
break;
|
|
case 'c':
|
|
i->battery_status[bat]=3;
|
|
break;
|
|
case 'u':
|
|
i->battery_status[bat]=0;
|
|
break;
|
|
}
|
|
}
|
|
if ((ptr = strstr (buf, "remaining capacity:"))) {
|
|
ptr += 25;
|
|
sscanf(ptr,"%d",&i->remain[bat]);
|
|
}
|
|
if ((ptr = strstr (buf, "present rate:"))) {
|
|
ptr += 25;
|
|
sscanf(ptr,"%d",&((*firstRateElem).rate[bat]));
|
|
}
|
|
}
|
|
|
|
i->battery_percentage[bat] = (((float)(i->remain[bat])*100)/cur_acpi_infos.currcap[bat]);
|
|
|
|
|
|
|
|
currRateElement = *firstRateElem;
|
|
if(currRateElement.rate[bat]!=0){
|
|
for(hist=0;hist<history_size;hist++){
|
|
if(currRateElement.rate[bat]!=0){
|
|
rate += currRateElement.rate[bat];
|
|
} else rate+= (*firstRateElem).rate[bat];
|
|
currRateElement = *currRateElement.next;
|
|
}
|
|
} else {
|
|
rate=0;
|
|
i->rate[bat]=0;
|
|
}
|
|
|
|
|
|
|
|
/* calc average */
|
|
rate = rate / history_size;
|
|
i->rate[bat] = rate;
|
|
}
|
|
|
|
if((i->battery_status[0]==1 || i->battery_status[1]==1) && (i->rate[0]+i->rate[1])>0){
|
|
time = (float)(i->remain[0]+i->remain[1])/(float)(i->rate[0]+i->rate[1]);
|
|
i->hours_left=(int)time;
|
|
i->minutes_left=(int)((time-(int)time)*60);
|
|
}
|
|
if(i->battery_status[0]==0 && i->battery_status[1]==0){
|
|
i->hours_left=0;
|
|
i->minutes_left=0;
|
|
}
|
|
if((i->battery_status[0]==3||i->battery_status[1]==3) && (i->rate[0]>0 || i->rate[1]>0)){
|
|
time = (float)(cur_acpi_infos.currcap[0] - i->remain[0] + cur_acpi_infos.currcap[1] - i->remain[1])/(float)(i->rate[0]+i->rate[1]);
|
|
i->hours_left=(int)time;
|
|
i->minutes_left=(int)(60*(time-(int)time));
|
|
}
|
|
for(bat=0;bat<number_of_batteries;bat++){
|
|
allremain += i->remain[bat];
|
|
allcapacity += cur_acpi_infos.currcap[bat];
|
|
}
|
|
|
|
cur_acpi_infos.low=0;
|
|
if(allcapacity>0){
|
|
if(((double)allremain/(double)allcapacity)*100<alarm_level){
|
|
cur_acpi_infos.low=1;
|
|
}
|
|
}
|
|
|
|
DEBUGSTRING("MID acpi_read()\n")
|
|
firstRateElem = ((*firstRateElem).next);
|
|
free(buf);
|
|
DEBUGSTRING("END acpi_read()\n")
|
|
return retcode;
|
|
}
|
|
#endif
|