2019-05-27 08:55:05 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2010-11-01 15:13:37 -04:00
/*
* Force feedback support for EMS Trio Linker Plus II
*
* Copyright ( c ) 2010 Ignaz Forster < ignaz . forster @ gmx . de >
*/
/*
*/
# include <linux/hid.h>
# include <linux/input.h>
2011-07-03 13:39:48 -04:00
# include <linux/module.h>
2010-11-01 15:13:37 -04:00
# include "hid-ids.h"
struct emsff_device {
struct hid_report * report ;
} ;
static int emsff_play ( struct input_dev * dev , void * data ,
struct ff_effect * effect )
{
struct hid_device * hid = input_get_drvdata ( dev ) ;
struct emsff_device * emsff = data ;
int weak , strong ;
weak = effect - > u . rumble . weak_magnitude ;
strong = effect - > u . rumble . strong_magnitude ;
dbg_hid ( " called with 0x%04x 0x%04x \n " , strong , weak ) ;
weak = weak * 0xff / 0xffff ;
strong = strong * 0xff / 0xffff ;
emsff - > report - > field [ 0 ] - > value [ 1 ] = weak ;
emsff - > report - > field [ 0 ] - > value [ 2 ] = strong ;
dbg_hid ( " running with 0x%02x 0x%02x \n " , strong , weak ) ;
2013-02-25 11:31:46 +01:00
hid_hw_request ( hid , emsff - > report , HID_REQ_SET_REPORT ) ;
2010-11-01 15:13:37 -04:00
return 0 ;
}
static int emsff_init ( struct hid_device * hid )
{
struct emsff_device * emsff ;
struct hid_report * report ;
2019-10-03 14:53:59 -04:00
struct hid_input * hidinput ;
2010-11-01 15:13:37 -04:00
struct list_head * report_list =
& hid - > report_enum [ HID_OUTPUT_REPORT ] . report_list ;
2019-10-03 14:53:59 -04:00
struct input_dev * dev ;
2010-11-01 15:13:37 -04:00
int error ;
2019-10-03 14:53:59 -04:00
if ( list_empty ( & hid - > inputs ) ) {
hid_err ( hid , " no inputs found \n " ) ;
return - ENODEV ;
}
hidinput = list_first_entry ( & hid - > inputs , struct hid_input , list ) ;
dev = hidinput - > input ;
2010-11-01 15:13:37 -04:00
if ( list_empty ( report_list ) ) {
2010-12-09 19:29:03 -08:00
hid_err ( hid , " no output reports found \n " ) ;
2010-11-01 15:13:37 -04:00
return - ENODEV ;
}
report = list_first_entry ( report_list , struct hid_report , list ) ;
if ( report - > maxfield < 1 ) {
2010-12-09 19:29:03 -08:00
hid_err ( hid , " no fields in the report \n " ) ;
2010-11-01 15:13:37 -04:00
return - ENODEV ;
}
if ( report - > field [ 0 ] - > report_count < 7 ) {
2010-12-09 19:29:03 -08:00
hid_err ( hid , " not enough values in the field \n " ) ;
2010-11-01 15:13:37 -04:00
return - ENODEV ;
}
emsff = kzalloc ( sizeof ( struct emsff_device ) , GFP_KERNEL ) ;
if ( ! emsff )
return - ENOMEM ;
set_bit ( FF_RUMBLE , dev - > ffbit ) ;
error = input_ff_create_memless ( dev , emsff , emsff_play ) ;
if ( error ) {
kfree ( emsff ) ;
return error ;
}
emsff - > report = report ;
emsff - > report - > field [ 0 ] - > value [ 0 ] = 0x01 ;
emsff - > report - > field [ 0 ] - > value [ 1 ] = 0x00 ;
emsff - > report - > field [ 0 ] - > value [ 2 ] = 0x00 ;
emsff - > report - > field [ 0 ] - > value [ 3 ] = 0x00 ;
emsff - > report - > field [ 0 ] - > value [ 4 ] = 0x00 ;
emsff - > report - > field [ 0 ] - > value [ 5 ] = 0x00 ;
emsff - > report - > field [ 0 ] - > value [ 6 ] = 0x00 ;
2013-02-25 11:31:46 +01:00
hid_hw_request ( hid , emsff - > report , HID_REQ_SET_REPORT ) ;
2010-11-01 15:13:37 -04:00
2010-12-09 19:29:03 -08:00
hid_info ( hid , " force feedback for EMS based devices by Ignaz Forster <ignaz.forster@gmx.de> \n " ) ;
2010-11-01 15:13:37 -04:00
return 0 ;
}
static int ems_probe ( struct hid_device * hdev , const struct hid_device_id * id )
{
int ret ;
ret = hid_parse ( hdev ) ;
if ( ret ) {
2010-12-09 19:29:03 -08:00
hid_err ( hdev , " parse failed \n " ) ;
2010-11-01 15:13:37 -04:00
goto err ;
}
ret = hid_hw_start ( hdev , HID_CONNECT_DEFAULT & ~ HID_CONNECT_FF ) ;
if ( ret ) {
2010-12-09 19:29:03 -08:00
hid_err ( hdev , " hw start failed \n " ) ;
2010-11-01 15:13:37 -04:00
goto err ;
}
2011-07-19 16:14:04 +08:00
ret = emsff_init ( hdev ) ;
if ( ret ) {
dev_err ( & hdev - > dev , " force feedback init failed \n " ) ;
hid_hw_stop ( hdev ) ;
goto err ;
}
2010-11-01 15:13:37 -04:00
return 0 ;
err :
return ret ;
}
static const struct hid_device_id ems_devices [ ] = {
2011-12-21 00:29:19 +01:00
{ HID_USB_DEVICE ( USB_VENDOR_ID_EMS , USB_DEVICE_ID_EMS_TRIO_LINKER_PLUS_II ) } ,
2010-11-01 15:13:37 -04:00
{ }
} ;
MODULE_DEVICE_TABLE ( hid , ems_devices ) ;
static struct hid_driver ems_driver = {
. name = " hkems " ,
. id_table = ems_devices ,
. probe = ems_probe ,
} ;
2012-12-17 15:28:26 -07:00
module_hid_driver ( ems_driver ) ;
2010-11-01 15:13:37 -04:00
MODULE_LICENSE ( " GPL " ) ;