[lm-sensors] Asus P5B Deluxe / Winbond 83627DHG
Jean Delvare
khali at linux-fr.org
Fri Dec 22 15:58:45 CET 2006
Hi David,
On Thu, 21 Dec 2006 13:13:02 -0800, David Hubbard wrote:
> This patch applies cleanly to linux-2.6.20-rc1, adding support for the
> w83627dhg. Whoever is least busy, please review. :-)
Here you go. Functionally, it looks correct to me, I only have comments
on implementation choices and details.
> --- old/drivers/hwmon/w83627ehf.c 2006-12-13 17:14:23.000000000 -0800
> +++ new/drivers/hwmon/w83627ehf.c 2006-12-21 13:53:34.000000000 -0800
> @@ -32,8 +32,10 @@
>
> Supports the following chips:
>
> - Chip #vin #fan #pwm #temp chip_id man_id
> - w83627ehf 10 5 4 3 0x88,0xa1 0x5ca3
> + Chip #vin #fan #pwm #temp chip IDs mfg ID
> + w83627ehf 10 5 4 3 0x8853 0x88 0x5ca3
> + 0x8863 0xa1
> + w83627dhg 9 5 4 3 0xa020 0xc1 0x5ca3
> */
The 4 LSB of the Super-I/O chip ID (4 LSB of LPC register 0x21) are
subject to change in the future, which is why we mask them out when
checking for device ID.
>
> #include <linux/module.h>
> @@ -55,8 +57,18 @@
> * Super-I/O constants and functions
> */
>
> +/*
> + * The three following globals are initialized in w83627ehf_find(), before
> + * the i2c-isa device is created. Otherwise, they could be stored in
> + * w83627ehf_data. This is ugly, but necessary. and when the driver is next
> + * updated to become a platform driver, the globals will disappear.
> + */
> static int REG; /* The register to read/write */
> static int VAL; /* The value to read/write */
> +/* The w83627ehf/ehg have 10 voltage inputs, but the w83627dhg has 9. This
> + * value is also used in w83627ehf_detect() to export a device name in sysfs
> + * (e.g. w83627ehf or w83627dhg) */
> +static int w83627ehf_num_in;
Strange choice. You store one specific difference between both chips
first, then deduce the chip name again based on this difference. A
saner approach would be to store the chip id (or kind) as a global, and
then in w83627ehf_detect(), use the value to set the number of voltage
inputs as a data structure member. It would be much more flexible that
the current approach. This will also make your job easier when
converting the driver to a platform driver.
You can take a look at the it87 driver, it does exactly this.
>
> #define W83627EHF_LD_HWM 0x0b
>
> @@ -65,8 +77,10 @@
> #define SIO_REG_ENABLE 0x30 /* Logical device enable */
> #define SIO_REG_ADDR 0x60 /* Logical device address (2 bytes) */
>
> -#define SIO_W83627EHF_ID 0x8840
> -#define SIO_ID_MASK 0xFFC0
> +#define SIO_W83627EHF_ID 0x8850
> +#define SIO_W83627EHG_ID 0x8860
> +#define SIO_W83627DHG_ID 0xa020
> +#define SIO_ID_MASK 0xFFF0
>
> static inline void
> superio_outb(int reg, int val)
> @@ -115,8 +129,12 @@
>
> #define W83627EHF_REG_BANK 0x4E
> #define W83627EHF_REG_CONFIG 0x40
> -#define W83627EHF_REG_CHIP_ID 0x49
Good catch, I had never noticed that this register definition was bogus.
> +
> +/* Not currently used. 0x4f does not make sense.
How that?
> + * REG_MAN_ID is 0x5ca3 for all supported chips.
> + * REG_CHIP_ID == 0x88/0xa1/0xc1 depending on chip model. */
> #define W83627EHF_REG_MAN_ID 0x4F
> +#define W83627EHF_REG_CHIP_ID 0x58
Why define them at all if we don't use them? Unused defines make
preprocessing more expensive.
>
> static const u16 W83627EHF_REG_FAN[] = { 0x28, 0x29, 0x2a, 0x3f, 0x553 };
> static const u16 W83627EHF_REG_FAN_MIN[] = { 0x3b, 0x3c, 0x3d, 0x3e, 0x55c };
> @@ -429,7 +447,7 @@
> }
>
> /* Measured voltages and limits */
> - for (i = 0; i < 10; i++) {
> + for (i = 0; i < w83627ehf_num_in; i++) {
> data->in[i] = w83627ehf_read_value(client,
> W83627EHF_REG_IN(i));
> data->in_min[i] = w83627ehf_read_value(client,
> @@ -1121,7 +1139,7 @@
> device_remove_file(dev, &sda_sf3_arrays[i].dev_attr);
> for (i = 0; i < ARRAY_SIZE(sda_sf3_arrays_fan4); i++)
> device_remove_file(dev, &sda_sf3_arrays_fan4[i].dev_attr);
> - for (i = 0; i < 10; i++) {
> + for (i = 0; i < w83627ehf_num_in; i++) {
> device_remove_file(dev, &sda_in_input[i].dev_attr);
> device_remove_file(dev, &sda_in_alarm[i].dev_attr);
> device_remove_file(dev, &sda_in_min[i].dev_attr);
> @@ -1196,7 +1214,11 @@
> client->flags = 0;
> dev = &client->dev;
>
> - strlcpy(client->name, "w83627ehf", I2C_NAME_SIZE);
> + if (w83627ehf_num_in == 9)
> + strlcpy(client->name, "w83627dhg", I2C_NAME_SIZE);
> + else /* just say ehf. 627EHG is 627EHF in lead-free packaging. */
> + strlcpy(client->name, "w83627ehf", I2C_NAME_SIZE);
> +
> data->valid = 0;
> mutex_init(&data->update_lock);
>
> @@ -1246,7 +1268,7 @@
> goto exit_remove;
> }
>
> - for (i = 0; i < 10; i++)
> + for (i = 0; i < w83627ehf_num_in; i++)
> if ((err = device_create_file(dev, &sda_in_input[i].dev_attr))
> || (err = device_create_file(dev,
> &sda_in_alarm[i].dev_attr))
> @@ -1340,7 +1362,17 @@
>
> val = (superio_inb(SIO_REG_DEVID) << 8)
> | superio_inb(SIO_REG_DEVID + 1);
> - if ((val & SIO_ID_MASK) != SIO_W83627EHF_ID) {
> + switch (val & SIO_ID_MASK) {
> + case SIO_W83627DHG_ID:
> + w83627ehf_num_in = 9;
> + break;
> + case SIO_W83627EHF_ID:
> + case SIO_W83627EHG_ID:
> + w83627ehf_num_in = 10;
> + break;
> + default:
> + printk(KERN_WARNING "w83627ehf: unknown ID: 0x%04x "
> + "(is this really a w83627?)\n", val);
It could be one of the other W83627 chips (HF and THF), which are not
(and will never be) supported by this driver, so this comment is a bit
misleading. Maybe just "Unsupported chip"?
> superio_exit();
> return -ENODEV;
> }
> --- old/Documentation/hwmon/w83627ehf 2006-12-13 17:14:23.000000000 -0800
> +++ new/Documentation/hwmon/w83627ehf 2006-12-10 16:23:42.000000000 -0800
> @@ -2,26 +2,29 @@
> =======================
>
> Supported chips:
> - * Winbond W83627EHF/EHG (ISA access ONLY)
> + * Winbond W83627EHF/EHG/DHG (ISA access ONLY)
> Prefix: 'w83627ehf'
> Addresses scanned: ISA address retrieved from Super I/O registers
> - Datasheet: http://www.winbond-usa.com/products/winbond_products/pdfs/PCIC/W83627EHF_%20W83627EHGb.pdf
> + Datasheet:
> + http://www.winbond-usa.com/products/winbond_products/pdfs/PCIC/W83627EHF_%20W83627EHGb.pdf
> + DHG datasheet confidential.
Broken alignment.
>
> Authors:
> Jean Delvare <khali at linux-fr.org>
> Yuan Mu (Winbond)
> Rudolf Marek <r.marek at assembler.cz>
> + David Hubbard <david.c.hubbard at gmail.com>
>
> Description
> -----------
>
> -This driver implements support for the Winbond W83627EHF and W83627EHG
> -super I/O chips. We will refer to them collectively as Winbond chips.
> +This driver implements support for the Winbond W83627EHF, W83627EHG, and
> +W83627DHG super I/O chips. We will refer to them collectively as Winbond chips.
>
> The chips implement three temperature sensors, five fan rotation
> -speed sensors, ten analog voltage sensors, alarms with beep warnings (control
> -unimplemented), and some automatic fan regulation strategies (plus manual
> -fan control mode).
> +speed sensors, ten analog voltage sensors (only nine for the 627DHG), alarms
> +with beep warnings (control unimplemented), and some automatic fan regulation
> +strategies (plus manual fan control mode).
>
> Temperatures are measured in degrees Celsius and measurement resolution is 1
> degC for temp1 and 0.5 degC for temp2 and temp3. An alarm is triggered when
> @@ -55,6 +58,9 @@
> /sys files
> ----------
>
> +name - this is a standard hwmon device entry. For the W83627EHF and W83627EHG,
> + it is set to "w83627ehf" and for the W83627DHG it is set to "w83627dhg."
Maybe move the final dot outside of the quotes to avoid unnecessaryt
confusion.
> +
> pwm[1-4] - this file stores PWM duty cycle or DC value (fan speed) in range:
> 0 (stop) to 255 (full)
>
> @@ -83,3 +89,36 @@
>
> Note: last two functions are influenced by other control bits, not yet exported
> by the driver, so a change might not have any effect.
> +
> +Implementation Details
> +----------------------
> +Future driver development should bear in mind that the following registers have
> +different functions on the 627EHF and the 627DHG. Some registers also have
> +different power-on default values, but BIOS should already be loading
> +appropriate defaults. Note that bank selection must be performed as is currently
> +done in the driver for all register addresses.
> +
> +0x49: only on DHG, selects temperature source for AUX fan, CPU fan0
> +0x4a: not completely documented for the EHF and the DHG documentation assigns
> + different behavior to bits 7 and 6, including extending the temperature
> + input selection to SmartFan I, not just SmartFan III. Testing on the EHF
> + will reveal whether they are compatible or not.
> +
> +0x58: Chip ID: 0xa1=EHF 0xc1=DHG
> +0x5e: only on DHG, has bits to enable "current mode" temperature detection and
> + critical temperature protection
> +0x45b: only on EHF, bit 3, vin4 alarm (EHF supports 10 inputs, only 9 on DHG)
> +0x552: only on EHF, vin4
> +0x558: only on EHF, vin4 high limit
> +0x559: only on EHF, vin4 low limit
> +0x6b: only on DHG, SYS fan critical temperature
> +0x6c: only on DHG, CPU fan0 critical temperature
> +0x6d: only on DHG, AUX fan critical temperature
> +0x6e: only on DHG, CPU fan1 critical temperature
> +
> +0x50-0x55 and 0x650-0x657 are marked "Test Register" for the EHF, but "Reserved
> + Register" for the DHG
This last paragraph isn't particularly helpful, "test" and "reserved"
really mean the same.
> +
> +The DHG also supports PECI, where the DHG queries Intel CPU temperatures, and
> +the ICH8 southbridge gets that data via PECI from the DHG, so that the
> +southbridge drives the fans. And the DHG supports SST, a one-wire serial bus.
Thanks,
--
Jean Delvare
More information about the lm-sensors
mailing list