linux-kernel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH] Support for OWI/Maplin USB Robotic Arm
@ 2012-10-08 21:29 Maxin B. John
  2012-10-08 23:55 ` Greg KH
  0 siblings, 1 reply; 4+ messages in thread
From: Maxin B. John @ 2012-10-08 21:29 UTC (permalink / raw)
  To: gregkh, devel; +Cc: linux-kernel

Hi Greg,

This driver helps us to control the OWI/Maplin USB Robotic Arm device.

I have tested this driver in my x86 Laptop and it works as expected
(Some clean-ups and modifications are in the pipeline, though).

After review process and the life in staging,I hope this driver will
find it's place in "drivers/usb/misc/" :)

The userspace utility program is available from the below URL:
https://github.com/maxinbjohn/robotic_arm_driver

OWI/Maplin USB Robotic arm is available from:
http://www.maplin.co.uk/robotic-arm-kit-with-usb-pc-interface-266257
http://www.owirobots.com/cart/html/owi-535pc-robotic-arm-kit-with-usb-pc-interface-assembled.html

Please let me know your comments.

Signed-off-by: Maxin B. John <maxin.john@gmail.com>
---
diff --git a/drivers/staging/Kconfig b/drivers/staging/Kconfig
index d805eef..a1eefa9 100644
--- a/drivers/staging/Kconfig
+++ b/drivers/staging/Kconfig
@@ -144,4 +144,6 @@ source "drivers/staging/imx-drm/Kconfig"
 
 source "drivers/staging/dgrp/Kconfig"
 
+source "drivers/staging/usb_robotic_arm/Kconfig"
+
 endif # STAGING
diff --git a/drivers/staging/Makefile b/drivers/staging/Makefile
index 76e2ebd..fd9b5f3 100644
--- a/drivers/staging/Makefile
+++ b/drivers/staging/Makefile
@@ -64,3 +64,4 @@ obj-$(CONFIG_NET_VENDOR_SILICOM)	+= silicom/
 obj-$(CONFIG_CED1401)		+= ced1401/
 obj-$(CONFIG_DRM_IMX)		+= imx-drm/
 obj-$(CONFIG_DGRP)		+= dgrp/
+obj-$(CONFIG_USB_ROBOTIC_ARM)	+= usb_robotic_arm/
diff --git a/drivers/staging/usb_robotic_arm/Kconfig b/drivers/staging/usb_robotic_arm/Kconfig
new file mode 100644
index 0000000..7ac4547
--- /dev/null
+++ b/drivers/staging/usb_robotic_arm/Kconfig
@@ -0,0 +1,15 @@
+config USB_ROBOTIC_ARM
+	tristate "OWI/MAPLIN USB Robotic Arm support"
+	depends on USB
+	default N
+	---help---
+	  This enables us to control the OWI/Maplin Robotic Arm
+	  device attached to the USB port.
+
+	  For more details, and to get the userspace utility
+	  programs, please see https://github.com/maxinbjohn/robotic_arm_driver
+
+	  To compile this as a module, choose M here: the module will
+	  be called robotic_arm.
+
+	  If unsure, say N.
diff --git a/drivers/staging/usb_robotic_arm/Makefile b/drivers/staging/usb_robotic_arm/Makefile
new file mode 100644
index 0000000..9509413
--- /dev/null
+++ b/drivers/staging/usb_robotic_arm/Makefile
@@ -0,0 +1 @@
+obj-$(CONFIG_USB_ROBOTIC_ARM)             += robotic_arm.o
diff --git a/drivers/staging/usb_robotic_arm/README b/drivers/staging/usb_robotic_arm/README
new file mode 100644
index 0000000..f7998ef
--- /dev/null
+++ b/drivers/staging/usb_robotic_arm/README
@@ -0,0 +1,59 @@
+
+The USB_ROBOTIC_ARM is a proof-of-concept driver for the OWI/Maplin USB
+Robotic Arm. I created it for fun. So, don't expect much from it.
+
+INSTALLATION:
+
+Select the 'USB ROBOTIC ARM' and build the kernel.
+
+If we build it as module, locate the robotic_arm.ko
+
+As 'root', insert the module:
+# insmod robotic_arm.ko
+# dmesg
+[ 8808.461441] usbcore: registered new interface driver robotic_arm
+
+Connect the Robotic Arm to the PC.
+
+# dmesg
+...
+[ 8718.729311] robotic_arm 5-1:1.0: USB ROBOTIC ARM now attached
+
+# cd /sys/bus/usb/drivers/robotic_arm/
+In my case,
+# ls
+5-1:1.0  bind  module  new_id  remove_id  uevent  unbind
+
+# cd 5-1:1.0
+# ls
+bAlternateSetting  bInterfaceNumber    bNumEndpoints  gripmotor  motor2  power                 uevent
+basemotor          bInterfaceProtocol  driver         led        motor3  subsystem
+bInterfaceClass    bInterfaceSubClass  ep_81          modalias   motor4  supports_autosuspend
+
+TESTING:
+
+# echo 1 > basemotor
+(This will rotate the base clockwise)
+
+# echo 2 > basemotor
+(This will rotate the base anti-clockwise)
+
+# echo 0 > basemotor
+(This will stop the base motor if it is on)
+
+# echo 1 > led
+(This will switch on the front side LED)
+
+# echo 1 > motor3
+(This will switch on motor3)
+
+HARDWARE AVAILABILITY:
+
+OWI/Maplin USB Robotic arm is available from:
+http://www.maplin.co.uk/robotic-arm-kit-with-usb-pc-interface-266257
+http://www.owirobots.com/cart/html/owi-535pc-robotic-arm-kit-with-usb-pc-interface-assembled.html
+
+REFERENCES:
+
+Reverse Engineering details of Robotic Arm's Protocol is available here:
+http://notbrainsurgery.livejournal.com/38622.html
diff --git a/drivers/staging/usb_robotic_arm/TODO b/drivers/staging/usb_robotic_arm/TODO
new file mode 100644
index 0000000..6c0fd7e
--- /dev/null
+++ b/drivers/staging/usb_robotic_arm/TODO
@@ -0,0 +1,8 @@
+TODO:
+	- provide independent control for motors in the hand
+	- various code clean-ups
+	- test on various platforms
+	- send to lkml for review
+
+Please send patches to Greg Kroah-Hartman <gregkh@linuxfoundation.org>
+and Cc: Maxin B. John <maxin.john@gmail.com>
diff --git a/drivers/staging/usb_robotic_arm/robotic_arm.c b/drivers/staging/usb_robotic_arm/robotic_arm.c
new file mode 100644
index 0000000..dbde911
--- /dev/null
+++ b/drivers/staging/usb_robotic_arm/robotic_arm.c
@@ -0,0 +1,278 @@
+/*
+ * USB ROBOTIC ARM driver
+ * robotic_arm - driver for Robotic ARM with USB interface
+ * from Maplin Electronics.This is an experimental driver.
+ * Use at your own risk.
+ *
+ * Copyright (C) 2012 Maxin B. John (maxin.john@gmail.com)
+ *
+ * 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, version 2.
+ *
+ * Derived from USB LED driver
+ * Copyright (c) 2001 Greg Kroah-Hartman (greg@kroah.com)
+ */
+
+#define DEBUG 1
+
+#include <linux/kernel.h>
+#include <linux/errno.h>
+#include <linux/init.h>
+#include <linux/slab.h>
+#include <linux/module.h>
+#include <linux/usb.h>
+
+/* Version Information */
+#define DRIVER_VERSION "v0.1"
+#define DRIVER_AUTHOR "Maxin B. John, maxin.john@gmail.com"
+#define DRIVER_DESC "USB ROBOTIC ARM Driver"
+
+#define VENDOR_ID       0x1267
+#define PRODUCT_ID      0x0000
+
+/* table of devices that work with this driver */
+static struct usb_device_id id_table[] = {
+	{.idVendor = 0x1267, .match_flags = USB_DEVICE_ID_MATCH_VENDOR,},
+	{},
+};
+
+MODULE_DEVICE_TABLE(usb, id_table);
+
+struct usb_robotic_arm {
+	struct usb_device *udev;
+	unsigned char led;
+	unsigned char gripmotor;
+	unsigned char motor2;
+	unsigned char motor3;
+	unsigned char motor4;
+	unsigned char basemotor;
+};
+
+static void update_robotic_arm(struct usb_robotic_arm *robotic_arm)
+{
+	int retval = 0;
+	u8 cmd[3];
+
+	cmd[0] = 0;
+	cmd[1] = 0;
+	cmd[2] = 0;
+
+	/*
+	 * Byte 2 controls LED light inside the grip.
+	 * If Bit 0 is 1, LED on, otherwise LED off
+	 *
+	 */
+
+	if (robotic_arm->led)
+		cmd[2] = 1;
+
+	/*
+	 * Byte 1 controls the Base Motor
+	 * If value is 1, Base Motor rotates clockwise
+	 * If value is 2, Base Motor rotates anti-clockwise
+	 * Otherwise, Base Motor is off
+	 *
+	 */
+
+	if (robotic_arm->basemotor == 1)
+		cmd[1] = 1;
+
+	else if (robotic_arm->basemotor == 2)
+		cmd[1] = 2;
+	else
+		cmd[1] = 0;
+
+	/*
+	 * Byte 0 controls the Grip
+	 * If value is 1, Grip tightens
+	 * If values is 2, Grip Looses
+	 *
+	 */
+
+	if (robotic_arm->gripmotor == 1)
+		cmd[0] = 1;
+	else if (robotic_arm->gripmotor == 2)
+		cmd[0] = 2;
+	/*
+	 * Byte 0 controls the Motor 2
+	 * If value is 1, Motor 2 (wrist) moves up
+	 * If value is 2, Motor 2 (wrist) moves down
+	 *
+	 */
+
+	else if (robotic_arm->motor2 == 1)
+		cmd[0] = 4;
+	else if (robotic_arm->motor2 == 2)
+		cmd[0] = 8;
+
+	/*
+	 * Byte 0 controls the Motor 3
+	 * If value is 1, Motor 3 (elbow) moves up
+	 * If value is 2, Motor 3 (elbow) moves down
+	 *
+	 */
+
+	else if (robotic_arm->motor3 == 1)
+		cmd[0] = 16;
+	else if (robotic_arm->motor3 == 2)
+		cmd[0] = 32;
+
+	/*
+	 * Byte 0 controls the Motor 4
+	 * If value is 1, Motor 4 (shoulder) moves up
+	 * If value is 2, Motor 4 (shoulder) moves down
+	 *
+	 */
+
+	else if (robotic_arm->motor4 == 1)
+		cmd[0] = 64;
+	else if (robotic_arm->motor4 == 2)
+		cmd[0] = 128;
+	else
+		cmd[0] = 0;
+
+	retval = usb_control_msg(robotic_arm->udev,
+				 usb_sndctrlpipe(robotic_arm->udev, 0),
+				 6, 0x40, 0x100, 0, cmd, 3, 0);
+
+
+	if (retval)
+		dev_dbg(&robotic_arm->udev->dev, "retval = %d\n", retval);
+
+}
+
+#define show_set(value) \
+static ssize_t show_##value(struct device *dev, struct device_attribute *attr,\
+			char *buf)					\
+{									\
+	struct usb_interface *intf = to_usb_interface(dev);		\
+	struct usb_robotic_arm *robotic_arm = usb_get_intfdata(intf);	\
+									\
+	return sprintf(buf, "%d\n", robotic_arm->value);		\
+}                                                                       \
+static ssize_t set_##value(struct device *dev, struct device_attribute *attr,\
+				const char *buf, size_t count)		\
+{                                                                       \
+	struct usb_interface *intf = to_usb_interface(dev);             \
+	struct usb_robotic_arm *robotic_arm = usb_get_intfdata(intf);   \
+	int ret = -ENXIO;						\
+	unsigned long val;						\
+									\
+	ret = kstrtoul(buf, 0, &val);					\
+	if (ret)							\
+		return ret;						\
+	robotic_arm->value = val;					\
+	update_robotic_arm(robotic_arm);                                \
+	return count;                                                   \
+}                                                                       \
+static DEVICE_ATTR(value, S_IRUGO | S_IWUSR, show_##value, set_##value);
+show_set(led);
+show_set(gripmotor);
+show_set(motor2);
+show_set(motor3);
+show_set(motor4);
+show_set(basemotor);
+
+static int robotic_arm_probe(struct usb_interface *interface,
+			     const struct usb_device_id *id)
+{
+	struct usb_device *udev;
+	struct usb_robotic_arm *dev;
+	int retval = -ENOMEM;
+
+	udev = interface_to_usbdev(interface);
+	dev = kzalloc(sizeof(struct usb_robotic_arm), GFP_KERNEL);
+	if (dev == NULL) {
+		dev_err(&interface->dev, "out of memory\n");
+		goto error_mem;
+	}
+
+	dev->udev = usb_get_dev(udev);
+
+	usb_set_intfdata(interface, dev);
+
+	retval = device_create_file(&interface->dev, &dev_attr_led);
+	if (retval)
+		goto error;
+	retval = device_create_file(&interface->dev, &dev_attr_gripmotor);
+	if (retval)
+		goto error;
+	retval = device_create_file(&interface->dev, &dev_attr_motor2);
+	if (retval)
+		goto error;
+	retval = device_create_file(&interface->dev, &dev_attr_motor3);
+	if (retval)
+		goto error;
+	retval = device_create_file(&interface->dev, &dev_attr_motor4);
+	if (retval)
+		goto error;
+	retval = device_create_file(&interface->dev, &dev_attr_basemotor);
+	if (retval)
+		goto error;
+
+	dev_info(&interface->dev, "USB ROBOTIC ARM now attached\n");
+	return 0;
+
+error:
+	device_remove_file(&interface->dev, &dev_attr_led);
+	device_remove_file(&interface->dev, &dev_attr_gripmotor);
+	device_remove_file(&interface->dev, &dev_attr_motor2);
+	device_remove_file(&interface->dev, &dev_attr_motor3);
+	device_remove_file(&interface->dev, &dev_attr_motor4);
+	device_remove_file(&interface->dev, &dev_attr_basemotor);
+
+	usb_set_intfdata(interface, NULL);
+	usb_put_dev(dev->udev);
+	kfree(dev);
+
+error_mem:
+	return retval;
+}
+
+static void robotic_arm_disconnect(struct usb_interface *interface)
+{
+	struct usb_robotic_arm *dev = NULL;
+
+	dev = usb_get_intfdata(interface);
+
+	device_remove_file(&interface->dev, &dev_attr_led);
+	device_remove_file(&interface->dev, &dev_attr_gripmotor);
+	device_remove_file(&interface->dev, &dev_attr_motor2);
+	device_remove_file(&interface->dev, &dev_attr_motor3);
+	device_remove_file(&interface->dev, &dev_attr_motor4);
+	device_remove_file(&interface->dev, &dev_attr_basemotor);
+
+	/* first remove the files, then set the pointer to NULL */
+	usb_set_intfdata(interface, NULL);
+
+	usb_put_dev(dev->udev);
+
+	kfree(dev);
+
+	dev_info(&interface->dev, "USB ROBOTIC ARM now disconnected\n");
+}
+
+static struct usb_driver robotic_arm_driver = {
+	.name = "robotic_arm",
+	.probe = robotic_arm_probe,
+	.disconnect = robotic_arm_disconnect,
+	.id_table = id_table,
+};
+
+static int __init usb_robotic_arm_init(void)
+{
+	return usb_register(&robotic_arm_driver);
+}
+
+static void __exit usb_robotic_arm_exit(void)
+{
+	usb_deregister(&robotic_arm_driver);
+}
+
+module_init(usb_robotic_arm_init);
+module_exit(usb_robotic_arm_exit);
+
+MODULE_AUTHOR(DRIVER_AUTHOR);
+MODULE_DESCRIPTION(DRIVER_DESC);
+MODULE_LICENSE("GPL");

^ permalink raw reply related	[flat|nested] 4+ messages in thread

* Re: [PATCH] Support for OWI/Maplin USB Robotic Arm
  2012-10-08 21:29 [PATCH] Support for OWI/Maplin USB Robotic Arm Maxin B. John
@ 2012-10-08 23:55 ` Greg KH
  2012-10-09  3:52   ` Maxin B John
  0 siblings, 1 reply; 4+ messages in thread
From: Greg KH @ 2012-10-08 23:55 UTC (permalink / raw)
  To: Maxin B. John; +Cc: devel, linux-kernel

On Mon, Oct 08, 2012 at 05:29:43PM -0400, Maxin B. John wrote:
> Hi Greg,
> 
> This driver helps us to control the OWI/Maplin USB Robotic Arm device.
> 
> I have tested this driver in my x86 Laptop and it works as expected
> (Some clean-ups and modifications are in the pipeline, though).
> 
> After review process and the life in staging,I hope this driver will
> find it's place in "drivers/usb/misc/" :)
> 
> The userspace utility program is available from the below URL:
> https://github.com/maxinbjohn/robotic_arm_driver
> 
> OWI/Maplin USB Robotic arm is available from:
> http://www.maplin.co.uk/robotic-arm-kit-with-usb-pc-interface-266257
> http://www.owirobots.com/cart/html/owi-535pc-robotic-arm-kit-with-usb-pc-interface-assembled.html
> 
> Please let me know your comments.

Very nice, but are there any real requirement to have this be a kernel
driver, and not just be driven from userspace through libusb/usbfs
instead?

thanks,

greg k-h

^ permalink raw reply	[flat|nested] 4+ messages in thread

* Re: [PATCH] Support for OWI/Maplin USB Robotic Arm
  2012-10-08 23:55 ` Greg KH
@ 2012-10-09  3:52   ` Maxin B John
  2012-10-22 22:40     ` Greg KH
  0 siblings, 1 reply; 4+ messages in thread
From: Maxin B John @ 2012-10-09  3:52 UTC (permalink / raw)
  To: Greg KH; +Cc: devel, linux-kernel

Hi Greg,

On Mon, Oct 8, 2012 at 7:55 PM, Greg KH <gregkh@linuxfoundation.org> wrote:
> On Mon, Oct 08, 2012 at 05:29:43PM -0400, Maxin B. John wrote:
>> Hi Greg,
>>
>> This driver helps us to control the OWI/Maplin USB Robotic Arm device.
>>
>> I have tested this driver in my x86 Laptop and it works as expected
>> (Some clean-ups and modifications are in the pipeline, though).
>>
>> After review process and the life in staging,I hope this driver will
>> find it's place in "drivers/usb/misc/" :)
>>
>> The userspace utility program is available from the below URL:
>> https://github.com/maxinbjohn/robotic_arm_driver
>>
>> OWI/Maplin USB Robotic arm is available from:
>> http://www.maplin.co.uk/robotic-arm-kit-with-usb-pc-interface-266257
>> http://www.owirobots.com/cart/html/owi-535pc-robotic-arm-kit-with-usb-pc-interface-assembled.html
>>
>> Please let me know your comments.
>
> Very nice, but are there any real requirement to have this be a kernel
> driver, and not just be driven from userspace through libusb/usbfs
> instead?

Thanks. A number of people are using the libusb based userspace
applications to control these devices. This was an attempt to remove
the libusb dependency and control the device easily like:
# echo 1 > /sys/bus/usb/drivers/robotic_arm/5-1:1.0/led

> thanks,
> greg k-h

Best Regards,
Maxin B. John

^ permalink raw reply	[flat|nested] 4+ messages in thread

* Re: [PATCH] Support for OWI/Maplin USB Robotic Arm
  2012-10-09  3:52   ` Maxin B John
@ 2012-10-22 22:40     ` Greg KH
  0 siblings, 0 replies; 4+ messages in thread
From: Greg KH @ 2012-10-22 22:40 UTC (permalink / raw)
  To: Maxin B John; +Cc: devel, linux-kernel

On Mon, Oct 08, 2012 at 11:52:56PM -0400, Maxin B John wrote:
> Hi Greg,
> 
> On Mon, Oct 8, 2012 at 7:55 PM, Greg KH <gregkh@linuxfoundation.org> wrote:
> > On Mon, Oct 08, 2012 at 05:29:43PM -0400, Maxin B. John wrote:
> >> Hi Greg,
> >>
> >> This driver helps us to control the OWI/Maplin USB Robotic Arm device.
> >>
> >> I have tested this driver in my x86 Laptop and it works as expected
> >> (Some clean-ups and modifications are in the pipeline, though).
> >>
> >> After review process and the life in staging,I hope this driver will
> >> find it's place in "drivers/usb/misc/" :)
> >>
> >> The userspace utility program is available from the below URL:
> >> https://github.com/maxinbjohn/robotic_arm_driver
> >>
> >> OWI/Maplin USB Robotic arm is available from:
> >> http://www.maplin.co.uk/robotic-arm-kit-with-usb-pc-interface-266257
> >> http://www.owirobots.com/cart/html/owi-535pc-robotic-arm-kit-with-usb-pc-interface-assembled.html
> >>
> >> Please let me know your comments.
> >
> > Very nice, but are there any real requirement to have this be a kernel
> > driver, and not just be driven from userspace through libusb/usbfs
> > instead?
> 
> Thanks. A number of people are using the libusb based userspace
> applications to control these devices. This was an attempt to remove
> the libusb dependency and control the device easily like:
> # echo 1 > /sys/bus/usb/drivers/robotic_arm/5-1:1.0/led

I really think that libusb is the best way to control this device.  We
removed a bunch of drivers from the kernel a number of years ago that
could be better controlled by userspace.  Because of that, I can't take
this driver either.

Sorry,

greg k-h

^ permalink raw reply	[flat|nested] 4+ messages in thread

end of thread, other threads:[~2012-10-22 22:40 UTC | newest]

Thread overview: 4+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2012-10-08 21:29 [PATCH] Support for OWI/Maplin USB Robotic Arm Maxin B. John
2012-10-08 23:55 ` Greg KH
2012-10-09  3:52   ` Maxin B John
2012-10-22 22:40     ` Greg KH

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).