2
0
mirror of https://github.com/lm-sensors/lm-sensors synced 2025-09-05 16:55:45 +00:00
Files
lm-sensors/lib/access.c

333 lines
12 KiB
C
Raw Normal View History

/*
access.c - Part of libsensors, a Linux library for reading sensor data.
Copyright (c) 1998 Frodo Looijaard <frodol@dds.nl>
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.
*/
#include <stddef.h>
#include <string.h>
#include "access.h"
#include "sensors.h"
#include "data.h"
#include "error.h"
#include "proc.h"
/* Compare two chips name descriptions, to see whether they could match.
Return 0 if it does not match, return 1 if it does match. */
int sensors_match_chip(sensors_chip_name chip1, sensors_chip_name chip2)
{
if ((chip1.prefix != SENSORS_CHIP_NAME_PREFIX_ANY) &&
(chip2.prefix != SENSORS_CHIP_NAME_PREFIX_ANY) &&
strcmp(chip1.prefix,chip2.prefix))
return 0;
if ((chip1.bus != SENSORS_CHIP_NAME_BUS_ANY) &&
(chip2.bus != SENSORS_CHIP_NAME_BUS_ANY) &&
(chip1.bus != chip2.bus)) {
if ((chip1.bus == SENSORS_CHIP_NAME_BUS_ISA) ||
(chip2.bus == SENSORS_CHIP_NAME_BUS_ISA))
return 0;
if ((chip1.bus != SENSORS_CHIP_NAME_BUS_ANY_I2C) &&
(chip2.bus != SENSORS_CHIP_NAME_BUS_ANY_I2C))
return 0;
}
if ((chip1.addr != chip2.addr) &&
(chip1.addr != SENSORS_CHIP_NAME_ADDR_ANY) &&
(chip2.addr != SENSORS_CHIP_NAME_ADDR_ANY))
return 0;
return 1;
}
/* Returns, one by one, a pointer to all sensor_chip structs of the
config file which match with the given chip name. Last should be
the value returned by the last call, or NULL if this is the first
call. Returns NULL if no more matches are found. Do not modify
the struct the return value points to!
Note that this visits the list of chips from last to first. Usually,
you want the match that was latest in the config file. */
sensors_chip *sensors_for_all_config_chips(sensors_chip_name chip_name,
const sensors_chip *last)
{
int nr,i;
sensors_chip_name_list chips;
for (nr = last?(last-sensors_config_chips)-1:sensors_config_chips_count-1;
nr >= 0; nr--) {
chips = sensors_config_chips[nr].chips;
for (i =0; i < chips.fits_count; i++) {
if (sensors_match_chip(chips.fits[i],chip_name))
return sensors_config_chips+nr;
}
}
return NULL;
}
/* Look up a resource in the intern chip list, and return a pointer to it.
Do not modify the struct the return value points to! Returns NULL if
not found.*/
const sensors_chip_feature *sensors_lookup_feature_nr(const char *prefix,
int feature)
{
int i,j;
const sensors_chip_feature *features;
for (i = 0; sensors_chip_features_list[i].prefix; i++)
if (!strcmp(sensors_chip_features_list[i].prefix,prefix)) {
features = sensors_chip_features_list[i].feature;
for (j=0; features[j].name; j++)
if (features[j].number == feature)
return features + j;
}
return NULL;
}
/* Look up a resource in the intern chip list, and return a pointer to it.
Do not modify the struct the return value points to! Returns NULL if
not found.*/
const sensors_chip_feature *sensors_lookup_feature_name(const char *prefix,
const char *feature)
{
int i,j;
const sensors_chip_feature *features;
for (i = 0; sensors_chip_features_list[i].prefix; i++)
if (!strcmp(sensors_chip_features_list[i].prefix,prefix)) {
features = sensors_chip_features_list[i].feature;
for (j=0; features[j].name; j++)
if (!strcmp(features[j].name,feature))
return features + j;
}
return NULL;
}
/* Check whether the chip name is an 'absolute' name, which can only match
one chip, or whether it has wildcards. Returns 0 if it is absolute, 1
if there are wildcards. */
int sensors_chip_name_has_wildcards(sensors_chip_name chip)
{
if ((chip.prefix == SENSORS_CHIP_NAME_PREFIX_ANY) ||
(chip.bus == SENSORS_CHIP_NAME_BUS_ANY) ||
(chip.bus == SENSORS_CHIP_NAME_BUS_ANY_I2C) ||
(chip.addr == SENSORS_CHIP_NAME_ADDR_ANY))
return 1;
else
return 0;
}
/* Look up the label which belongs to this chip. Note that chip should not
contain wildcard values! *result is newly allocated (free it yourself).
This function will return 0 on success, and <0 on failure. */
int sensors_get_label(sensors_chip_name name, int feature, char **result)
{
const sensors_chip *chip;
const sensors_chip_feature *featureptr;
int i;
if (sensors_chip_name_has_wildcards(name))
return -SENSORS_ERR_WILDCARDS;
if (! (featureptr = sensors_lookup_feature_nr(name.prefix,feature)))
return -SENSORS_ERR_NO_ENTRY;
for (chip = NULL; (chip = sensors_for_all_config_chips(name,chip));)
for (i = 0; i < chip->labels_count; i++)
if (!strcmp(featureptr->name, chip->labels[i].name)) {
if (! (*result = strdup(chip->labels[i].value)))
sensors_fatal_error("sensors_get_label","Allocating label text");
return 0;
}
if (! (*result = strdup(featureptr->name)))
sensors_fatal_error("sensors_get_label","Allocating label text");
return 0;
}
/* Read the value of a feature of a certain chip. Note that chip should not
contain wildcard values! This function will return 0 on success, and <0
on failure. */
int sensors_get_feature(sensors_chip_name name, int feature, double *result)
{
const sensors_chip_feature *main_feature;
const sensors_chip_feature *alt_feature;
const sensors_chip *chip;
const sensors_expr *expr = NULL;
double val;
int res,i;
int final_expr=0;
if (sensors_chip_name_has_wildcards(name))
return -SENSORS_ERR_WILDCARDS;
if (! (main_feature = sensors_lookup_feature_nr(name.prefix,feature)))
return -SENSORS_ERR_NO_ENTRY;
if (main_feature->compute_mapping == SENSORS_NO_MAPPING)
alt_feature = NULL;
else if (! (alt_feature =
sensors_lookup_feature_nr(name.prefix,
main_feature->compute_mapping)))
return -SENSORS_ERR_NO_ENTRY;
if (! (main_feature->mode && SENSORS_MODE_R))
return -SENSORS_ERR_ACCESS;
for (chip = NULL; !expr && (chip = sensors_for_all_config_chips(name,chip));)
for (i = 0; !final_expr && (i < chip->computes_count); i++) {
if (!strcmp(main_feature->name,chip->computes[i].name)) {
expr = chip->computes[i].from_proc;
final_expr = 1;
} else if (alt_feature &&
!strcmp(alt_feature->name,chip->computes[i].name))
expr = chip->computes[i].from_proc;
}
if (sensors_read_proc(name,feature,&val))
return -SENSORS_ERR_PROC;
if (! expr)
*result = val;
else if ((res = sensors_eval_expr(name,expr,val,result)))
return res;
return 0;
}
/* Set the value of a feature of a certain chip. Note that chip should not
contain wildcard values! This function will return 0 on success, and <0
on failure. BUGGY! */
int sensors_set_feature(sensors_chip_name name, int feature, double value)
{
const sensors_chip_feature *featureptr;
const sensors_chip *chip;
const sensors_expr *expr = NULL;
int i,res;
if (sensors_chip_name_has_wildcards(name))
return -SENSORS_ERR_WILDCARDS;
if (! (featureptr = sensors_lookup_feature_nr(name.prefix,feature)))
return -SENSORS_ERR_NO_ENTRY;
if (! (featureptr->mode && SENSORS_MODE_W))
return -SENSORS_ERR_ACCESS;
for (chip = NULL; !expr && (chip = sensors_for_all_config_chips(name,chip));)
for (i = 0; !expr && (i < chip->computes_count); i++)
if (!strcmp(featureptr->name,chip->computes->name))
expr = chip->computes->to_proc;
if (expr)
if ((res = sensors_eval_expr(name,expr,value,&value)))
return res;
if (sensors_write_proc(name,feature,value))
return -SENSORS_ERR_PROC;
return 0;
}
const sensors_chip_name *sensors_get_detected_chips (int *nr)
{
const sensors_chip_name *res;
res = *nr >= sensors_proc_chips_count?NULL:&sensors_proc_chips[*nr].name;
(*nr)++;
return res;
}
const char *sensors_get_adapter_name(int bus_nr)
{
int i;
if (bus_nr == SENSORS_CHIP_NAME_BUS_ISA)
return "ISA adapter";
for (i=0; i < sensors_proc_bus_count; i++)
if (sensors_proc_bus[i].number == bus_nr)
return sensors_proc_bus[i].adapter;
return NULL;
}
const char *sensors_get_algorithm_name(int bus_nr)
{
int i;
if (bus_nr == SENSORS_CHIP_NAME_BUS_ISA)
return "ISA algorithm";
for (i=0; i < sensors_proc_bus_count; i++)
if (sensors_proc_bus[i].number == bus_nr)
return sensors_proc_bus[i].algorithm;
return NULL;
}
/* nr1-1 is the last main feature found; nr2-1 is the last subfeature found */
const sensors_feature_data *sensors_get_all_features (sensors_chip_name name,
int *nr1, int*nr2)
{
sensors_chip_feature *feature_list;
int i;
for (i = 0; sensors_chip_features_list[i].prefix; i++)
if (!strcmp(sensors_chip_features_list[i].prefix,name.prefix)) {
feature_list=sensors_chip_features_list[i].feature;
if (!*nr1 && !*nr2) { /* Return the first entry */
if (!feature_list[0].name) /* The list may be empty */
return NULL;
*nr1 = *nr2 = 1;
return (sensors_feature_data *) (feature_list + 0);
}
for ((*nr2)++; feature_list[*nr2-1].name; (*nr2)++)
if (feature_list[*nr2-1].logical_mapping == feature_list[*nr1-1].number)
return (sensors_feature_data *) (feature_list + *nr2 - 1);
for ((*nr1)++; feature_list[*nr1-1].name &&
(feature_list[*nr1-1].logical_mapping != SENSORS_NO_MAPPING);
(*nr1)++);
*nr2 = *nr1;
if (! feature_list[*nr1-1].name)
return NULL;
return (sensors_feature_data *) (feature_list + *nr1 - 1);
}
return NULL;
}
int sensors_eval_expr(sensors_chip_name chipname, const sensors_expr *expr,
double val, double *result)
{
double res1,res2;
int res;
const sensors_chip_feature *feature;
if (expr->kind == sensors_kind_val) {
*result = expr->data.val;
return 0;
}
if (expr->kind == sensors_kind_source) {
*result = val;
return 0;
}
if (expr->kind == sensors_kind_var) {
if (! (feature = sensors_lookup_feature_name(chipname.prefix,
expr->data.var)))
return SENSORS_ERR_NO_ENTRY;
if (! (res = sensors_get_feature(chipname,feature->number,result)))
return res;
return 0;
}
if ((res = sensors_eval_expr(chipname,expr->data.subexpr.sub1,val,&res1)))
return res;
if (expr->data.subexpr.sub2 &&
(res = sensors_eval_expr(chipname,expr->data.subexpr.sub2,val,&res2)))
return res;
switch(expr->data.subexpr.op) {
case sensors_add:
*result = res1 + res2;
return 0;
case sensors_sub:
*result = res1 - res2;
return 0;
case sensors_multiply:
*result = res1 * res2;
return 0;
case sensors_divide:
if (res2 == 0.0)
return -SENSORS_ERR_DIV_ZERO;
*result = res1 / res2;
return 0;
case sensors_negate:
*result = -res1;
return 0;
}
return 0;
}