2010-08-29 12:30:18 +02:00
/*
* Roccat Pyra driver for Linux
*
* Copyright ( c ) 2010 Stefan Achatz < erazor_de @ users . sourceforge . net >
*/
/*
* 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 .
*/
/*
* Roccat Pyra is a mobile gamer mouse which comes in wired and wireless
* variant . Wireless variant is not tested .
* Userland tools can be found at http : //sourceforge.net/projects/roccat
*/
# include <linux/device.h>
# include <linux/input.h>
# include <linux/hid.h>
# include <linux/usb.h>
# include <linux/module.h>
# include <linux/slab.h>
# include "hid-ids.h"
# include "hid-roccat.h"
# include "hid-roccat-pyra.h"
2010-11-26 19:57:38 +00:00
static uint profile_numbers [ 5 ] = { 0 , 1 , 2 , 3 , 4 } ;
2010-11-26 19:57:33 +00:00
/* pyra_class is used for creating sysfs attributes via roccat char device */
static struct class * pyra_class ;
2010-08-29 12:30:18 +02:00
static void profile_activated ( struct pyra_device * pyra ,
unsigned int new_profile )
{
pyra - > actual_profile = new_profile ;
pyra - > actual_cpi = pyra - > profile_settings [ pyra - > actual_profile ] . y_cpi ;
}
static int pyra_send_control ( struct usb_device * usb_dev , int value ,
enum pyra_control_requests request )
{
int len ;
struct pyra_control control ;
if ( ( request = = PYRA_CONTROL_REQUEST_PROFILE_SETTINGS | |
request = = PYRA_CONTROL_REQUEST_PROFILE_BUTTONS ) & &
( value < 0 | | value > 4 ) )
return - EINVAL ;
control . command = PYRA_COMMAND_CONTROL ;
control . value = value ;
control . request = request ;
len = usb_control_msg ( usb_dev , usb_sndctrlpipe ( usb_dev , 0 ) ,
USB_REQ_SET_CONFIGURATION ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT ,
PYRA_USB_COMMAND_CONTROL , 0 , ( char * ) & control ,
sizeof ( struct pyra_control ) ,
USB_CTRL_SET_TIMEOUT ) ;
if ( len ! = sizeof ( struct pyra_control ) )
return len ;
return 0 ;
}
static int pyra_receive_control_status ( struct usb_device * usb_dev )
{
int len ;
struct pyra_control control ;
do {
msleep ( 10 ) ;
len = usb_control_msg ( usb_dev , usb_rcvctrlpipe ( usb_dev , 0 ) ,
USB_REQ_CLEAR_FEATURE ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE |
USB_DIR_IN ,
PYRA_USB_COMMAND_CONTROL , 0 , ( char * ) & control ,
sizeof ( struct pyra_control ) ,
USB_CTRL_SET_TIMEOUT ) ;
/* requested too early, try again */
} while ( len = = - EPROTO ) ;
if ( len = = sizeof ( struct pyra_control ) & &
control . command = = PYRA_COMMAND_CONTROL & &
control . request = = PYRA_CONTROL_REQUEST_STATUS & &
control . value = = 1 )
return 0 ;
else {
2010-12-09 19:29:03 -08:00
hid_err ( usb_dev , " receive control status: unknown response 0x%x 0x%x \n " ,
control . request , control . value ) ;
2010-08-29 12:30:18 +02:00
return - EINVAL ;
}
}
static int pyra_get_profile_settings ( struct usb_device * usb_dev ,
struct pyra_profile_settings * buf , int number )
{
int retval ;
retval = pyra_send_control ( usb_dev , number ,
PYRA_CONTROL_REQUEST_PROFILE_SETTINGS ) ;
if ( retval )
return retval ;
retval = usb_control_msg ( usb_dev , usb_rcvctrlpipe ( usb_dev , 0 ) ,
USB_REQ_CLEAR_FEATURE ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN ,
PYRA_USB_COMMAND_PROFILE_SETTINGS , 0 , ( char * ) buf ,
sizeof ( struct pyra_profile_settings ) ,
USB_CTRL_SET_TIMEOUT ) ;
if ( retval ! = sizeof ( struct pyra_profile_settings ) )
return retval ;
return 0 ;
}
static int pyra_get_profile_buttons ( struct usb_device * usb_dev ,
struct pyra_profile_buttons * buf , int number )
{
int retval ;
retval = pyra_send_control ( usb_dev , number ,
PYRA_CONTROL_REQUEST_PROFILE_BUTTONS ) ;
if ( retval )
return retval ;
retval = usb_control_msg ( usb_dev , usb_rcvctrlpipe ( usb_dev , 0 ) ,
USB_REQ_CLEAR_FEATURE ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN ,
PYRA_USB_COMMAND_PROFILE_BUTTONS , 0 , ( char * ) buf ,
sizeof ( struct pyra_profile_buttons ) ,
USB_CTRL_SET_TIMEOUT ) ;
if ( retval ! = sizeof ( struct pyra_profile_buttons ) )
return retval ;
return 0 ;
}
static int pyra_get_settings ( struct usb_device * usb_dev ,
struct pyra_settings * buf )
{
int len ;
len = usb_control_msg ( usb_dev , usb_rcvctrlpipe ( usb_dev , 0 ) ,
USB_REQ_CLEAR_FEATURE ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN ,
PYRA_USB_COMMAND_SETTINGS , 0 , buf ,
sizeof ( struct pyra_settings ) , USB_CTRL_SET_TIMEOUT ) ;
if ( len ! = sizeof ( struct pyra_settings ) )
return - EIO ;
return 0 ;
}
static int pyra_get_info ( struct usb_device * usb_dev , struct pyra_info * buf )
{
int len ;
len = usb_control_msg ( usb_dev , usb_rcvctrlpipe ( usb_dev , 0 ) ,
USB_REQ_CLEAR_FEATURE ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN ,
PYRA_USB_COMMAND_INFO , 0 , buf ,
sizeof ( struct pyra_info ) , USB_CTRL_SET_TIMEOUT ) ;
if ( len ! = sizeof ( struct pyra_info ) )
return - EIO ;
return 0 ;
}
static int pyra_set_profile_settings ( struct usb_device * usb_dev ,
struct pyra_profile_settings const * settings )
{
int len ;
len = usb_control_msg ( usb_dev , usb_sndctrlpipe ( usb_dev , 0 ) ,
USB_REQ_SET_CONFIGURATION ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT ,
PYRA_USB_COMMAND_PROFILE_SETTINGS , 0 , ( char * ) settings ,
sizeof ( struct pyra_profile_settings ) ,
USB_CTRL_SET_TIMEOUT ) ;
if ( len ! = sizeof ( struct pyra_profile_settings ) )
return - EIO ;
if ( pyra_receive_control_status ( usb_dev ) )
return - EIO ;
return 0 ;
}
static int pyra_set_profile_buttons ( struct usb_device * usb_dev ,
struct pyra_profile_buttons const * buttons )
{
int len ;
len = usb_control_msg ( usb_dev , usb_sndctrlpipe ( usb_dev , 0 ) ,
USB_REQ_SET_CONFIGURATION ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT ,
PYRA_USB_COMMAND_PROFILE_BUTTONS , 0 , ( char * ) buttons ,
sizeof ( struct pyra_profile_buttons ) ,
USB_CTRL_SET_TIMEOUT ) ;
if ( len ! = sizeof ( struct pyra_profile_buttons ) )
return - EIO ;
if ( pyra_receive_control_status ( usb_dev ) )
return - EIO ;
return 0 ;
}
static int pyra_set_settings ( struct usb_device * usb_dev ,
struct pyra_settings const * settings )
{
int len ;
len = usb_control_msg ( usb_dev , usb_sndctrlpipe ( usb_dev , 0 ) ,
USB_REQ_SET_CONFIGURATION ,
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT ,
PYRA_USB_COMMAND_SETTINGS , 0 , ( char * ) settings ,
sizeof ( struct pyra_settings ) , USB_CTRL_SET_TIMEOUT ) ;
if ( len ! = sizeof ( struct pyra_settings ) )
return - EIO ;
if ( pyra_receive_control_status ( usb_dev ) )
return - EIO ;
return 0 ;
}
static ssize_t pyra_sysfs_read_profilex_settings ( struct file * fp ,
struct kobject * kobj , struct bin_attribute * attr , char * buf ,
2010-11-26 19:57:38 +00:00
loff_t off , size_t count )
2010-08-29 12:30:18 +02:00
{
2010-11-26 19:57:33 +00:00
struct device * dev =
container_of ( kobj , struct device , kobj ) - > parent - > parent ;
2010-08-29 12:30:18 +02:00
struct pyra_device * pyra = hid_get_drvdata ( dev_get_drvdata ( dev ) ) ;
if ( off > = sizeof ( struct pyra_profile_settings ) )
return 0 ;
if ( off + count > sizeof ( struct pyra_profile_settings ) )
count = sizeof ( struct pyra_profile_settings ) - off ;
mutex_lock ( & pyra - > pyra_lock ) ;
2010-11-26 19:57:38 +00:00
memcpy ( buf , ( ( char const * ) & pyra - > profile_settings [ * ( uint * ) ( attr - > private ) ] ) + off ,
2010-08-29 12:30:18 +02:00
count ) ;
mutex_unlock ( & pyra - > pyra_lock ) ;
return count ;
}
static ssize_t pyra_sysfs_read_profilex_buttons ( struct file * fp ,
struct kobject * kobj , struct bin_attribute * attr , char * buf ,
2010-11-26 19:57:38 +00:00
loff_t off , size_t count )
2010-08-29 12:30:18 +02:00
{
2010-11-26 19:57:33 +00:00
struct device * dev =
container_of ( kobj , struct device , kobj ) - > parent - > parent ;
2010-08-29 12:30:18 +02:00
struct pyra_device * pyra = hid_get_drvdata ( dev_get_drvdata ( dev ) ) ;
if ( off > = sizeof ( struct pyra_profile_buttons ) )
return 0 ;
if ( off + count > sizeof ( struct pyra_profile_buttons ) )
count = sizeof ( struct pyra_profile_buttons ) - off ;
mutex_lock ( & pyra - > pyra_lock ) ;
2010-11-26 19:57:38 +00:00
memcpy ( buf , ( ( char const * ) & pyra - > profile_buttons [ * ( uint * ) ( attr - > private ) ] ) + off ,
2010-08-29 12:30:18 +02:00
count ) ;
mutex_unlock ( & pyra - > pyra_lock ) ;
return count ;
}
static ssize_t pyra_sysfs_write_profile_settings ( struct file * fp ,
struct kobject * kobj , struct bin_attribute * attr , char * buf ,
loff_t off , size_t count )
{
2010-11-26 19:57:33 +00:00
struct device * dev =
container_of ( kobj , struct device , kobj ) - > parent - > parent ;
2010-08-29 12:30:18 +02:00
struct pyra_device * pyra = hid_get_drvdata ( dev_get_drvdata ( dev ) ) ;
struct usb_device * usb_dev = interface_to_usbdev ( to_usb_interface ( dev ) ) ;
int retval = 0 ;
int difference ;
int profile_number ;
struct pyra_profile_settings * profile_settings ;
if ( off ! = 0 | | count ! = sizeof ( struct pyra_profile_settings ) )
return - EINVAL ;
profile_number = ( ( struct pyra_profile_settings const * ) buf ) - > number ;
profile_settings = & pyra - > profile_settings [ profile_number ] ;
mutex_lock ( & pyra - > pyra_lock ) ;
difference = memcmp ( buf , profile_settings ,
sizeof ( struct pyra_profile_settings ) ) ;
if ( difference ) {
retval = pyra_set_profile_settings ( usb_dev ,
( struct pyra_profile_settings const * ) buf ) ;
if ( ! retval )
memcpy ( profile_settings , buf ,
sizeof ( struct pyra_profile_settings ) ) ;
}
mutex_unlock ( & pyra - > pyra_lock ) ;
if ( retval )
return retval ;
return sizeof ( struct pyra_profile_settings ) ;
}
static ssize_t pyra_sysfs_write_profile_buttons ( struct file * fp ,
struct kobject * kobj , struct bin_attribute * attr , char * buf ,
loff_t off , size_t count )
{
2010-11-26 19:57:33 +00:00
struct device * dev =
container_of ( kobj , struct device , kobj ) - > parent - > parent ;
2010-08-29 12:30:18 +02:00
struct pyra_device * pyra = hid_get_drvdata ( dev_get_drvdata ( dev ) ) ;
struct usb_device * usb_dev = interface_to_usbdev ( to_usb_interface ( dev ) ) ;
int retval = 0 ;
int difference ;
int profile_number ;
struct pyra_profile_buttons * profile_buttons ;
if ( off ! = 0 | | count ! = sizeof ( struct pyra_profile_buttons ) )
return - EINVAL ;
profile_number = ( ( struct pyra_profile_buttons const * ) buf ) - > number ;
profile_buttons = & pyra - > profile_buttons [ profile_number ] ;
mutex_lock ( & pyra - > pyra_lock ) ;
difference = memcmp ( buf , profile_buttons ,
sizeof ( struct pyra_profile_buttons ) ) ;
if ( difference ) {
retval = pyra_set_profile_buttons ( usb_dev ,
( struct pyra_profile_buttons const * ) buf ) ;
if ( ! retval )
memcpy ( profile_buttons , buf ,
sizeof ( struct pyra_profile_buttons ) ) ;
}
mutex_unlock ( & pyra - > pyra_lock ) ;
if ( retval )
return retval ;
return sizeof ( struct pyra_profile_buttons ) ;
}
static ssize_t pyra_sysfs_read_settings ( struct file * fp ,
struct kobject * kobj , struct bin_attribute * attr , char * buf ,
loff_t off , size_t count )
{
2010-11-26 19:57:33 +00:00
struct device * dev =
container_of ( kobj , struct device , kobj ) - > parent - > parent ;
2010-08-29 12:30:18 +02:00
struct pyra_device * pyra = hid_get_drvdata ( dev_get_drvdata ( dev ) ) ;
if ( off > = sizeof ( struct pyra_settings ) )
return 0 ;
if ( off + count > sizeof ( struct pyra_settings ) )
count = sizeof ( struct pyra_settings ) - off ;
mutex_lock ( & pyra - > pyra_lock ) ;
memcpy ( buf , ( ( char const * ) & pyra - > settings ) + off , count ) ;
mutex_unlock ( & pyra - > pyra_lock ) ;
return count ;
}
static ssize_t pyra_sysfs_write_settings ( struct file * fp ,
struct kobject * kobj , struct bin_attribute * attr , char * buf ,
loff_t off , size_t count )
{
2010-11-26 19:57:33 +00:00
struct device * dev =
container_of ( kobj , struct device , kobj ) - > parent - > parent ;
2010-08-29 12:30:18 +02:00
struct pyra_device * pyra = hid_get_drvdata ( dev_get_drvdata ( dev ) ) ;
struct usb_device * usb_dev = interface_to_usbdev ( to_usb_interface ( dev ) ) ;
int retval = 0 ;
int difference ;
if ( off ! = 0 | | count ! = sizeof ( struct pyra_settings ) )
return - EINVAL ;
mutex_lock ( & pyra - > pyra_lock ) ;
difference = memcmp ( buf , & pyra - > settings , sizeof ( struct pyra_settings ) ) ;
if ( difference ) {
retval = pyra_set_settings ( usb_dev ,
( struct pyra_settings const * ) buf ) ;
if ( ! retval )
memcpy ( & pyra - > settings , buf ,
sizeof ( struct pyra_settings ) ) ;
}
mutex_unlock ( & pyra - > pyra_lock ) ;
if ( retval )
return retval ;
profile_activated ( pyra , pyra - > settings . startup_profile ) ;
return sizeof ( struct pyra_settings ) ;
}
static ssize_t pyra_sysfs_show_actual_cpi ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
2010-11-26 19:57:33 +00:00
struct pyra_device * pyra =
hid_get_drvdata ( dev_get_drvdata ( dev - > parent - > parent ) ) ;
2010-08-29 12:30:18 +02:00
return snprintf ( buf , PAGE_SIZE , " %d \n " , pyra - > actual_cpi ) ;
}
static ssize_t pyra_sysfs_show_actual_profile ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
2010-11-26 19:57:33 +00:00
struct pyra_device * pyra =
hid_get_drvdata ( dev_get_drvdata ( dev - > parent - > parent ) ) ;
2010-08-29 12:30:18 +02:00
return snprintf ( buf , PAGE_SIZE , " %d \n " , pyra - > actual_profile ) ;
}
static ssize_t pyra_sysfs_show_firmware_version ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
2010-11-26 19:57:33 +00:00
struct pyra_device * pyra =
hid_get_drvdata ( dev_get_drvdata ( dev - > parent - > parent ) ) ;
2010-08-29 12:30:18 +02:00
return snprintf ( buf , PAGE_SIZE , " %d \n " , pyra - > firmware_version ) ;
}
static ssize_t pyra_sysfs_show_startup_profile ( struct device * dev ,
struct device_attribute * attr , char * buf )
{
2010-11-26 19:57:33 +00:00
struct pyra_device * pyra =
hid_get_drvdata ( dev_get_drvdata ( dev - > parent - > parent ) ) ;
2010-08-29 12:30:18 +02:00
return snprintf ( buf , PAGE_SIZE , " %d \n " , pyra - > settings . startup_profile ) ;
}
2010-11-26 19:57:33 +00:00
static struct device_attribute pyra_attributes [ ] = {
__ATTR ( actual_cpi , 0440 , pyra_sysfs_show_actual_cpi , NULL ) ,
__ATTR ( actual_profile , 0440 , pyra_sysfs_show_actual_profile , NULL ) ,
__ATTR ( firmware_version , 0440 ,
pyra_sysfs_show_firmware_version , NULL ) ,
__ATTR ( startup_profile , 0440 ,
pyra_sysfs_show_startup_profile , NULL ) ,
__ATTR_NULL
2010-08-29 12:30:18 +02:00
} ;
2010-11-26 19:57:33 +00:00
static struct bin_attribute pyra_bin_attributes [ ] = {
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile_settings " , . mode = 0220 } ,
. size = sizeof ( struct pyra_profile_settings ) ,
. write = pyra_sysfs_write_profile_settings
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile1_settings " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_settings ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_settings ,
. private = & profile_numbers [ 0 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile2_settings " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_settings ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_settings ,
. private = & profile_numbers [ 1 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile3_settings " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_settings ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_settings ,
. private = & profile_numbers [ 2 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile4_settings " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_settings ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_settings ,
. private = & profile_numbers [ 3 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile5_settings " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_settings ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_settings ,
. private = & profile_numbers [ 4 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile_buttons " , . mode = 0220 } ,
. size = sizeof ( struct pyra_profile_buttons ) ,
. write = pyra_sysfs_write_profile_buttons
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile1_buttons " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_buttons ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_buttons ,
. private = & profile_numbers [ 0 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile2_buttons " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_buttons ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_buttons ,
. private = & profile_numbers [ 1 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile3_buttons " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_buttons ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_buttons ,
. private = & profile_numbers [ 2 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile4_buttons " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_buttons ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_buttons ,
. private = & profile_numbers [ 3 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " profile5_buttons " , . mode = 0440 } ,
. size = sizeof ( struct pyra_profile_buttons ) ,
2010-11-26 19:57:38 +00:00
. read = pyra_sysfs_read_profilex_buttons ,
. private = & profile_numbers [ 4 ]
2010-11-26 19:57:33 +00:00
} ,
{
2010-08-29 12:30:18 +02:00
. attr = { . name = " settings " , . mode = 0660 } ,
. size = sizeof ( struct pyra_settings ) ,
. read = pyra_sysfs_read_settings ,
. write = pyra_sysfs_write_settings
2010-11-26 19:57:33 +00:00
} ,
__ATTR_NULL
2010-08-29 12:30:18 +02:00
} ;
static int pyra_init_pyra_device_struct ( struct usb_device * usb_dev ,
struct pyra_device * pyra )
{
struct pyra_info * info ;
int retval , i ;
mutex_init ( & pyra - > pyra_lock ) ;
info = kmalloc ( sizeof ( struct pyra_info ) , GFP_KERNEL ) ;
if ( ! info )
return - ENOMEM ;
retval = pyra_get_info ( usb_dev , info ) ;
if ( retval ) {
kfree ( info ) ;
return retval ;
}
pyra - > firmware_version = info - > firmware_version ;
kfree ( info ) ;
retval = pyra_get_settings ( usb_dev , & pyra - > settings ) ;
if ( retval )
return retval ;
for ( i = 0 ; i < 5 ; + + i ) {
retval = pyra_get_profile_settings ( usb_dev ,
& pyra - > profile_settings [ i ] , i ) ;
if ( retval )
return retval ;
retval = pyra_get_profile_buttons ( usb_dev ,
& pyra - > profile_buttons [ i ] , i ) ;
if ( retval )
return retval ;
}
profile_activated ( pyra , pyra - > settings . startup_profile ) ;
return 0 ;
}
static int pyra_init_specials ( struct hid_device * hdev )
{
struct usb_interface * intf = to_usb_interface ( hdev - > dev . parent ) ;
struct usb_device * usb_dev = interface_to_usbdev ( intf ) ;
struct pyra_device * pyra ;
int retval ;
if ( intf - > cur_altsetting - > desc . bInterfaceProtocol
= = USB_INTERFACE_PROTOCOL_MOUSE ) {
pyra = kzalloc ( sizeof ( * pyra ) , GFP_KERNEL ) ;
if ( ! pyra ) {
2010-12-09 19:29:03 -08:00
hid_err ( hdev , " can't alloc device descriptor \n " ) ;
2010-08-29 12:30:18 +02:00
return - ENOMEM ;
}
hid_set_drvdata ( hdev , pyra ) ;
retval = pyra_init_pyra_device_struct ( usb_dev , pyra ) ;
if ( retval ) {
2010-12-09 19:29:03 -08:00
hid_err ( hdev , " couldn't init struct pyra_device \n " ) ;
2010-08-29 12:30:18 +02:00
goto exit_free ;
}
2010-11-26 19:57:33 +00:00
retval = roccat_connect ( pyra_class , hdev ) ;
2010-08-29 12:30:18 +02:00
if ( retval < 0 ) {
2010-12-09 19:29:03 -08:00
hid_err ( hdev , " couldn't init char dev \n " ) ;
2010-08-29 12:30:18 +02:00
} else {
pyra - > chrdev_minor = retval ;
pyra - > roccat_claimed = 1 ;
}
} else {
hid_set_drvdata ( hdev , NULL ) ;
}
return 0 ;
exit_free :
kfree ( pyra ) ;
return retval ;
}
static void pyra_remove_specials ( struct hid_device * hdev )
{
struct usb_interface * intf = to_usb_interface ( hdev - > dev . parent ) ;
struct pyra_device * pyra ;
if ( intf - > cur_altsetting - > desc . bInterfaceProtocol
= = USB_INTERFACE_PROTOCOL_MOUSE ) {
pyra = hid_get_drvdata ( hdev ) ;
if ( pyra - > roccat_claimed )
roccat_disconnect ( pyra - > chrdev_minor ) ;
kfree ( hid_get_drvdata ( hdev ) ) ;
}
}
static int pyra_probe ( struct hid_device * hdev , const struct hid_device_id * id )
{
int retval ;
retval = hid_parse ( hdev ) ;
if ( retval ) {
2010-12-09 19:29:03 -08:00
hid_err ( hdev , " parse failed \n " ) ;
2010-08-29 12:30:18 +02:00
goto exit ;
}
retval = hid_hw_start ( hdev , HID_CONNECT_DEFAULT ) ;
if ( retval ) {
2010-12-09 19:29:03 -08:00
hid_err ( hdev , " hw start failed \n " ) ;
2010-08-29 12:30:18 +02:00
goto exit ;
}
retval = pyra_init_specials ( hdev ) ;
if ( retval ) {
2010-12-09 19:29:03 -08:00
hid_err ( hdev , " couldn't install mouse \n " ) ;
2010-08-29 12:30:18 +02:00
goto exit_stop ;
}
return 0 ;
exit_stop :
hid_hw_stop ( hdev ) ;
exit :
return retval ;
}
static void pyra_remove ( struct hid_device * hdev )
{
pyra_remove_specials ( hdev ) ;
hid_hw_stop ( hdev ) ;
}
static void pyra_keep_values_up_to_date ( struct pyra_device * pyra ,
u8 const * data )
{
struct pyra_mouse_event_button const * button_event ;
switch ( data [ 0 ] ) {
case PYRA_MOUSE_REPORT_NUMBER_BUTTON :
button_event = ( struct pyra_mouse_event_button const * ) data ;
switch ( button_event - > type ) {
case PYRA_MOUSE_EVENT_BUTTON_TYPE_PROFILE_2 :
profile_activated ( pyra , button_event - > data1 - 1 ) ;
break ;
case PYRA_MOUSE_EVENT_BUTTON_TYPE_CPI :
pyra - > actual_cpi = button_event - > data1 ;
break ;
}
break ;
}
}
static void pyra_report_to_chrdev ( struct pyra_device const * pyra ,
u8 const * data )
{
struct pyra_roccat_report roccat_report ;
struct pyra_mouse_event_button const * button_event ;
if ( data [ 0 ] ! = PYRA_MOUSE_REPORT_NUMBER_BUTTON )
return ;
button_event = ( struct pyra_mouse_event_button const * ) data ;
switch ( button_event - > type ) {
case PYRA_MOUSE_EVENT_BUTTON_TYPE_PROFILE_2 :
case PYRA_MOUSE_EVENT_BUTTON_TYPE_CPI :
roccat_report . type = button_event - > type ;
roccat_report . value = button_event - > data1 ;
roccat_report . key = 0 ;
roccat_report_event ( pyra - > chrdev_minor ,
( uint8_t const * ) & roccat_report ,
sizeof ( struct pyra_roccat_report ) ) ;
break ;
case PYRA_MOUSE_EVENT_BUTTON_TYPE_MACRO :
case PYRA_MOUSE_EVENT_BUTTON_TYPE_SHORTCUT :
case PYRA_MOUSE_EVENT_BUTTON_TYPE_QUICKLAUNCH :
if ( button_event - > data2 = = PYRA_MOUSE_EVENT_BUTTON_PRESS ) {
roccat_report . type = button_event - > type ;
roccat_report . key = button_event - > data1 ;
2010-09-01 12:42:23 +02:00
/*
* pyra reports profile numbers with range 1 - 5.
* Keeping this behaviour .
*/
roccat_report . value = pyra - > actual_profile + 1 ;
2010-08-29 12:30:18 +02:00
roccat_report_event ( pyra - > chrdev_minor ,
( uint8_t const * ) & roccat_report ,
sizeof ( struct pyra_roccat_report ) ) ;
}
break ;
}
}
static int pyra_raw_event ( struct hid_device * hdev , struct hid_report * report ,
u8 * data , int size )
{
struct usb_interface * intf = to_usb_interface ( hdev - > dev . parent ) ;
struct pyra_device * pyra = hid_get_drvdata ( hdev ) ;
if ( intf - > cur_altsetting - > desc . bInterfaceProtocol
! = USB_INTERFACE_PROTOCOL_MOUSE )
return 0 ;
pyra_keep_values_up_to_date ( pyra , data ) ;
if ( pyra - > roccat_claimed )
pyra_report_to_chrdev ( pyra , data ) ;
return 0 ;
}
static const struct hid_device_id pyra_devices [ ] = {
{ HID_USB_DEVICE ( USB_VENDOR_ID_ROCCAT ,
USB_DEVICE_ID_ROCCAT_PYRA_WIRED ) } ,
/* TODO add USB_DEVICE_ID_ROCCAT_PYRA_WIRELESS after testing */
{ }
} ;
MODULE_DEVICE_TABLE ( hid , pyra_devices ) ;
static struct hid_driver pyra_driver = {
. name = " pyra " ,
. id_table = pyra_devices ,
. probe = pyra_probe ,
. remove = pyra_remove ,
. raw_event = pyra_raw_event
} ;
static int __init pyra_init ( void )
{
2010-11-26 19:57:33 +00:00
int retval ;
/* class name has to be same as driver name */
pyra_class = class_create ( THIS_MODULE , " pyra " ) ;
if ( IS_ERR ( pyra_class ) )
return PTR_ERR ( pyra_class ) ;
pyra_class - > dev_attrs = pyra_attributes ;
pyra_class - > dev_bin_attrs = pyra_bin_attributes ;
retval = hid_register_driver ( & pyra_driver ) ;
if ( retval )
class_destroy ( pyra_class ) ;
return retval ;
2010-08-29 12:30:18 +02:00
}
static void __exit pyra_exit ( void )
{
2010-11-26 19:57:33 +00:00
class_destroy ( pyra_class ) ;
2010-08-29 12:30:18 +02:00
hid_unregister_driver ( & pyra_driver ) ;
}
module_init ( pyra_init ) ;
module_exit ( pyra_exit ) ;
MODULE_AUTHOR ( " Stefan Achatz " ) ;
MODULE_DESCRIPTION ( " USB Roccat Pyra driver " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;