| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 1 | /* | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 2 |     Copyright (c) 1998 - 2002  Frodo Looijaard <frodol@dds.nl>, | 
| Jan Engelhardt | 96de0e2 | 2007-10-19 23:21:04 +0200 | [diff] [blame] | 3 |     Philip Edelbrock <phil@netroedge.com>, Kyösti Mälkki <kmalkki@cc.hut.fi>, | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 4 |     Mark D. Studebaker <mdsxyz123@yahoo.com> | 
| Jean Delvare | 0d227a7 | 2008-01-27 18:14:51 +0100 | [diff] [blame] | 5 |     Copyright (C) 2005 - 2008  Jean Delvare <khali@linux-fr.org> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 6 |  | 
 | 7 |     This program is free software; you can redistribute it and/or modify | 
 | 8 |     it under the terms of the GNU General Public License as published by | 
 | 9 |     the Free Software Foundation; either version 2 of the License, or | 
 | 10 |     (at your option) any later version. | 
 | 11 |  | 
 | 12 |     This program is distributed in the hope that it will be useful, | 
 | 13 |     but WITHOUT ANY WARRANTY; without even the implied warranty of | 
 | 14 |     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
 | 15 |     GNU General Public License for more details. | 
 | 16 |  | 
 | 17 |     You should have received a copy of the GNU General Public License | 
 | 18 |     along with this program; if not, write to the Free Software | 
 | 19 |     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | 
 | 20 | */ | 
 | 21 |  | 
 | 22 | /* | 
| Jean Delvare | aaf7f14 | 2005-09-22 22:09:07 +0200 | [diff] [blame] | 23 |    Supports the following VIA south bridges: | 
 | 24 |  | 
 | 25 |    Chip name          PCI ID  REV     I2C block | 
 | 26 |    VT82C596A          0x3050             no | 
 | 27 |    VT82C596B          0x3051             no | 
 | 28 |    VT82C686A          0x3057  0x30       no | 
 | 29 |    VT82C686B          0x3057  0x40       yes | 
 | 30 |    VT8231             0x8235             no? | 
 | 31 |    VT8233             0x3074             yes | 
 | 32 |    VT8233A            0x3147             yes? | 
 | 33 |    VT8235             0x3177             yes | 
 | 34 |    VT8237R            0x3227             yes | 
| Rudolf Marek | c243353 | 2006-09-03 22:35:21 +0200 | [diff] [blame] | 35 |    VT8237A            0x3337             yes | 
| Jean Delvare | 0d227a7 | 2008-01-27 18:14:51 +0100 | [diff] [blame] | 36 |    VT8237S            0x3372             yes | 
| Rudolf Marek | c243353 | 2006-09-03 22:35:21 +0200 | [diff] [blame] | 37 |    VT8251             0x3287             yes | 
| Jean Delvare | ab6a6ed | 2007-02-13 22:09:02 +0100 | [diff] [blame] | 38 |    CX700              0x8324             yes | 
| Rudolf Marek | b806a71 | 2008-10-14 17:30:03 +0200 | [diff] [blame] | 39 |    VX800/VX820        0x8353             yes | 
| Harald Welte | a231591 | 2009-06-15 18:01:49 +0200 | [diff] [blame] | 40 |    VX855/VX875        0x8409             yes | 
| Jean Delvare | aaf7f14 | 2005-09-22 22:09:07 +0200 | [diff] [blame] | 41 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 42 |    Note: we assume there can only be one device, with one SMBus interface. | 
 | 43 | */ | 
 | 44 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 45 | #include <linux/module.h> | 
 | 46 | #include <linux/delay.h> | 
 | 47 | #include <linux/pci.h> | 
 | 48 | #include <linux/kernel.h> | 
 | 49 | #include <linux/stddef.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 50 | #include <linux/ioport.h> | 
 | 51 | #include <linux/i2c.h> | 
 | 52 | #include <linux/init.h> | 
