2 * Roccat common functions for device specific drivers
4 * Copyright (c) 2011 Stefan Achatz <erazor_de@users.sourceforge.net>
8 * This program is free software; you can redistribute it and/or modify it
9 * under the terms of the GNU General Public License as published by the Free
10 * Software Foundation; either version 2 of the License, or (at your option)
14 #include <linux/hid.h>
15 #include <linux/slab.h>
16 #include <linux/module.h>
17 #include "hid-roccat-common.h"
19 static inline uint16_t roccat_common2_feature_report(uint8_t report_id
)
21 return 0x300 | report_id
;
24 int roccat_common2_receive(struct usb_device
*usb_dev
, uint report_id
,
25 void *data
, uint size
)
30 buf
= kmalloc(size
, GFP_KERNEL
);
34 len
= usb_control_msg(usb_dev
, usb_rcvctrlpipe(usb_dev
, 0),
36 USB_TYPE_CLASS
| USB_RECIP_INTERFACE
| USB_DIR_IN
,
37 roccat_common2_feature_report(report_id
),
38 0, buf
, size
, USB_CTRL_SET_TIMEOUT
);
40 memcpy(data
, buf
, size
);
42 return ((len
< 0) ? len
: ((len
!= size
) ? -EIO
: 0));
44 EXPORT_SYMBOL_GPL(roccat_common2_receive
);
46 int roccat_common2_send(struct usb_device
*usb_dev
, uint report_id
,
47 void const *data
, uint size
)
52 buf
= kmemdup(data
, size
, GFP_KERNEL
);
56 len
= usb_control_msg(usb_dev
, usb_sndctrlpipe(usb_dev
, 0),
58 USB_TYPE_CLASS
| USB_RECIP_INTERFACE
| USB_DIR_OUT
,
59 roccat_common2_feature_report(report_id
),
60 0, buf
, size
, USB_CTRL_SET_TIMEOUT
);
63 return ((len
< 0) ? len
: ((len
!= size
) ? -EIO
: 0));
65 EXPORT_SYMBOL_GPL(roccat_common2_send
);
67 enum roccat_common2_control_states
{
68 ROCCAT_COMMON_CONTROL_STATUS_OVERLOAD
= 0,
69 ROCCAT_COMMON_CONTROL_STATUS_OK
= 1,
70 ROCCAT_COMMON_CONTROL_STATUS_INVALID
= 2,
71 ROCCAT_COMMON_CONTROL_STATUS_WAIT
= 3,
74 static int roccat_common2_receive_control_status(struct usb_device
*usb_dev
)
77 struct roccat_common2_control control
;
81 retval
= roccat_common2_receive(usb_dev
,
82 ROCCAT_COMMON_COMMAND_CONTROL
,
83 &control
, sizeof(struct roccat_common2_control
));
88 switch (control
.value
) {
89 case ROCCAT_COMMON_CONTROL_STATUS_OK
:
91 case ROCCAT_COMMON_CONTROL_STATUS_WAIT
:
94 case ROCCAT_COMMON_CONTROL_STATUS_INVALID
:
96 case ROCCAT_COMMON_CONTROL_STATUS_OVERLOAD
:
97 /* seems to be critical - replug necessary */
100 dev_err(&usb_dev
->dev
,
101 "roccat_common2_receive_control_status: "
102 "unknown response value 0x%x\n",
110 int roccat_common2_send_with_status(struct usb_device
*usb_dev
,
111 uint command
, void const *buf
, uint size
)
115 retval
= roccat_common2_send(usb_dev
, command
, buf
, size
);
121 return roccat_common2_receive_control_status(usb_dev
);
123 EXPORT_SYMBOL_GPL(roccat_common2_send_with_status
);
125 MODULE_AUTHOR("Stefan Achatz");
126 MODULE_DESCRIPTION("USB Roccat common driver");
127 MODULE_LICENSE("GPL v2");