Fix handling of 6bit VID register (6th bit takes place of 12VIn sensor).

Move VID out of sysctl tree - make it a separate sensor of type INTEGER.

Clean-up some of the chip-descriptors.
This commit is contained in:
pgoyette 2010-02-24 23:37:45 +00:00
parent 197d9d4ff8
commit c138fa58da
2 changed files with 71 additions and 71 deletions

View File

@ -1,4 +1,4 @@
/* $NetBSD: dbcool.c,v 1.15 2010/02/24 22:37:57 dyoung Exp $ */
/* $NetBSD: dbcool.c,v 1.16 2010/02/24 23:37:45 pgoyette Exp $ */
/*-
* Copyright (c) 2008 The NetBSD Foundation, Inc.
@ -49,7 +49,7 @@
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: dbcool.c,v 1.15 2010/02/24 22:37:57 dyoung Exp $");
__KERNEL_RCSID(0, "$NetBSD: dbcool.c,v 1.16 2010/02/24 23:37:45 pgoyette Exp $");
#include <sys/param.h>
#include <sys/systm.h>
@ -89,7 +89,6 @@ static int sysctl_dbcool_volt_limit(SYSCTLFN_PROTO);
static int sysctl_dbcool_temp_limit(SYSCTLFN_PROTO);
static int sysctl_dbcool_fan_limit(SYSCTLFN_PROTO);
static int sysctl_dbcool_thyst(SYSCTLFN_PROTO);
static int sysctl_dbcool_vid(SYSCTLFN_PROTO);
/* Set-up subroutines */
static void dbcool_setup_controllers(struct dbcool_softc *,
@ -155,7 +154,7 @@ static struct dbc_sysctl_info dbc_sysctl_table[] = {
static const char *dbc_sensor_names[] = {
"l_temp", "r1_temp", "r2_temp", "Vccp", "Vcc", "fan1",
"fan2", "fan3", "fan4", "AIN1", "AIN2", "V2dot5",
"V5", "V12", "Vtt", "Imon"
"V5", "V12", "Vtt", "Imon", "VID"
};
/*
@ -220,6 +219,9 @@ struct dbcool_sensor ADT7490_sensor_table[] = {
{ DBC_FAN, { DBCOOL_FAN4_TACH_LSB,
DBCOOL_NO_REG,
DBCOOL_TACH4_MIN_LSB }, 8, 0, 0 },
{ DBC_VID, { DBCOOL_VID_REG,
DBCOOL_NO_REG,
DBCOOL_NO_REG }, 16, 0, 0 },
{ DBC_CTL, { DBCOOL_LOCAL_TMIN,
DBCOOL_NO_REG,
DBCOOL_NO_REG }, 0, 5, 0 },
@ -287,6 +289,9 @@ struct dbcool_sensor ADT7476_sensor_table[] = {
{ DBC_FAN, { DBCOOL_FAN4_TACH_LSB,
DBCOOL_NO_REG,
DBCOOL_TACH4_MIN_LSB }, 8, 0, 0 },
{ DBC_VID, { DBCOOL_VID_REG,
DBCOOL_NO_REG,
DBCOOL_NO_REG }, 16, 0, 0 },
{ DBC_CTL, { DBCOOL_LOCAL_TMIN,
DBCOOL_NO_REG,
DBCOOL_NO_REG }, 0, 5, 0 },
@ -454,6 +459,9 @@ struct dbcool_sensor ADM1027_sensor_table[] = {
{ DBC_FAN, { DBCOOL_FAN4_TACH_LSB,
DBCOOL_NO_REG,
DBCOOL_TACH4_MIN_LSB }, 8, 0, 0 },
{ DBC_VID, { DBCOOL_VID_REG,
DBCOOL_NO_REG,
DBCOOL_NO_REG }, 16, 0, 0 },
{ DBC_CTL, { DBCOOL_LOCAL_TMIN,
DBCOOL_NO_REG,
DBCOOL_NO_REG }, 0, 5, 0 },
@ -524,13 +532,12 @@ struct dbcool_power_control ADM1030_power_table[] = {
struct chip_id chip_table[] = {
{ DBCOOL_COMPANYID, ADT7490_DEVICEID, ADT7490_REV_ID,
ADT7475_sensor_table, ADT7475_power_table,
DBCFLAG_TEMPOFFSET | DBCFLAG_HAS_MAXDUTY | DBCFLAG_HAS_VID |
DBCFLAG_HAS_PECI,
ADT7490_sensor_table, ADT7475_power_table,
DBCFLAG_TEMPOFFSET | DBCFLAG_HAS_MAXDUTY | DBCFLAG_HAS_PECI,
90000 * 60, "ADT7490" },
{ DBCOOL_COMPANYID, ADT7476_DEVICEID, 0xff,
ADT7476_sensor_table, ADT7475_power_table,
DBCFLAG_TEMPOFFSET | DBCFLAG_HAS_MAXDUTY | DBCFLAG_HAS_VID,
DBCFLAG_TEMPOFFSET | DBCFLAG_HAS_MAXDUTY,
90000 * 60, "ADT7476" },
{ DBCOOL_COMPANYID, ADT7475_DEVICEID, 0xff,
ADT7475_sensor_table, ADT7475_power_table,
@ -547,7 +554,7 @@ struct chip_id chip_table[] = {
{ DBCOOL_COMPANYID, ADT7468_DEVICEID, 0xff,
ADT7476_sensor_table, ADT7475_power_table,
DBCFLAG_TEMPOFFSET | DBCFLAG_MULTI_VCC | DBCFLAG_HAS_MAXDUTY |
DBCFLAG_4BIT_VER | DBCFLAG_HAS_SHDN | DBCFLAG_HAS_VID,
DBCFLAG_4BIT_VER | DBCFLAG_HAS_SHDN,
90000 * 60, "ADT7467/ADT7468" },
{ DBCOOL_COMPANYID, ADT7466_DEVICEID, 0xff,
ADT7466_sensor_table, NULL,
@ -555,21 +562,20 @@ struct chip_id chip_table[] = {
82000 * 60, "ADT7466" },
{ DBCOOL_COMPANYID, ADT7463_DEVICEID, ADT7463_REV_ID1,
ADM1027_sensor_table, ADT7475_power_table,
DBCFLAG_MULTI_VCC | DBCFLAG_4BIT_VER | DBCFLAG_HAS_SHDN |
DBCFLAG_ADM1027 | DBCFLAG_HAS_VID,
DBCFLAG_MULTI_VCC | DBCFLAG_4BIT_VER | DBCFLAG_HAS_SHDN,
90000 * 60, "ADT7463" },
{ DBCOOL_COMPANYID, ADT7463_DEVICEID, ADT7463_REV_ID2,
ADM1027_sensor_table, ADT7475_power_table,
DBCFLAG_MULTI_VCC | DBCFLAG_4BIT_VER | DBCFLAG_HAS_SHDN |
DBCFLAG_HAS_VID | DBCFLAG_HAS_VID_SEL,
DBCFLAG_HAS_VID_SEL,
90000 * 60, "ADT7463" },
{ DBCOOL_COMPANYID, ADM1027_DEVICEID, ADM1027_REV_ID,
ADM1027_sensor_table, ADT7475_power_table,
DBCFLAG_MULTI_VCC | DBCFLAG_4BIT_VER | DBCFLAG_HAS_VID,
DBCFLAG_MULTI_VCC | DBCFLAG_4BIT_VER,
90000 * 60, "ADM1027" },
{ DBCOOL_COMPANYID, ADM1030_DEVICEID, 0xff,
ADM1030_sensor_table, ADM1030_power_table,
DBCFLAG_ADM1030,
DBCFLAG_ADM1030 | DBCFLAG_NO_READBYTE,
11250 * 60, "ADM1030" },
{ 0, 0, 0, NULL, NULL, 0, 0, NULL }
};
@ -703,7 +709,7 @@ dbcool_readreg(struct dbcool_chipset *dc, uint8_t reg)
if (iic_acquire_bus(dc->dc_tag, 0) != 0)
return data;
if (dc->dc_chip == NULL || dc->dc_chip->flags & DBCFLAG_ADM1027) {
if (dc->dc_chip == NULL || dc->dc_chip->flags & DBCFLAG_NO_READBYTE) {
/* ADM1027 doesn't support i2c read_byte protocol */
if (iic_smbus_send_byte(dc->dc_tag, dc->dc_addr, reg, 0) != 0)
goto bad;
@ -1339,35 +1345,6 @@ sysctl_dbcool_fan_limit(SYSCTLFN_ARGS)
return 0;
}
static int
sysctl_dbcool_vid(SYSCTLFN_ARGS)
{
struct sysctlnode node;
struct dbcool_softc *sc;
int reg, error;
uint8_t chipreg, newreg;
node = *rnode;
sc = (struct dbcool_softc *)node.sysctl_data;
chipreg = node.sysctl_num;
/* retrieve 5- or 6-bit value */
newreg = sc->sc_dc.dc_readreg(&sc->sc_dc, chipreg);
if ((sc->sc_dc.dc_chip->flags & DBCFLAG_HAS_VID_SEL) &&
(reg & 0x80))
reg = newreg & 0x3f;
else
reg = newreg & 0x1f;
node.sysctl_data = &reg;
error = sysctl_lookup(SYSCTLFN_CALL(&node));
if (error == 0 && newp != NULL)
error = EINVAL;
return error;
}
static int
sysctl_dbcool_thyst(SYSCTLFN_ARGS)
{
@ -1518,16 +1495,6 @@ dbcool_setup(device_t self)
CTLTYPE_NODE, device_xname(self), NULL,
NULL, 0, NULL, 0,
CTL_HW, CTL_CREATE, CTL_EOL);
if (sc->sc_dc.dc_chip->flags & DBCFLAG_HAS_VID) {
ret = sysctl_createv(NULL, 0, NULL,
(const struct sysctlnode **)&node,
CTLFLAG_READONLY, CTLTYPE_INT, "CPU_VID_bits", NULL,
sysctl_dbcool_vid,
0, sc, sizeof(int),
CTL_HW, me->sysctl_num, DBCOOL_VID_REG, CTL_EOL);
if (node != NULL)
node->sysctl_data = sc;
}
#ifdef DBCOOL_DEBUG
ret = sysctl_createv(NULL, 0, NULL,
@ -1596,33 +1563,62 @@ dbcool_setup_sensors(struct dbcool_softc *sc, const struct sysctlnode *me,
{
int i, j, ret;
int error = 0;
uint8_t sysctl_reg;
uint8_t sysctl_reg, vid_reg, vid_val;
struct sysctlnode *node = NULL;
struct chip_id *chip = sc->sc_dc.dc_chip;
int sysctl_index, sysctl_num;
char name[SYSCTL_NAMELEN];
for (i=0; sc->sc_dc.dc_chip->table[i].type != DBC_EOF; i++) {
for (i=0; chip->table[i].type != DBC_EOF; i++) {
if (i >= DBCOOL_MAXSENSORS &&
sc->sc_dc.dc_chip->table[i].type != DBC_CTL) {
chip->table[i].type != DBC_CTL) {
aprint_normal_dev(sc->sc_dev, "chip table too big!\n");
break;
}
switch (sc->sc_dc.dc_chip->table[i].type) {
switch (chip->table[i].type) {
case DBC_TEMP:
sc->sc_sensor[i].units = ENVSYS_STEMP;
sc->sc_sensor[i].flags |= ENVSYS_FMONLIMITS;
error = dbcool_attach_sensor(sc, me, i,
sysctl_dbcool_temp_limit);
break;
case DBC_VOLT:
/*
* If 12V-In pin has been reconfigured as 6th bit
* of VID code, don't create a 12V-In sensor
*/
if ((chip->flags & DBCFLAG_HAS_VID_SEL) &&
(chip->table[i].reg.val_reg == DBCOOL_12VIN) &&
(sc->sc_dc.dc_readreg(&sc->sc_dc, DBCOOL_VID_REG) &
0x80))
break;
sc->sc_sensor[i].units = ENVSYS_SVOLTS_DC;
sc->sc_sensor[i].flags |= ENVSYS_FMONLIMITS;
error = dbcool_attach_sensor(sc, me, i,
sysctl_dbcool_volt_limit);
break;
case DBC_FAN:
sc->sc_sensor[i].units = ENVSYS_SFANRPM;
sc->sc_sensor[i].flags |= ENVSYS_FMONLIMITS;
error = dbcool_attach_sensor(sc, me, i,
sysctl_dbcool_fan_limit);
break;
case DBC_VID:
sc->sc_sensor[i].units = ENVSYS_INTEGER;
sc->sc_sensor[i].flags |= ENVSYS_FMONNOTSUPP;
/* retrieve 5- or 6-bit value */
vid_reg = chip->table[i].reg.val_reg;
vid_val = sc->sc_dc.dc_readreg(&sc->sc_dc, vid_reg);
if (chip->flags & DBCFLAG_HAS_VID_SEL)
vid_val &= 0x3f;
else
vid_val &= 0x1f;
sc->sc_sensor[i].value_cur = vid_val;
error = dbcool_attach_sensor(sc, me, i, NULL);
break;
case DBC_CTL:
/*
* Search for the corresponding temp sensor
@ -1631,18 +1627,18 @@ dbcool_setup_sensors(struct dbcool_softc *sc, const struct sysctlnode *me,
sysctl_num = -1;
for (j = 0; j < i; j++) {
if (j > DBCOOL_MAXSENSORS ||
sc->sc_dc.dc_chip->table[j].type != DBC_TEMP)
chip->table[j].type != DBC_TEMP)
continue;
if (sc->sc_dc.dc_chip->table[j].name_index ==
sc->sc_dc.dc_chip->table[i].name_index) {
if (chip->table[j].name_index ==
chip->table[i].name_index) {
sysctl_num = sc->sc_sysctl_num[j];
break;
}
}
if (sysctl_num == -1)
break;
sysctl_index = sc->sc_dc.dc_chip->table[i].sysctl_index;
sysctl_reg = sc->sc_dc.dc_chip->table[i].reg.val_reg;
sysctl_index = chip->table[i].sysctl_index;
sysctl_reg = chip->table[i].reg.val_reg;
strlcpy(name, dbc_sysctl_table[sysctl_index].name,
sizeof(name));
ret = sysctl_createv(NULL, 0, NULL,
@ -1661,7 +1657,7 @@ dbcool_setup_sensors(struct dbcool_softc *sc, const struct sysctlnode *me,
default:
aprint_error_dev(sc->sc_dev,
"sensor_table index %d has bad type %d\n",
i, sc->sc_dc.dc_chip->table[i].type);
i, chip->table[i].type);
break;
}
if (error)
@ -1687,12 +1683,13 @@ dbcool_attach_sensor(struct dbcool_softc *sc, const struct sysctlnode *me,
sc->sc_regs[idx] = &sc->sc_dc.dc_chip->table[idx].reg;
sc->sc_nom_volt[idx] = sc->sc_dc.dc_chip->table[idx].nom_volt_index;
sc->sc_sensor[idx].flags |= ENVSYS_FMONLIMITS;
error = sysmon_envsys_sensor_attach(sc->sc_sme, &sc->sc_sensor[idx]);
if (error)
return error;
/* VIDs do not have any limits */
if (sc->sc_dc.dc_chip->table[idx].type == DBC_VID)
return 0;
/*
* create sysctl node for the sensor, and the nodes for
* the sensor's high and low limit values
@ -1810,6 +1807,8 @@ dbcool_refresh(struct sysmon_envsys *sme, envsys_data_t *edata)
low = dbcool_read_rpm(sc, reg->lo_lim_reg);
hi = 1 << 16;
break;
case ENVSYS_INTEGER:
return;
default:
edata->state = ENVSYS_SINVALID;
return;

View File

@ -1,4 +1,4 @@
/* $NetBSD: dbcool_var.h,v 1.8 2010/02/24 22:37:57 dyoung Exp $ */
/* $NetBSD: dbcool_var.h,v 1.9 2010/02/24 23:37:45 pgoyette Exp $ */
/*-
* Copyright (c) 2008 The NetBSD Foundation, Inc.
@ -41,7 +41,7 @@
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: dbcool_var.h,v 1.8 2010/02/24 22:37:57 dyoung Exp $");
__KERNEL_RCSID(0, "$NetBSD: dbcool_var.h,v 1.9 2010/02/24 23:37:45 pgoyette Exp $");
#include <dev/i2c/i2cvar.h>
@ -62,6 +62,7 @@ enum dbc_sensor_type {
DBC_TEMP,
DBC_VOLT,
DBC_FAN,
DBC_VID,
DBC_EOF
};
@ -70,15 +71,15 @@ enum dbc_sensor_type {
#define DBCFLAG_HAS_SHDN 0x0004
#define DBCFLAG_MULTI_VCC 0x0008
#define DBCFLAG_4BIT_VER 0x0010
#define DBCFLAG_HAS_VID 0x0020
#define DBCFLAG_OBSOLETE 0x0020 /* was DBCFLAG_HAS_VID */
#define DBCFLAG_HAS_VID_SEL 0x0040
#define DBCFLAG_HAS_PECI 0x0080
#define DBCFLAG_ADM1027 0x1000
#define DBCFLAG_NO_READBYTE 0x1000
#define DBCFLAG_ADM1030 0x2000
#define DBCFLAG_ADT7466 0x4000
/* Maximum sensors for any dbCool device */
#define DBCOOL_MAXSENSORS 15
#define DBCOOL_MAXSENSORS 16
struct reg_list {
uint8_t val_reg;