aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/usb
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/usb')
-rw-r--r--drivers/usb/misc/berry_charge.c35
1 files changed, 35 insertions, 0 deletions
diff --git a/drivers/usb/misc/berry_charge.c b/drivers/usb/misc/berry_charge.c
index b15f2fd8dab4..92c1d2768df9 100644
--- a/drivers/usb/misc/berry_charge.c
+++ b/drivers/usb/misc/berry_charge.c
@@ -26,8 +26,11 @@
26 26
27#define RIM_VENDOR 0x0fca 27#define RIM_VENDOR 0x0fca
28#define BLACKBERRY 0x0001 28#define BLACKBERRY 0x0001
29#define BLACKBERRY_PEARL_DUAL 0x0004
30#define BLACKBERRY_PEARL 0x0006
29 31
30static int debug; 32static int debug;
33static int pearl_dual_mode = 1;
31 34
32#ifdef dbg 35#ifdef dbg
33#undef dbg 36#undef dbg
@@ -38,6 +41,8 @@ static int debug;
38 41
39static struct usb_device_id id_table [] = { 42static struct usb_device_id id_table [] = {
40 { USB_DEVICE(RIM_VENDOR, BLACKBERRY) }, 43 { USB_DEVICE(RIM_VENDOR, BLACKBERRY) },
44 { USB_DEVICE(RIM_VENDOR, BLACKBERRY_PEARL) },
45 { USB_DEVICE(RIM_VENDOR, BLACKBERRY_PEARL_DUAL) },
41 { }, /* Terminating entry */ 46 { }, /* Terminating entry */
42}; 47};
43MODULE_DEVICE_TABLE(usb, id_table); 48MODULE_DEVICE_TABLE(usb, id_table);
@@ -86,6 +91,30 @@ static int magic_charge(struct usb_device *udev)
86 return retval; 91 return retval;
87} 92}
88 93
94static int magic_dual_mode(struct usb_device *udev)
95{
96 char *dummy_buffer = kzalloc(2, GFP_KERNEL);
97 int retval;
98
99 if (!dummy_buffer)
100 return -ENOMEM;
101
102 /* send magic command so that the Blackberry Pearl device exposes
103 * two interfaces: both the USB mass-storage one and one which can
104 * be used for database access. */
105 dbg(&udev->dev, "Sending magic pearl command\n");
106 retval = usb_control_msg(udev, usb_rcvctrlpipe(udev, 0),
107 0xa9, 0xc0, 1, 1, dummy_buffer, 2, 100);
108 dbg(&udev->dev, "Magic pearl command returned %d\n", retval);
109
110 dbg(&udev->dev, "Calling set_configuration\n");
111 retval = usb_driver_set_configuration(udev, 1);
112 if (retval)
113 dev_err(&udev->dev, "Set Configuration failed :%d.\n", retval);
114
115 return retval;
116}
117
89static int berry_probe(struct usb_interface *intf, 118static int berry_probe(struct usb_interface *intf,
90 const struct usb_device_id *id) 119 const struct usb_device_id *id)
91{ 120{
@@ -105,6 +134,10 @@ static int berry_probe(struct usb_interface *intf,
105 /* turn the power on */ 134 /* turn the power on */
106 magic_charge(udev); 135 magic_charge(udev);
107 136
137 if ((le16_to_cpu(udev->descriptor.idProduct) == BLACKBERRY_PEARL) &&
138 (pearl_dual_mode))
139 magic_dual_mode(udev);
140
108 /* we don't really want to bind to the device, userspace programs can 141 /* we don't really want to bind to the device, userspace programs can
109 * handle the syncing just fine, so get outta here. */ 142 * handle the syncing just fine, so get outta here. */
110 return -ENODEV; 143 return -ENODEV;
@@ -138,3 +171,5 @@ MODULE_LICENSE("GPL");
138MODULE_AUTHOR("Greg Kroah-Hartman <gregkh@suse.de>"); 171MODULE_AUTHOR("Greg Kroah-Hartman <gregkh@suse.de>");
139module_param(debug, bool, S_IRUGO | S_IWUSR); 172module_param(debug, bool, S_IRUGO | S_IWUSR);
140MODULE_PARM_DESC(debug, "Debug enabled or not"); 173MODULE_PARM_DESC(debug, "Debug enabled or not");
174module_param(pearl_dual_mode, bool, S_IRUGO | S_IWUSR);
175MODULE_PARM_DESC(pearl_dual_mode, "Change Blackberry Pearl to run in dual mode");