Commit 67b671bc authored by Jean Delvare's avatar Jean Delvare Committed by Mark M. Hoffman

hwmon: Let the user override the detected Super-I/O device ID

While it is possible to force SMBus-based hardware monitoring chip
drivers to drive a not officially supported device, we do not have this
possibility for Super-I/O-based drivers. That's unfortunate because
sometimes newer chips are fully compatible and just forcing the driver
to load would work. Instead of that we have to tell the users to
recompile the kernel driver, which isn't an easy task for everyone.

So, I propose that we add a module parameter to all Super-I/O based
hardware monitoring drivers, letting advanced users force the driver
to load on their machine. The user has to provide the device ID of a
supposedly compatible device. This requires looking at the source code or
a datasheet, so I am confident that users can't randomly force a driver
without knowing what they are doing. Thus this should be relatively safe.

As you can see from the code, the implementation is pretty simple and
unintrusive.
Signed-off-by: default avatarJean Delvare <khali@linux-fr.org>
Acked-by: default avatarHans de Goede <j.w.r.degoede@hhs.nl>
Signed-off-by: default avatarMark M. Hoffman <mhoffman@lightlink.com>
parent b20ff13a
...@@ -44,6 +44,10 @@ static int force_start; ...@@ -44,6 +44,10 @@ static int force_start;
module_param(force_start, bool, 0); module_param(force_start, bool, 0);
MODULE_PARM_DESC(force_start, "Force the chip to start monitoring inputs"); MODULE_PARM_DESC(force_start, "Force the chip to start monitoring inputs");
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
/* Addresses to scan */ /* Addresses to scan */
static unsigned short normal_i2c[] = {0x2c, 0x2d, 0x2e, I2C_CLIENT_END}; static unsigned short normal_i2c[] = {0x2c, 0x2d, 0x2e, I2C_CLIENT_END};
...@@ -2191,7 +2195,7 @@ static int __init dme1737_isa_detect(int sio_cip, unsigned short *addr) ...@@ -2191,7 +2195,7 @@ static int __init dme1737_isa_detect(int sio_cip, unsigned short *addr)
/* Check device ID /* Check device ID
* We currently know about SCH3112 (0x7c), SCH3114 (0x7d), and * We currently know about SCH3112 (0x7c), SCH3114 (0x7d), and
* SCH3116 (0x7f). */ * SCH3116 (0x7f). */
reg = dme1737_sio_inb(sio_cip, 0x20); reg = force_id ? force_id : dme1737_sio_inb(sio_cip, 0x20);
if (!(reg == 0x7c || reg == 0x7d || reg == 0x7f)) { if (!(reg == 0x7c || reg == 0x7d || reg == 0x7f)) {
err = -ENODEV; err = -ENODEV;
goto exit; goto exit;
......
...@@ -41,6 +41,10 @@ ...@@ -41,6 +41,10 @@
#include <linux/ioport.h> #include <linux/ioport.h>
#include <asm/io.h> #include <asm/io.h>
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
static struct platform_device *pdev; static struct platform_device *pdev;
#define DRVNAME "f71805f" #define DRVNAME "f71805f"
...@@ -1497,7 +1501,7 @@ static int __init f71805f_find(int sioaddr, unsigned short *address, ...@@ -1497,7 +1501,7 @@ static int __init f71805f_find(int sioaddr, unsigned short *address,
if (devid != SIO_FINTEK_ID) if (devid != SIO_FINTEK_ID)
goto exit; goto exit;
devid = superio_inw(sioaddr, SIO_REG_DEVID); devid = force_id ? force_id : superio_inw(sioaddr, SIO_REG_DEVID);
switch (devid) { switch (devid) {
case SIO_F71805F_ID: case SIO_F71805F_ID:
sio_data->kind = f71805f; sio_data->kind = f71805f;
......
...@@ -74,6 +74,10 @@ ...@@ -74,6 +74,10 @@
#define FAN_MIN_DETECT 366 /* Lowest detectable fanspeed */ #define FAN_MIN_DETECT 366 /* Lowest detectable fanspeed */
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
static struct platform_device *f71882fg_pdev = NULL; static struct platform_device *f71882fg_pdev = NULL;
/* Super-I/O Function prototypes */ /* Super-I/O Function prototypes */
...@@ -843,7 +847,7 @@ static int __init f71882fg_find(int sioaddr, unsigned short *address) ...@@ -843,7 +847,7 @@ static int __init f71882fg_find(int sioaddr, unsigned short *address)
goto exit; goto exit;
} }
devid = superio_inw(sioaddr, SIO_REG_DEVID); devid = force_id ? force_id : superio_inw(sioaddr, SIO_REG_DEVID);
if (devid != SIO_F71882_ID) { if (devid != SIO_F71882_ID) {
printk(KERN_INFO DRVNAME ": Unsupported Fintek device\n"); printk(KERN_INFO DRVNAME ": Unsupported Fintek device\n");
goto exit; goto exit;
......
...@@ -52,6 +52,10 @@ ...@@ -52,6 +52,10 @@
enum chips { it87, it8712, it8716, it8718 }; enum chips { it87, it8712, it8716, it8718 };
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
static struct platform_device *pdev; static struct platform_device *pdev;
#define REG 0x2e /* The register to read/write */ #define REG 0x2e /* The register to read/write */
...@@ -906,7 +910,7 @@ static int __init it87_find(unsigned short *address, ...@@ -906,7 +910,7 @@ static int __init it87_find(unsigned short *address,
u16 chip_type; u16 chip_type;
superio_enter(); superio_enter();
chip_type = superio_inw(DEVID); chip_type = force_id ? force_id : superio_inw(DEVID);
switch (chip_type) { switch (chip_type) {
case IT8705F_DEVID: case IT8705F_DEVID:
......
...@@ -59,6 +59,10 @@ MODULE_PARM_DESC(init, ...@@ -59,6 +59,10 @@ MODULE_PARM_DESC(init,
" 2: Forcibly enable all voltage and temperature channels, except in9\n" " 2: Forcibly enable all voltage and temperature channels, except in9\n"
" 3: Forcibly enable all voltage and temperature channels, including in9"); " 3: Forcibly enable all voltage and temperature channels, including in9");
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
/* /*
* Super-I/O registers and operations * Super-I/O registers and operations
*/ */
...@@ -826,7 +830,7 @@ static int __init pc87360_find(int sioaddr, u8 *devid, unsigned short *addresses ...@@ -826,7 +830,7 @@ static int __init pc87360_find(int sioaddr, u8 *devid, unsigned short *addresses
/* No superio_enter */ /* No superio_enter */
/* Identify device */ /* Identify device */
val = superio_inb(sioaddr, DEVID); val = force_id ? force_id : superio_inb(sioaddr, DEVID);
switch (val) { switch (val) {
case 0xE1: /* PC87360 */ case 0xE1: /* PC87360 */
case 0xE8: /* PC87363 */ case 0xE8: /* PC87363 */
......
...@@ -34,6 +34,10 @@ ...@@ -34,6 +34,10 @@
#include <linux/ioport.h> #include <linux/ioport.h>
#include <asm/io.h> #include <asm/io.h>
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
static struct platform_device *pdev; static struct platform_device *pdev;
#define DRVNAME "pc87427" #define DRVNAME "pc87427"
...@@ -555,7 +559,7 @@ static int __init pc87427_find(int sioaddr, unsigned short *address) ...@@ -555,7 +559,7 @@ static int __init pc87427_find(int sioaddr, unsigned short *address)
int i, err = 0; int i, err = 0;
/* Identify device */ /* Identify device */
val = superio_inb(sioaddr, SIOREG_DEVID); val = force_id ? force_id : superio_inb(sioaddr, SIOREG_DEVID);
if (val != 0xf2) { /* PC87427 */ if (val != 0xf2) { /* PC87427 */
err = -ENODEV; err = -ENODEV;
goto exit; goto exit;
......
...@@ -38,6 +38,10 @@ ...@@ -38,6 +38,10 @@
#include <linux/mutex.h> #include <linux/mutex.h>
#include <asm/io.h> #include <asm/io.h>
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
static struct platform_device *pdev; static struct platform_device *pdev;
#define DRVNAME "smsc47b397" #define DRVNAME "smsc47b397"
...@@ -333,7 +337,7 @@ static int __init smsc47b397_find(unsigned short *addr) ...@@ -333,7 +337,7 @@ static int __init smsc47b397_find(unsigned short *addr)
u8 id, rev; u8 id, rev;
superio_enter(); superio_enter();
id = superio_inb(SUPERIO_REG_DEVID); id = force_id ? force_id : superio_inb(SUPERIO_REG_DEVID);
if ((id != 0x6f) && (id != 0x81) && (id != 0x85)) { if ((id != 0x6f) && (id != 0x81) && (id != 0x85)) {
superio_exit(); superio_exit();
......
...@@ -39,6 +39,10 @@ ...@@ -39,6 +39,10 @@
#include <linux/sysfs.h> #include <linux/sysfs.h>
#include <asm/io.h> #include <asm/io.h>
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
static struct platform_device *pdev; static struct platform_device *pdev;
#define DRVNAME "smsc47m1" #define DRVNAME "smsc47m1"
...@@ -399,7 +403,7 @@ static int __init smsc47m1_find(unsigned short *addr, ...@@ -399,7 +403,7 @@ static int __init smsc47m1_find(unsigned short *addr,
u8 val; u8 val;
superio_enter(); superio_enter();
val = superio_inb(SUPERIO_REG_DEVID); val = force_id ? force_id : superio_inb(SUPERIO_REG_DEVID);
/* /*
* SMSC LPC47M10x/LPC47M112/LPC47M13x (device id 0x59), LPC47M14x * SMSC LPC47M10x/LPC47M112/LPC47M13x (device id 0x59), LPC47M14x
......
...@@ -42,6 +42,10 @@ static int int_mode = -1; ...@@ -42,6 +42,10 @@ static int int_mode = -1;
module_param(int_mode, int, 0); module_param(int_mode, int, 0);
MODULE_PARM_DESC(int_mode, "Force the temperature interrupt mode"); MODULE_PARM_DESC(int_mode, "Force the temperature interrupt mode");
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
static struct platform_device *pdev; static struct platform_device *pdev;
#define DRVNAME "vt1211" #define DRVNAME "vt1211"
...@@ -1280,10 +1284,12 @@ static int __init vt1211_device_add(unsigned short address) ...@@ -1280,10 +1284,12 @@ static int __init vt1211_device_add(unsigned short address)
static int __init vt1211_find(int sio_cip, unsigned short *address) static int __init vt1211_find(int sio_cip, unsigned short *address)
{ {
int err = -ENODEV; int err = -ENODEV;
int devid;
superio_enter(sio_cip); superio_enter(sio_cip);
if (superio_inb(sio_cip, SIO_VT1211_DEVID) != SIO_VT1211_ID) { devid = force_id ? force_id : superio_inb(sio_cip, SIO_VT1211_DEVID);
if (devid != SIO_VT1211_ID) {
goto EXIT; goto EXIT;
} }
......
...@@ -59,6 +59,10 @@ static const char * w83627ehf_device_names[] = { ...@@ -59,6 +59,10 @@ static const char * w83627ehf_device_names[] = {
"w83627dhg", "w83627dhg",
}; };
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
#define DRVNAME "w83627ehf" #define DRVNAME "w83627ehf"
/* /*
...@@ -1445,6 +1449,9 @@ static int __init w83627ehf_find(int sioaddr, unsigned short *addr, ...@@ -1445,6 +1449,9 @@ static int __init w83627ehf_find(int sioaddr, unsigned short *addr,
superio_enter(sioaddr); superio_enter(sioaddr);
if (force_id)
val = force_id;
else
val = (superio_inb(sioaddr, SIO_REG_DEVID) << 8) val = (superio_inb(sioaddr, SIO_REG_DEVID) << 8)
| superio_inb(sioaddr, SIO_REG_DEVID + 1); | superio_inb(sioaddr, SIO_REG_DEVID + 1);
switch (val & SIO_ID_MASK) { switch (val & SIO_ID_MASK) {
......
...@@ -75,6 +75,10 @@ static int init = 1; ...@@ -75,6 +75,10 @@ static int init = 1;
module_param(init, bool, 0); module_param(init, bool, 0);
MODULE_PARM_DESC(init, "Set to zero to bypass chip initialization"); MODULE_PARM_DESC(init, "Set to zero to bypass chip initialization");
static unsigned short force_id;
module_param(force_id, ushort, 0);
MODULE_PARM_DESC(force_id, "Override the detected device ID");
/* modified from kernel/include/traps.c */ /* modified from kernel/include/traps.c */
static int REG; /* The register to read/write */ static int REG; /* The register to read/write */
#define DEV 0x07 /* Register: Logical device select */ #define DEV 0x07 /* Register: Logical device select */
...@@ -1014,7 +1018,7 @@ static int __init w83627hf_find(int sioaddr, unsigned short *addr, ...@@ -1014,7 +1018,7 @@ static int __init w83627hf_find(int sioaddr, unsigned short *addr,
VAL = sioaddr + 1; VAL = sioaddr + 1;
superio_enter(); superio_enter();
val= superio_inb(DEVID); val = force_id ? force_id : superio_inb(DEVID);
switch (val) { switch (val) {
case W627_DEVID: case W627_DEVID:
sio_data->type = w83627hf; sio_data->type = w83627hf;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment