aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb/misc/phidgetservo.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/usb/misc/phidgetservo.c')
-rw-r--r--drivers/usb/misc/phidgetservo.c118
1 files changed, 78 insertions, 40 deletions
diff --git a/drivers/usb/misc/phidgetservo.c b/drivers/usb/misc/phidgetservo.c
index 5a040c205ee..7163f05c5b2 100644
--- a/drivers/usb/misc/phidgetservo.c
+++ b/drivers/usb/misc/phidgetservo.c
@@ -1,7 +1,7 @@
1/* 1/*
2 * USB PhidgetServo driver 1.0 2 * USB PhidgetServo driver 1.0
3 * 3 *
4 * Copyright (C) 2004 Sean Young <sean@mess.org> 4 * Copyright (C) 2004, 2006 Sean Young <sean@mess.org>
5 * 5 *
6 * This program is free software; you can redistribute it and/or modify 6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by 7 * it under the terms of the GNU General Public License as published by
@@ -15,17 +15,8 @@
15 * 15 *
16 * CAUTION: Generally you should use 0 < degrees < 180 as anything else 16 * CAUTION: Generally you should use 0 < degrees < 180 as anything else
17 * is probably beyond the range of your servo and may damage it. 17 * is probably beyond the range of your servo and may damage it.
18 *
19 * Jun 16, 2004: Sean Young <sean@mess.org>
20 * - cleanups
21 * - was using memory after kfree()
22 * Aug 8, 2004: Sean Young <sean@mess.org>
23 * - set the highest angle as high as the hardware allows, there are
24 * some odd servos out there
25 *
26 */ 18 */
27 19
28#include <linux/config.h>
29#include <linux/kernel.h> 20#include <linux/kernel.h>
30#include <linux/errno.h> 21#include <linux/errno.h>
31#include <linux/init.h> 22#include <linux/init.h>
@@ -33,6 +24,8 @@
33#include <linux/module.h> 24#include <linux/module.h>
34#include <linux/usb.h> 25#include <linux/usb.h>
35 26
27#include "phidget.h"
28
36#define DRIVER_AUTHOR "Sean Young <sean@mess.org>" 29#define DRIVER_AUTHOR "Sean Young <sean@mess.org>"
37#define DRIVER_DESC "USB PhidgetServo Driver" 30#define DRIVER_DESC "USB PhidgetServo Driver"
38 31
@@ -71,8 +64,12 @@ static struct usb_device_id id_table[] = {
71 64
72MODULE_DEVICE_TABLE(usb, id_table); 65MODULE_DEVICE_TABLE(usb, id_table);
73 66
67static int unsigned long device_no;
68
74struct phidget_servo { 69struct phidget_servo {
75 struct usb_device *udev; 70 struct usb_device *udev;
71 struct device *dev;
72 int dev_no;
76 ulong type; 73 ulong type;
77 int pulse[4]; 74 int pulse[4];
78 int degrees[4]; 75 int degrees[4];
@@ -204,16 +201,16 @@ change_position_v20(struct phidget_servo *servo, int servo_no, int degrees,
204} 201}
205 202
206#define show_set(value) \ 203#define show_set(value) \
207static ssize_t set_servo##value (struct device *dev, struct device_attribute *attr, \ 204static ssize_t set_servo##value (struct device *dev, \
205 struct device_attribute *attr, \
208 const char *buf, size_t count) \ 206 const char *buf, size_t count) \
209{ \ 207{ \
210 int degrees, minutes, retval; \ 208 int degrees, minutes, retval; \
211 struct usb_interface *intf = to_usb_interface (dev); \ 209 struct phidget_servo *servo = dev_get_drvdata(dev); \
212 struct phidget_servo *servo = usb_get_intfdata (intf); \
213 \ 210 \
214 minutes = 0; \ 211 minutes = 0; \
215 /* must at least convert degrees */ \ 212 /* must at least convert degrees */ \
216 if (sscanf (buf, "%d.%d", &degrees, &minutes) < 1) { \ 213 if (sscanf(buf, "%d.%d", &degrees, &minutes) < 1) { \
217 return -EINVAL; \ 214 return -EINVAL; \
218 } \ 215 } \
219 \ 216 \
@@ -221,86 +218,127 @@ static ssize_t set_servo##value (struct device *dev, struct device_attribute *at
221 return -EINVAL; \ 218 return -EINVAL; \
222 \ 219 \
223 if (servo->type & SERVO_VERSION_30) \ 220 if (servo->type & SERVO_VERSION_30) \
224 retval = change_position_v30 (servo, value, degrees, \ 221 retval = change_position_v30(servo, value, degrees, \
225 minutes); \ 222 minutes); \
226 else \ 223 else \
227 retval = change_position_v20 (servo, value, degrees, \ 224 retval = change_position_v20(servo, value, degrees, \
228 minutes); \ 225 minutes); \
229 \ 226 \
230 return retval < 0 ? retval : count; \ 227 return retval < 0 ? retval : count; \
231} \ 228} \
232 \ 229 \
233static ssize_t show_servo##value (struct device *dev, struct device_attribute *attr, char *buf) \ 230static ssize_t show_servo##value (struct device *dev, \
231 struct device_attribute *attr, \
232 char *buf) \
234{ \ 233{ \
235 struct usb_interface *intf = to_usb_interface (dev); \ 234 struct phidget_servo *servo = dev_get_drvdata(dev); \
236 struct phidget_servo *servo = usb_get_intfdata (intf); \
237 \ 235 \
238 return sprintf (buf, "%d.%02d\n", servo->degrees[value], \ 236 return sprintf(buf, "%d.%02d\n", servo->degrees[value], \
239 servo->minutes[value]); \ 237 servo->minutes[value]); \
240} \ 238}
241static DEVICE_ATTR(servo##value, S_IWUGO | S_IRUGO, \
242 show_servo##value, set_servo##value);
243 239
240#define servo_attr(value) \
241 __ATTR(servo##value, S_IWUGO | S_IRUGO, \
242 show_servo##value, set_servo##value)
244show_set(0); 243show_set(0);
245show_set(1); 244show_set(1);
246show_set(2); 245show_set(2);
247show_set(3); 246show_set(3);
248 247
248static struct device_attribute dev_attrs[] = {
249 servo_attr(0), servo_attr(1), servo_attr(2), servo_attr(3)
250};
251
249static int 252static int
250servo_probe(struct usb_interface *interface, const struct usb_device_id *id) 253servo_probe(struct usb_interface *interface, const struct usb_device_id *id)
251{ 254{
252 struct usb_device *udev = interface_to_usbdev(interface); 255 struct usb_device *udev = interface_to_usbdev(interface);
253 struct phidget_servo *dev; 256 struct phidget_servo *dev;
257 int bit, value, rc;
258 int servo_count, i;
254 259
255 dev = kzalloc(sizeof (struct phidget_servo), GFP_KERNEL); 260 dev = kzalloc(sizeof (struct phidget_servo), GFP_KERNEL);
256 if (dev == NULL) { 261 if (dev == NULL) {
257 dev_err(&interface->dev, "%s - out of memory\n", __FUNCTION__); 262 dev_err(&interface->dev, "%s - out of memory\n", __FUNCTION__);
258 return -ENOMEM; 263 rc = -ENOMEM;
264 goto out;
259 } 265 }
260 266
261 dev->udev = usb_get_dev(udev); 267 dev->udev = usb_get_dev(udev);
262 dev->type = id->driver_info; 268 dev->type = id->driver_info;
269 dev->dev_no = -1;
263 usb_set_intfdata(interface, dev); 270 usb_set_intfdata(interface, dev);
264 271
265 device_create_file(&interface->dev, &dev_attr_servo0); 272 do {
266 if (dev->type & SERVO_COUNT_QUAD) { 273 bit = find_first_zero_bit(&device_no, sizeof(device_no));
267 device_create_file(&interface->dev, &dev_attr_servo1); 274 value = test_and_set_bit(bit, &device_no);
268 device_create_file(&interface->dev, &dev_attr_servo2); 275 } while (value);
269 device_create_file(&interface->dev, &dev_attr_servo3); 276 dev->dev_no = bit;
277
278 dev->dev = device_create(phidget_class, &dev->udev->dev, 0,
279 "servo%d", dev->dev_no);
280 if (IS_ERR(dev->dev)) {
281 rc = PTR_ERR(dev->dev);
282 dev->dev = NULL;
283 goto out;
284 }
285
286 servo_count = dev->type & SERVO_COUNT_QUAD ? 4 : 1;
287
288 for (i=0; i<servo_count; i++) {
289 rc = device_create_file(dev->dev, &dev_attrs[i]);
290 if (rc)
291 goto out2;
270 } 292 }
271 293
272 dev_info(&interface->dev, "USB %d-Motor PhidgetServo v%d.0 attached\n", 294 dev_info(&interface->dev, "USB %d-Motor PhidgetServo v%d.0 attached\n",
273 dev->type & SERVO_COUNT_QUAD ? 4 : 1, 295 servo_count, dev->type & SERVO_VERSION_30 ? 3 : 2);
274 dev->type & SERVO_VERSION_30 ? 3 : 2);
275 296
276 if(!(dev->type & SERVO_VERSION_30)) 297 if (!(dev->type & SERVO_VERSION_30))
277 dev_info(&interface->dev, 298 dev_info(&interface->dev,
278 "WARNING: v2.0 not tested! Please report if it works.\n"); 299 "WARNING: v2.0 not tested! Please report if it works.\n");
279 300
280 return 0; 301 return 0;
302out2:
303 while (i-- > 0)
304 device_remove_file(dev->dev, &dev_attrs[i]);
305out:
306 if (dev) {
307 if (dev->dev)
308 device_unregister(dev->dev);
309 if (dev->dev_no >= 0)
310 clear_bit(dev->dev_no, &device_no);
311
312 kfree(dev);
313 }
314
315 return rc;
281} 316}
282 317
283static void 318static void
284servo_disconnect(struct usb_interface *interface) 319servo_disconnect(struct usb_interface *interface)
285{ 320{
286 struct phidget_servo *dev; 321 struct phidget_servo *dev;
322 int servo_count, i;
287 323
288 dev = usb_get_intfdata(interface); 324 dev = usb_get_intfdata(interface);
289 usb_set_intfdata(interface, NULL); 325 usb_set_intfdata(interface, NULL);
290 326
291 device_remove_file(&interface->dev, &dev_attr_servo0); 327 if (!dev)
292 if (dev->type & SERVO_COUNT_QUAD) { 328 return;
293 device_remove_file(&interface->dev, &dev_attr_servo1); 329
294 device_remove_file(&interface->dev, &dev_attr_servo2); 330 servo_count = dev->type & SERVO_COUNT_QUAD ? 4 : 1;
295 device_remove_file(&interface->dev, &dev_attr_servo3); 331
296 } 332 for (i=0; i<servo_count; i++)
333 device_remove_file(dev->dev, &dev_attrs[i]);
297 334
335 device_unregister(dev->dev);
298 usb_put_dev(dev->udev); 336 usb_put_dev(dev->udev);
299 337
300 dev_info(&interface->dev, "USB %d-Motor PhidgetServo v%d.0 detached\n", 338 dev_info(&interface->dev, "USB %d-Motor PhidgetServo v%d.0 detached\n",
301 dev->type & SERVO_COUNT_QUAD ? 4 : 1, 339 servo_count, dev->type & SERVO_VERSION_30 ? 3 : 2);
302 dev->type & SERVO_VERSION_30 ? 3 : 2);
303 340
341 clear_bit(dev->dev_no, &device_no);
304 kfree(dev); 342 kfree(dev);
305} 343}
306 344