| Jean Delvare | 54fb4a05 | 2008-07-14 22:38:33 +0200 | [diff] [blame] | 53 | #include <linux/acpi.h> | 
| H Hartley Sweeten | 2178218 | 2010-05-21 18:41:01 +0200 | [diff] [blame] | 54 | #include <linux/io.h> | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 55 |  | 
 | 56 | static struct pci_dev *vt596_pdev; | 
 | 57 |  | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 58 | #define SMBBA1		0x90 | 
 | 59 | #define SMBBA2		0x80 | 
 | 60 | #define SMBBA3		0xD0 | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 61 |  | 
 | 62 | /* SMBus address offsets */ | 
 | 63 | static unsigned short vt596_smba; | 
 | 64 | #define SMBHSTSTS	(vt596_smba + 0) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 65 | #define SMBHSTCNT	(vt596_smba + 2) | 
 | 66 | #define SMBHSTCMD	(vt596_smba + 3) | 
 | 67 | #define SMBHSTADD	(vt596_smba + 4) | 
 | 68 | #define SMBHSTDAT0	(vt596_smba + 5) | 
 | 69 | #define SMBHSTDAT1	(vt596_smba + 6) | 
 | 70 | #define SMBBLKDAT	(vt596_smba + 7) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 71 |  | 
 | 72 | /* PCI Address Constants */ | 
 | 73 |  | 
 | 74 | /* SMBus data in configuration space can be found in two places, | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 75 |    We try to select the better one */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 76 |  | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 77 | static unsigned short SMBHSTCFG = 0xD2; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 78 |  | 
 | 79 | /* Other settings */ | 
 | 80 | #define MAX_TIMEOUT	500 | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 81 |  | 
 | 82 | /* VT82C596 constants */ | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 83 | #define VT596_QUICK		0x00 | 
 | 84 | #define VT596_BYTE		0x04 | 
 | 85 | #define VT596_BYTE_DATA		0x08 | 
 | 86 | #define VT596_WORD_DATA		0x0C | 
| Prakash Mortha | a05f2c5 | 2008-10-14 17:30:06 +0200 | [diff] [blame] | 87 | #define VT596_PROC_CALL		0x10 | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 88 | #define VT596_BLOCK_DATA	0x14 | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 89 | #define VT596_I2C_BLOCK_DATA	0x34 | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 90 |  | 
 | 91 |  | 
 | 92 | /* If force is set to anything different from 0, we forcibly enable the | 
 | 93 |    VT596. DANGEROUS! */ | 
 | 94 | static int force; | 
 | 95 | module_param(force, bool, 0); | 
 | 96 | MODULE_PARM_DESC(force, "Forcibly enable the SMBus. DANGEROUS!"); | 
 | 97 |  | 
 | 98 | /* If force_addr is set to anything different from 0, we forcibly enable | 
 | 99 |    the VT596 at the given address. VERY DANGEROUS! */ | 
 | 100 | static u16 force_addr; | 
 | 101 | module_param(force_addr, ushort, 0); | 
 | 102 | MODULE_PARM_DESC(force_addr, | 
 | 103 | 		 "Forcibly enable the SMBus at the given address. " | 
 | 104 | 		 "EXTREMELY DANGEROUS!"); | 
 | 105 |  | 
 | 106 |  | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 107 | static struct pci_driver vt596_driver; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 108 | static struct i2c_adapter vt596_adapter; | 
 | 109 |  | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 110 | #define FEATURE_I2CBLOCK	(1<<0) | 
 | 111 | static unsigned int vt596_features; | 
 | 112 |  | 
| Jean Delvare | ed5453e | 2005-09-22 22:23:32 +0200 | [diff] [blame] | 113 | #ifdef DEBUG | 
 | 114 | static void vt596_dump_regs(const char *msg, u8 size) | 
 | 115 | { | 
 | 116 | 	dev_dbg(&vt596_adapter.dev, "%s: STS=%02x CNT=%02x CMD=%02x ADD=%02x " | 
 | 117 | 		"DAT=%02x,%02x\n", msg, inb_p(SMBHSTSTS), inb_p(SMBHSTCNT), | 
 | 118 | 		inb_p(SMBHSTCMD), inb_p(SMBHSTADD), inb_p(SMBHSTDAT0), | 
 | 119 | 		inb_p(SMBHSTDAT1)); | 
 | 120 |  | 
 | 121 | 	if (size == VT596_BLOCK_DATA | 
 | 122 | 	 || size == VT596_I2C_BLOCK_DATA) { | 
 | 123 | 		int i; | 
 | 124 |  | 
 | 125 | 		dev_dbg(&vt596_adapter.dev, "BLK="); | 
 | 126 | 		for (i = 0; i < I2C_SMBUS_BLOCK_MAX / 2; i++) | 
 | 127 | 			printk("%02x,", inb_p(SMBBLKDAT)); | 
 | 128 | 		printk("\n"); | 
 | 129 | 		dev_dbg(&vt596_adapter.dev, "    "); | 
 | 130 | 		for (; i < I2C_SMBUS_BLOCK_MAX - 1; i++) | 
 | 131 | 			printk("%02x,", inb_p(SMBBLKDAT)); | 
 | 132 | 		printk("%02x\n", inb_p(SMBBLKDAT)); | 
 | 133 | 	} | 
 | 134 | } | 
