diff options
| -rw-r--r-- | drivers/hid/Kconfig | 7 | ||||
| -rw-r--r-- | drivers/hid/Makefile | 1 | ||||
| -rw-r--r-- | drivers/hid/hid-core.c | 1 | ||||
| -rw-r--r-- | drivers/hid/hid-dummy.c | 3 | ||||
| -rw-r--r-- | drivers/hid/hid-sony.c | 111 | ||||
| -rw-r--r-- | drivers/hid/usbhid/hid-core.c | 30 | ||||
| -rw-r--r-- | drivers/hid/usbhid/hid-quirks.c | 2 | ||||
| -rw-r--r-- | include/linux/hid.h | 1 | 
8 files changed, 123 insertions, 33 deletions
| diff --git a/drivers/hid/Kconfig b/drivers/hid/Kconfig index 52b617bf2b9..7ced6476026 100644 --- a/drivers/hid/Kconfig +++ b/drivers/hid/Kconfig @@ -182,6 +182,13 @@ config HID_SAMSUNG  	---help---  	Support for Samsung IR remote. +config HID_SONY +	tristate "Sony" +	default m +	depends on USB_HID +	---help--- +	Support for Sony PS3 controller. +  config HID_SUNPLUS  	tristate "Sunplus"  	default m diff --git a/drivers/hid/Makefile b/drivers/hid/Makefile index 0141ff88008..4f39b9431eb 100644 --- a/drivers/hid/Makefile +++ b/drivers/hid/Makefile @@ -25,6 +25,7 @@ obj-$(CONFIG_HID_MICROSOFT)	+= hid-microsoft.o  obj-$(CONFIG_HID_MONTEREY)	+= hid-monterey.o  obj-$(CONFIG_HID_PETALYNX)	+= hid-petalynx.o  obj-$(CONFIG_HID_SAMSUNG)	+= hid-samsung.o +obj-$(CONFIG_HID_SONY)		+= hid-sony.o  obj-$(CONFIG_HID_SUNPLUS)	+= hid-sunplus.o  obj-$(CONFIG_USB_HID)		+= usbhid/ diff --git a/drivers/hid/hid-core.c b/drivers/hid/hid-core.c index 1b43af072ce..f527e332b59 100644 --- a/drivers/hid/hid-core.c +++ b/drivers/hid/hid-core.c @@ -1195,6 +1195,7 @@ static const struct hid_device_id hid_blacklist[] = {  	{ HID_USB_DEVICE(USB_VENDOR_ID_MONTEREY, USB_DEVICE_ID_GENIUS_KB29E) },  	{ HID_USB_DEVICE(USB_VENDOR_ID_PETALYNX, USB_DEVICE_ID_PETALYNX_MAXTER_REMOTE) },  	{ HID_USB_DEVICE(USB_VENDOR_ID_SAMSUNG, USB_DEVICE_ID_SAMSUNG_IR_REMOTE) }, +	{ HID_USB_DEVICE(USB_VENDOR_ID_SONY, USB_DEVICE_ID_SONY_PS3_CONTROLLER) },  	{ HID_USB_DEVICE(USB_VENDOR_ID_SUNPLUS, USB_DEVICE_ID_SUNPLUS_WDESKTOP) },  	{ HID_BLUETOOTH_DEVICE(USB_VENDOR_ID_APPLE, 0x030c) }, diff --git a/drivers/hid/hid-dummy.c b/drivers/hid/hid-dummy.c index cd21dcc4181..1ef3111f7fe 100644 --- a/drivers/hid/hid-dummy.c +++ b/drivers/hid/hid-dummy.c @@ -43,6 +43,9 @@ static int __init hid_dummy_init(void)  #ifdef CONFIG_HID_SAMSUNG_MODULE  	HID_COMPAT_CALL_DRIVER(samsung);  #endif +#ifdef CONFIG_HID_SONY_MODULE +	HID_COMPAT_CALL_DRIVER(sony); +#endif  #ifdef CONFIG_HID_SUNPLUS_MODULE  	HID_COMPAT_CALL_DRIVER(sunplus);  #endif diff --git a/drivers/hid/hid-sony.c b/drivers/hid/hid-sony.c new file mode 100644 index 00000000000..97668c68f0a --- /dev/null +++ b/drivers/hid/hid-sony.c @@ -0,0 +1,111 @@ +/* + *  HID driver for some sony "special" devices + * + *  Copyright (c) 1999 Andreas Gal + *  Copyright (c) 2000-2005 Vojtech Pavlik <vojtech@suse.cz> + *  Copyright (c) 2005 Michael Haboustak <mike-@cinci.rr.com> for Concept2, Inc + *  Copyright (c) 2006-2007 Jiri Kosina + *  Copyright (c) 2007 Paul Walmsley + *  Copyright (c) 2008 Jiri Slaby + */ + +/* + * 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; either version 2 of the License, or (at your option) + * any later version. + */ + +#include <linux/device.h> +#include <linux/hid.h> +#include <linux/module.h> +#include <linux/usb.h> + +#include "hid-ids.h" + +/* + * Sending HID_REQ_GET_REPORT changes the operation mode of the ps3 controller + * to "operational".  Without this, the ps3 controller will not report any + * events. + */ +static int sony_set_operational(struct hid_device *hdev) +{ +	struct usb_interface *intf = to_usb_interface(hdev->dev.parent); +	struct usb_device *dev = interface_to_usbdev(intf); +	__u16 ifnum = intf->cur_altsetting->desc.bInterfaceNumber; +	int ret; +	char *buf = kmalloc(18, GFP_KERNEL); + +	if (!buf) +		return -ENOMEM; + +	ret = usb_control_msg(dev, usb_rcvctrlpipe(dev, 0), +				 HID_REQ_GET_REPORT, +				 USB_DIR_IN | USB_TYPE_CLASS | +				 USB_RECIP_INTERFACE, +				 (3 << 8) | 0xf2, ifnum, buf, 17, +				 USB_CTRL_GET_TIMEOUT); +	if (ret < 0) +		dev_err(&hdev->dev, "can't set operational mode\n"); + +	kfree(buf); + +	return ret; +} + +static int sony_probe(struct hid_device *hdev, const struct hid_device_id *id) +{ +	int ret; + +	hdev->quirks |= HID_QUIRK_HIDDEV; + +	ret = hid_parse(hdev); +	if (ret) { +		dev_err(&hdev->dev, "parse failed\n"); +		goto err_free; +	} + +	ret = hid_hw_start(hdev); +	if (ret) { +		dev_err(&hdev->dev, "hw start failed\n"); +		goto err_free; +	} + +	ret = sony_set_operational(hdev); +	if (ret) +		goto err_stop; + +	return 0; +err_stop: +	hid_hw_stop(hdev); +err_free: +	return ret; +} + +static const struct hid_device_id sony_devices[] = { +	{ HID_USB_DEVICE(USB_VENDOR_ID_SONY, USB_DEVICE_ID_SONY_PS3_CONTROLLER) }, +	{ } +}; +MODULE_DEVICE_TABLE(hid, sony_devices); + +static struct hid_driver sony_driver = { +	.name = "sony", +	.id_table = sony_devices, +	.probe = sony_probe, +}; + +static int sony_init(void) +{ +	return hid_register_driver(&sony_driver); +} + +static void sony_exit(void) +{ +	hid_unregister_driver(&sony_driver); +} + +module_init(sony_init); +module_exit(sony_exit); +MODULE_LICENSE("GPL"); + +HID_COMPAT_LOAD_DRIVER(sony); diff --git a/drivers/hid/usbhid/hid-core.c b/drivers/hid/usbhid/hid-core.c index e900d597bc2..b41d0110a75 100644 --- a/drivers/hid/usbhid/hid-core.c +++ b/drivers/hid/usbhid/hid-core.c @@ -670,32 +670,6 @@ static void hid_free_buffers(struct usb_device *dev, struct hid_device *hid)  	usb_buffer_free(dev, usbhid->bufsize, usbhid->ctrlbuf, usbhid->ctrlbuf_dma);  } -/* - * Sending HID_REQ_GET_REPORT changes the operation mode of the ps3 controller - * to "operational".  Without this, the ps3 controller will not report any - * events. - */ -static void hid_fixup_sony_ps3_controller(struct usb_device *dev, int ifnum) -{ -	int result; -	char *buf = kmalloc(18, GFP_KERNEL); - -	if (!buf) -		return; - -	result = usb_control_msg(dev, usb_rcvctrlpipe(dev, 0), -				 HID_REQ_GET_REPORT, -				 USB_DIR_IN | USB_TYPE_CLASS | -				 USB_RECIP_INTERFACE, -				 (3 << 8) | 0xf2, ifnum, buf, 17, -				 USB_CTRL_GET_TIMEOUT); - -	if (result < 0) -		err_hid("%s failed: %d\n", __func__, result); - -	kfree(buf); -} -  static int usbhid_start_finish(struct hid_device *hid)  {  	struct usb_interface *intf = to_usb_interface(hid->dev.parent); @@ -723,10 +697,6 @@ static int usbhid_start_finish(struct hid_device *hid)  	if ((hid->claimed & HID_CLAIMED_INPUT))  		hid_ff_init(hid); -	if (hid->quirks & HID_QUIRK_SONY_PS3_CONTROLLER) -		hid_fixup_sony_ps3_controller(interface_to_usbdev(intf), -			intf->cur_altsetting->desc.bInterfaceNumber); -  	printk(KERN_INFO);  	if (hid->claimed & HID_CLAIMED_INPUT) diff --git a/drivers/hid/usbhid/hid-quirks.c b/drivers/hid/usbhid/hid-quirks.c index f66d2e43b5d..a154a7dc1e6 100644 --- a/drivers/hid/usbhid/hid-quirks.c +++ b/drivers/hid/usbhid/hid-quirks.c @@ -46,8 +46,6 @@ static const struct hid_blacklist {  	{ USB_VENDOR_ID_PANTHERLORD, USB_DEVICE_ID_PANTHERLORD_TWIN_USB_JOYSTICK, HID_QUIRK_MULTI_INPUT | HID_QUIRK_SKIP_OUTPUT_REPORTS },  	{ USB_VENDOR_ID_PLAYDOTCOM, USB_DEVICE_ID_PLAYDOTCOM_EMS_USBII, HID_QUIRK_MULTI_INPUT }, -	{ USB_VENDOR_ID_SONY, USB_DEVICE_ID_SONY_PS3_CONTROLLER, HID_QUIRK_SONY_PS3_CONTROLLER | HID_QUIRK_HIDDEV }, -  	{ USB_VENDOR_ID_ATEN, USB_DEVICE_ID_ATEN_UC100KM, HID_QUIRK_NOGET },  	{ USB_VENDOR_ID_ATEN, USB_DEVICE_ID_ATEN_CS124U, HID_QUIRK_NOGET },  	{ USB_VENDOR_ID_ATEN, USB_DEVICE_ID_ATEN_2PORTKVM, HID_QUIRK_NOGET }, diff --git a/include/linux/hid.h b/include/linux/hid.h index 9eac330a1df..43aa51a7fa9 100644 --- a/include/linux/hid.h +++ b/include/linux/hid.h @@ -262,7 +262,6 @@ struct hid_item {  #define HID_QUIRK_BADPAD			0x00000020  #define HID_QUIRK_MULTI_INPUT			0x00000040  #define HID_QUIRK_SKIP_OUTPUT_REPORTS		0x00010000 -#define HID_QUIRK_SONY_PS3_CONTROLLER		0x00040000  #define HID_QUIRK_RESET_LEDS			0x00100000  #define HID_QUIRK_HIDINPUT			0x00200000  #define HID_QUIRK_IGNORE_HIDINPUT		0x01000000 | 
