2011-01-30 20:38:23 +08:00
|
|
|
/*
|
|
|
|
* Roccat common functions for device specific drivers
|
|
|
|
*
|
|
|
|
* Copyright (c) 2011 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.
|
|
|
|
*/
|
|
|
|
|
2011-06-01 21:54:17 +08:00
|
|
|
#include <linux/hid.h>
|
2011-01-30 20:38:23 +08:00
|
|
|
#include <linux/slab.h>
|
2011-07-04 01:39:48 +08:00
|
|
|
#include <linux/module.h>
|
2011-01-30 20:38:23 +08:00
|
|
|
#include "hid-roccat-common.h"
|
|
|
|
|
2011-06-01 21:54:17 +08:00
|
|
|
static inline uint16_t roccat_common_feature_report(uint8_t report_id)
|
|
|
|
{
|
|
|
|
return 0x300 | report_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
int roccat_common_receive(struct usb_device *usb_dev, uint report_id,
|
2011-01-30 20:38:23 +08:00
|
|
|
void *data, uint size)
|
|
|
|
{
|
|
|
|
char *buf;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
buf = kmalloc(size, GFP_KERNEL);
|
|
|
|
if (buf == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
len = usb_control_msg(usb_dev, usb_rcvctrlpipe(usb_dev, 0),
|
2011-06-01 21:54:17 +08:00
|
|
|
HID_REQ_GET_REPORT,
|
2011-01-30 20:38:23 +08:00
|
|
|
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_IN,
|
2011-06-01 21:54:17 +08:00
|
|
|
roccat_common_feature_report(report_id),
|
|
|
|
0, buf, size, USB_CTRL_SET_TIMEOUT);
|
2011-01-30 20:38:23 +08:00
|
|
|
|
|
|
|
memcpy(data, buf, size);
|
|
|
|
kfree(buf);
|
|
|
|
return ((len < 0) ? len : ((len != size) ? -EIO : 0));
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(roccat_common_receive);
|
|
|
|
|
2011-06-01 21:54:17 +08:00
|
|
|
int roccat_common_send(struct usb_device *usb_dev, uint report_id,
|
2011-01-30 20:38:23 +08:00
|
|
|
void const *data, uint size)
|
|
|
|
{
|
|
|
|
char *buf;
|
|
|
|
int len;
|
|
|
|
|
2011-11-18 06:43:40 +08:00
|
|
|
buf = kmemdup(data, size, GFP_KERNEL);
|
2011-01-30 20:38:23 +08:00
|
|
|
if (buf == NULL)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
len = usb_control_msg(usb_dev, usb_sndctrlpipe(usb_dev, 0),
|
2011-06-01 21:54:17 +08:00
|
|
|
HID_REQ_SET_REPORT,
|
2011-01-30 20:38:23 +08:00
|
|
|
USB_TYPE_CLASS | USB_RECIP_INTERFACE | USB_DIR_OUT,
|
2011-06-01 21:54:17 +08:00
|
|
|
roccat_common_feature_report(report_id),
|
|
|
|
0, buf, size, USB_CTRL_SET_TIMEOUT);
|
2011-01-30 20:38:23 +08:00
|
|
|
|
|
|
|
kfree(buf);
|
|
|
|
return ((len < 0) ? len : ((len != size) ? -EIO : 0));
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(roccat_common_send);
|
|
|
|
|
|
|
|
MODULE_AUTHOR("Stefan Achatz");
|
|
|
|
MODULE_DESCRIPTION("USB Roccat common driver");
|
|
|
|
MODULE_LICENSE("GPL v2");
|