aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDavid Fries <david@fries.net>2008-10-16 01:04:42 -0400
committerLinus Torvalds <torvalds@linux-foundation.org>2008-10-16 14:21:49 -0400
commit6a158c0de791a81eb761ccf26ead1bd0834abac2 (patch)
treec4a35705bbeb2f90b81a5e5d44e9a7b45c2f666a
parent3c52e4e627896b42152cc6ff98216c302932227e (diff)
W1: feature, enable hardware strong pullup
Add a strong pullup option to the w1 system. This supplies extra power for parasite powered devices. There is a w1_master_pullup sysfs entry and enable_pullup module parameter to enable or disable the strong pullup. The one wire bus requires at a minimum one wire and ground. The common wire is used for sending and receiving data as well as supplying power to devices that are parasite powered of which temperature sensors can be one example. The bus must be idle and left high while a temperature conversion is in progress, in addition the normal pullup resister on larger networks or even higher temperatures might not supply enough power. The pullup resister can't provide too much pullup current, because devices need to pull the bus down to write a value. This enables the strong pullup for supported hardware, which can supply more current when requested. Unsupported hardware will just delay with the bus high. The hardware USB 2490 one wire bus master has a bit on some commands which will enable the strong pullup as soon as the command finishes executing. To use strong pullup, call the new w1_next_pullup function to register the duration. The next write command will call set_pullup before sending the data, and reset the duration to zero once it returns. Switched from simple_strtol to strict_strtol. Signed-off-by: David Fries <david@fries.net> Cc: Evgeniy Polyakov <johnpol@2ka.mipt.ru> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r--drivers/w1/w1.c40
-rw-r--r--drivers/w1/w1.h12
-rw-r--r--drivers/w1/w1_int.c16
-rw-r--r--drivers/w1/w1_io.c68
4 files changed, 131 insertions, 5 deletions
diff --git a/drivers/w1/w1.c b/drivers/w1/w1.c
index 9b5c11701c37..32418d4e555a 100644
--- a/drivers/w1/w1.c
+++ b/drivers/w1/w1.c
@@ -246,10 +246,14 @@ static ssize_t w1_master_attribute_store_search(struct device * dev,
246 struct device_attribute *attr, 246 struct device_attribute *attr,
247 const char * buf, size_t count) 247 const char * buf, size_t count)
248{ 248{
249 long tmp;
249 struct w1_master *md = dev_to_w1_master(dev); 250 struct w1_master *md = dev_to_w1_master(dev);
250 251
252 if (strict_strtol(buf, 0, &tmp) == -EINVAL)
253 return -EINVAL;
254
251 mutex_lock(&md->mutex); 255 mutex_lock(&md->mutex);
252 md->search_count = simple_strtol(buf, NULL, 0); 256 md->search_count = tmp;
253 mutex_unlock(&md->mutex); 257 mutex_unlock(&md->mutex);
254 wake_up_process(md->thread); 258 wake_up_process(md->thread);
255 259
@@ -270,6 +274,38 @@ static ssize_t w1_master_attribute_show_search(struct device *dev,
270 return count; 274 return count;
271} 275}
272 276
277static ssize_t w1_master_attribute_store_pullup(struct device *dev,
278 struct device_attribute *attr,
279 const char *buf, size_t count)
280{
281 long tmp;
282 struct w1_master *md = dev_to_w1_master(dev);
283
284 if (strict_strtol(buf, 0, &tmp) == -EINVAL)
285 return -EINVAL;
286
287 mutex_lock(&md->mutex);
288 md->enable_pullup = tmp;
289 mutex_unlock(&md->mutex);
290 wake_up_process(md->thread);
291
292 return count;
293}
294
295static ssize_t w1_master_attribute_show_pullup(struct device *dev,
296 struct device_attribute *attr,
297 char *buf)
298{
299 struct w1_master *md = dev_to_w1_master(dev);
300 ssize_t count;
301
302 mutex_lock(&md->mutex);
303 count = sprintf(buf, "%d\n", md->enable_pullup);
304 mutex_unlock(&md->mutex);
305
306 return count;
307}
308
273static ssize_t w1_master_attribute_show_pointer(struct device *dev, struct device_attribute *attr, char *buf) 309static ssize_t w1_master_attribute_show_pointer(struct device *dev, struct device_attribute *attr, char *buf)
274{ 310{
275 struct w1_master *md = dev_to_w1_master(dev); 311 struct w1_master *md = dev_to_w1_master(dev);
@@ -365,6 +401,7 @@ static W1_MASTER_ATTR_RO(attempts, S_IRUGO);
365static W1_MASTER_ATTR_RO(timeout, S_IRUGO); 401static W1_MASTER_ATTR_RO(timeout, S_IRUGO);
366static W1_MASTER_ATTR_RO(pointer, S_IRUGO); 402static W1_MASTER_ATTR_RO(pointer, S_IRUGO);
367static W1_MASTER_ATTR_RW(search, S_IRUGO | S_IWUGO); 403static W1_MASTER_ATTR_RW(search, S_IRUGO | S_IWUGO);
404static W1_MASTER_ATTR_RW(pullup, S_IRUGO | S_IWUGO);
368 405
369static struct attribute *w1_master_default_attrs[] = { 406static struct attribute *w1_master_default_attrs[] = {
370 &w1_master_attribute_name.attr, 407 &w1_master_attribute_name.attr,
@@ -375,6 +412,7 @@ static struct attribute *w1_master_default_attrs[] = {
375 &w1_master_attribute_timeout.attr, 412 &w1_master_attribute_timeout.attr,
376 &w1_master_attribute_pointer.attr, 413 &w1_master_attribute_pointer.attr,
377 &w1_master_attribute_search.attr, 414 &w1_master_attribute_search.attr,
415 &w1_master_attribute_pullup.attr,
378 NULL 416 NULL
379}; 417};
380 418
diff --git a/drivers/w1/w1.h b/drivers/w1/w1.h
index 34ee01e008ad..00b84ab22808 100644
--- a/drivers/w1/w1.h
+++ b/drivers/w1/w1.h
@@ -142,6 +142,12 @@ struct w1_bus_master
142 */ 142 */
143 u8 (*reset_bus)(void *); 143 u8 (*reset_bus)(void *);
144 144
145 /**
146 * Put out a strong pull-up pulse of the specified duration.
147 * @return -1=Error, 0=completed
148 */
149 u8 (*set_pullup)(void *, int);
150
145 /** Really nice hardware can handles the different types of ROM search 151 /** Really nice hardware can handles the different types of ROM search
146 * w1_master* is passed to the slave found callback. 152 * w1_master* is passed to the slave found callback.
147 */ 153 */
@@ -167,6 +173,11 @@ struct w1_master
167 void *priv; 173 void *priv;
168 int priv_size; 174 int priv_size;
169 175
176 /** 5V strong pullup enabled flag, 1 enabled, zero disabled. */
177 int enable_pullup;
178 /** 5V strong pullup duration in milliseconds, zero disabled. */
179 int pullup_duration;
180
170 struct task_struct *thread; 181 struct task_struct *thread;
171 struct mutex mutex; 182 struct mutex mutex;
172 183
@@ -201,6 +212,7 @@ u8 w1_calc_crc8(u8 *, int);
201void w1_write_block(struct w1_master *, const u8 *, int); 212void w1_write_block(struct w1_master *, const u8 *, int);
202u8 w1_read_block(struct w1_master *, u8 *, int); 213u8 w1_read_block(struct w1_master *, u8 *, int);
203int w1_reset_select_slave(struct w1_slave *sl); 214int w1_reset_select_slave(struct w1_slave *sl);
215void w1_next_pullup(struct w1_master *, int);
204 216
205static inline struct w1_slave* dev_to_w1_slave(struct device *dev) 217static inline struct w1_slave* dev_to_w1_slave(struct device *dev)
206{ 218{
diff --git a/drivers/w1/w1_int.c b/drivers/w1/w1_int.c
index bd877b24ce42..9d723efdf915 100644
--- a/drivers/w1/w1_int.c
+++ b/drivers/w1/w1_int.c
@@ -31,6 +31,9 @@
31 31
32static u32 w1_ids = 1; 32static u32 w1_ids = 1;
33 33
34static int w1_enable_pullup = 1;
35module_param_named(enable_pullup, w1_enable_pullup, int, 0);
36
34static struct w1_master * w1_alloc_dev(u32 id, int slave_count, int slave_ttl, 37static struct w1_master * w1_alloc_dev(u32 id, int slave_count, int slave_ttl,
35 struct device_driver *driver, 38 struct device_driver *driver,
36 struct device *device) 39 struct device *device)
@@ -59,6 +62,7 @@ static struct w1_master * w1_alloc_dev(u32 id, int slave_count, int slave_ttl,
59 dev->initialized = 0; 62 dev->initialized = 0;
60 dev->id = id; 63 dev->id = id;
61 dev->slave_ttl = slave_ttl; 64 dev->slave_ttl = slave_ttl;
65 dev->enable_pullup = w1_enable_pullup;
62 dev->search_count = -1; /* continual scan */ 66 dev->search_count = -1; /* continual scan */
63 67
64 /* 1 for w1_process to decrement 68 /* 1 for w1_process to decrement
@@ -107,6 +111,18 @@ int w1_add_master_device(struct w1_bus_master *master)
107 printk(KERN_ERR "w1_add_master_device: invalid function set\n"); 111 printk(KERN_ERR "w1_add_master_device: invalid function set\n");
108 return(-EINVAL); 112 return(-EINVAL);
109 } 113 }
114 /* While it would be electrically possible to make a device that
115 * generated a strong pullup in bit bang mode, only hardare that
116 * controls 1-wire time frames are even expected to support a strong
117 * pullup. w1_io.c would need to support calling set_pullup before
118 * the last write_bit operation of a w1_write_8 which it currently
119 * doesn't.
120 */
121 if (!master->write_byte && !master->touch_bit && master->set_pullup) {
122 printk(KERN_ERR "w1_add_master_device: set_pullup requires "
123 "write_byte or touch_bit, disabling\n");
124 master->set_pullup = NULL;
125 }
110 126
111 dev = w1_alloc_dev(w1_ids++, w1_max_slave_count, w1_max_slave_ttl, &w1_master_driver, &w1_master_device); 127 dev = w1_alloc_dev(w1_ids++, w1_max_slave_count, w1_max_slave_ttl, &w1_master_driver, &w1_master_device);
112 if (!dev) 128 if (!dev)
diff --git a/drivers/w1/w1_io.c b/drivers/w1/w1_io.c
index 0056ef69009b..97b338a16abc 100644
--- a/drivers/w1/w1_io.c
+++ b/drivers/w1/w1_io.c
@@ -93,6 +93,40 @@ static void w1_write_bit(struct w1_master *dev, int bit)
93} 93}
94 94
95/** 95/**
96 * Pre-write operation, currently only supporting strong pullups.
97 * Program the hardware for a strong pullup, if one has been requested and
98 * the hardware supports it.
99 *
100 * @param dev the master device
101 */
102static void w1_pre_write(struct w1_master *dev)
103{
104 if (dev->pullup_duration &&
105 dev->enable_pullup && dev->bus_master->set_pullup) {
106 dev->bus_master->set_pullup(dev->bus_master->data,
107 dev->pullup_duration);
108 }
109}
110
111/**
112 * Post-write operation, currently only supporting strong pullups.
113 * If a strong pullup was requested, clear it if the hardware supports
114 * them, or execute the delay otherwise, in either case clear the request.
115 *
116 * @param dev the master device
117 */
118static void w1_post_write(struct w1_master *dev)
119{
120 if (dev->pullup_duration) {
121 if (dev->enable_pullup && dev->bus_master->set_pullup)
122 dev->bus_master->set_pullup(dev->bus_master->data, 0);
123 else
124 msleep(dev->pullup_duration);
125 dev->pullup_duration = 0;
126 }
127}
128
129/**
96 * Writes 8 bits. 130 * Writes 8 bits.
97 * 131 *
98 * @param dev the master device 132 * @param dev the master device
@@ -102,11 +136,17 @@ void w1_write_8(struct w1_master *dev, u8 byte)
102{ 136{
103 int i; 137 int i;
104 138
105 if (dev->bus_master->write_byte) 139 if (dev->bus_master->write_byte) {
140 w1_pre_write(dev);
106 dev->bus_master->write_byte(dev->bus_master->data, byte); 141 dev->bus_master->write_byte(dev->bus_master->data, byte);
142 }
107 else 143 else
108 for (i = 0; i < 8; ++i) 144 for (i = 0; i < 8; ++i) {
145 if (i == 7)
146 w1_pre_write(dev);
109 w1_touch_bit(dev, (byte >> i) & 0x1); 147 w1_touch_bit(dev, (byte >> i) & 0x1);
148 }
149 w1_post_write(dev);
110} 150}
111EXPORT_SYMBOL_GPL(w1_write_8); 151EXPORT_SYMBOL_GPL(w1_write_8);
112 152
@@ -203,11 +243,14 @@ void w1_write_block(struct w1_master *dev, const u8 *buf, int len)
203{ 243{
204 int i; 244 int i;
205 245
206 if (dev->bus_master->write_block) 246 if (dev->bus_master->write_block) {
247 w1_pre_write(dev);
207 dev->bus_master->write_block(dev->bus_master->data, buf, len); 248 dev->bus_master->write_block(dev->bus_master->data, buf, len);
249 }
208 else 250 else
209 for (i = 0; i < len; ++i) 251 for (i = 0; i < len; ++i)
210 w1_write_8(dev, buf[i]); 252 w1_write_8(dev, buf[i]); /* calls w1_pre_write */
253 w1_post_write(dev);
211} 254}
212EXPORT_SYMBOL_GPL(w1_write_block); 255EXPORT_SYMBOL_GPL(w1_write_block);
213 256
@@ -306,3 +349,20 @@ int w1_reset_select_slave(struct w1_slave *sl)
306 return 0; 349 return 0;
307} 350}
308EXPORT_SYMBOL_GPL(w1_reset_select_slave); 351EXPORT_SYMBOL_GPL(w1_reset_select_slave);
352
353/**
354 * Put out a strong pull-up of the specified duration after the next write
355 * operation. Not all hardware supports strong pullups. Hardware that
356 * doesn't support strong pullups will sleep for the given time after the
357 * write operation without a strong pullup. This is a one shot request for
358 * the next write, specifying zero will clear a previous request.
359 * The w1 master lock must be held.
360 *
361 * @param delay time in milliseconds
362 * @return 0=success, anything else=error
363 */
364void w1_next_pullup(struct w1_master *dev, int delay)
365{
366 dev->pullup_duration = delay;
367}
368EXPORT_SYMBOL_GPL(w1_next_pullup);