| Greg KH | ca68f11 | 2005-09-22 22:23:32 +0200 | [diff] [blame] | 135 | #else | 
 | 136 | static inline void vt596_dump_regs(const char *msg, u8 size) { } | 
| Jean Delvare | ed5453e | 2005-09-22 22:23:32 +0200 | [diff] [blame] | 137 | #endif | 
 | 138 |  | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 139 | /* Return -1 on error, 0 on success */ | 
| Jean Delvare | 50c1cc3 | 2005-09-22 22:15:53 +0200 | [diff] [blame] | 140 | static int vt596_transaction(u8 size) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 141 | { | 
 | 142 | 	int temp; | 
 | 143 | 	int result = 0; | 
 | 144 | 	int timeout = 0; | 
 | 145 |  | 
| Jean Delvare | ed5453e | 2005-09-22 22:23:32 +0200 | [diff] [blame] | 146 | 	vt596_dump_regs("Transaction (pre)", size); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 147 |  | 
 | 148 | 	/* Make sure the SMBus host is ready to start transmitting */ | 
 | 149 | 	if ((temp = inb_p(SMBHSTSTS)) & 0x1F) { | 
 | 150 | 		dev_dbg(&vt596_adapter.dev, "SMBus busy (0x%02x). " | 
| Jean Delvare | 8750197 | 2005-10-31 18:51:21 +0100 | [diff] [blame] | 151 | 			"Resetting...\n", temp); | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 152 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 153 | 		outb_p(temp, SMBHSTSTS); | 
 | 154 | 		if ((temp = inb_p(SMBHSTSTS)) & 0x1F) { | 
| Jean Delvare | 8750197 | 2005-10-31 18:51:21 +0100 | [diff] [blame] | 155 | 			dev_err(&vt596_adapter.dev, "SMBus reset failed! " | 
 | 156 | 				"(0x%02x)\n", temp); | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 157 | 			return -EBUSY; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 158 | 		} | 
 | 159 | 	} | 
 | 160 |  | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 161 | 	/* Start the transaction by setting bit 6 */ | 
| Jean Delvare | 8750197 | 2005-10-31 18:51:21 +0100 | [diff] [blame] | 162 | 	outb_p(0x40 | size, SMBHSTCNT); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 163 |  | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 164 | 	/* We will always wait for a fraction of a second */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 165 | 	do { | 
 | 166 | 		msleep(1); | 
 | 167 | 		temp = inb_p(SMBHSTSTS); | 
| Roel Kluin | b6a3195 | 2010-01-16 20:43:12 +0100 | [diff] [blame] | 168 | 	} while ((temp & 0x01) && (++timeout < MAX_TIMEOUT)); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 169 |  | 
 | 170 | 	/* If the SMBus is still busy, we give up */ | 
| Roel Kluin | b6a3195 | 2010-01-16 20:43:12 +0100 | [diff] [blame] | 171 | 	if (timeout == MAX_TIMEOUT) { | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 172 | 		result = -ETIMEDOUT; | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 173 | 		dev_err(&vt596_adapter.dev, "SMBus timeout!\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 174 | 	} | 
 | 175 |  | 
 | 176 | 	if (temp & 0x10) { | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 177 | 		result = -EIO; | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 178 | 		dev_err(&vt596_adapter.dev, "Transaction failed (0x%02x)\n", | 
| Jean Delvare | 8750197 | 2005-10-31 18:51:21 +0100 | [diff] [blame] | 179 | 			size); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 180 | 	} | 
 | 181 |  | 
 | 182 | 	if (temp & 0x08) { | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 183 | 		result = -EIO; | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 184 | 		dev_err(&vt596_adapter.dev, "SMBus collision!\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 185 | 	} | 
 | 186 |  | 
 | 187 | 	if (temp & 0x04) { | 
| Jean Delvare | 8750197 | 2005-10-31 18:51:21 +0100 | [diff] [blame] | 188 | 		int read = inb_p(SMBHSTADD) & 0x01; | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 189 | 		result = -ENXIO; | 
| Jean Delvare | 8750197 | 2005-10-31 18:51:21 +0100 | [diff] [blame] | 190 | 		/* The quick and receive byte commands are used to probe | 
 | 191 | 		   for chips, so errors are expected, and we don't want | 
 | 192 | 		   to frighten the user. */ | 
 | 193 | 		if (!((size == VT596_QUICK && !read) || | 
 | 194 | 		      (size == VT596_BYTE && read))) | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 195 | 			dev_err(&vt596_adapter.dev, "Transaction error!\n"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 196 | 	} | 
 | 197 |  | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 198 | 	/* Resetting status register */ | 
 | 199 | 	if (temp & 0x1F) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 200 | 		outb_p(temp, SMBHSTSTS); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 201 |  | 
| Jean Delvare | ed5453e | 2005-09-22 22:23:32 +0200 | [diff] [blame] | 202 | 	vt596_dump_regs("Transaction (post)", size); | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 203 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 204 | 	return result; | 
 | 205 | } | 
 | 206 |  | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 207 | /* Return negative errno on error, 0 on success */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 208 | static s32 vt596_access(struct i2c_adapter *adap, u16 addr, | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 209 | 		unsigned short flags, char read_write, u8 command, | 
 | 210 | 		int size, union i2c_smbus_data *data) | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 211 | { | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 212 | 	int i; | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 213 | 	int status; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 214 |  | 
 | 215 | 	switch (size) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 216 | 	case I2C_SMBUS_QUICK: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 217 | 		size = VT596_QUICK; | 
 | 218 | 		break; | 
 | 219 | 	case I2C_SMBUS_BYTE: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 220 | 		if (read_write == I2C_SMBUS_WRITE) | 
 | 221 | 			outb_p(command, SMBHSTCMD); | 
 | 222 | 		size = VT596_BYTE; | 
 | 223 | 		break; | 
 | 224 | 	case I2C_SMBUS_BYTE_DATA: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 225 | 		outb_p(command, SMBHSTCMD); | 
 | 226 | 		if (read_write == I2C_SMBUS_WRITE) | 
 | 227 | 			outb_p(data->byte, SMBHSTDAT0); | 
 | 228 | 		size = VT596_BYTE_DATA; | 
 | 229 | 		break; | 
 | 230 | 	case I2C_SMBUS_WORD_DATA: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 231 | 		outb_p(command, SMBHSTCMD); | 
 | 232 | 		if (read_write == I2C_SMBUS_WRITE) { | 
 | 233 | 			outb_p(data->word & 0xff, SMBHSTDAT0); | 
 | 234 | 			outb_p((data->word & 0xff00) >> 8, SMBHSTDAT1); | 
 | 235 | 		} | 
 | 236 | 		size = VT596_WORD_DATA; | 
 | 237 | 		break; | 
| Prakash Mortha | a05f2c5 | 2008-10-14 17:30:06 +0200 | [diff] [blame] | 238 | 	case I2C_SMBUS_PROC_CALL: | 
 | 239 | 		outb_p(command, SMBHSTCMD); | 
 | 240 | 		outb_p(data->word & 0xff, SMBHSTDAT0); | 
 | 241 | 		outb_p((data->word & 0xff00) >> 8, SMBHSTDAT1); | 
 | 242 | 		size = VT596_PROC_CALL; | 
 | 243 | 		break; | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 244 | 	case I2C_SMBUS_I2C_BLOCK_DATA: | 
 | 245 | 		if (!(vt596_features & FEATURE_I2CBLOCK)) | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 246 | 			goto exit_unsupported; | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 247 | 		if (read_write == I2C_SMBUS_READ) | 
| Jean Delvare | 4b2643d | 2007-07-12 14:12:29 +0200 | [diff] [blame] | 248 | 			outb_p(data->block[0], SMBHSTDAT0); | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 249 | 		/* Fall through */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 250 | 	case I2C_SMBUS_BLOCK_DATA: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 251 | 		outb_p(command, SMBHSTCMD); | 
 | 252 | 		if (read_write == I2C_SMBUS_WRITE) { | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 253 | 			u8 len = data->block[0]; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 254 | 			if (len > I2C_SMBUS_BLOCK_MAX) | 
 | 255 | 				len = I2C_SMBUS_BLOCK_MAX; | 
 | 256 | 			outb_p(len, SMBHSTDAT0); | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 257 | 			inb_p(SMBHSTCNT);	/* Reset SMBBLKDAT */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 258 | 			for (i = 1; i <= len; i++) | 
 | 259 | 				outb_p(data->block[i], SMBBLKDAT); | 
 | 260 | 		} | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 261 | 		size = (size == I2C_SMBUS_I2C_BLOCK_DATA) ? | 
 | 262 | 		       VT596_I2C_BLOCK_DATA : VT596_BLOCK_DATA; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 263 | 		break; | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 264 | 	default: | 
 | 265 | 		goto exit_unsupported; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 266 | 	} | 
 | 267 |  | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 268 | 	outb_p(((addr & 0x7f) << 1) | read_write, SMBHSTADD); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 269 |  | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 270 | 	status = vt596_transaction(size); | 
 | 271 | 	if (status) | 
 | 272 | 		return status; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 273 |  | 
| Prakash Mortha | a05f2c5 | 2008-10-14 17:30:06 +0200 | [diff] [blame] | 274 | 	if (size == VT596_PROC_CALL) | 
 | 275 | 		read_write = I2C_SMBUS_READ; | 
 | 276 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 277 | 	if ((read_write == I2C_SMBUS_WRITE) || (size == VT596_QUICK)) | 
 | 278 | 		return 0; | 
 | 279 |  | 
 | 280 | 	switch (size) { | 
 | 281 | 	case VT596_BYTE: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 282 | 	case VT596_BYTE_DATA: | 
 | 283 | 		data->byte = inb_p(SMBHSTDAT0); | 
 | 284 | 		break; | 
 | 285 | 	case VT596_WORD_DATA: | 
| Prakash Mortha | a05f2c5 | 2008-10-14 17:30:06 +0200 | [diff] [blame] | 286 | 	case VT596_PROC_CALL: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 287 | 		data->word = inb_p(SMBHSTDAT0) + (inb_p(SMBHSTDAT1) << 8); | 
 | 288 | 		break; | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 289 | 	case VT596_I2C_BLOCK_DATA: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 290 | 	case VT596_BLOCK_DATA: | 
 | 291 | 		data->block[0] = inb_p(SMBHSTDAT0); | 
 | 292 | 		if (data->block[0] > I2C_SMBUS_BLOCK_MAX) | 
 | 293 | 			data->block[0] = I2C_SMBUS_BLOCK_MAX; | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 294 | 		inb_p(SMBHSTCNT);	/* Reset SMBBLKDAT */ | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 295 | 		for (i = 1; i <= data->block[0]; i++) | 
 | 296 | 			data->block[i] = inb_p(SMBBLKDAT); | 
 | 297 | 		break; | 
 | 298 | 	} | 
 | 299 | 	return 0; | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 300 |  | 
 | 301 | exit_unsupported: | 
| Jean Delvare | ac7fc4f | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 302 | 	dev_warn(&vt596_adapter.dev, "Unsupported transaction %d\n", | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 303 | 		 size); | 
| David Brownell | 9714034 | 2008-07-14 22:38:25 +0200 | [diff] [blame] | 304 | 	return -EOPNOTSUPP; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 305 | } | 
 | 306 |  | 
 | 307 | static u32 vt596_func(struct i2c_adapter *adapter) | 
 | 308 | { | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 309 | 	u32 func = I2C_FUNC_SMBUS_QUICK | I2C_FUNC_SMBUS_BYTE | | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 310 | 	    I2C_FUNC_SMBUS_BYTE_DATA | I2C_FUNC_SMBUS_WORD_DATA | | 
| Prakash Mortha | a05f2c5 | 2008-10-14 17:30:06 +0200 | [diff] [blame] | 311 | 	    I2C_SMBUS_PROC_CALL | I2C_FUNC_SMBUS_BLOCK_DATA; | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 312 |  | 
 | 313 | 	if (vt596_features & FEATURE_I2CBLOCK) | 
 | 314 | 		func |= I2C_FUNC_SMBUS_I2C_BLOCK; | 
 | 315 | 	return func; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 316 | } | 
 | 317 |  | 
| Jean Delvare | 8f9082c | 2006-09-03 22:39:46 +0200 | [diff] [blame] | 318 | static const struct i2c_algorithm smbus_algorithm = { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 319 | 	.smbus_xfer	= vt596_access, | 
 | 320 | 	.functionality	= vt596_func, | 
 | 321 | }; | 
 | 322 |  | 
 | 323 | static struct i2c_adapter vt596_adapter = { | 
 | 324 | 	.owner		= THIS_MODULE, | 
| Jean Delvare | 3401b2f | 2008-07-14 22:38:29 +0200 | [diff] [blame] | 325 | 	.class		= I2C_CLASS_HWMON | I2C_CLASS_SPD, | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 326 | 	.algo		= &smbus_algorithm, | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 327 | }; | 
 | 328 |  | 
 | 329 | static int __devinit vt596_probe(struct pci_dev *pdev, | 
 | 330 | 				 const struct pci_device_id *id) | 
 | 331 | { | 
 | 332 | 	unsigned char temp; | 
 | 333 | 	int error = -ENODEV; | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 334 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 335 | 	/* Determine the address of the SMBus areas */ | 
 | 336 | 	if (force_addr) { | 
 | 337 | 		vt596_smba = force_addr & 0xfff0; | 
 | 338 | 		force = 0; | 
 | 339 | 		goto found; | 
 | 340 | 	} | 
 | 341 |  | 
 | 342 | 	if ((pci_read_config_word(pdev, id->driver_data, &vt596_smba)) || | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 343 | 	    !(vt596_smba & 0x0001)) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 344 | 		/* try 2nd address and config reg. for 596 */ | 
 | 345 | 		if (id->device == PCI_DEVICE_ID_VIA_82C596_3 && | 
 | 346 | 		    !pci_read_config_word(pdev, SMBBA2, &vt596_smba) && | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 347 | 		    (vt596_smba & 0x0001)) { | 
 | 348 | 			SMBHSTCFG = 0x84; | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 349 | 		} else { | 
 | 350 | 			/* no matches at all */ | 
 | 351 | 			dev_err(&pdev->dev, "Cannot configure " | 
 | 352 | 				"SMBus I/O Base address\n"); | 
 | 353 | 			return -ENODEV; | 
 | 354 | 		} | 
 | 355 | 	} | 
 | 356 |  | 
 | 357 | 	vt596_smba &= 0xfff0; | 
 | 358 | 	if (vt596_smba == 0) { | 
 | 359 | 		dev_err(&pdev->dev, "SMBus base address " | 
 | 360 | 			"uninitialized - upgrade BIOS or use " | 
 | 361 | 			"force_addr=0xaddr\n"); | 
 | 362 | 		return -ENODEV; | 
 | 363 | 	} | 
 | 364 |  | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 365 | found: | 
| Jean Delvare | 54fb4a05 | 2008-07-14 22:38:33 +0200 | [diff] [blame] | 366 | 	error = acpi_check_region(vt596_smba, 8, vt596_driver.name); | 
 | 367 | 	if (error) | 
| Jean Delvare | 18669ea | 2009-10-04 22:53:45 +0200 | [diff] [blame] | 368 | 		return -ENODEV; | 
| Jean Delvare | 54fb4a05 | 2008-07-14 22:38:33 +0200 | [diff] [blame] | 369 |  | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 370 | 	if (!request_region(vt596_smba, 8, vt596_driver.name)) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 371 | 		dev_err(&pdev->dev, "SMBus region 0x%x already in use!\n", | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 372 | 			vt596_smba); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 373 | 		return -ENODEV; | 
 | 374 | 	} | 
 | 375 |  | 
 | 376 | 	pci_read_config_byte(pdev, SMBHSTCFG, &temp); | 
 | 377 | 	/* If force_addr is set, we program the new address here. Just to make | 
 | 378 | 	   sure, we disable the VT596 first. */ | 
 | 379 | 	if (force_addr) { | 
 | 380 | 		pci_write_config_byte(pdev, SMBHSTCFG, temp & 0xfe); | 
 | 381 | 		pci_write_config_word(pdev, id->driver_data, vt596_smba); | 
 | 382 | 		pci_write_config_byte(pdev, SMBHSTCFG, temp | 0x01); | 
 | 383 | 		dev_warn(&pdev->dev, "WARNING: SMBus interface set to new " | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 384 | 			 "address 0x%04x!\n", vt596_smba); | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 385 | 	} else if (!(temp & 0x01)) { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 386 | 		if (force) { | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 387 | 			/* NOTE: This assumes I/O space and other allocations | 
 | 388 | 			 * WERE done by the Bios!  Don't complain if your | 
 | 389 | 			 * hardware does weird things after enabling this. | 
 | 390 | 			 * :') Check for Bios updates before resorting to | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 391 | 			 * this. | 
 | 392 | 			 */ | 
| Jean Delvare | c2f559d | 2005-09-22 22:01:07 +0200 | [diff] [blame] | 393 | 			pci_write_config_byte(pdev, SMBHSTCFG, temp | 0x01); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 394 | 			dev_info(&pdev->dev, "Enabling SMBus device\n"); | 
 | 395 | 		} else { | 
 | 396 | 			dev_err(&pdev->dev, "SMBUS: Error: Host SMBus " | 
 | 397 | 				"controller not enabled! - upgrade BIOS or " | 
 | 398 | 				"use force=1\n"); | 
 | 399 | 			goto release_region; | 
 | 400 | 		} | 
 | 401 | 	} | 
 | 402 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 403 | 	dev_dbg(&pdev->dev, "VT596_smba = 0x%X\n", vt596_smba); | 
 | 404 |  | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 405 | 	switch (pdev->device) { | 
| Jean Delvare | ab6a6ed | 2007-02-13 22:09:02 +0100 | [diff] [blame] | 406 | 	case PCI_DEVICE_ID_VIA_CX700: | 
| Rudolf Marek | b806a71 | 2008-10-14 17:30:03 +0200 | [diff] [blame] | 407 | 	case PCI_DEVICE_ID_VIA_VX800: | 
| Harald Welte | a231591 | 2009-06-15 18:01:49 +0200 | [diff] [blame] | 408 | 	case PCI_DEVICE_ID_VIA_VX855: | 
| Rudolf Marek | c243353 | 2006-09-03 22:35:21 +0200 | [diff] [blame] | 409 | 	case PCI_DEVICE_ID_VIA_8251: | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 410 | 	case PCI_DEVICE_ID_VIA_8237: | 
| Rudolf Marek | c243353 | 2006-09-03 22:35:21 +0200 | [diff] [blame] | 411 | 	case PCI_DEVICE_ID_VIA_8237A: | 
| Jean Delvare | 0d227a7 | 2008-01-27 18:14:51 +0100 | [diff] [blame] | 412 | 	case PCI_DEVICE_ID_VIA_8237S: | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 413 | 	case PCI_DEVICE_ID_VIA_8235: | 
 | 414 | 	case PCI_DEVICE_ID_VIA_8233A: | 
 | 415 | 	case PCI_DEVICE_ID_VIA_8233_0: | 
 | 416 | 		vt596_features |= FEATURE_I2CBLOCK; | 
 | 417 | 		break; | 
 | 418 | 	case PCI_DEVICE_ID_VIA_82C686_4: | 
 | 419 | 		/* The VT82C686B (rev 0x40) does support I2C block | 
 | 420 | 		   transactions, but the VT82C686A (rev 0x30) doesn't */ | 
| Auke Kok | 44c1013 | 2007-06-08 15:46:36 -0700 | [diff] [blame] | 421 | 		if (pdev->revision >= 0x40) | 
| Jean Delvare | f118301 | 2005-09-22 21:58:41 +0200 | [diff] [blame] | 422 | 			vt596_features |= FEATURE_I2CBLOCK; | 
 | 423 | 		break; | 
 | 424 | 	} | 
 | 425 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 426 | 	vt596_adapter.dev.parent = &pdev->dev; | 
| David Brownell | 2096b95 | 2007-05-01 23:26:28 +0200 | [diff] [blame] | 427 | 	snprintf(vt596_adapter.name, sizeof(vt596_adapter.name), | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 428 | 		 "SMBus Via Pro adapter at %04x", vt596_smba); | 
 | 429 |  | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 430 | 	vt596_pdev = pci_dev_get(pdev); | 
 | 431 | 	if (i2c_add_adapter(&vt596_adapter)) { | 
 | 432 | 		pci_dev_put(vt596_pdev); | 
 | 433 | 		vt596_pdev = NULL; | 
 | 434 | 	} | 
 | 435 |  | 
 | 436 | 	/* Always return failure here.  This is to allow other drivers to bind | 
 | 437 | 	 * to this pci device.  We don't really want to have control over the | 
 | 438 | 	 * pci device, we only wanted to read as few register values from it. | 
 | 439 | 	 */ | 
 | 440 | 	return -ENODEV; | 
 | 441 |  | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 442 | release_region: | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 443 | 	release_region(vt596_smba, 8); | 
 | 444 | 	return error; | 
 | 445 | } | 
 | 446 |  | 
| Márton Németh | 4111ecd | 2010-03-02 12:23:37 +0100 | [diff] [blame] | 447 | static const struct pci_device_id vt596_ids[] = { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 448 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C596_3), | 
 | 449 | 	  .driver_data = SMBBA1 }, | 
 | 450 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C596B_3), | 
 | 451 | 	  .driver_data = SMBBA1 }, | 
 | 452 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C686_4), | 
 | 453 | 	  .driver_data = SMBBA1 }, | 
 | 454 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8233_0), | 
 | 455 | 	  .driver_data = SMBBA3 }, | 
 | 456 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8233A), | 
 | 457 | 	  .driver_data = SMBBA3 }, | 
 | 458 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8235), | 
 | 459 | 	  .driver_data = SMBBA3 }, | 
 | 460 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8237), | 
 | 461 | 	  .driver_data = SMBBA3 }, | 
