patch-2.3.34 linux/drivers/i2c/i2c-core.c
Next file: linux/drivers/i2c/i2c-dev.c
Previous file: linux/drivers/i2c/i2c-algo-pcf.c
Back to the patch index
Back to the overall index
- Lines: 1354
- Date:
Thu Dec 16 13:59:38 1999
- Orig file:
v2.3.33/linux/drivers/i2c/i2c-core.c
- Orig date:
Wed Dec 31 16:00:00 1969
diff -u --recursive --new-file v2.3.33/linux/drivers/i2c/i2c-core.c linux/drivers/i2c/i2c-core.c
@@ -0,0 +1,1353 @@
+/* i2c-core.c - a device driver for the iic-bus interface */
+/* ------------------------------------------------------------------------- */
+/* Copyright (C) 1995-99 Simon G. Vogl
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */
+/* ------------------------------------------------------------------------- */
+#define RCSID "$Id: i2c-core.c,v 1.42 1999/11/30 20:06:42 frodo Exp $"
+/* ------------------------------------------------------------------------- */
+
+/* With some changes from Ky�sti M�lkki <kmalkki@cc.hut.fi>.
+ All SMBus-related things are written by Frodo Looijaard <frodol@dds.nl> */
+
+#include <linux/module.h>
+#include <linux/kernel.h>
+#include <linux/errno.h>
+#include <linux/malloc.h>
+#include <linux/proc_fs.h>
+#include <linux/config.h>
+
+#include <linux/i2c.h>
+
+/* ----- compatibility stuff ----------------------------------------------- */
+
+/* 2.0.0 kernel compatibility */
+#if LINUX_VERSION_CODE < 0x020100
+#define MODULE_AUTHOR(noone)
+#define MODULE_DESCRIPTION(none)
+#define MODULE_PARM(no,param)
+#define MODULE_PARM_DESC(no,description)
+#define EXPORT_SYMBOL(noexport)
+#define EXPORT_NO_SYMBOLS
+#endif
+
+#include <linux/version.h>
+#ifndef KERNEL_VERSION
+#define KERNEL_VERSION(a,b,c) (((a) << 16) | ((b) << 8) | (c))
+#endif
+
+#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,53)
+#include <linux/init.h>
+#else
+#define __init
+#endif
+
+#if LINUX_VERSION_CODE < KERNEL_VERSION(2,3,1)
+#define init_MUTEX(s) do { *(s) = MUTEX; } while(0)
+#endif
+
+#if (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,4))
+#define copy_from_user memcpy_fromfs
+#define copy_to_user memcpy_tofs
+#else
+#include <asm/uaccess.h>
+#endif
+
+/* ----- global defines ---------------------------------------------------- */
+
+/* exclusive access to the bus */
+#define I2C_LOCK(adap) down(&adap->lock)
+#define I2C_UNLOCK(adap) up(&adap->lock)
+
+#define ADAP_LOCK() down(&adap_lock)
+#define ADAP_UNLOCK() up(&adap_lock)
+
+#define DRV_LOCK() down(&driver_lock)
+#define DRV_UNLOCK() up(&driver_lock)
+
+#define DEB(x) if (i2c_debug>=1) x;
+#define DEB2(x) if (i2c_debug>=2) x;
+
+/* ----- global variables -------------------------------------------------- */
+
+/**** lock for writing to global variables: the adapter & driver list */
+struct semaphore adap_lock;
+struct semaphore driver_lock;
+
+/**** adapter list */
+static struct i2c_adapter *adapters[I2C_ADAP_MAX];
+static int adap_count;
+
+/**** drivers list */
+static struct i2c_driver *drivers[I2C_DRIVER_MAX];
+static int driver_count;
+
+/**** debug level */
+static int i2c_debug=1;
+static void i2c_dummy_adapter(struct i2c_adapter *adapter);
+static void i2c_dummy_client(struct i2c_client *client);
+
+/* ---------------------------------------------------
+ * /proc entry declarations
+ *----------------------------------------------------
+ */
+
+/* Note that quite some things changed within the 2.1 kernel series.
+ Some things below are somewhat difficult to read because of this. */
+
+#ifdef CONFIG_PROC_FS
+
+static int i2cproc_init(void);
+static int i2cproc_cleanup(void);
+
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,58)) && \
+ (LINUX_VERSION_CODE <= KERNEL_VERSION(2,3,27))
+static void monitor_bus_i2c(struct inode *inode, int fill);
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,58)) */
+
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+
+static ssize_t i2cproc_bus_read(struct file * file, char * buf,size_t count,
+ loff_t *ppos);
+static int read_bus_i2c(char *buf, char **start, off_t offset, int len,
+ int *eof , void *private);
+
+#else /* (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,29)) */
+
+static int i2cproc_bus_read(struct inode * inode, struct file * file,
+ char * buf, int count);
+static int read_bus_i2c(char *buf, char **start, off_t offset, int len,
+ int unused);
+
+static struct proc_dir_entry proc_bus_dir =
+ {
+ /* low_ino */ 0, /* Set by proc_register_dynamic */
+ /* namelen */ 3,
+ /* name */ "bus",
+ /* mode */ S_IRUGO | S_IXUGO | S_IFDIR,
+ /* nlink */ 2, /* Corrected by proc_register[_dynamic] */
+ /* uid */ 0,
+ /* gid */ 0,
+ /* size */ 0,
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,0,36))
+ /* ops */ &proc_dir_inode_operations,
+#endif
+ };
+
+static struct proc_dir_entry proc_bus_i2c_dir =
+ {
+ /* low_ino */ 0, /* Set by proc_register_dynamic */
+ /* namelen */ 3,
+ /* name */ "i2c",
+ /* mode */ S_IRUGO | S_IFREG,
+ /* nlink */ 1,
+ /* uid */ 0,
+ /* gid */ 0,
+ /* size */ 0,
+ /* ops */ NULL,
+ /* get_info */ &read_bus_i2c
+ };
+
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29)) */
+
+/* To implement the dynamic /proc/bus/i2c-? files, we need our own
+ implementation of the read hook */
+static struct file_operations i2cproc_operations = {
+ NULL,
+ i2cproc_bus_read,
+};
+
+static struct inode_operations i2cproc_inode_operations = {
+ &i2cproc_operations
+};
+
+static int i2cproc_initialized = 0;
+
+#else /* undef CONFIG_PROC_FS */
+
+#define i2cproc_init()
+#define i2cproc_cleanup()
+
+#endif /* CONFIG_PROC_FS */
+
+
+/* ---------------------------------------------------
+ * registering functions
+ * ---------------------------------------------------
+ */
+
+/* -----
+ * i2c_add_adapter is called from within the algorithm layer,
+ * when a new hw adapter registers. A new device is register to be
+ * available for clients.
+ */
+int i2c_add_adapter(struct i2c_adapter *adap)
+{
+ int i,j;
+
+ ADAP_LOCK();
+ for (i = 0; i < I2C_ADAP_MAX; i++)
+ if (NULL == adapters[i])
+ break;
+ if (I2C_ADAP_MAX == i) {
+ printk(KERN_WARNING
+ " i2c-core.o: register_adapter(%s) - enlarge I2C_ADAP_MAX.\n",
+ adap->name);
+ ADAP_UNLOCK();
+ return -ENOMEM;
+ }
+
+ adapters[i] = adap;
+ adap_count++;
+ ADAP_UNLOCK();
+
+ /* init data types */
+ init_MUTEX(&adap->lock);
+
+ i2c_dummy_adapter(adap); /* actually i2c_dummy->add_adapter */
+#ifdef CONFIG_PROC_FS
+
+ if (i2cproc_initialized) {
+ char name[8];
+ struct proc_dir_entry *proc_entry;
+#if (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,29))
+ int res;
+#endif
+
+ sprintf(name,"i2c-%d", i);
+
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+ proc_entry = create_proc_entry(name,0,proc_bus);
+ if (! proc_entry) {
+ printk("i2c-core.o: Could not create /proc/bus/%s\n",
+ name);
+ return -ENOENT;
+ }
+ proc_entry->ops = &i2cproc_inode_operations;
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,27))
+ proc_entry->owner = THIS_MODULE;
+#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,58))
+ proc_entry->fill_inode = &monitor_bus_i2c;
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,58)) */
+#else /* (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,29)) */
+ adap->proc_entry = NULL;
+ if (!(proc_entry = kmalloc(sizeof(struct proc_dir_entry)+
+ strlen(name)+1, GFP_KERNEL))) {
+ printk("i2c-core.o: Out of memory!\n");
+ return -ENOMEM;
+ }
+ memset(proc_entry,0,sizeof(struct proc_dir_entry));
+ proc_entry->namelen = strlen(name);
+ proc_entry->name = (char *) (proc_entry + 1);
+ proc_entry->mode = S_IRUGO | S_IFREG;
+ proc_entry->nlink = 1;
+ proc_entry->ops = &i2cproc_inode_operations;
+
+ /* Nasty stuff to keep GCC satisfied */
+ {
+ char *procname;
+ (const char *) procname = proc_entry->name;
+ strcpy (procname,name);
+ }
+
+ if ((res = proc_register_dynamic(&proc_bus_dir, proc_entry))) {
+ printk("i2c-core.o: Could not create %s.\n",name);
+ kfree(proc_entry);
+ return res;
+ }
+
+ adap->proc_entry = proc_entry;
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29)) */
+
+ adap->inode = proc_entry->low_ino;
+ }
+
+#endif /* def CONFIG_PROC_FS */
+
+ /* inform drivers of new adapters */
+ DRV_LOCK();
+ for (j=0;j<I2C_DRIVER_MAX;j++)
+ if (drivers[j]!=NULL && drivers[j]->flags&I2C_DF_NOTIFY)
+ drivers[j]->attach_adapter(adap);
+ DRV_UNLOCK();
+
+ DEB(printk("i2c-core.o: adapter %s registered as adapter %d.\n",adap->name,i));
+
+ return 0;
+}
+
+
+int i2c_del_adapter(struct i2c_adapter *adap)
+{
+ int i,j;
+ ADAP_LOCK();
+ for (i = 0; i < I2C_ADAP_MAX; i++)
+ if (adap == adapters[i])
+ break;
+ if (I2C_ADAP_MAX == i) {
+ printk( " i2c-core.o: unregister_adapter adap [%s] not found.\n",
+ adap->name);
+ ADAP_UNLOCK();
+ return -ENODEV;
+ }
+
+ i2c_dummy_adapter(adap); /* actually i2c_dummy->del_adapter */
+#ifdef CONFIG_PROC_FS
+ if (i2cproc_initialized) {
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+ char name[8];
+ sprintf(name,"i2c-%d", i);
+ remove_proc_entry(name,proc_bus);
+#else /* (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,29)) */
+ int res;
+ if (adapters[i]->proc_entry) {
+ if ((res = proc_unregister(&proc_bus_dir,
+ adapters[i]->proc_entry->low_ino))) {
+ printk("i2c-core.o: Deregistration of /proc "
+ "entry failed\n");
+ ADAP_UNLOCK();
+ return res;
+ }
+ kfree(adapters[i]->proc_entry);
+ }
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29)) */
+ }
+#endif /* def CONFIG_PROC_FS */
+
+ /* detach any active clients */
+ for (j=0;j<I2C_CLIENT_MAX;j++) {
+ struct i2c_client *client = adap->clients[j];
+ if ( (client!=NULL)
+ /* && (client->driver->flags & I2C_DF_NOTIFY) */ )
+ /* detaching devices is unconditional of the set notify
+ * flag, as _all_ clients that reside on the adapter
+ * must be deleted, as this would cause invalid states.
+ */
+ client->driver->detach_client(client);
+ /* i2c_detach_client(client); --- frodo */
+ }
+ /* all done, now unregister */
+ adapters[i] = NULL;
+ adap_count--;
+
+ ADAP_UNLOCK();
+ DEB(printk("i2c-core.o: adapter unregistered: %s\n",adap->name));
+ return 0;
+}
+
+
+/* -----
+ * What follows is the "upwards" interface: commands for talking to clients,
+ * which implement the functions to access the physical information of the
+ * chips.
+ */
+
+int i2c_add_driver(struct i2c_driver *driver)
+{
+ int i,j;
+ DRV_LOCK();
+ for (i = 0; i < I2C_DRIVER_MAX; i++)
+ if (NULL == drivers[i])
+ break;
+ if (I2C_DRIVER_MAX == i) {
+ printk(KERN_WARNING
+ " i2c-core.o: register_driver(%s) - enlarge I2C_DRIVER_MAX.\n",
+ driver->name);
+ DRV_UNLOCK();
+ return -ENOMEM;
+ }
+
+ drivers[i] = driver;
+ driver_count++;
+
+ DRV_UNLOCK(); /* driver was successfully added */
+
+ DEB(printk("i2c-core.o: driver %s registered.\n",driver->name));
+
+ /* Notify all existing adapters and clients to dummy driver */
+ ADAP_LOCK();
+ if (driver->flags&I2C_DF_DUMMY) {
+ for (i=0; i<I2C_ADAP_MAX; i++) {
+ if (adapters[i]) {
+ driver->attach_adapter(adapters[i]);
+ for (j=0; j<I2C_CLIENT_MAX; j++)
+ if (adapters[i]->clients[j])
+ driver->detach_client(adapters[i]->clients[j]);
+ }
+ }
+ ADAP_UNLOCK();
+ return 0;
+ }
+
+ /* now look for instances of driver on our adapters
+ */
+ if ( driver->flags&I2C_DF_NOTIFY ) {
+ for (i=0;i<I2C_ADAP_MAX;i++)
+ if (adapters[i]!=NULL)
+ driver->attach_adapter(adapters[i]);
+ }
+ ADAP_UNLOCK();
+ return 0;
+}
+
+int i2c_del_driver(struct i2c_driver *driver)
+{
+ int i,j,k;
+
+ DRV_LOCK();
+ for (i = 0; i < I2C_DRIVER_MAX; i++)
+ if (driver == drivers[i])
+ break;
+ if (I2C_DRIVER_MAX == i) {
+ printk(KERN_WARNING " i2c-core.o: unregister_driver: [%s] not found\n",
+ driver->name);
+ DRV_UNLOCK();
+ return -ENODEV;
+ }
+ /* Have a look at each adapter, if clients of this driver are still
+ * attached. If so, detach them to be able to kill the driver
+ * afterwards.
+ */
+ DEB2(printk("i2c-core.o: unregister_driver - looking for clients.\n"));
+ /* removing clients does not depend on the notify flag, else
+ * invalid operation might (will!) result, when using stale client
+ * pointers.
+ */
+ ADAP_LOCK(); /* should be moved inside the if statement... */
+ if ((driver->flags&I2C_DF_DUMMY)==0)
+ for (k=0;k<I2C_ADAP_MAX;k++) {
+ struct i2c_adapter *adap = adapters[k];
+ if (adap == NULL) /* skip empty entries. */
+ continue;
+ DEB2(printk("i2c-core.o: examining adapter %s:\n",adap->name));
+ for (j=0;j<I2C_CLIENT_MAX;j++) {
+ struct i2c_client *client = adap->clients[j];
+ if (client != NULL && client->driver == driver) {
+ DEB2(printk("i2c-core.o: detaching client %s:\n",
+ client->name));
+ /*i2c_detach_client(client);*/
+ driver->detach_client(client);
+ }
+ }
+ }
+ ADAP_UNLOCK();
+ drivers[i] = NULL;
+ driver_count--;
+ DRV_UNLOCK();
+
+ DEB(printk("i2c-core.o: driver unregistered: %s\n",driver->name));
+ return 0;
+}
+
+
+int i2c_attach_client(struct i2c_client *client)
+{
+ struct i2c_adapter *adapter = client->adapter;
+ int i;
+
+ for (i = 0; i < I2C_CLIENT_MAX; i++)
+ if (NULL == adapter->clients[i])
+ break;
+ if (I2C_CLIENT_MAX == i) {
+ printk(KERN_WARNING
+ " i2c-core.o: attach_client(%s) - enlarge I2C_CLIENT_MAX.\n",
+ client->name);
+ return -ENOMEM;
+ }
+
+ adapter->clients[i] = client;
+ adapter->client_count++;
+ i2c_dummy_client(client);
+
+ if (adapter->client_register != NULL)
+ adapter->client_register(client);
+ DEB(printk("i2c-core.o: client [%s] registered to adapter [%s](pos. %d).\n",
+ client->name, adapter->name,i));
+ return 0;
+}
+
+
+int i2c_detach_client(struct i2c_client *client)
+{
+ struct i2c_adapter *adapter = client->adapter;
+ int i;
+
+ for (i = 0; i < I2C_CLIENT_MAX; i++)
+ if (client == adapter->clients[i])
+ break;
+ if (I2C_CLIENT_MAX == i) {
+ printk(KERN_WARNING " i2c-core.o: unregister_client [%s] not found\n",
+ client->name);
+ return -ENODEV;
+ }
+
+ if (adapter->client_unregister != NULL)
+ adapter->client_unregister(client);
+ /* client->driver->detach_client(client);*/
+
+ adapter->clients[i] = NULL;
+ adapter->client_count--;
+ i2c_dummy_client(client);
+
+ DEB(printk("i2c-core.o: client [%s] unregistered.\n",client->name));
+ return 0;
+}
+
+void i2c_inc_use_client(struct i2c_client *client)
+{
+
+ if (client->driver->inc_use != NULL)
+ client->driver->inc_use(client);
+
+ if (client->adapter->inc_use != NULL)
+ client->adapter->inc_use(client->adapter);
+}
+
+void i2c_dec_use_client(struct i2c_client *client)
+{
+
+ if (client->driver->dec_use != NULL)
+ client->driver->dec_use(client);
+
+ if (client->adapter->dec_use != NULL)
+ client->adapter->dec_use(client->adapter);
+}
+
+/* ----------------------------------------------------
+ * The /proc functions
+ * ----------------------------------------------------
+ */
+
+#ifdef CONFIG_PROC_FS
+
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,58)) && \
+ (LINUX_VERSION_CODE <= KERNEL_VERSION(2,3,27))
+/* Monitor access to /proc/bus/i2c*; make unloading i2c-proc impossible
+ if some process still uses it or some file in it */
+void monitor_bus_i2c(struct inode *inode, int fill)
+{
+ if (fill)
+ MOD_INC_USE_COUNT;
+ else
+ MOD_DEC_USE_COUNT;
+}
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,58)) */
+
+/* This function generates the output for /proc/bus/i2c */
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+int read_bus_i2c(char *buf, char **start, off_t offset, int len, int *eof,
+ void *private)
+#else /* (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,29)) */
+int read_bus_i2c(char *buf, char **start, off_t offset, int len, int unused)
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29)) */
+{
+ int i;
+ int nr = 0;
+ /* Note that it is safe to write a `little' beyond len. Yes, really. */
+ for (i = 0; (i < I2C_ADAP_MAX) && (nr < len); i++)
+ if (adapters[i]) {
+ nr += sprintf(buf+nr, "i2c-%d\t", i);
+ if (adapters[i]->algo->smbus_xfer) {
+ if (adapters[i]->algo->master_xfer)
+ nr += sprintf(buf+nr,"smbus/i2c");
+ else
+ nr += sprintf(buf+nr,"smbus ");
+ } else if (adapters[i]->algo->master_xfer)
+ nr += sprintf(buf+nr,"i2c ");
+ else
+ nr += sprintf(buf+nr,"dummy ");
+ nr += sprintf(buf+nr,"\t%-32s\t%-32s\n",
+ adapters[i]->name,
+ adapters[i]->algo->name);
+ }
+ return nr;
+}
+
+/* This function generates the output for /proc/bus/i2c-? */
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+ssize_t i2cproc_bus_read(struct file * file, char * buf,size_t count,
+ loff_t *ppos)
+{
+ struct inode * inode = file->f_dentry->d_inode;
+#else (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,29))
+int i2cproc_bus_read(struct inode * inode, struct file * file,char * buf,
+ int count)
+{
+#endif (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+ char *kbuf;
+ struct i2c_client *client;
+ int i,j,len=0;
+
+ if (count < 0)
+ return -EINVAL;
+ if (count > 4000)
+ count = 4000;
+ for (i = 0; i < I2C_ADAP_MAX; i++)
+ if (adapters[i]->inode == inode->i_ino) {
+ /* We need a bit of slack in the kernel buffer; this makes the
+ sprintf safe. */
+ if (! (kbuf = kmalloc(count + 80,GFP_KERNEL)))
+ return -ENOMEM;
+ for (j = 0; j < I2C_CLIENT_MAX; j++)
+ if ((client = adapters[i]->clients[j]))
+ /* Filter out dummy clients */
+ if (client->driver->id != I2C_DRIVERID_I2CDEV)
+ len += sprintf(kbuf+len,"%02x\t%-32s\t%-32s\n",
+ client->addr,
+ client->name,client->driver->name);
+ if (file->f_pos+len > count)
+ len = count - file->f_pos;
+ len = len - file->f_pos;
+ if (len < 0)
+ len = 0;
+ copy_to_user (buf,kbuf+file->f_pos,len);
+ file->f_pos += len;
+ kfree(kbuf);
+ return len;
+ }
+ return -ENOENT;
+}
+
+int i2cproc_init(void)
+{
+
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+ struct proc_dir_entry *proc_bus_i2c;
+#else
+ int res;
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29)) */
+
+ i2cproc_initialized = 0;
+
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+ if (! proc_bus) {
+ printk("i2c-core.o: /proc/bus/ does not exist");
+ i2cproc_cleanup();
+ return -ENOENT;
+ }
+ proc_bus_i2c = create_proc_entry("i2c",0,proc_bus);
+ if (!proc_bus_i2c) {
+ printk("i2c-core.o: Could not create /proc/bus/i2c");
+ i2cproc_cleanup();
+ return -ENOENT;
+ }
+ proc_bus_i2c->read_proc = &read_bus_i2c;
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,3,27))
+ proc_bus_i2c->owner = THIS_MODULE;
+#elif (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,58))
+ proc_bus_i2c->fill_inode = &monitor_bus_i2c;
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29)) */
+ i2cproc_initialized += 2;
+#else /* (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,29)) */
+ /* In Linux 2.0.x, there is no /proc/bus! But I hope no other module
+ introduced it, or we are fucked. And 2.0.35 and earlier does not
+ export proc_dir_inode_operations, so we grab it from proc_net,
+ which also uses it. Not nice. */
+ proc_bus_dir.ops = proc_net.ops;
+ if ((res = proc_register_dynamic(&proc_root, &proc_bus_dir))) {
+ printk("i2c-core.o: Could not create /proc/bus/");
+ i2cproc_cleanup();
+ return res;
+ }
+ i2cproc_initialized ++;
+ if ((res = proc_register_dynamic(&proc_bus_dir, &proc_bus_i2c_dir))) {
+ printk("i2c-core.o: Could not create /proc/bus/i2c\n");
+ i2cproc_cleanup();
+ return res;
+ }
+ i2cproc_initialized ++;
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29)) */
+ return 0;
+}
+
+int i2cproc_cleanup(void)
+{
+
+ if (i2cproc_initialized >= 1) {
+#if (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29))
+ remove_proc_entry("i2c",proc_bus);
+ i2cproc_initialized -= 2;
+#else /* (LINUX_VERSION_CODE < KERNEL_VERSION(2,1,29)) */
+ int res;
+ if (i2cproc_initialized >= 2) {
+ if ((res = proc_unregister(&proc_bus_dir,
+ proc_bus_i2c_dir.low_ino))) {
+ printk("i2c-core.o: could not delete "
+ "/proc/bus/i2c, module not removed.");
+ return res;
+ }
+ i2cproc_initialized --;
+ }
+ if ((res = proc_unregister(&proc_root,proc_bus_dir.low_ino))) {
+ printk("i2c-core.o: could not delete /proc/bus/, "
+ "module not removed.");
+ return res;
+ }
+ i2cproc_initialized --;
+#endif /* (LINUX_VERSION_CODE >= KERNEL_VERSION(2,1,29)) */
+ }
+ return 0;
+}
+
+
+#endif /* def CONFIG_PROC_FS */
+
+/* ---------------------------------------------------
+ * dummy driver notification
+ * ---------------------------------------------------
+ */
+
+static void i2c_dummy_adapter(struct i2c_adapter *adap)
+{
+ int i;
+ for (i=0; i<I2C_DRIVER_MAX; i++)
+ if (drivers[i] && (drivers[i]->flags & I2C_DF_DUMMY))
+ drivers[i]->attach_adapter(adap);
+}
+
+static void i2c_dummy_client(struct i2c_client *client)
+{
+ int i;
+ for (i=0; i<I2C_DRIVER_MAX; i++)
+ if (drivers[i] && (drivers[i]->flags & I2C_DF_DUMMY))
+ drivers[i]->detach_client(client);
+}
+
+
+/* ----------------------------------------------------
+ * the functional interface to the i2c busses.
+ * ----------------------------------------------------
+ */
+
+int i2c_transfer(struct i2c_adapter * adap, struct i2c_msg msgs[],int num)
+{
+ int ret;
+
+ if (adap->algo->master_xfer) {
+ DEB2(printk("i2c-core.o: master_xfer: %s with %d msgs.\n",adap->name,num));
+
+ I2C_LOCK(adap);
+ ret = adap->algo->master_xfer(adap,msgs,num);
+ I2C_UNLOCK(adap);
+
+ return ret;
+ } else {
+ printk("i2c-core.o: I2C adapter %04x: I2C level transfers not supported\n",
+ adap->id);
+ return -ENOSYS;
+ }
+}
+
+int i2c_master_send(struct i2c_client *client,const char *buf ,int count)
+{
+ int ret;
+ struct i2c_adapter *adap=client->adapter;
+ struct i2c_msg msg;
+
+ if (client->adapter->algo->master_xfer) {
+ msg.addr = client->addr;
+ msg.flags = client->flags & I2C_M_TEN;
+ msg.len = count;
+ (const char *)msg.buf = buf;
+
+ DEB2(printk("i2c-core.o: master_send: writing %d bytes on %s.\n",
+ count,client->adapter->name));
+
+ I2C_LOCK(adap);
+ ret = adap->algo->master_xfer(adap,&msg,1);
+ I2C_UNLOCK(adap);
+
+ /* if everything went ok (i.e. 1 msg transmitted), return #bytes
+ * transmitted, else error code.
+ */
+ return (ret == 1 )? count : ret;
+ } else {
+ printk("i2c-core.o: I2C adapter %04x: I2C level transfers not supported\n",
+ client->adapter->id);
+ return -ENOSYS;
+ }
+}
+
+int i2c_master_recv(struct i2c_client *client, char *buf ,int count)
+{
+ struct i2c_adapter *adap=client->adapter;
+ struct i2c_msg msg;
+ int ret;
+ if (client->adapter->algo->master_xfer) {
+ msg.addr = client->addr;
+ msg.flags = client->flags & I2C_M_TEN;
+ msg.flags |= I2C_M_RD;
+ msg.len = count;
+ msg.buf = buf;
+
+ DEB2(printk("i2c-core.o: master_recv: reading %d bytes on %s.\n",
+ count,client->adapter->name));
+
+ I2C_LOCK(adap);
+ ret = adap->algo->master_xfer(adap,&msg,1);
+ I2C_UNLOCK(adap);
+
+ DEB2(printk("i2c-core.o: master_recv: return:%d (count:%d, addr:0x%02x)\n",
+ ret, count, client->addr));
+
+ /* if everything went ok (i.e. 1 msg transmitted), return #bytes
+ * transmitted, else error code.
+ */
+ return (ret == 1 )? count : ret;
+ } else {
+ printk("i2c-core.o: I2C adapter %04x: I2C level transfers not supported\n",
+ client->adapter->id);
+ return -ENOSYS;
+ }
+}
+
+
+int i2c_control(struct i2c_client *client,
+ unsigned int cmd, unsigned long arg)
+{
+ int ret = 0;
+ struct i2c_adapter *adap = client->adapter;
+
+ DEB2(printk("i2c-core.o: i2c ioctl, cmd: 0x%x, arg: %#lx\n", cmd, arg));
+ switch ( cmd ) {
+ case I2C_RETRIES:
+ adap->retries = arg;
+ break;
+ case I2C_TIMEOUT:
+ adap->timeout = arg;
+ break;
+ default:
+ if (adap->algo->algo_control!=NULL)
+ ret = adap->algo->algo_control(adap,cmd,arg);
+ }
+ return ret;
+}
+
+/* ----------------------------------------------------
+ * the i2c address scanning function
+ * Will not work for 10-bit addresses!
+ * ----------------------------------------------------
+ */
+int i2c_probe(struct i2c_adapter *adapter,
+ struct i2c_client_address_data *address_data,
+ i2c_client_found_addr_proc *found_proc)
+{
+ int addr,i,found,err;
+ int adap_id = i2c_adapter_id(adapter);
+
+ /* Forget it if we can't probe using SMBUS_QUICK */
+ if (! i2c_check_functionality(adapter,I2C_FUNC_SMBUS_QUICK))
+ return -1;
+
+ for (addr = 0x00;
+ addr <= 0x7f;
+ addr++) {
+
+ /* If it is in one of the force entries, we don't do any detection
+ at all */
+ found = 0;
+
+ for (i = 0;
+ !found && (address_data->force[i] != I2C_CLIENT_END);
+ i += 3) {
+ if (((adap_id == address_data->force[i]) ||
+ (address_data->force[i] == ANY_I2C_BUS)) &&
+ (addr == address_data->force[i+1])) {
+ DEB2(printk("i2c-core.o: found force parameter for adapter %d, addr %04x\n",
+ adap_id,addr));
+ if ((err = found_proc(adapter,addr,0,0)))
+ return err;
+ found = 1;
+ }
+ }
+ if (found)
+ continue;
+
+ /* If this address is in one of the ignores, we can forget about it
+ right now */
+ for (i = 0;
+ !found && (address_data->ignore[i] != I2C_CLIENT_END);
+ i += 2) {
+ if (((adap_id == address_data->ignore[i]) ||
+ ((address_data->ignore[i] == ANY_I2C_BUS))) &&
+ (addr == address_data->ignore[i+1])) {
+ DEB2(printk("i2c-core.o: found ignore parameter for adapter %d, "
+ "addr %04x\n", adap_id ,addr));
+ found = 1;
+ }
+ }
+ for (i = 0;
+ !found && (address_data->ignore_range[i] != I2C_CLIENT_END);
+ i += 3) {
+ if (((adap_id == address_data->ignore_range[i]) ||
+ ((address_data->ignore_range[i]==ANY_I2C_BUS))) &&
+ (addr >= address_data->ignore_range[i+1]) &&
+ (addr <= address_data->ignore_range[i+2])) {
+ DEB2(printk("i2c-core.o: found ignore_range parameter for adapter %d, "
+ "addr %04x\n", adap_id,addr));
+ found = 1;
+ }
+ }
+ if (found)
+ continue;
+
+ /* Now, we will do a detection, but only if it is in the normal or
+ probe entries */
+ for (i = 0;
+ !found && (address_data->normal_i2c[i] != I2C_CLIENT_END);
+ i += 1) {
+ if (addr == address_data->normal_i2c[i]) {
+ found = 1;
+ DEB2(printk("i2c-core.o: found normal i2c entry for adapter %d, "
+ "addr %02x", adap_id,addr));
+ }
+ }
+
+ for (i = 0;
+ !found && (address_data->normal_i2c_range[i] != I2C_CLIENT_END);
+ i += 2) {
+ if ((addr >= address_data->normal_i2c_range[i]) &&
+ (addr <= address_data->normal_i2c_range[i+1])) {
+ found = 1;
+ DEB2(printk("i2c-core.o: found normal i2c_range entry for adapter %d, "
+ "addr %04x\n", adap_id,addr));
+ }
+ }
+
+ for (i = 0;
+ !found && (address_data->probe[i] != I2C_CLIENT_END);
+ i += 2) {
+ if (((adap_id == address_data->probe[i]) ||
+ ((address_data->probe[i] == ANY_I2C_BUS))) &&
+ (addr == address_data->probe[i+1])) {
+ found = 1;
+ DEB2(printk("i2c-core.o: found probe parameter for adapter %d, "
+ "addr %04x\n", adap_id,addr));
+ }
+ }
+ for (i = 0;
+ !found && (address_data->probe_range[i] != I2C_CLIENT_END);
+ i += 3) {
+ if (((adap_id == address_data->probe_range[i]) ||
+ (address_data->probe_range[i] == ANY_I2C_BUS)) &&
+ (addr >= address_data->probe_range[i+1]) &&
+ (addr <= address_data->probe_range[i+2])) {
+ found = 1;
+ DEB2(printk("i2c-core.o: found probe_range parameter for adapter %d, "
+ "addr %04x\n", adap_id,addr));
+ }
+ }
+ if (!found)
+ continue;
+
+ /* OK, so we really should examine this address. First check
+ whether there is some client here at all! */
+ if (i2c_smbus_xfer(adapter,addr,0,0,0,I2C_SMBUS_QUICK,NULL) >= 0)
+ if ((err = found_proc(adapter,addr,0,-1)))
+ return err;
+ }
+ return 0;
+}
+
+/* +++ frodo
+ * return id number for a specific adapter
+ */
+int i2c_adapter_id(struct i2c_adapter *adap)
+{
+ int i;
+ for (i = 0; i < I2C_ADAP_MAX; i++)
+ if (adap == adapters[i])
+ return i;
+ return -1;
+}
+
+/* The SMBus parts */
+
+extern s32 i2c_smbus_write_quick(struct i2c_client * client, u8 value)
+{
+ return i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ value,0,I2C_SMBUS_QUICK,NULL);
+}
+
+extern s32 i2c_smbus_read_byte(struct i2c_client * client)
+{
+ union i2c_smbus_data data;
+ if (i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_READ,0,I2C_SMBUS_BYTE, &data))
+ return -1;
+ else
+ return 0x0FF & data.byte;
+}
+
+extern s32 i2c_smbus_write_byte(struct i2c_client * client, u8 value)
+{
+ return i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_WRITE,value, I2C_SMBUS_BYTE,NULL);
+}
+
+extern s32 i2c_smbus_read_byte_data(struct i2c_client * client, u8 command)
+{
+ union i2c_smbus_data data;
+ if (i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_READ,command, I2C_SMBUS_BYTE_DATA,&data))
+ return -1;
+ else
+ return 0x0FF & data.byte;
+}
+
+extern s32 i2c_smbus_write_byte_data(struct i2c_client * client,
+ u8 command, u8 value)
+{
+ union i2c_smbus_data data;
+ data.byte = value;
+ return i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_WRITE,command,
+ I2C_SMBUS_BYTE_DATA,&data);
+}
+
+extern s32 i2c_smbus_read_word_data(struct i2c_client * client, u8 command)
+{
+ union i2c_smbus_data data;
+ if (i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_READ,command, I2C_SMBUS_WORD_DATA, &data))
+ return -1;
+ else
+ return 0x0FFFF & data.word;
+}
+
+extern s32 i2c_smbus_write_word_data(struct i2c_client * client,
+ u8 command, u16 value)
+{
+ union i2c_smbus_data data;
+ data.word = value;
+ return i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_WRITE,command,
+ I2C_SMBUS_WORD_DATA,&data);
+}
+
+extern s32 i2c_smbus_process_call(struct i2c_client * client,
+ u8 command, u16 value)
+{
+ union i2c_smbus_data data;
+ data.word = value;
+ if (i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_WRITE,command,
+ I2C_SMBUS_PROC_CALL, &data))
+ return -1;
+ else
+ return 0x0FFFF & data.word;
+}
+
+/* Returns the number of read bytes */
+extern s32 i2c_smbus_read_block_data(struct i2c_client * client,
+ u8 command, u8 *values)
+{
+ union i2c_smbus_data data;
+ int i;
+ if (i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_READ,command,
+ I2C_SMBUS_BLOCK_DATA,&data))
+ return -1;
+ else {
+ for (i = 1; i <= data.block[0]; i++)
+ values[i-1] = data.block[i];
+ return data.block[0];
+ }
+}
+
+extern s32 i2c_smbus_write_block_data(struct i2c_client * client,
+ u8 command, u8 length, u8 *values)
+{
+ union i2c_smbus_data data;
+ int i;
+ if (length > 32)
+ length = 32;
+ for (i = 1; i <= length; i++)
+ data.block[i] = values[i-1];
+ data.block[0] = length;
+ return i2c_smbus_xfer(client->adapter,client->addr,client->flags,
+ I2C_SMBUS_WRITE,command,
+ I2C_SMBUS_BLOCK_DATA,&data);
+}
+
+/* Simulate a SMBus command using the i2c protocol
+ No checking of parameters is done! */
+static s32 i2c_smbus_xfer_emulated(struct i2c_adapter * adapter, u16 addr,
+ unsigned short flags,
+ char read_write, u8 command, int size,
+ union i2c_smbus_data * data)
+{
+ /* So we need to generate a series of msgs. In the case of writing, we
+ need to use only one message; when reading, we need two. We initialize
+ most things with sane defaults, to keep the code below somewhat
+ simpler. */
+ unsigned char msgbuf0[33];
+ unsigned char msgbuf1[33];
+ int num = read_write == I2C_SMBUS_READ?2:1;
+ struct i2c_msg msg[2] = { { addr, flags, 1, msgbuf0 },
+ { addr, flags | I2C_M_RD, 0, msgbuf1 }
+ };
+ int i;
+
+ msgbuf0[0] = command;
+ switch(size) {
+ case I2C_SMBUS_QUICK:
+ msg[0].len = 0;
+ /* Special case: The read/write field is used as data */
+ msg[0].flags = flags | (read_write==I2C_SMBUS_READ)?I2C_M_RD:0;
+ num = 1;
+ break;
+ case I2C_SMBUS_BYTE:
+ if (read_write == I2C_SMBUS_READ) {
+ /* Special case: only a read! */
+ msg[0].flags = I2C_M_RD | flags;
+ num = 1;
+ }
+ break;
+ case I2C_SMBUS_BYTE_DATA:
+ if (read_write == I2C_SMBUS_READ)
+ msg[1].len = 1;
+ else {
+ msg[0].len = 2;
+ msgbuf0[1] = data->byte;
+ }
+ break;
+ case I2C_SMBUS_WORD_DATA:
+ if (read_write == I2C_SMBUS_READ)
+ msg[1].len = 2;
+ else {
+ msg[0].len=3;
+ msgbuf0[1] = data->word & 0xff;
+ msgbuf0[2] = (data->word >> 8) & 0xff;
+ }
+ break;
+ case I2C_SMBUS_PROC_CALL:
+ num = 2; /* Special case */
+ msg[0].len = 3;
+ msg[1].len = 2;
+ msgbuf0[1] = data->word & 0xff;
+ msgbuf0[2] = (data->word >> 8) & 0xff;
+ break;
+ case I2C_SMBUS_BLOCK_DATA:
+ if (read_write == I2C_SMBUS_READ) {
+ printk("i2c-core.o: Block read not supported under "
+ "I2C emulation!\n");
+ return -1;
+ } else {
+ msg[1].len = data->block[0] + 1;
+ if (msg[1].len > 32) {
+ printk("i2c-core.o: smbus_access called with "
+ "invalid block write size (%d)\n",
+ msg[1].len);
+ return -1;
+ }
+ for (i = 1; i <= msg[1].len; i++)
+ msgbuf0[i] = data->block[i];
+ }
+ break;
+ default:
+ printk("i2c-core.o: smbus_access called with invalid size (%d)\n",
+ size);
+ return -1;
+ }
+
+ if (i2c_transfer(adapter, msg, num) < 0)
+ return -1;
+
+ if (read_write == I2C_SMBUS_READ)
+ switch(size) {
+ case I2C_SMBUS_BYTE:
+ data->byte = msgbuf0[0];
+ break;
+ case I2C_SMBUS_BYTE_DATA:
+ data->byte = msgbuf1[0];
+ break;
+ case I2C_SMBUS_WORD_DATA:
+ case I2C_SMBUS_PROC_CALL:
+ data->word = msgbuf1[0] | (msgbuf1[1] << 8);
+ break;
+ }
+ return 0;
+}
+
+
+s32 i2c_smbus_xfer(struct i2c_adapter * adapter, u16 addr, unsigned short flags,
+ char read_write, u8 command, int size,
+ union i2c_smbus_data * data)
+{
+ s32 res;
+ flags = flags & I2C_M_TEN;
+ if (adapter->algo->smbus_xfer) {
+ I2C_LOCK(adapter);
+ res = adapter->algo->smbus_xfer(adapter,addr,flags,read_write,
+ command,size,data);
+ I2C_UNLOCK(adapter);
+ } else
+ res = i2c_smbus_xfer_emulated(adapter,addr,flags,read_write,
+ command,size,data);
+ return res;
+}
+
+
+/* You should always define `functionality'; the 'else' is just for
+ backward compatibility. */
+u32 i2c_get_functionality (struct i2c_adapter *adap)
+{
+ if (adap->algo->functionality)
+ return adap->algo->functionality(adap);
+ else
+ return 0xffffffff;
+}
+
+int i2c_check_functionality (struct i2c_adapter *adap, u32 func)
+{
+ u32 adap_func = i2c_get_functionality (adap);
+ return (func & adap_func) == func;
+}
+
+
+static int __init i2c_init(void)
+{
+ printk("i2c-core.o: i2c core module\n");
+ memset(adapters,0,sizeof(adapters));
+ memset(drivers,0,sizeof(drivers));
+ adap_count=0;
+ driver_count=0;
+
+ init_MUTEX(&adap_lock);
+ init_MUTEX(&driver_lock);
+
+ i2cproc_init();
+
+ return 0;
+}
+
+#ifndef MODULE
+#ifdef CONFIG_I2C_CHARDEV
+ extern int i2c_dev_init(void);
+#endif
+#ifdef CONFIG_I2C_ALGOBIT
+ extern int algo_bit_init(void);
+#endif
+#ifdef CONFIG_I2C_BITLP
+ extern int bitlp_init(void);
+#endif
+#ifdef CONFIG_I2C_BITELV
+ extern int bitelv_init(void);
+#endif
+#ifdef CONFIG_I2C_BITVELLE
+ extern int bitvelle_init(void);
+#endif
+#ifdef CONFIG_I2C_BITVIA
+ extern int bitvia_init(void);
+#endif
+
+#ifdef CONFIG_I2C_ALGOPCF
+ extern int algo_pcf_init(void);
+#endif
+#ifdef CONFIG_I2C_PCFISA
+ extern int pcfisa_init(void);
+#endif
+
+/* This is needed for automatic patch generation: sensors code starts here */
+/* This is needed for automatic patch generation: sensors code ends here */
+
+int __init i2c_init_all(void)
+{
+ /* --------------------- global ----- */
+ i2c_init();
+
+#ifdef CONFIG_I2C_CHARDEV
+ i2c_dev_init();
+#endif
+ /* --------------------- bit -------- */
+#ifdef CONFIG_I2C_ALGOBIT
+ i2c_algo_bit_init();
+#endif
+#ifdef CONFIG_I2C_PHILIPSPAR
+ i2c_bitlp_init();
+#endif
+#ifdef CONFIG_I2C_ELV
+ i2c_bitelv_init();
+#endif
+#ifdef CONFIG_I2C_VELLEMAN
+ i2c_bitvelle_init();
+#endif
+
+ /* --------------------- pcf -------- */
+#ifdef CONFIG_I2C_ALGOPCF
+ i2c_algo_pcf_init();
+#endif
+#ifdef CONFIG_I2C_ELEKTOR
+ i2c_pcfisa_init();
+#endif
+/* This is needed for automatic patch generation: sensors code starts here */
+/* This is needed for automatic patch generation: sensors code ends here */
+
+ return 0;
+}
+
+#endif
+
+
+
+EXPORT_SYMBOL(i2c_add_adapter);
+EXPORT_SYMBOL(i2c_del_adapter);
+EXPORT_SYMBOL(i2c_add_driver);
+EXPORT_SYMBOL(i2c_del_driver);
+EXPORT_SYMBOL(i2c_attach_client);
+EXPORT_SYMBOL(i2c_detach_client);
+EXPORT_SYMBOL(i2c_inc_use_client);
+EXPORT_SYMBOL(i2c_dec_use_client);
+
+
+EXPORT_SYMBOL(i2c_master_send);
+EXPORT_SYMBOL(i2c_master_recv);
+EXPORT_SYMBOL(i2c_control);
+EXPORT_SYMBOL(i2c_transfer);
+EXPORT_SYMBOL(i2c_adapter_id);
+EXPORT_SYMBOL(i2c_probe);
+
+EXPORT_SYMBOL(i2c_smbus_xfer);
+EXPORT_SYMBOL(i2c_smbus_write_quick);
+EXPORT_SYMBOL(i2c_smbus_read_byte);
+EXPORT_SYMBOL(i2c_smbus_write_byte);
+EXPORT_SYMBOL(i2c_smbus_read_byte_data);
+EXPORT_SYMBOL(i2c_smbus_write_byte_data);
+EXPORT_SYMBOL(i2c_smbus_read_word_data);
+EXPORT_SYMBOL(i2c_smbus_write_word_data);
+EXPORT_SYMBOL(i2c_smbus_process_call);
+EXPORT_SYMBOL(i2c_smbus_read_block_data);
+EXPORT_SYMBOL(i2c_smbus_write_block_data);
+
+EXPORT_SYMBOL(i2c_get_functionality);
+EXPORT_SYMBOL(i2c_check_functionality);
+
+#ifdef MODULE
+MODULE_AUTHOR("Simon G. Vogl <simon@tk.uni-linz.ac.at>");
+MODULE_DESCRIPTION("I2C-Bus main module");
+MODULE_PARM(i2c_debug, "i");
+MODULE_PARM_DESC(i2c_debug,"debug level");
+
+int init_module(void)
+{
+ return i2c_init();
+}
+
+void cleanup_module(void)
+{
+ i2cproc_cleanup();
+}
+#endif
FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen (who was at: slshen@lbl.gov)