blob: c3e1d3e888d7b25c20d90ae4a7ecb8f5be420b98 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
Jean Delvare400c4552005-07-19 23:48:43 +02002 i2c-isa.c - an i2c-core-like thing for ISA hardware monitoring chips
3 Copyright (C) 2005 Jean Delvare <khali@linux-fr.org>
4
5 Based on the i2c-isa pseudo-adapter from the lm_sensors project
Linus Torvalds1da177e2005-04-16 15:20:36 -07006 Copyright (c) 1998, 1999 Frodo Looijaard <frodol@dds.nl>
7
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2 of the License, or
11 (at your option) any later version.
12
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
17
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
21*/
22
Jean Delvare400c4552005-07-19 23:48:43 +020023/* This implements an i2c-core-like thing for ISA hardware monitoring
24 chips. Such chips are linked to the i2c subsystem for historical
25 reasons (because the early ISA hardware monitoring chips such as the
26 LM78 had both an I2C and an ISA interface). They used to be
27 registered with the main i2c-core, but as a first step in the
28 direction of a clean separation between I2C and ISA chip drivers,
29 we now have this separate core for ISA ones. It is significantly
30 more simple than the real one, of course, because we don't have to
31 handle multiple busses: there is only one (fake) ISA adapter.
32 It is worth noting that we still rely on i2c-core for some things
33 at the moment - but hopefully this won't last. */
Linus Torvalds1da177e2005-04-16 15:20:36 -070034
Linus Torvalds1da177e2005-04-16 15:20:36 -070035#include <linux/init.h>
36#include <linux/module.h>
37#include <linux/kernel.h>
38#include <linux/errno.h>
39#include <linux/i2c.h>
Jean Delvare400c4552005-07-19 23:48:43 +020040#include <linux/i2c-isa.h>
Russell Kingd052d1b2005-10-29 19:07:23 +010041#include <linux/platform_device.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070042
43static u32 isa_func(struct i2c_adapter *adapter);
44
45/* This is the actual algorithm we define */
46static struct i2c_algorithm isa_algorithm = {
Linus Torvalds1da177e2005-04-16 15:20:36 -070047 .functionality = isa_func,
48};
49
50/* There can only be one... */
51static struct i2c_adapter isa_adapter = {
52 .owner = THIS_MODULE,
Jean Delvarec7a46532005-08-11 23:41:56 +020053 .id = I2C_HW_ISA,
Linus Torvalds1da177e2005-04-16 15:20:36 -070054 .class = I2C_CLASS_HWMON,
55 .algo = &isa_algorithm,
56 .name = "ISA main adapter",
57};
58
59/* We can't do a thing... */
60static u32 isa_func(struct i2c_adapter *adapter)
61{
62 return 0;
63}
64
Jean Delvare400c4552005-07-19 23:48:43 +020065
66/* Copied from i2c-core */
67static ssize_t show_adapter_name(struct device *dev,
68 struct device_attribute *attr, char *buf)
69{
70 struct i2c_adapter *adap = dev_to_i2c_adapter(dev);
71 return sprintf(buf, "%s\n", adap->name);
72}
73static DEVICE_ATTR(name, S_IRUGO, show_adapter_name, NULL);
74
Jean Delvare400c4552005-07-19 23:48:43 +020075
76/* We implement an interface which resembles i2c_{add,del}_driver,
77 but for i2c-isa drivers. We don't have to remember and handle lists
78 of drivers and adapters so this is much more simple, of course. */
79
80int i2c_isa_add_driver(struct i2c_driver *driver)
81{
82 int res;
83
84 /* Add the driver to the list of i2c drivers in the driver core */
Jean Delvare400c4552005-07-19 23:48:43 +020085 driver->driver.bus = &i2c_bus_type;
Jean Delvare400c4552005-07-19 23:48:43 +020086 res = driver_register(&driver->driver);
87 if (res)
88 return res;
Greg Kroah-Hartmane1c489b2005-11-30 16:35:09 -080089 dev_dbg(&isa_adapter.dev, "Driver %s registered\n", driver->driver.name);
Jean Delvare400c4552005-07-19 23:48:43 +020090
91 /* Now look for clients */
92 driver->attach_adapter(&isa_adapter);
93
94 return 0;
95}
96
97int i2c_isa_del_driver(struct i2c_driver *driver)
98{
99 struct list_head *item, *_n;
100 struct i2c_client *client;
101 int res;
102
103 /* Detach all clients belonging to this one driver */
104 list_for_each_safe(item, _n, &isa_adapter.clients) {
105 client = list_entry(item, struct i2c_client, list);
106 if (client->driver != driver)
107 continue;
108 dev_dbg(&isa_adapter.dev, "Detaching client %s at 0x%x\n",
109 client->name, client->addr);
110 if ((res = driver->detach_client(client))) {
111 dev_err(&isa_adapter.dev, "Failed, driver "
112 "%s not unregistered!\n",
Laurent Riffard35d8b2e2005-11-26 20:34:05 +0100113 driver->driver.name);
Jean Delvare400c4552005-07-19 23:48:43 +0200114 return res;
115 }
116 }
117
118 /* Get the driver off the core list */
119 driver_unregister(&driver->driver);
Greg Kroah-Hartmane1c489b2005-11-30 16:35:09 -0800120 dev_dbg(&isa_adapter.dev, "Driver %s unregistered\n", driver->driver.name);
Jean Delvare400c4552005-07-19 23:48:43 +0200121
122 return 0;
123}
124
125
Linus Torvalds1da177e2005-04-16 15:20:36 -0700126static int __init i2c_isa_init(void)
127{
Ingo Molnar5c085d32006-01-18 23:16:04 +0100128 mutex_init(&isa_adapter.clist_lock);
Jean Delvare400c4552005-07-19 23:48:43 +0200129 INIT_LIST_HEAD(&isa_adapter.clients);
130
131 isa_adapter.nr = ANY_I2C_ISA_BUS;
132 isa_adapter.dev.parent = &platform_bus;
133 sprintf(isa_adapter.dev.bus_id, "i2c-%d", isa_adapter.nr);
134 isa_adapter.dev.driver = &i2c_adapter_driver;
135 isa_adapter.dev.release = &i2c_adapter_dev_release;
136 device_register(&isa_adapter.dev);
137 device_create_file(&isa_adapter.dev, &dev_attr_name);
138
139 /* Add this adapter to the i2c_adapter class */
140 memset(&isa_adapter.class_dev, 0x00, sizeof(struct class_device));
141 isa_adapter.class_dev.dev = &isa_adapter.dev;
142 isa_adapter.class_dev.class = &i2c_adapter_class;
143 strlcpy(isa_adapter.class_dev.class_id, isa_adapter.dev.bus_id,
144 BUS_ID_SIZE);
145 class_device_register(&isa_adapter.class_dev);
146
147 dev_dbg(&isa_adapter.dev, "%s registered\n", isa_adapter.name);
148
149 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700150}
151
152static void __exit i2c_isa_exit(void)
153{
Jean Delvare400c4552005-07-19 23:48:43 +0200154#ifdef DEBUG
155 struct list_head *item, *_n;
156 struct i2c_client *client = NULL;
157#endif
158
159 /* There should be no more active client */
160#ifdef DEBUG
161 dev_dbg(&isa_adapter.dev, "Looking for clients\n");
162 list_for_each_safe(item, _n, &isa_adapter.clients) {
163 client = list_entry(item, struct i2c_client, list);
164 dev_err(&isa_adapter.dev, "Driver %s still has an active "
Greg Kroah-Hartmane1c489b2005-11-30 16:35:09 -0800165 "ISA client at 0x%x\n", client->driver->driver.name,
Jean Delvare400c4552005-07-19 23:48:43 +0200166 client->addr);
167 }
168 if (client != NULL)
169 return;
170#endif
171
172 /* Clean up the sysfs representation */
173 dev_dbg(&isa_adapter.dev, "Unregistering from sysfs\n");
174 init_completion(&isa_adapter.dev_released);
175 init_completion(&isa_adapter.class_dev_released);
176 class_device_unregister(&isa_adapter.class_dev);
177 device_remove_file(&isa_adapter.dev, &dev_attr_name);
178 device_unregister(&isa_adapter.dev);
179
180 /* Wait for sysfs to drop all references */
181 dev_dbg(&isa_adapter.dev, "Waiting for sysfs completion\n");
182 wait_for_completion(&isa_adapter.dev_released);
183 wait_for_completion(&isa_adapter.class_dev_released);
184
185 dev_dbg(&isa_adapter.dev, "%s unregistered\n", isa_adapter.name);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700186}
187
Jean Delvare400c4552005-07-19 23:48:43 +0200188EXPORT_SYMBOL(i2c_isa_add_driver);
189EXPORT_SYMBOL(i2c_isa_del_driver);
190
191MODULE_AUTHOR("Jean Delvare <khali@linux-fr.org>");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700192MODULE_DESCRIPTION("ISA bus access through i2c");
193MODULE_LICENSE("GPL");
194
195module_init(i2c_isa_init);
196module_exit(i2c_isa_exit);