| Rudolf Marek | c243353 | 2006-09-03 22:35:21 +0200 | [diff] [blame] | 462 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8237A), | 
 | 463 | 	  .driver_data = SMBBA3 }, | 
| Jean Delvare | 0d227a7 | 2008-01-27 18:14:51 +0100 | [diff] [blame] | 464 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8237S), | 
 | 465 | 	  .driver_data = SMBBA3 }, | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 466 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8231_4), | 
 | 467 | 	  .driver_data = SMBBA1 }, | 
| Rudolf Marek | c243353 | 2006-09-03 22:35:21 +0200 | [diff] [blame] | 468 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_8251), | 
 | 469 | 	  .driver_data = SMBBA3 }, | 
| Jean Delvare | ab6a6ed | 2007-02-13 22:09:02 +0100 | [diff] [blame] | 470 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_CX700), | 
 | 471 | 	  .driver_data = SMBBA3 }, | 
| Rudolf Marek | b806a71 | 2008-10-14 17:30:03 +0200 | [diff] [blame] | 472 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_VX800), | 
 | 473 | 	  .driver_data = SMBBA3 }, | 
| Harald Welte | a231591 | 2009-06-15 18:01:49 +0200 | [diff] [blame] | 474 | 	{ PCI_DEVICE(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_VX855), | 
 | 475 | 	  .driver_data = SMBBA3 }, | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 476 | 	{ 0, } | 
 | 477 | }; | 
 | 478 |  | 
