NetBSD/sys/dev/isa/it.c

418 lines
11 KiB
C

/* $NetBSD: it.c,v 1.13 2007/07/20 07:23:47 xtraeme Exp $ */
/* $OpenBSD: it.c,v 1.19 2006/04/10 00:57:54 deraadt Exp $ */
/*
* Copyright (c) 2006-2007 Juan Romero Pardines <juan@xtrarom.org>
* Copyright (c) 2003 Julien Bordet <zejames@greyhats.org>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITD TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITD TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/*
* Driver for the iTE IT8705/IT871[26]F Super I/O hardware monitor.
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: it.c,v 1.13 2007/07/20 07:23:47 xtraeme Exp $");
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/proc.h>
#include <sys/device.h>
#include <sys/malloc.h>
#include <sys/errno.h>
#include <sys/conf.h>
#include <sys/envsys.h>
#include <sys/time.h>
#include <machine/bus.h>
#include <machine/intr.h>
#include <dev/isa/isareg.h>
#include <dev/isa/isavar.h>
#include <dev/sysmon/sysmonvar.h>
#include <dev/isa/itvar.h>
#define IT_VOLTSTART_IDX 3 /* voltage start index */
#define IT_FANSTART_IDX 12 /* fan start index */
#if defined(ITDEBUG)
#define DPRINTF(x) do { printf x; } while (0)
#else
#define DPRINTF(x)
#endif
/*
* IT87-compatible chips can typically measure voltages up to 4.096 V.
* To measure higher voltages the input is attenuated with (external)
* resistors. Negative voltages are measured using a reference
* voltage. So we have to convert the sensor values back to real
* voltages by applying the appropriate resistor factor.
*/
#define RFACT_NONE 10000
#define RFACT(x, y) (RFACT_NONE * ((x) + (y)) / (y))
/* autoconf(9) functions */
static int it_isa_match(struct device *, struct cfdata *, void *);
static void it_isa_attach(struct device *, struct device *, void *);
CFATTACH_DECL(it_isa, sizeof(struct it_softc),
it_isa_match, it_isa_attach, NULL, NULL);
/* driver functions */
static int it_check(bus_space_tag_t, int);
static uint8_t it_readreg(struct it_softc *, int);
static void it_writereg(struct it_softc *, int, int);
/* envsys(9) glue */
static void it_setup_sensors(struct it_softc *);
static void it_refresh_temp(struct it_softc *, envsys_data_t *);
static void it_refresh_volts(struct it_softc *, envsys_data_t *);
static void it_refresh_fans(struct it_softc *, envsys_data_t *);
static int it_gtredata(struct sysmon_envsys *, envsys_data_t *);
/* rfact values for voltage sensors */
static const int it_vrfact[] = {
RFACT_NONE, /* VCORE_A */
RFACT_NONE, /* VCORE_B */
RFACT_NONE, /* +3.3V */
RFACT(68, 100), /* +5V */
RFACT(30, 10), /* +12V */
RFACT(21, 10), /* -12V */
RFACT(83, 20), /* -5V */
RFACT(68, 100), /* STANDBY */
RFACT_NONE /* VBAT */
};
static int
it_isa_match(struct device *parent, struct cfdata *match, void *aux)
{
struct isa_attach_args *ia = aux;
int rv = 0;
/* Must supply an address */
if (ia->ia_nio < 1)
return 0;
if (ISA_DIRECT_CONFIG(ia))
return 0;
if (ia->ia_io[0].ir_addr == ISA_UNKNOWN_PORT)
return 0;
rv = it_check(ia->ia_iot, ia->ia_io[0].ir_addr);
if (rv) {
ia->ia_nio = 1;
ia->ia_io[0].ir_size = 8;
ia->ia_niomem = 0;
ia->ia_nirq = 0;
ia->ia_ndrq = 0;
}
return rv;
}
static void
it_isa_attach(struct device *parent, struct device *self, void *aux)
{
struct it_softc *sc = (struct it_softc *)self;
struct isa_attach_args *ia = aux;
int i;
uint8_t cr;
ia->ia_iot = sc->sc_iot;
if (bus_space_map(sc->sc_iot, ia->ia_io[0].ir_addr, 8, 0,
&sc->sc_ioh)) {
aprint_error(": can't map i/o space\n");
return;
}
sc->sc_idr = it_readreg(sc, IT_COREID);
if (sc->sc_idr == IT_REV_8712)
aprint_normal(": IT871[26]F Hardware monitor\n");
else {
sc->sc_idr = it_readreg(sc, IT_VENDORID);
if (sc->sc_idr == IT_REV_8705)
aprint_normal(": IT8705F Hardware monitor\n");
else
aprint_normal(": iTE unknown vendor id (0x%x)\n",
sc->sc_idr);
}
/* Activate monitoring */
cr = it_readreg(sc, IT_CONFIG);
SET(cr, 0x01);
it_writereg(sc, IT_CONFIG, cr);
#ifdef notyet
/* Enable beep alarms */
br = it_readreg(sc, IT_BEEPEER);
SET(br, 0x02); /* Voltage exceeds limit */
SET(br, 0x04); /* Temperature exceeds limit */
it_writereg(sc, IT_BEEPEER, br);
#endif
/* Initialize sensors */
for (i = 0; i < IT_NUM_SENSORS; ++i) {
sc->sc_data[i].sensor = i;
sc->sc_data[i].state = ENVSYS_SVALID;
}
it_setup_sensors(sc);
/*
* Hook into the system monitor.
*/
sc->sc_sysmon.sme_name = sc->sc_dev.dv_xname;
sc->sc_sysmon.sme_sensor_data = sc->sc_data;
sc->sc_sysmon.sme_cookie = sc;
sc->sc_sysmon.sme_gtredata = it_gtredata;
sc->sc_sysmon.sme_nsensors = IT_NUM_SENSORS;
if (sysmon_envsys_register(&sc->sc_sysmon))
printf("%s: unable to register with sysmon\n",
sc->sc_dev.dv_xname);
}
static int
it_check(bus_space_tag_t iot, int base)
{
bus_space_handle_t ioh;
int rv = 0;
uint8_t cr0, cr1;
if (bus_space_map(iot, base, 8, 0, &ioh))
return 0;
bus_space_write_1(iot, ioh, IT_ADDR, IT_RES48);
cr0 = bus_space_read_1(iot, ioh, IT_DATA);
bus_space_write_1(iot, ioh, IT_ADDR, IT_RES52);
cr1 = bus_space_read_1(iot, ioh, IT_DATA);
if (cr0 == IT_RES48_DEFAULT && cr1 == IT_RES52_DEFAULT)
rv = 1;
bus_space_unmap(iot, ioh, 8);
return rv;
}
static uint8_t
it_readreg(struct it_softc *sc, int reg)
{
bus_space_write_1(sc->sc_iot, sc->sc_ioh, IT_ADDR, reg);
return bus_space_read_1(sc->sc_iot, sc->sc_ioh, IT_DATA);
}
static void
it_writereg(struct it_softc *sc, int reg, int val)
{
bus_space_write_1(sc->sc_iot, sc->sc_ioh, IT_ADDR, reg);
bus_space_write_1(sc->sc_iot, sc->sc_ioh, IT_DATA, val);
}
#define COPYDESCR(x, y) \
do { \
strlcpy((x), (y), sizeof(x)); \
} while (0)
static void
it_setup_sensors(struct it_softc *sc)
{
int i;
/* temperatures */
for (i = 0; i < IT_VOLTSTART_IDX; i++)
sc->sc_data[i].units = ENVSYS_STEMP;
COPYDESCR(sc->sc_data[0].desc, "CPU Temp");
COPYDESCR(sc->sc_data[1].desc, "System Temp");
COPYDESCR(sc->sc_data[2].desc, "Aux Temp");
/* voltages */
for (i = IT_VOLTSTART_IDX; i < IT_FANSTART_IDX; i++) {
sc->sc_data[i].units = ENVSYS_SVOLTS_DC;
sc->sc_data[i].flags = ENVSYS_FCHANGERFACT;
}
COPYDESCR(sc->sc_data[3].desc, "VCORE_A");
COPYDESCR(sc->sc_data[4].desc, "VCORE_B");
COPYDESCR(sc->sc_data[5].desc, "+3.3V");
COPYDESCR(sc->sc_data[6].desc, "+5V");
COPYDESCR(sc->sc_data[7].desc, "+12V");
COPYDESCR(sc->sc_data[8].desc, "-12V");
COPYDESCR(sc->sc_data[9].desc, "-5V");
COPYDESCR(sc->sc_data[10].desc, "STANDBY");
COPYDESCR(sc->sc_data[11].desc, "VBAT");
/* fans */
for (i = IT_FANSTART_IDX; i < IT_NUM_SENSORS; i++)
sc->sc_data[i].units = ENVSYS_SFANRPM;
COPYDESCR(sc->sc_data[12].desc, "CPU Fan");
COPYDESCR(sc->sc_data[13].desc, "System Fan");
COPYDESCR(sc->sc_data[14].desc, "Aux Fan");
}
#undef COPYDESCR
static void
it_refresh_temp(struct it_softc *sc, envsys_data_t *edata)
{
int sdata;
sdata = it_readreg(sc, IT_SENSORTEMPBASE + edata->sensor);
/* sensor is not connected or reporting invalid data */
if (sdata == 0 || sdata >= 0xfa) {
edata->state = ENVSYS_SINVALID;
return;
}
DPRINTF(("%s: sdata[temp%d] 0x%x\n", __func__, edata->sensor, sdata));
/* Convert temperature to uK */
edata->value_cur = sdata * 1000000 + 273150000;
edata->state = ENVSYS_SVALID;
}
static void
it_refresh_volts(struct it_softc *sc, envsys_data_t *edata)
{
uint8_t vbatcr = 0;
int i, sdata;
i = edata->sensor - IT_VOLTSTART_IDX;
sdata = it_readreg(sc, IT_SENSORVOLTBASE + i);
/* not connected */
if (sdata == 0) {
edata->state = ENVSYS_SINVALID;
return;
}
/*
* update VBAT voltage reading every time we read it, to get
* latest value.
*/
if (i == 8) {
vbatcr = it_readreg(sc, IT_CONFIG);
SET(vbatcr, IT_UPDATEVBAT);
it_writereg(sc, IT_CONFIG, vbatcr);
}
DPRINTF(("%s: sdata[volt%d] 0x%x\n", __func__, i, sdata));
/* voltage returned as (mV << 4) */
edata->value_cur = (sdata << 4);
/* rfact is (factor * 10^4) */
edata->value_cur *= it_vrfact[i];
if (edata->rfact)
edata->value_cur += edata->rfact;
else
edata->rfact = it_vrfact[i];
/* division by 10 gets us back to uVDC */
edata->value_cur /= 10;
edata->state = ENVSYS_SVALID;
}
static void
it_refresh_fans(struct it_softc *sc, envsys_data_t *edata)
{
int i, mode, sdata, divisor, odivisor, ndivisor;
i = edata->sensor - IT_FANSTART_IDX;
sdata = divisor = odivisor = ndivisor = 0;
#define FANDATA() it_readreg(sc, IT_SENSORFANEXTBASE + i)
mode = it_readreg(sc, IT_FAN16);
if (sc->sc_idr == IT_REV_8705) {
odivisor = ndivisor = divisor = it_readreg(sc, IT_FAN);
divisor >>= 3;
sdata = it_readreg(sc, IT_SENSORFANBASE + i);
if (mode & (1 << i))
sdata += (FANDATA() << 8);
if (!(mode & (1 << i)) && sdata == 0xff) {
edata->state = ENVSYS_SINVALID;
if (i == 2)
ndivisor |= 0x40;
else if ((divisor & 7) != 7) {
ndivisor &= ~(7 << (i * 3));
ndivisor |= ((divisor + 1) & 7) << (i * 3);
}
} else {
if (i == 2)
divisor = divisor & 1 ? 3 : 1;
if ((sdata << (divisor & 7)) == 0)
edata->state = ENVSYS_SINVALID;
else {
edata->value_cur =
1350000 / (sdata << (divisor & 7));
edata->state = ENVSYS_SVALID;
}
}
DPRINTF(("%s: sdata[fan%d] 0x%x div: 0x%x\n", __func__,
i, sdata, divisor));
if (ndivisor != odivisor)
it_writereg(sc, IT_FAN, ndivisor);
} else {
/* IT8712F, IT8716F */
sdata = it_readreg(sc, IT_SENSORFANBASE + i);
if (mode & (1 << i)) /* 16-bit mode enabled */
sdata += (FANDATA() << 8);
edata->state = ENVSYS_SVALID;
if (sdata == 0 ||
sdata == ((mode & (1 << i)) ? 0xffff : 0xff))
edata->state = ENVSYS_SINVALID;
else {
edata->value_cur = 1350000 / 2 / sdata;
edata->state = ENVSYS_SVALID;
}
DPRINTF(("%s: sdata[fan%d] 0x%x div: 0x%x\n", __func__,
i, sdata, divisor));
}
#undef FANDATA
}
static int
it_gtredata(struct sysmon_envsys *sme, struct envsys_data *edata)
{
struct it_softc *sc = sme->sme_cookie;
if (edata->sensor < IT_VOLTSTART_IDX)
it_refresh_temp(sc, edata);
else if (edata->sensor >= IT_VOLTSTART_IDX &&
edata->sensor < IT_FANSTART_IDX)
it_refresh_volts(sc, edata);
else if (edata->sensor >= IT_FANSTART_IDX)
it_refresh_fans(sc, edata);
else
return 0;
return 0;
}