Files
archived-lcd4linux/plugin_i2c_sensors.c
reinelt 9ef4282e9e [lcd4linux @ 2004-03-03 03:47:04 by reinelt]
big patch from Martin Hejl:
- use qprintf() where appropriate
- save CPU cycles on gettimeofday()
- add quit() functions to free allocated memory
- fixed lots of memory leaks

git-svn-id: https://ssl.bulix.org/svn/lcd4linux/trunk@384 3ae390bd-cb1e-0410-b409-cd5a39f66f1f
2004-03-03 03:47:04 +00:00

402 lines
11 KiB
C
Raw Blame History

/* $Id: plugin_i2c_sensors.c,v 1.13 2004/03/03 03:47:04 reinelt Exp $
*
* I2C sensors plugin
*
* Copyright 2003,2004 Xavier Vello <xavier66@free.fr>
* Copyright 2004 The LCD4Linux Team <lcd4linux-devel@users.sourceforge.net>
*
* This file is part of LCD4Linux.
*
* LCD4Linux 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, or (at your option)
* any later version.
*
* LCD4Linux 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.
*
*
* $Log: plugin_i2c_sensors.c,v $
* Revision 1.13 2004/03/03 03:47:04 reinelt
* big patch from Martin Hejl:
* - use qprintf() where appropriate
* - save CPU cycles on gettimeofday()
* - add quit() functions to free allocated memory
* - fixed lots of memory leaks
*
* Revision 1.12 2004/02/16 08:19:44 reinelt
* i2c_sensors patch from Xavier
*
* Revision 1.11 2004/02/15 21:43:43 reinelt
* T6963 driver nearly finished
* framework for graphic displays done
* i2c_sensors patch from Xavier
* some more old generation files removed
*
* Revision 1.10 2004/02/14 12:07:27 nicowallmeier
* minor bugfix
*
* Revision 1.9 2004/02/14 11:56:17 reinelt
* M50530 driver ported
* changed lots of 'char' to 'unsigned char'
*
* Revision 1.8 2004/02/14 10:09:50 reinelt
* I2C Sensors for 2.4 kernels (/proc instead of /sysfs)
*
* Revision 1.7 2004/01/30 20:57:56 reinelt
* HD44780 patch from Martin Hejl
* dmalloc integrated
*
* Revision 1.6 2004/01/30 07:12:35 reinelt
* HD44780 busy-flag support from Martin Hejl
* loadavg() uClibc replacement from Martin Heyl
* round() uClibc replacement from Martin Hejl
* warning in i2c_sensors fixed
*
* Revision 1.5 2004/01/29 05:55:30 reinelt
* check for /sys mounted
*
* Revision 1.4 2004/01/29 04:40:02 reinelt
* every .c file includes "config.h" now
*
* Revision 1.3 2004/01/27 08:13:39 reinelt
* ported PPP token to plugin_ppp
*
* Revision 1.2 2004/01/27 05:06:10 reinelt
* i2c update from Xavier
*
* Revision 1.1 2004/01/10 17:36:56 reinelt
*
* I2C Sensors plugin from Xavier added
*
*/
/*
* exported functions:
*
* int plugin_init_i2c_sensors (void)
* adds function i2c_sensors() to retrieve informations from
* the i2c sensors via sysfs or procfs interface
*
* -- WARNING --
* This plugin should detect where your sensors are at startup.
* If you can't get any token to work, ensure you don't get
* an error message with "lcd4linux -Fvvv".
*
* If so, try to force the path to your sensors in the conf like this :
* for sysfs: i2c_sensors-path '/sys/bus/i2c/devices/0-6000/'
* for procfs: i2c_sensors-path '/proc/sys/dev/sensors/via686a-isa-6000'
* /!\ these path are for my system, change the last dir according to yours
*/
/*
* Available tokens : # represents an int from 1 to 3 (or more)
* temp_input# -> temperature of sensor # (in <20>C)
* temp_max# and temp_hyst# -> max and min of sensor #
* in_input#, in_min# and in_max# -> voltages
* fan_input# -> speed (in RPM) of fan #
* fan_min# and fan_div#
*
* Tokens avaible only via sysfs if suported by your sensors:
* curr_input#, curr_min# and curr_max# -> value of current (in amps)
* pwm#
* temp_crit# -> critical value of sensor #
* vid -> cpu core voltage
* and maybe others
* (see /usr/src/linux/Documentation/i2c/sysfs-interface on linux 2.6)
*/
#include "config.h"
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <dirent.h>
#include "debug.h"
#include "plugin.h"
#include "cfg.h"
#include "hash.h"
#ifdef WITH_DMALLOC
#include <dmalloc.h>
#endif
static char *path=NULL;
static HASH I2Csensors = { 0, };
static const char *procfs_tokens[4][3] = {
{"temp_hyst", "temp_max", "temp_input"}, // for temp#
{"in_min", "in_max", "in_input"}, // for in#
{"fan_div1", "fan_div2", "fan_div3"}, // for fan_div
{"fan_min", "fan_input", ""} // for fan#
};
static int (*parse_i2c_sensors)(char *key);
/***********************************************\
* Parsing for new 2.6 kernels 'sysfs' interface *
\***********************************************/
static int parse_i2c_sensors_sysfs(char *key)
{
double value;
char val[32];
char buffer[32];
char file[64];
FILE *stream;
strcpy(file, path);
strcat(file, key);
stream=fopen(file, "r");
if (stream==NULL) {
error ("fopen(%s) failed",file);
return -1;
}
fgets (buffer, sizeof(buffer), stream);
fclose (stream);
if (!buffer) {
error ("%s empty ?!",file);
return -1;
}
// now the formating stuff, depending on the file :
// Some values must be divided by 1000, the others
// are parsed directly (we just remove the \n).
if (!strncmp(key, "temp_", 5) ||
!strncmp(key, "curr_", 5) ||
!strncmp(key, "in_", 3) ||
!strncmp(key, "vid", 3)) {
value = strtod(buffer, NULL);
// FIXME: any way to do this without converting to double ?
value /= 1000.0;
sprintf(val, "%f", value);
} else {
sprintf(val, "%s", buffer);
// we supress this nasty \n at the end
val[strlen(val)-1]='\0';
}
hash_set (&I2Csensors, key, val);
return 0;
}
/************************************************\
* Parsing for old 2.4 kernels 'procfs' interface *
\************************************************/
static int parse_i2c_sensors_procfs(char *key)
{
char file[64];
FILE *stream;
char buffer[32];
char *value;
char *running;
int pos=0;
const char delim[3]=" \n";
char final_key[32];
char *number = &key[strlen(key)-1];
int tokens_index;
//debug("%s -> %s", key, number);
strcpy(file, path);
if (!strncmp(key, "temp_", 5)) {
tokens_index=0;
strcat(file, "temp");
strcat(file, number);
} else if (!strncmp(key, "in_", 3)) {
tokens_index=1;
strcat(file, "in");
strcat(file, number);
} else if (!strncmp(key, "fan_div", 7)) {
tokens_index=2;
strcat(file, "fan_div");
number = "";
} else if (!strncmp(key, "fan_", 4)) {
tokens_index=3;
strcat(file, "fan");
strcat(file, number);
} else {
return -1;
}
stream=fopen(file, "r");
if (stream==NULL) {
error ("fopen(%s) failed",file);
return -1;
}
fgets (buffer, sizeof(buffer), stream);
fclose (stream);
if (!buffer) {
error ("%s empty ?!",file);
return -1;
}
running=strdupa(buffer);
while(1) {
value = strsep (&running, delim);
debug("%s pos %i -> %s", file, pos , value);
if (!value || !strcmp(value, "")) {
debug("%s pos %i -> BREAK", file, pos);
break;
} else {
sprintf (final_key, "%s%s", procfs_tokens[tokens_index][pos], number);
debug ("%s -> %s", final_key, value);
hash_set (&I2Csensors, final_key, value);
pos++;
}
}
return 0;
}
/*****************************************\
* Common functions (path search and init) *
\*****************************************/
void my_i2c_sensors(RESULT *result, RESULT *arg)
{
int age;
char *val;
char *key=R2S(arg);
age=hash_age(&I2Csensors, key, &val);
if (age<0 || age>250) {
parse_i2c_sensors(key);
val=hash_get(&I2Csensors, key);
}
if (val) {
SetResult(&result, R_STRING, val);
} else {
SetResult(&result, R_STRING, "??");
}
}
void my_i2c_sensors_path(char *method)
{
struct dirent *dir;
struct dirent *file;
const char *base;
char dname[64];
DIR *fd1;
DIR *fd2;
int done;
if (!strcmp(method, "sysfs")) {
base="/sys/bus/i2c/devices/";
} else if (!strcmp(method, "procfs")) {
base="/proc/sys/dev/sensors/";
//base="/sensors_2.4/"; // fake dir to test without rebooting 2.4 ;)
} else {
return;
}
fd1 = opendir(base);
if (!fd1) {
if (!strcmp(method, "sysfs")) {
error("[i2c_sensors] Impossible to open %s! Is /sys mounted?", base);
} else if (!strcmp(method, "procfs")) {
error("[i2c_sensors] Impossible to open %s! Is i2c_proc loaded ?", base);
}
return;
}
while((dir = readdir(fd1))) {
// Skip non-directories and '.' and '..'
if ((dir->d_type!=DT_DIR && dir->d_type!=DT_LNK) ||
strcmp(dir->d_name, "." )==0 ||
strcmp(dir->d_name, "..")==0) {
continue;
}
// dname is the absolute path
strcpy(dname, base);
strcat(dname, dir->d_name);
strcat(dname, "/");
fd2 = opendir(dname);
done = 0;
while((file = readdir(fd2))) {
if (!strcmp(file->d_name, "temp_input1") || !strcmp(file->d_name, "temp1")) { // FIXME : do all sensors have a temp_input1 ?
path = realloc(path, strlen(dname)+1);
strcpy(path, dname);
done=1;
break;
}
}
closedir(fd2);
if (done) break;
}
closedir(fd1);
}
int plugin_init_i2c_sensors (void)
{
char *path_cfg=cfg_get(NULL, "i2c_sensors-path", "");
if (strncmp(path_cfg, "/", 1)) {
//debug("No path to i2c sensors found in the conf, calling my_i2c_sensors_path()");
my_i2c_sensors_path("sysfs");
if (!path)
my_i2c_sensors_path("procfs");
if (!path) {
error("[i2c_sensors] No i2c sensors found via the i2c interface !");
error("[i2c_sensors] Try to specify the path to the sensors !");
} else {
debug("Your i2c sensors are probably in %s", path);
debug("if i2c_sensors doesn't work, try to specify the path in your conf");
}
} else {
if (path_cfg[strlen(path_cfg)-1] != '/') {
// the headless user forgot the trailing slash :/
debug("adding a trailing slash at the end of the path");
path_cfg = realloc(path_cfg, strlen(path_cfg)+2);
strcat(path_cfg, "/");
}
debug("Path to i2c sensors from the conf : %s", path_cfg);
debug("if i2c_sensors doesn't work, double check this value !");
path = realloc(path, strlen(path_cfg)+1);
strcpy(path, path_cfg);
}
free(path_cfg);
// we activate the function only if there's a possibly path found
if (path!=NULL) {
if (strncmp(path, "/sys", 4)==0) {
parse_i2c_sensors=parse_i2c_sensors_sysfs;
AddFunction ("i2c_sensors", 1, my_i2c_sensors);
} else if (strncmp(path, "/proc", 5)==0) {
parse_i2c_sensors=parse_i2c_sensors_procfs;
AddFunction ("i2c_sensors", 1, my_i2c_sensors);
} else {
error("[i2c_sensors] unknown path %s, should start with /sys or /proc");
}
}
return 0;
}
void plugin_exit_i2c_sensors(void)
{
hash_destroy(&I2Csensors);
}