| Jean Delvare | 5f49ef8 | 2005-09-22 21:50:47 +0200 | [diff] [blame] | 479 | MODULE_DEVICE_TABLE(pci, vt596_ids); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 480 |  | 
 | 481 | static struct pci_driver vt596_driver = { | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 482 | 	.name		= "vt596_smbus", | 
 | 483 | 	.id_table	= vt596_ids, | 
 | 484 | 	.probe		= vt596_probe, | 
 | 485 | }; | 
 | 486 |  | 
 | 487 | static int __init i2c_vt596_init(void) | 
 | 488 | { | 
 | 489 | 	return pci_register_driver(&vt596_driver); | 
 | 490 | } | 
 | 491 |  | 
 | 492 |  | 
 | 493 | static void __exit i2c_vt596_exit(void) | 
 | 494 | { | 
 | 495 | 	pci_unregister_driver(&vt596_driver); | 
 | 496 | 	if (vt596_pdev != NULL) { | 
 | 497 | 		i2c_del_adapter(&vt596_adapter); | 
 | 498 | 		release_region(vt596_smba, 8); | 
 | 499 | 		pci_dev_put(vt596_pdev); | 
 | 500 | 		vt596_pdev = NULL; | 
 | 501 | 	} | 
 | 502 | } | 
 | 503 |  | 
| Jean Delvare | 8750197 | 2005-10-31 18:51:21 +0100 | [diff] [blame] | 504 | MODULE_AUTHOR("Kyosti Malkki <kmalkki@cc.hut.fi>, " | 
 | 505 | 	      "Mark D. Studebaker <mdsxyz123@yahoo.com> and " | 
 | 506 | 	      "Jean Delvare <khali@linux-fr.org>"); | 
| Linus Torvalds | 1da177e | 2005-04-16 15:20:36 -0700 | [diff] [blame] | 507 | MODULE_DESCRIPTION("vt82c596 SMBus driver"); | 
 | 508 | MODULE_LICENSE("GPL"); | 
 | 509 |  | 
 | 510 | module_init(i2c_vt596_init); | 
 | 511 | module_exit(i2c_vt596_exit); |