cachepc-linux

Fork of AMDESE/linux with modifications for CachePC side-channel attack
git clone https://git.sinitax.com/sinitax/cachepc-linux
Log | Files | Refs | README | LICENSE | sfeed.txt

nct6775-platform.c (31469B)


      1// SPDX-License-Identifier: GPL-2.0-or-later
      2/*
      3 * nct6775 - Platform driver for the hardware monitoring
      4 *	     functionality of Nuvoton NCT677x Super-I/O chips
      5 *
      6 * Copyright (C) 2012  Guenter Roeck <linux@roeck-us.net>
      7 */
      8
      9#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
     10
     11#include <linux/acpi.h>
     12#include <linux/dmi.h>
     13#include <linux/hwmon-sysfs.h>
     14#include <linux/hwmon-vid.h>
     15#include <linux/init.h>
     16#include <linux/io.h>
     17#include <linux/module.h>
     18#include <linux/platform_device.h>
     19#include <linux/regmap.h>
     20#include <linux/wmi.h>
     21
     22#include "nct6775.h"
     23
     24enum sensor_access { access_direct, access_asuswmi };
     25
     26static const char * const nct6775_sio_names[] __initconst = {
     27	"NCT6106D",
     28	"NCT6116D",
     29	"NCT6775F",
     30	"NCT6776D/F",
     31	"NCT6779D",
     32	"NCT6791D",
     33	"NCT6792D",
     34	"NCT6793D",
     35	"NCT6795D",
     36	"NCT6796D",
     37	"NCT6797D",
     38	"NCT6798D",
     39};
     40
     41static unsigned short force_id;
     42module_param(force_id, ushort, 0);
     43MODULE_PARM_DESC(force_id, "Override the detected device ID");
     44
     45static unsigned short fan_debounce;
     46module_param(fan_debounce, ushort, 0);
     47MODULE_PARM_DESC(fan_debounce, "Enable debouncing for fan RPM signal");
     48
     49#define DRVNAME "nct6775"
     50
     51#define NCT6775_PORT_CHIPID	0x58
     52
     53/*
     54 * ISA constants
     55 */
     56
     57#define IOREGION_ALIGNMENT	(~7)
     58#define IOREGION_OFFSET		5
     59#define IOREGION_LENGTH		2
     60#define ADDR_REG_OFFSET		0
     61#define DATA_REG_OFFSET		1
     62
     63/*
     64 * Super-I/O constants and functions
     65 */
     66
     67#define NCT6775_LD_ACPI		0x0a
     68#define NCT6775_LD_HWM		0x0b
     69#define NCT6775_LD_VID		0x0d
     70#define NCT6775_LD_12		0x12
     71
     72#define SIO_REG_LDSEL		0x07	/* Logical device select */
     73#define SIO_REG_DEVID		0x20	/* Device ID (2 bytes) */
     74#define SIO_REG_ENABLE		0x30	/* Logical device enable */
     75#define SIO_REG_ADDR		0x60	/* Logical device address (2 bytes) */
     76
     77#define SIO_NCT6106_ID		0xc450
     78#define SIO_NCT6116_ID		0xd280
     79#define SIO_NCT6775_ID		0xb470
     80#define SIO_NCT6776_ID		0xc330
     81#define SIO_NCT6779_ID		0xc560
     82#define SIO_NCT6791_ID		0xc800
     83#define SIO_NCT6792_ID		0xc910
     84#define SIO_NCT6793_ID		0xd120
     85#define SIO_NCT6795_ID		0xd350
     86#define SIO_NCT6796_ID		0xd420
     87#define SIO_NCT6797_ID		0xd450
     88#define SIO_NCT6798_ID		0xd428
     89#define SIO_ID_MASK		0xFFF8
     90
     91/*
     92 * Control registers
     93 */
     94#define NCT6775_REG_CR_FAN_DEBOUNCE	0xf0
     95
     96struct nct6775_sio_data {
     97	int sioreg;
     98	int ld;
     99	enum kinds kind;
    100	enum sensor_access access;
    101
    102	/* superio_() callbacks  */
    103	void (*sio_outb)(struct nct6775_sio_data *sio_data, int reg, int val);
    104	int (*sio_inb)(struct nct6775_sio_data *sio_data, int reg);
    105	void (*sio_select)(struct nct6775_sio_data *sio_data, int ld);
    106	int (*sio_enter)(struct nct6775_sio_data *sio_data);
    107	void (*sio_exit)(struct nct6775_sio_data *sio_data);
    108};
    109
    110#define ASUSWMI_MONITORING_GUID		"466747A0-70EC-11DE-8A39-0800200C9A66"
    111#define ASUSWMI_METHODID_RSIO		0x5253494F
    112#define ASUSWMI_METHODID_WSIO		0x5753494F
    113#define ASUSWMI_METHODID_RHWM		0x5248574D
    114#define ASUSWMI_METHODID_WHWM		0x5748574D
    115#define ASUSWMI_UNSUPPORTED_METHOD	0xFFFFFFFE
    116
    117static int nct6775_asuswmi_evaluate_method(u32 method_id, u8 bank, u8 reg, u8 val, u32 *retval)
    118{
    119#if IS_ENABLED(CONFIG_ACPI_WMI)
    120	u32 args = bank | (reg << 8) | (val << 16);
    121	struct acpi_buffer input = { (acpi_size) sizeof(args), &args };
    122	struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL };
    123	acpi_status status;
    124	union acpi_object *obj;
    125	u32 tmp = ASUSWMI_UNSUPPORTED_METHOD;
    126
    127	status = wmi_evaluate_method(ASUSWMI_MONITORING_GUID, 0,
    128				     method_id, &input, &output);
    129
    130	if (ACPI_FAILURE(status))
    131		return -EIO;
    132
    133	obj = output.pointer;
    134	if (obj && obj->type == ACPI_TYPE_INTEGER)
    135		tmp = obj->integer.value;
    136
    137	if (retval)
    138		*retval = tmp;
    139
    140	kfree(obj);
    141
    142	if (tmp == ASUSWMI_UNSUPPORTED_METHOD)
    143		return -ENODEV;
    144	return 0;
    145#else
    146	return -EOPNOTSUPP;
    147#endif
    148}
    149
    150static inline int nct6775_asuswmi_write(u8 bank, u8 reg, u8 val)
    151{
    152	return nct6775_asuswmi_evaluate_method(ASUSWMI_METHODID_WHWM, bank,
    153					      reg, val, NULL);
    154}
    155
    156static inline int nct6775_asuswmi_read(u8 bank, u8 reg, u8 *val)
    157{
    158	u32 ret, tmp = 0;
    159
    160	ret = nct6775_asuswmi_evaluate_method(ASUSWMI_METHODID_RHWM, bank,
    161					      reg, 0, &tmp);
    162	*val = tmp;
    163	return ret;
    164}
    165
    166static int superio_wmi_inb(struct nct6775_sio_data *sio_data, int reg)
    167{
    168	int tmp = 0;
    169
    170	nct6775_asuswmi_evaluate_method(ASUSWMI_METHODID_RSIO, sio_data->ld,
    171					reg, 0, &tmp);
    172	return tmp;
    173}
    174
    175static void superio_wmi_outb(struct nct6775_sio_data *sio_data, int reg, int val)
    176{
    177	nct6775_asuswmi_evaluate_method(ASUSWMI_METHODID_WSIO, sio_data->ld,
    178					reg, val, NULL);
    179}
    180
    181static void superio_wmi_select(struct nct6775_sio_data *sio_data, int ld)
    182{
    183	sio_data->ld = ld;
    184}
    185
    186static int superio_wmi_enter(struct nct6775_sio_data *sio_data)
    187{
    188	return 0;
    189}
    190
    191static void superio_wmi_exit(struct nct6775_sio_data *sio_data)
    192{
    193}
    194
    195static void superio_outb(struct nct6775_sio_data *sio_data, int reg, int val)
    196{
    197	int ioreg = sio_data->sioreg;
    198
    199	outb(reg, ioreg);
    200	outb(val, ioreg + 1);
    201}
    202
    203static int superio_inb(struct nct6775_sio_data *sio_data, int reg)
    204{
    205	int ioreg = sio_data->sioreg;
    206
    207	outb(reg, ioreg);
    208	return inb(ioreg + 1);
    209}
    210
    211static void superio_select(struct nct6775_sio_data *sio_data, int ld)
    212{
    213	int ioreg = sio_data->sioreg;
    214
    215	outb(SIO_REG_LDSEL, ioreg);
    216	outb(ld, ioreg + 1);
    217}
    218
    219static int superio_enter(struct nct6775_sio_data *sio_data)
    220{
    221	int ioreg = sio_data->sioreg;
    222
    223	/*
    224	 * Try to reserve <ioreg> and <ioreg + 1> for exclusive access.
    225	 */
    226	if (!request_muxed_region(ioreg, 2, DRVNAME))
    227		return -EBUSY;
    228
    229	outb(0x87, ioreg);
    230	outb(0x87, ioreg);
    231
    232	return 0;
    233}
    234
    235static void superio_exit(struct nct6775_sio_data *sio_data)
    236{
    237	int ioreg = sio_data->sioreg;
    238
    239	outb(0xaa, ioreg);
    240	outb(0x02, ioreg);
    241	outb(0x02, ioreg + 1);
    242	release_region(ioreg, 2);
    243}
    244
    245static inline void nct6775_wmi_set_bank(struct nct6775_data *data, u16 reg)
    246{
    247	u8 bank = reg >> 8;
    248
    249	data->bank = bank;
    250}
    251
    252static int nct6775_wmi_reg_read(void *ctx, unsigned int reg, unsigned int *val)
    253{
    254	struct nct6775_data *data = ctx;
    255	int err, word_sized = nct6775_reg_is_word_sized(data, reg);
    256	u8 tmp = 0;
    257	u16 res;
    258
    259	nct6775_wmi_set_bank(data, reg);
    260
    261	err = nct6775_asuswmi_read(data->bank, reg & 0xff, &tmp);
    262	if (err)
    263		return err;
    264
    265	res = tmp;
    266	if (word_sized) {
    267		err = nct6775_asuswmi_read(data->bank, (reg & 0xff) + 1, &tmp);
    268		if (err)
    269			return err;
    270
    271		res = (res << 8) + tmp;
    272	}
    273	*val = res;
    274	return 0;
    275}
    276
    277static int nct6775_wmi_reg_write(void *ctx, unsigned int reg, unsigned int value)
    278{
    279	struct nct6775_data *data = ctx;
    280	int res, word_sized = nct6775_reg_is_word_sized(data, reg);
    281
    282	nct6775_wmi_set_bank(data, reg);
    283
    284	if (word_sized) {
    285		res = nct6775_asuswmi_write(data->bank, reg & 0xff, value >> 8);
    286		if (res)
    287			return res;
    288
    289		res = nct6775_asuswmi_write(data->bank, (reg & 0xff) + 1, value);
    290	} else {
    291		res = nct6775_asuswmi_write(data->bank, reg & 0xff, value);
    292	}
    293
    294	return res;
    295}
    296
    297/*
    298 * On older chips, only registers 0x50-0x5f are banked.
    299 * On more recent chips, all registers are banked.
    300 * Assume that is the case and set the bank number for each access.
    301 * Cache the bank number so it only needs to be set if it changes.
    302 */
    303static inline void nct6775_set_bank(struct nct6775_data *data, u16 reg)
    304{
    305	u8 bank = reg >> 8;
    306
    307	if (data->bank != bank) {
    308		outb_p(NCT6775_REG_BANK, data->addr + ADDR_REG_OFFSET);
    309		outb_p(bank, data->addr + DATA_REG_OFFSET);
    310		data->bank = bank;
    311	}
    312}
    313
    314static int nct6775_reg_read(void *ctx, unsigned int reg, unsigned int *val)
    315{
    316	struct nct6775_data *data = ctx;
    317	int word_sized = nct6775_reg_is_word_sized(data, reg);
    318
    319	nct6775_set_bank(data, reg);
    320	outb_p(reg & 0xff, data->addr + ADDR_REG_OFFSET);
    321	*val = inb_p(data->addr + DATA_REG_OFFSET);
    322	if (word_sized) {
    323		outb_p((reg & 0xff) + 1,
    324		       data->addr + ADDR_REG_OFFSET);
    325		*val = (*val << 8) + inb_p(data->addr + DATA_REG_OFFSET);
    326	}
    327	return 0;
    328}
    329
    330static int nct6775_reg_write(void *ctx, unsigned int reg, unsigned int value)
    331{
    332	struct nct6775_data *data = ctx;
    333	int word_sized = nct6775_reg_is_word_sized(data, reg);
    334
    335	nct6775_set_bank(data, reg);
    336	outb_p(reg & 0xff, data->addr + ADDR_REG_OFFSET);
    337	if (word_sized) {
    338		outb_p(value >> 8, data->addr + DATA_REG_OFFSET);
    339		outb_p((reg & 0xff) + 1,
    340		       data->addr + ADDR_REG_OFFSET);
    341	}
    342	outb_p(value & 0xff, data->addr + DATA_REG_OFFSET);
    343	return 0;
    344}
    345
    346static void nct6791_enable_io_mapping(struct nct6775_sio_data *sio_data)
    347{
    348	int val;
    349
    350	val = sio_data->sio_inb(sio_data, NCT6791_REG_HM_IO_SPACE_LOCK_ENABLE);
    351	if (val & 0x10) {
    352		pr_info("Enabling hardware monitor logical device mappings.\n");
    353		sio_data->sio_outb(sio_data, NCT6791_REG_HM_IO_SPACE_LOCK_ENABLE,
    354			       val & ~0x10);
    355	}
    356}
    357
    358static int __maybe_unused nct6775_suspend(struct device *dev)
    359{
    360	int err;
    361	u16 tmp;
    362	struct nct6775_data *data = dev_get_drvdata(dev);
    363
    364	if (IS_ERR(data))
    365		return PTR_ERR(data);
    366
    367	mutex_lock(&data->update_lock);
    368	err = nct6775_read_value(data, data->REG_VBAT, &tmp);
    369	if (err)
    370		goto out;
    371	data->vbat = tmp;
    372	if (data->kind == nct6775) {
    373		err = nct6775_read_value(data, NCT6775_REG_FANDIV1, &tmp);
    374		if (err)
    375			goto out;
    376		data->fandiv1 = tmp;
    377
    378		err = nct6775_read_value(data, NCT6775_REG_FANDIV2, &tmp);
    379		if (err)
    380			goto out;
    381		data->fandiv2 = tmp;
    382	}
    383out:
    384	mutex_unlock(&data->update_lock);
    385
    386	return err;
    387}
    388
    389static int __maybe_unused nct6775_resume(struct device *dev)
    390{
    391	struct nct6775_data *data = dev_get_drvdata(dev);
    392	struct nct6775_sio_data *sio_data = dev_get_platdata(dev);
    393	int i, j, err = 0;
    394	u8 reg;
    395
    396	mutex_lock(&data->update_lock);
    397	data->bank = 0xff;		/* Force initial bank selection */
    398
    399	err = sio_data->sio_enter(sio_data);
    400	if (err)
    401		goto abort;
    402
    403	sio_data->sio_select(sio_data, NCT6775_LD_HWM);
    404	reg = sio_data->sio_inb(sio_data, SIO_REG_ENABLE);
    405	if (reg != data->sio_reg_enable)
    406		sio_data->sio_outb(sio_data, SIO_REG_ENABLE, data->sio_reg_enable);
    407
    408	if (data->kind == nct6791 || data->kind == nct6792 ||
    409	    data->kind == nct6793 || data->kind == nct6795 ||
    410	    data->kind == nct6796 || data->kind == nct6797 ||
    411	    data->kind == nct6798)
    412		nct6791_enable_io_mapping(sio_data);
    413
    414	sio_data->sio_exit(sio_data);
    415
    416	/* Restore limits */
    417	for (i = 0; i < data->in_num; i++) {
    418		if (!(data->have_in & BIT(i)))
    419			continue;
    420
    421		err = nct6775_write_value(data, data->REG_IN_MINMAX[0][i], data->in[i][1]);
    422		if (err)
    423			goto abort;
    424		err = nct6775_write_value(data, data->REG_IN_MINMAX[1][i], data->in[i][2]);
    425		if (err)
    426			goto abort;
    427	}
    428
    429	for (i = 0; i < ARRAY_SIZE(data->fan_min); i++) {
    430		if (!(data->has_fan_min & BIT(i)))
    431			continue;
    432
    433		err = nct6775_write_value(data, data->REG_FAN_MIN[i], data->fan_min[i]);
    434		if (err)
    435			goto abort;
    436	}
    437
    438	for (i = 0; i < NUM_TEMP; i++) {
    439		if (!(data->have_temp & BIT(i)))
    440			continue;
    441
    442		for (j = 1; j < ARRAY_SIZE(data->reg_temp); j++)
    443			if (data->reg_temp[j][i]) {
    444				err = nct6775_write_temp(data, data->reg_temp[j][i],
    445							 data->temp[j][i]);
    446				if (err)
    447					goto abort;
    448			}
    449	}
    450
    451	/* Restore other settings */
    452	err = nct6775_write_value(data, data->REG_VBAT, data->vbat);
    453	if (err)
    454		goto abort;
    455	if (data->kind == nct6775) {
    456		err = nct6775_write_value(data, NCT6775_REG_FANDIV1, data->fandiv1);
    457		if (err)
    458			goto abort;
    459		err = nct6775_write_value(data, NCT6775_REG_FANDIV2, data->fandiv2);
    460	}
    461
    462abort:
    463	/* Force re-reading all values */
    464	data->valid = false;
    465	mutex_unlock(&data->update_lock);
    466
    467	return err;
    468}
    469
    470static SIMPLE_DEV_PM_OPS(nct6775_dev_pm_ops, nct6775_suspend, nct6775_resume);
    471
    472static void
    473nct6775_check_fan_inputs(struct nct6775_data *data, struct nct6775_sio_data *sio_data)
    474{
    475	bool fan3pin = false, fan4pin = false, fan4min = false;
    476	bool fan5pin = false, fan6pin = false, fan7pin = false;
    477	bool pwm3pin = false, pwm4pin = false, pwm5pin = false;
    478	bool pwm6pin = false, pwm7pin = false;
    479
    480	/* Store SIO_REG_ENABLE for use during resume */
    481	sio_data->sio_select(sio_data, NCT6775_LD_HWM);
    482	data->sio_reg_enable = sio_data->sio_inb(sio_data, SIO_REG_ENABLE);
    483
    484	/* fan4 and fan5 share some pins with the GPIO and serial flash */
    485	if (data->kind == nct6775) {
    486		int cr2c = sio_data->sio_inb(sio_data, 0x2c);
    487
    488		fan3pin = cr2c & BIT(6);
    489		pwm3pin = cr2c & BIT(7);
    490
    491		/* On NCT6775, fan4 shares pins with the fdc interface */
    492		fan4pin = !(sio_data->sio_inb(sio_data, 0x2A) & 0x80);
    493	} else if (data->kind == nct6776) {
    494		bool gpok = sio_data->sio_inb(sio_data, 0x27) & 0x80;
    495		const char *board_vendor, *board_name;
    496
    497		board_vendor = dmi_get_system_info(DMI_BOARD_VENDOR);
    498		board_name = dmi_get_system_info(DMI_BOARD_NAME);
    499
    500		if (board_name && board_vendor &&
    501		    !strcmp(board_vendor, "ASRock")) {
    502			/*
    503			 * Auxiliary fan monitoring is not enabled on ASRock
    504			 * Z77 Pro4-M if booted in UEFI Ultra-FastBoot mode.
    505			 * Observed with BIOS version 2.00.
    506			 */
    507			if (!strcmp(board_name, "Z77 Pro4-M")) {
    508				if ((data->sio_reg_enable & 0xe0) != 0xe0) {
    509					data->sio_reg_enable |= 0xe0;
    510					sio_data->sio_outb(sio_data, SIO_REG_ENABLE,
    511						     data->sio_reg_enable);
    512				}
    513			}
    514		}
    515
    516		if (data->sio_reg_enable & 0x80)
    517			fan3pin = gpok;
    518		else
    519			fan3pin = !(sio_data->sio_inb(sio_data, 0x24) & 0x40);
    520
    521		if (data->sio_reg_enable & 0x40)
    522			fan4pin = gpok;
    523		else
    524			fan4pin = sio_data->sio_inb(sio_data, 0x1C) & 0x01;
    525
    526		if (data->sio_reg_enable & 0x20)
    527			fan5pin = gpok;
    528		else
    529			fan5pin = sio_data->sio_inb(sio_data, 0x1C) & 0x02;
    530
    531		fan4min = fan4pin;
    532		pwm3pin = fan3pin;
    533	} else if (data->kind == nct6106) {
    534		int cr24 = sio_data->sio_inb(sio_data, 0x24);
    535
    536		fan3pin = !(cr24 & 0x80);
    537		pwm3pin = cr24 & 0x08;
    538	} else if (data->kind == nct6116) {
    539		int cr1a = sio_data->sio_inb(sio_data, 0x1a);
    540		int cr1b = sio_data->sio_inb(sio_data, 0x1b);
    541		int cr24 = sio_data->sio_inb(sio_data, 0x24);
    542		int cr2a = sio_data->sio_inb(sio_data, 0x2a);
    543		int cr2b = sio_data->sio_inb(sio_data, 0x2b);
    544		int cr2f = sio_data->sio_inb(sio_data, 0x2f);
    545
    546		fan3pin = !(cr2b & 0x10);
    547		fan4pin = (cr2b & 0x80) ||			// pin 1(2)
    548			(!(cr2f & 0x10) && (cr1a & 0x04));	// pin 65(66)
    549		fan5pin = (cr2b & 0x80) ||			// pin 126(127)
    550			(!(cr1b & 0x03) && (cr2a & 0x02));	// pin 94(96)
    551
    552		pwm3pin = fan3pin && (cr24 & 0x08);
    553		pwm4pin = fan4pin;
    554		pwm5pin = fan5pin;
    555	} else {
    556		/*
    557		 * NCT6779D, NCT6791D, NCT6792D, NCT6793D, NCT6795D, NCT6796D,
    558		 * NCT6797D, NCT6798D
    559		 */
    560		int cr1a = sio_data->sio_inb(sio_data, 0x1a);
    561		int cr1b = sio_data->sio_inb(sio_data, 0x1b);
    562		int cr1c = sio_data->sio_inb(sio_data, 0x1c);
    563		int cr1d = sio_data->sio_inb(sio_data, 0x1d);
    564		int cr2a = sio_data->sio_inb(sio_data, 0x2a);
    565		int cr2b = sio_data->sio_inb(sio_data, 0x2b);
    566		int cr2d = sio_data->sio_inb(sio_data, 0x2d);
    567		int cr2f = sio_data->sio_inb(sio_data, 0x2f);
    568		bool dsw_en = cr2f & BIT(3);
    569		bool ddr4_en = cr2f & BIT(4);
    570		int cre0;
    571		int creb;
    572		int cred;
    573
    574		sio_data->sio_select(sio_data, NCT6775_LD_12);
    575		cre0 = sio_data->sio_inb(sio_data, 0xe0);
    576		creb = sio_data->sio_inb(sio_data, 0xeb);
    577		cred = sio_data->sio_inb(sio_data, 0xed);
    578
    579		fan3pin = !(cr1c & BIT(5));
    580		fan4pin = !(cr1c & BIT(6));
    581		fan5pin = !(cr1c & BIT(7));
    582
    583		pwm3pin = !(cr1c & BIT(0));
    584		pwm4pin = !(cr1c & BIT(1));
    585		pwm5pin = !(cr1c & BIT(2));
    586
    587		switch (data->kind) {
    588		case nct6791:
    589			fan6pin = cr2d & BIT(1);
    590			pwm6pin = cr2d & BIT(0);
    591			break;
    592		case nct6792:
    593			fan6pin = !dsw_en && (cr2d & BIT(1));
    594			pwm6pin = !dsw_en && (cr2d & BIT(0));
    595			break;
    596		case nct6793:
    597			fan5pin |= cr1b & BIT(5);
    598			fan5pin |= creb & BIT(5);
    599
    600			fan6pin = !dsw_en && (cr2d & BIT(1));
    601			fan6pin |= creb & BIT(3);
    602
    603			pwm5pin |= cr2d & BIT(7);
    604			pwm5pin |= (creb & BIT(4)) && !(cr2a & BIT(0));
    605
    606			pwm6pin = !dsw_en && (cr2d & BIT(0));
    607			pwm6pin |= creb & BIT(2);
    608			break;
    609		case nct6795:
    610			fan5pin |= cr1b & BIT(5);
    611			fan5pin |= creb & BIT(5);
    612
    613			fan6pin = (cr2a & BIT(4)) &&
    614					(!dsw_en || (cred & BIT(4)));
    615			fan6pin |= creb & BIT(3);
    616
    617			pwm5pin |= cr2d & BIT(7);
    618			pwm5pin |= (creb & BIT(4)) && !(cr2a & BIT(0));
    619
    620			pwm6pin = (cr2a & BIT(3)) && (cred & BIT(2));
    621			pwm6pin |= creb & BIT(2);
    622			break;
    623		case nct6796:
    624			fan5pin |= cr1b & BIT(5);
    625			fan5pin |= (cre0 & BIT(3)) && !(cr1b & BIT(0));
    626			fan5pin |= creb & BIT(5);
    627
    628			fan6pin = (cr2a & BIT(4)) &&
    629					(!dsw_en || (cred & BIT(4)));
    630			fan6pin |= creb & BIT(3);
    631
    632			fan7pin = !(cr2b & BIT(2));
    633
    634			pwm5pin |= cr2d & BIT(7);
    635			pwm5pin |= (cre0 & BIT(4)) && !(cr1b & BIT(0));
    636			pwm5pin |= (creb & BIT(4)) && !(cr2a & BIT(0));
    637
    638			pwm6pin = (cr2a & BIT(3)) && (cred & BIT(2));
    639			pwm6pin |= creb & BIT(2);
    640
    641			pwm7pin = !(cr1d & (BIT(2) | BIT(3)));
    642			break;
    643		case nct6797:
    644			fan5pin |= !ddr4_en && (cr1b & BIT(5));
    645			fan5pin |= creb & BIT(5);
    646
    647			fan6pin = cr2a & BIT(4);
    648			fan6pin |= creb & BIT(3);
    649
    650			fan7pin = cr1a & BIT(1);
    651
    652			pwm5pin |= (creb & BIT(4)) && !(cr2a & BIT(0));
    653			pwm5pin |= !ddr4_en && (cr2d & BIT(7));
    654
    655			pwm6pin = creb & BIT(2);
    656			pwm6pin |= cred & BIT(2);
    657
    658			pwm7pin = cr1d & BIT(4);
    659			break;
    660		case nct6798:
    661			fan6pin = !(cr1b & BIT(0)) && (cre0 & BIT(3));
    662			fan6pin |= cr2a & BIT(4);
    663			fan6pin |= creb & BIT(5);
    664
    665			fan7pin = cr1b & BIT(5);
    666			fan7pin |= !(cr2b & BIT(2));
    667			fan7pin |= creb & BIT(3);
    668
    669			pwm6pin = !(cr1b & BIT(0)) && (cre0 & BIT(4));
    670			pwm6pin |= !(cred & BIT(2)) && (cr2a & BIT(3));
    671			pwm6pin |= (creb & BIT(4)) && !(cr2a & BIT(0));
    672
    673			pwm7pin = !(cr1d & (BIT(2) | BIT(3)));
    674			pwm7pin |= cr2d & BIT(7);
    675			pwm7pin |= creb & BIT(2);
    676			break;
    677		default:	/* NCT6779D */
    678			break;
    679		}
    680
    681		fan4min = fan4pin;
    682	}
    683
    684	/* fan 1 and 2 (0x03) are always present */
    685	data->has_fan = 0x03 | (fan3pin << 2) | (fan4pin << 3) |
    686		(fan5pin << 4) | (fan6pin << 5) | (fan7pin << 6);
    687	data->has_fan_min = 0x03 | (fan3pin << 2) | (fan4min << 3) |
    688		(fan5pin << 4) | (fan6pin << 5) | (fan7pin << 6);
    689	data->has_pwm = 0x03 | (pwm3pin << 2) | (pwm4pin << 3) |
    690		(pwm5pin << 4) | (pwm6pin << 5) | (pwm7pin << 6);
    691}
    692
    693static ssize_t
    694cpu0_vid_show(struct device *dev, struct device_attribute *attr, char *buf)
    695{
    696	struct nct6775_data *data = dev_get_drvdata(dev);
    697
    698	return sprintf(buf, "%d\n", vid_from_reg(data->vid, data->vrm));
    699}
    700
    701static DEVICE_ATTR_RO(cpu0_vid);
    702
    703/* Case open detection */
    704
    705static const u8 NCT6775_REG_CR_CASEOPEN_CLR[] = { 0xe6, 0xee };
    706static const u8 NCT6775_CR_CASEOPEN_CLR_MASK[] = { 0x20, 0x01 };
    707
    708static ssize_t
    709clear_caseopen(struct device *dev, struct device_attribute *attr,
    710	       const char *buf, size_t count)
    711{
    712	struct nct6775_data *data = dev_get_drvdata(dev);
    713	struct nct6775_sio_data *sio_data = data->driver_data;
    714	int nr = to_sensor_dev_attr(attr)->index - INTRUSION_ALARM_BASE;
    715	unsigned long val;
    716	u8 reg;
    717	int ret;
    718
    719	if (kstrtoul(buf, 10, &val) || val != 0)
    720		return -EINVAL;
    721
    722	mutex_lock(&data->update_lock);
    723
    724	/*
    725	 * Use CR registers to clear caseopen status.
    726	 * The CR registers are the same for all chips, and not all chips
    727	 * support clearing the caseopen status through "regular" registers.
    728	 */
    729	ret = sio_data->sio_enter(sio_data);
    730	if (ret) {
    731		count = ret;
    732		goto error;
    733	}
    734
    735	sio_data->sio_select(sio_data, NCT6775_LD_ACPI);
    736	reg = sio_data->sio_inb(sio_data, NCT6775_REG_CR_CASEOPEN_CLR[nr]);
    737	reg |= NCT6775_CR_CASEOPEN_CLR_MASK[nr];
    738	sio_data->sio_outb(sio_data, NCT6775_REG_CR_CASEOPEN_CLR[nr], reg);
    739	reg &= ~NCT6775_CR_CASEOPEN_CLR_MASK[nr];
    740	sio_data->sio_outb(sio_data, NCT6775_REG_CR_CASEOPEN_CLR[nr], reg);
    741	sio_data->sio_exit(sio_data);
    742
    743	data->valid = false;	/* Force cache refresh */
    744error:
    745	mutex_unlock(&data->update_lock);
    746	return count;
    747}
    748
    749static SENSOR_DEVICE_ATTR(intrusion0_alarm, 0644, nct6775_show_alarm,
    750			  clear_caseopen, INTRUSION_ALARM_BASE);
    751static SENSOR_DEVICE_ATTR(intrusion1_alarm, 0644, nct6775_show_alarm,
    752			  clear_caseopen, INTRUSION_ALARM_BASE + 1);
    753static SENSOR_DEVICE_ATTR(intrusion0_beep, 0644, nct6775_show_beep,
    754			  nct6775_store_beep, INTRUSION_ALARM_BASE);
    755static SENSOR_DEVICE_ATTR(intrusion1_beep, 0644, nct6775_show_beep,
    756			  nct6775_store_beep, INTRUSION_ALARM_BASE + 1);
    757static SENSOR_DEVICE_ATTR(beep_enable, 0644, nct6775_show_beep,
    758			  nct6775_store_beep, BEEP_ENABLE_BASE);
    759
    760static umode_t nct6775_other_is_visible(struct kobject *kobj,
    761					struct attribute *attr, int index)
    762{
    763	struct device *dev = kobj_to_dev(kobj);
    764	struct nct6775_data *data = dev_get_drvdata(dev);
    765
    766	if (index == 0 && !data->have_vid)
    767		return 0;
    768
    769	if (index == 1 || index == 2) {
    770		if (data->ALARM_BITS[INTRUSION_ALARM_BASE + index - 1] < 0)
    771			return 0;
    772	}
    773
    774	if (index == 3 || index == 4) {
    775		if (data->BEEP_BITS[INTRUSION_ALARM_BASE + index - 3] < 0)
    776			return 0;
    777	}
    778
    779	return nct6775_attr_mode(data, attr);
    780}
    781
    782/*
    783 * nct6775_other_is_visible uses the index into the following array
    784 * to determine if attributes should be created or not.
    785 * Any change in order or content must be matched.
    786 */
    787static struct attribute *nct6775_attributes_other[] = {
    788	&dev_attr_cpu0_vid.attr,				/* 0 */
    789	&sensor_dev_attr_intrusion0_alarm.dev_attr.attr,	/* 1 */
    790	&sensor_dev_attr_intrusion1_alarm.dev_attr.attr,	/* 2 */
    791	&sensor_dev_attr_intrusion0_beep.dev_attr.attr,		/* 3 */
    792	&sensor_dev_attr_intrusion1_beep.dev_attr.attr,		/* 4 */
    793	&sensor_dev_attr_beep_enable.dev_attr.attr,		/* 5 */
    794
    795	NULL
    796};
    797
    798static const struct attribute_group nct6775_group_other = {
    799	.attrs = nct6775_attributes_other,
    800	.is_visible = nct6775_other_is_visible,
    801};
    802
    803static int nct6775_platform_probe_init(struct nct6775_data *data)
    804{
    805	int err;
    806	u8 cr2a;
    807	struct nct6775_sio_data *sio_data = data->driver_data;
    808
    809	err = sio_data->sio_enter(sio_data);
    810	if (err)
    811		return err;
    812
    813	cr2a = sio_data->sio_inb(sio_data, 0x2a);
    814	switch (data->kind) {
    815	case nct6775:
    816		data->have_vid = (cr2a & 0x40);
    817		break;
    818	case nct6776:
    819		data->have_vid = (cr2a & 0x60) == 0x40;
    820		break;
    821	case nct6106:
    822	case nct6116:
    823	case nct6779:
    824	case nct6791:
    825	case nct6792:
    826	case nct6793:
    827	case nct6795:
    828	case nct6796:
    829	case nct6797:
    830	case nct6798:
    831		break;
    832	}
    833
    834	/*
    835	 * Read VID value
    836	 * We can get the VID input values directly at logical device D 0xe3.
    837	 */
    838	if (data->have_vid) {
    839		sio_data->sio_select(sio_data, NCT6775_LD_VID);
    840		data->vid = sio_data->sio_inb(sio_data, 0xe3);
    841		data->vrm = vid_which_vrm();
    842	}
    843
    844	if (fan_debounce) {
    845		u8 tmp;
    846
    847		sio_data->sio_select(sio_data, NCT6775_LD_HWM);
    848		tmp = sio_data->sio_inb(sio_data,
    849				    NCT6775_REG_CR_FAN_DEBOUNCE);
    850		switch (data->kind) {
    851		case nct6106:
    852		case nct6116:
    853			tmp |= 0xe0;
    854			break;
    855		case nct6775:
    856			tmp |= 0x1e;
    857			break;
    858		case nct6776:
    859		case nct6779:
    860			tmp |= 0x3e;
    861			break;
    862		case nct6791:
    863		case nct6792:
    864		case nct6793:
    865		case nct6795:
    866		case nct6796:
    867		case nct6797:
    868		case nct6798:
    869			tmp |= 0x7e;
    870			break;
    871		}
    872		sio_data->sio_outb(sio_data, NCT6775_REG_CR_FAN_DEBOUNCE,
    873			     tmp);
    874		pr_info("Enabled fan debounce for chip %s\n", data->name);
    875	}
    876
    877	nct6775_check_fan_inputs(data, sio_data);
    878
    879	sio_data->sio_exit(sio_data);
    880
    881	return nct6775_add_attr_group(data, &nct6775_group_other);
    882}
    883
    884static const struct regmap_config nct6775_regmap_config = {
    885	.reg_bits = 16,
    886	.val_bits = 16,
    887	.reg_read = nct6775_reg_read,
    888	.reg_write = nct6775_reg_write,
    889};
    890
    891static const struct regmap_config nct6775_wmi_regmap_config = {
    892	.reg_bits = 16,
    893	.val_bits = 16,
    894	.reg_read = nct6775_wmi_reg_read,
    895	.reg_write = nct6775_wmi_reg_write,
    896};
    897
    898static int nct6775_platform_probe(struct platform_device *pdev)
    899{
    900	struct device *dev = &pdev->dev;
    901	struct nct6775_sio_data *sio_data = dev_get_platdata(dev);
    902	struct nct6775_data *data;
    903	struct resource *res;
    904	const struct regmap_config *regmapcfg;
    905
    906	if (sio_data->access == access_direct) {
    907		res = platform_get_resource(pdev, IORESOURCE_IO, 0);
    908		if (!devm_request_region(&pdev->dev, res->start, IOREGION_LENGTH, DRVNAME))
    909			return -EBUSY;
    910	}
    911
    912	data = devm_kzalloc(&pdev->dev, sizeof(*data), GFP_KERNEL);
    913	if (!data)
    914		return -ENOMEM;
    915
    916	data->kind = sio_data->kind;
    917	data->sioreg = sio_data->sioreg;
    918
    919	if (sio_data->access == access_direct) {
    920		data->addr = res->start;
    921		regmapcfg = &nct6775_regmap_config;
    922	} else {
    923		regmapcfg = &nct6775_wmi_regmap_config;
    924	}
    925
    926	platform_set_drvdata(pdev, data);
    927
    928	data->driver_data = sio_data;
    929	data->driver_init = nct6775_platform_probe_init;
    930
    931	return nct6775_probe(&pdev->dev, data, regmapcfg);
    932}
    933
    934static struct platform_driver nct6775_driver = {
    935	.driver = {
    936		.name	= DRVNAME,
    937		.pm	= &nct6775_dev_pm_ops,
    938	},
    939	.probe		= nct6775_platform_probe,
    940};
    941
    942/* nct6775_find() looks for a '627 in the Super-I/O config space */
    943static int __init nct6775_find(int sioaddr, struct nct6775_sio_data *sio_data)
    944{
    945	u16 val;
    946	int err;
    947	int addr;
    948
    949	sio_data->access = access_direct;
    950	sio_data->sioreg = sioaddr;
    951
    952	err = sio_data->sio_enter(sio_data);
    953	if (err)
    954		return err;
    955
    956	val = (sio_data->sio_inb(sio_data, SIO_REG_DEVID) << 8) |
    957		sio_data->sio_inb(sio_data, SIO_REG_DEVID + 1);
    958	if (force_id && val != 0xffff)
    959		val = force_id;
    960
    961	switch (val & SIO_ID_MASK) {
    962	case SIO_NCT6106_ID:
    963		sio_data->kind = nct6106;
    964		break;
    965	case SIO_NCT6116_ID:
    966		sio_data->kind = nct6116;
    967		break;
    968	case SIO_NCT6775_ID:
    969		sio_data->kind = nct6775;
    970		break;
    971	case SIO_NCT6776_ID:
    972		sio_data->kind = nct6776;
    973		break;
    974	case SIO_NCT6779_ID:
    975		sio_data->kind = nct6779;
    976		break;
    977	case SIO_NCT6791_ID:
    978		sio_data->kind = nct6791;
    979		break;
    980	case SIO_NCT6792_ID:
    981		sio_data->kind = nct6792;
    982		break;
    983	case SIO_NCT6793_ID:
    984		sio_data->kind = nct6793;
    985		break;
    986	case SIO_NCT6795_ID:
    987		sio_data->kind = nct6795;
    988		break;
    989	case SIO_NCT6796_ID:
    990		sio_data->kind = nct6796;
    991		break;
    992	case SIO_NCT6797_ID:
    993		sio_data->kind = nct6797;
    994		break;
    995	case SIO_NCT6798_ID:
    996		sio_data->kind = nct6798;
    997		break;
    998	default:
    999		if (val != 0xffff)
   1000			pr_debug("unsupported chip ID: 0x%04x\n", val);
   1001		sio_data->sio_exit(sio_data);
   1002		return -ENODEV;
   1003	}
   1004
   1005	/* We have a known chip, find the HWM I/O address */
   1006	sio_data->sio_select(sio_data, NCT6775_LD_HWM);
   1007	val = (sio_data->sio_inb(sio_data, SIO_REG_ADDR) << 8)
   1008	    | sio_data->sio_inb(sio_data, SIO_REG_ADDR + 1);
   1009	addr = val & IOREGION_ALIGNMENT;
   1010	if (addr == 0) {
   1011		pr_err("Refusing to enable a Super-I/O device with a base I/O port 0\n");
   1012		sio_data->sio_exit(sio_data);
   1013		return -ENODEV;
   1014	}
   1015
   1016	/* Activate logical device if needed */
   1017	val = sio_data->sio_inb(sio_data, SIO_REG_ENABLE);
   1018	if (!(val & 0x01)) {
   1019		pr_warn("Forcibly enabling Super-I/O. Sensor is probably unusable.\n");
   1020		sio_data->sio_outb(sio_data, SIO_REG_ENABLE, val | 0x01);
   1021	}
   1022
   1023	if (sio_data->kind == nct6791 || sio_data->kind == nct6792 ||
   1024	    sio_data->kind == nct6793 || sio_data->kind == nct6795 ||
   1025	    sio_data->kind == nct6796 || sio_data->kind == nct6797 ||
   1026	    sio_data->kind == nct6798)
   1027		nct6791_enable_io_mapping(sio_data);
   1028
   1029	sio_data->sio_exit(sio_data);
   1030	pr_info("Found %s or compatible chip at %#x:%#x\n",
   1031		nct6775_sio_names[sio_data->kind], sioaddr, addr);
   1032
   1033	return addr;
   1034}
   1035
   1036/*
   1037 * when Super-I/O functions move to a separate file, the Super-I/O
   1038 * bus will manage the lifetime of the device and this module will only keep
   1039 * track of the nct6775 driver. But since we use platform_device_alloc(), we
   1040 * must keep track of the device
   1041 */
   1042static struct platform_device *pdev[2];
   1043
   1044static const char * const asus_wmi_boards[] = {
   1045	"PRO H410T",
   1046	"ProArt X570-CREATOR WIFI",
   1047	"Pro B550M-C",
   1048	"Pro WS X570-ACE",
   1049	"PRIME B360-PLUS",
   1050	"PRIME B460-PLUS",
   1051	"PRIME B550-PLUS",
   1052	"PRIME B550M-A",
   1053	"PRIME B550M-A (WI-FI)",
   1054	"PRIME H410M-R",
   1055	"PRIME X570-P",
   1056	"PRIME X570-PRO",
   1057	"ROG CROSSHAIR VIII DARK HERO",
   1058	"ROG CROSSHAIR VIII FORMULA",
   1059	"ROG CROSSHAIR VIII HERO",
   1060	"ROG CROSSHAIR VIII IMPACT",
   1061	"ROG STRIX B550-A GAMING",
   1062	"ROG STRIX B550-E GAMING",
   1063	"ROG STRIX B550-F GAMING",
   1064	"ROG STRIX B550-F GAMING (WI-FI)",
   1065	"ROG STRIX B550-F GAMING WIFI II",
   1066	"ROG STRIX B550-I GAMING",
   1067	"ROG STRIX B550-XE GAMING (WI-FI)",
   1068	"ROG STRIX X570-E GAMING",
   1069	"ROG STRIX X570-E GAMING WIFI II",
   1070	"ROG STRIX X570-F GAMING",
   1071	"ROG STRIX X570-I GAMING",
   1072	"ROG STRIX Z390-E GAMING",
   1073	"ROG STRIX Z390-F GAMING",
   1074	"ROG STRIX Z390-H GAMING",
   1075	"ROG STRIX Z390-I GAMING",
   1076	"ROG STRIX Z490-A GAMING",
   1077	"ROG STRIX Z490-E GAMING",
   1078	"ROG STRIX Z490-F GAMING",
   1079	"ROG STRIX Z490-G GAMING",
   1080	"ROG STRIX Z490-G GAMING (WI-FI)",
   1081	"ROG STRIX Z490-H GAMING",
   1082	"ROG STRIX Z490-I GAMING",
   1083	"TUF GAMING B550M-PLUS",
   1084	"TUF GAMING B550M-PLUS (WI-FI)",
   1085	"TUF GAMING B550-PLUS",
   1086	"TUF GAMING B550-PRO",
   1087	"TUF GAMING X570-PLUS",
   1088	"TUF GAMING X570-PLUS (WI-FI)",
   1089	"TUF GAMING X570-PRO (WI-FI)",
   1090	"TUF GAMING Z490-PLUS",
   1091	"TUF GAMING Z490-PLUS (WI-FI)",
   1092};
   1093
   1094static int __init sensors_nct6775_platform_init(void)
   1095{
   1096	int i, err;
   1097	bool found = false;
   1098	int address;
   1099	struct resource res;
   1100	struct nct6775_sio_data sio_data;
   1101	int sioaddr[2] = { 0x2e, 0x4e };
   1102	enum sensor_access access = access_direct;
   1103	const char *board_vendor, *board_name;
   1104	u8 tmp;
   1105
   1106	err = platform_driver_register(&nct6775_driver);
   1107	if (err)
   1108		return err;
   1109
   1110	board_vendor = dmi_get_system_info(DMI_BOARD_VENDOR);
   1111	board_name = dmi_get_system_info(DMI_BOARD_NAME);
   1112
   1113	if (board_name && board_vendor &&
   1114	    !strcmp(board_vendor, "ASUSTeK COMPUTER INC.")) {
   1115		err = match_string(asus_wmi_boards, ARRAY_SIZE(asus_wmi_boards),
   1116				   board_name);
   1117		if (err >= 0) {
   1118			/* if reading chip id via WMI succeeds, use WMI */
   1119			if (!nct6775_asuswmi_read(0, NCT6775_PORT_CHIPID, &tmp) && tmp) {
   1120				pr_info("Using Asus WMI to access %#x chip.\n", tmp);
   1121				access = access_asuswmi;
   1122			} else {
   1123				pr_err("Can't read ChipID by Asus WMI.\n");
   1124			}
   1125		}
   1126	}
   1127
   1128	/*
   1129	 * initialize sio_data->kind and sio_data->sioreg.
   1130	 *
   1131	 * when Super-I/O functions move to a separate file, the Super-I/O
   1132	 * driver will probe 0x2e and 0x4e and auto-detect the presence of a
   1133	 * nct6775 hardware monitor, and call probe()
   1134	 */
   1135	for (i = 0; i < ARRAY_SIZE(pdev); i++) {
   1136		sio_data.sio_outb = superio_outb;
   1137		sio_data.sio_inb = superio_inb;
   1138		sio_data.sio_select = superio_select;
   1139		sio_data.sio_enter = superio_enter;
   1140		sio_data.sio_exit = superio_exit;
   1141
   1142		address = nct6775_find(sioaddr[i], &sio_data);
   1143		if (address <= 0)
   1144			continue;
   1145
   1146		found = true;
   1147
   1148		sio_data.access = access;
   1149
   1150		if (access == access_asuswmi) {
   1151			sio_data.sio_outb = superio_wmi_outb;
   1152			sio_data.sio_inb = superio_wmi_inb;
   1153			sio_data.sio_select = superio_wmi_select;
   1154			sio_data.sio_enter = superio_wmi_enter;
   1155			sio_data.sio_exit = superio_wmi_exit;
   1156		}
   1157
   1158		pdev[i] = platform_device_alloc(DRVNAME, address);
   1159		if (!pdev[i]) {
   1160			err = -ENOMEM;
   1161			goto exit_device_unregister;
   1162		}
   1163
   1164		err = platform_device_add_data(pdev[i], &sio_data,
   1165					       sizeof(struct nct6775_sio_data));
   1166		if (err)
   1167			goto exit_device_put;
   1168
   1169		if (sio_data.access == access_direct) {
   1170			memset(&res, 0, sizeof(res));
   1171			res.name = DRVNAME;
   1172			res.start = address + IOREGION_OFFSET;
   1173			res.end = address + IOREGION_OFFSET + IOREGION_LENGTH - 1;
   1174			res.flags = IORESOURCE_IO;
   1175
   1176			err = acpi_check_resource_conflict(&res);
   1177			if (err) {
   1178				platform_device_put(pdev[i]);
   1179				pdev[i] = NULL;
   1180				continue;
   1181			}
   1182
   1183			err = platform_device_add_resources(pdev[i], &res, 1);
   1184			if (err)
   1185				goto exit_device_put;
   1186		}
   1187
   1188		/* platform_device_add calls probe() */
   1189		err = platform_device_add(pdev[i]);
   1190		if (err)
   1191			goto exit_device_put;
   1192	}
   1193	if (!found) {
   1194		err = -ENODEV;
   1195		goto exit_unregister;
   1196	}
   1197
   1198	return 0;
   1199
   1200exit_device_put:
   1201	platform_device_put(pdev[i]);
   1202exit_device_unregister:
   1203	while (--i >= 0) {
   1204		if (pdev[i])
   1205			platform_device_unregister(pdev[i]);
   1206	}
   1207exit_unregister:
   1208	platform_driver_unregister(&nct6775_driver);
   1209	return err;
   1210}
   1211
   1212static void __exit sensors_nct6775_platform_exit(void)
   1213{
   1214	int i;
   1215
   1216	for (i = 0; i < ARRAY_SIZE(pdev); i++) {
   1217		if (pdev[i])
   1218			platform_device_unregister(pdev[i]);
   1219	}
   1220	platform_driver_unregister(&nct6775_driver);
   1221}
   1222
   1223MODULE_AUTHOR("Guenter Roeck <linux@roeck-us.net>");
   1224MODULE_DESCRIPTION("Platform driver for NCT6775F and compatible chips");
   1225MODULE_LICENSE("GPL");
   1226MODULE_IMPORT_NS(HWMON_NCT6775);
   1227
   1228module_init(sensors_nct6775_platform_init);
   1229module_exit(sensors_nct6775_platform_exit);