2019-05-27 09:55:01 +03:00
// SPDX-License-Identifier: GPL-2.0-or-later
2014-04-07 21:39:33 +04:00
/*
* Copyright ( c ) 2013 Andrew Duggan < aduggan @ synaptics . com >
* Copyright ( c ) 2013 Synaptics Incorporated
* Copyright ( c ) 2014 Benjamin Tissoires < benjamin . tissoires @ gmail . com >
* Copyright ( c ) 2014 Red Hat , Inc
*/
# include <linux/kernel.h>
# include <linux/hid.h>
# include <linux/input.h>
# include <linux/input/mt.h>
2017-01-05 11:48:58 +03:00
# include <linux/irq.h>
# include <linux/irqdomain.h>
2014-04-07 21:39:33 +04:00
# include <linux/module.h>
# include <linux/pm.h>
# include <linux/slab.h>
# include <linux/wait.h>
# include <linux/sched.h>
2017-01-05 11:48:58 +03:00
# include <linux/rmi.h>
2014-04-07 21:39:33 +04:00
# include "hid-ids.h"
# define RMI_MOUSE_REPORT_ID 0x01 /* Mouse emulation Report */
# define RMI_WRITE_REPORT_ID 0x09 /* Output Report */
# define RMI_READ_ADDR_REPORT_ID 0x0a /* Output Report */
# define RMI_READ_DATA_REPORT_ID 0x0b /* Input Report */
# define RMI_ATTN_REPORT_ID 0x0c /* Input Report */
# define RMI_SET_RMI_MODE_REPORT_ID 0x0f /* Feature Report */
/* flags */
2015-05-14 11:32:56 +03:00
# define RMI_READ_REQUEST_PENDING 0
# define RMI_READ_DATA_PENDING 1
# define RMI_STARTED 2
2014-04-07 21:39:33 +04:00
2014-12-20 01:45:41 +03:00
/* device flags */
# define RMI_DEVICE BIT(0)
2015-01-09 01:51:34 +03:00
# define RMI_DEVICE_HAS_PHYS_BUTTONS BIT(1)
2019-04-20 22:19:51 +03:00
# define RMI_DEVICE_OUTPUT_SET_REPORT BIT(2)
2014-12-20 01:45:41 +03:00
2015-07-10 22:48:21 +03:00
/*
* retrieve the ctrl registers
* the ctrl register has a size of 20 but a fw bug split it into 16 + 4 ,
* and there is no way to know if the first 20 bytes are here or not .
* We use only the first 12 bytes , so get only them .
*/
# define RMI_F11_CTRL_REG_COUNT 12
2014-04-07 21:39:33 +04:00
enum rmi_mode_type {
RMI_MODE_OFF = 0 ,
RMI_MODE_ATTN_REPORTS = 1 ,
RMI_MODE_NO_PACKED_ATTN_REPORTS = 2 ,
} ;
/**
* struct rmi_data - stores information for hid communication
*
* @ page_mutex : Locks current page to avoid changing pages in unexpected ways .
* @ page : Keeps track of the current virtual page
2017-01-05 11:48:58 +03:00
* @ xport : transport device to be registered with the RMI4 core .
2014-04-07 21:39:33 +04:00
*
* @ wait : Used for waiting for read data
*
* @ writeReport : output buffer when writing RMI registers
* @ readReport : input buffer when reading RMI registers
*
* @ input_report_size : size of an input report ( advertised by HID )
* @ output_report_size : size of an output report ( advertised by HID )
*
* @ flags : flags for the current device ( started , reading , etc . . . )
*
* @ reset_work : worker which will be called in case of a mouse report
* @ hdev : pointer to the struct hid_device
2017-01-05 11:48:58 +03:00
*
* @ device_flags : flags which describe the device
*
* @ domain : the IRQ domain allocated for this RMI4 device
* @ rmi_irq : the irq that will be used to generate events to rmi - core
2014-04-07 21:39:33 +04:00
*/
struct rmi_data {
struct mutex page_mutex ;
int page ;
2017-01-05 11:48:58 +03:00
struct rmi_transport_dev xport ;
2014-04-07 21:39:33 +04:00
wait_queue_head_t wait ;
u8 * writeReport ;
u8 * readReport ;
2018-02-03 18:57:15 +03:00
u32 input_report_size ;
u32 output_report_size ;
2014-04-07 21:39:33 +04:00
unsigned long flags ;
struct work_struct reset_work ;
struct hid_device * hdev ;
2014-12-20 01:45:41 +03:00
unsigned long device_flags ;
2015-07-07 02:48:31 +03:00
2017-01-05 11:48:58 +03:00
struct irq_domain * domain ;
int rmi_irq ;
2014-04-07 21:39:33 +04:00
} ;
# define RMI_PAGE(addr) (((addr) >> 8) & 0xff)
static int rmi_write_report ( struct hid_device * hdev , u8 * report , int len ) ;
/**
* rmi_set_page - Set RMI page
* @ hdev : The pointer to the hid_device struct
* @ page : The new page address .
*
* RMI devices have 16 - bit addressing , but some of the physical
* implementations ( like SMBus ) only have 8 - bit addressing . So RMI implements
* a page address at 0xff of every page so we can reliable page addresses
* every 256 registers .
*
* The page_mutex lock must be held when this function is entered .
*
* Returns zero on success , non - zero on failure .
*/
static int rmi_set_page ( struct hid_device * hdev , u8 page )
{
struct rmi_data * data = hid_get_drvdata ( hdev ) ;
int retval ;
data - > writeReport [ 0 ] = RMI_WRITE_REPORT_ID ;
data - > writeReport [ 1 ] = 1 ;
data - > writeReport [ 2 ] = 0xFF ;
data - > writeReport [ 4 ] = page ;
retval = rmi_write_report ( hdev , data - > writeReport ,
data - > output_report_size ) ;
if ( retval ! = data - > output_report_size ) {
dev_err ( & hdev - > dev ,
" %s: set page failed: %d. " , __func__ , retval ) ;
return retval ;
}
data - > page = page ;
return 0 ;
}
static int rmi_set_mode ( struct hid_device * hdev , u8 mode )
{
int ret ;
2016-11-21 13:48:42 +03:00
const u8 txbuf [ 2 ] = { RMI_SET_RMI_MODE_REPORT_ID , mode } ;
u8 * buf ;
2014-04-07 21:39:33 +04:00
2016-11-21 13:48:42 +03:00
buf = kmemdup ( txbuf , sizeof ( txbuf ) , GFP_KERNEL ) ;
if ( ! buf )
return - ENOMEM ;
ret = hid_hw_raw_request ( hdev , RMI_SET_RMI_MODE_REPORT_ID , buf ,
2014-04-07 21:39:33 +04:00
sizeof ( txbuf ) , HID_FEATURE_REPORT , HID_REQ_SET_REPORT ) ;
2016-11-21 13:48:42 +03:00
kfree ( buf ) ;
2014-04-07 21:39:33 +04:00
if ( ret < 0 ) {
dev_err ( & hdev - > dev , " unable to set rmi mode to %d (%d) \n " , mode ,
ret ) ;
return ret ;
}
return 0 ;
}
static int rmi_write_report ( struct hid_device * hdev , u8 * report , int len )
{
2019-04-20 22:19:51 +03:00
struct rmi_data * data = hid_get_drvdata ( hdev ) ;
2014-04-07 21:39:33 +04:00
int ret ;
2019-04-20 22:19:51 +03:00
if ( data - > device_flags & RMI_DEVICE_OUTPUT_SET_REPORT ) {
/*
* Talk to device by using SET_REPORT requests instead .
*/
ret = hid_hw_raw_request ( hdev , report [ 0 ] , report ,
len , HID_OUTPUT_REPORT , HID_REQ_SET_REPORT ) ;
} else {
ret = hid_hw_output_report ( hdev , ( void * ) report , len ) ;
}
2014-04-07 21:39:33 +04:00
if ( ret < 0 ) {
dev_err ( & hdev - > dev , " failed to write hid report (%d) \n " , ret ) ;
return ret ;
}
return ret ;
}
2017-01-05 11:48:58 +03:00
static int rmi_hid_read_block ( struct rmi_transport_dev * xport , u16 addr ,
void * buf , size_t len )
2014-04-07 21:39:33 +04:00
{
2017-01-05 11:48:58 +03:00
struct rmi_data * data = container_of ( xport , struct rmi_data , xport ) ;
struct hid_device * hdev = data - > hdev ;
2014-04-07 21:39:33 +04:00
int ret ;
int bytes_read ;
int bytes_needed ;
int retries ;
int read_input_count ;
mutex_lock ( & data - > page_mutex ) ;
if ( RMI_PAGE ( addr ) ! = data - > page ) {
ret = rmi_set_page ( hdev , RMI_PAGE ( addr ) ) ;
if ( ret < 0 )
goto exit ;
}
for ( retries = 5 ; retries > 0 ; retries - - ) {
data - > writeReport [ 0 ] = RMI_READ_ADDR_REPORT_ID ;
data - > writeReport [ 1 ] = 0 ; /* old 1 byte read count */
data - > writeReport [ 2 ] = addr & 0xFF ;
data - > writeReport [ 3 ] = ( addr > > 8 ) & 0xFF ;
data - > writeReport [ 4 ] = len & 0xFF ;
data - > writeReport [ 5 ] = ( len > > 8 ) & 0xFF ;
set_bit ( RMI_READ_REQUEST_PENDING , & data - > flags ) ;
ret = rmi_write_report ( hdev , data - > writeReport ,
data - > output_report_size ) ;
if ( ret ! = data - > output_report_size ) {
clear_bit ( RMI_READ_REQUEST_PENDING , & data - > flags ) ;
dev_err ( & hdev - > dev ,
" failed to write request output report (%d) \n " ,
ret ) ;
goto exit ;
}
bytes_read = 0 ;
bytes_needed = len ;
while ( bytes_read < len ) {
if ( ! wait_event_timeout ( data - > wait ,
test_bit ( RMI_READ_DATA_PENDING , & data - > flags ) ,
msecs_to_jiffies ( 1000 ) ) ) {
hid_warn ( hdev , " %s: timeout elapsed \n " ,
__func__ ) ;
ret = - EAGAIN ;
break ;
}
read_input_count = data - > readReport [ 1 ] ;
memcpy ( buf + bytes_read , & data - > readReport [ 2 ] ,
read_input_count < bytes_needed ?
read_input_count : bytes_needed ) ;
bytes_read + = read_input_count ;
bytes_needed - = read_input_count ;
clear_bit ( RMI_READ_DATA_PENDING , & data - > flags ) ;
}
if ( ret > = 0 ) {
ret = 0 ;
break ;
}
}
exit :
clear_bit ( RMI_READ_REQUEST_PENDING , & data - > flags ) ;
mutex_unlock ( & data - > page_mutex ) ;
return ret ;
}
2017-01-05 11:48:58 +03:00
static int rmi_hid_write_block ( struct rmi_transport_dev * xport , u16 addr ,
const void * buf , size_t len )
2015-02-25 04:36:48 +03:00
{
2017-01-05 11:48:58 +03:00
struct rmi_data * data = container_of ( xport , struct rmi_data , xport ) ;
struct hid_device * hdev = data - > hdev ;
2015-02-25 04:36:48 +03:00
int ret ;
mutex_lock ( & data - > page_mutex ) ;
if ( RMI_PAGE ( addr ) ! = data - > page ) {
ret = rmi_set_page ( hdev , RMI_PAGE ( addr ) ) ;
if ( ret < 0 )
goto exit ;
}
data - > writeReport [ 0 ] = RMI_WRITE_REPORT_ID ;
data - > writeReport [ 1 ] = len ;
data - > writeReport [ 2 ] = addr & 0xFF ;
data - > writeReport [ 3 ] = ( addr > > 8 ) & 0xFF ;
memcpy ( & data - > writeReport [ 4 ] , buf , len ) ;
ret = rmi_write_report ( hdev , data - > writeReport ,
data - > output_report_size ) ;
if ( ret < 0 ) {
dev_err ( & hdev - > dev ,
" failed to write request output report (%d) \n " ,
ret ) ;
goto exit ;
}
ret = 0 ;
exit :
mutex_unlock ( & data - > page_mutex ) ;
return ret ;
}
2015-07-17 03:14:00 +03:00
static int rmi_reset_attn_mode ( struct hid_device * hdev )
{
struct rmi_data * data = hid_get_drvdata ( hdev ) ;
2017-01-05 11:48:58 +03:00
struct rmi_device * rmi_dev = data - > xport . rmi_dev ;
2015-07-17 03:14:00 +03:00
int ret ;
ret = rmi_set_mode ( hdev , RMI_MODE_ATTN_REPORTS ) ;
if ( ret )
return ret ;
2017-01-05 11:48:58 +03:00
if ( test_bit ( RMI_STARTED , & data - > flags ) )
ret = rmi_dev - > driver - > reset_handler ( rmi_dev ) ;
2015-07-17 03:14:00 +03:00
2017-01-05 11:48:58 +03:00
return ret ;
2015-07-17 03:14:00 +03:00
}
2014-04-07 21:39:33 +04:00
static void rmi_reset_work ( struct work_struct * work )
{
struct rmi_data * hdata = container_of ( work , struct rmi_data ,
reset_work ) ;
/* switch the device to RMI if we receive a generic mouse report */
2015-07-17 03:14:00 +03:00
rmi_reset_attn_mode ( hdata - > hdev ) ;
2014-04-07 21:39:33 +04:00
}
2017-01-05 11:48:58 +03:00
static int rmi_input_event ( struct hid_device * hdev , u8 * data , int size )
2014-04-07 21:39:33 +04:00
{
struct rmi_data * hdata = hid_get_drvdata ( hdev ) ;
2017-01-05 11:48:58 +03:00
struct rmi_device * rmi_dev = hdata - > xport . rmi_dev ;
unsigned long flags ;
2014-04-07 21:39:33 +04:00
2017-01-05 11:48:58 +03:00
if ( ! ( test_bit ( RMI_STARTED , & hdata - > flags ) ) )
2014-04-07 21:39:33 +04:00
return 0 ;
2017-01-05 11:48:58 +03:00
local_irq_save ( flags ) ;
2014-09-11 05:02:37 +04:00
2017-01-05 11:48:58 +03:00
rmi_set_attn_data ( rmi_dev , data [ 1 ] , & data [ 2 ] , size - 2 ) ;
2014-04-07 21:39:33 +04:00
2017-01-05 11:48:58 +03:00
generic_handle_irq ( hdata - > rmi_irq ) ;
2014-04-07 21:39:33 +04:00
2017-01-05 11:48:58 +03:00
local_irq_restore ( flags ) ;
2014-04-07 21:39:33 +04:00
return 1 ;
}
static int rmi_read_data_event ( struct hid_device * hdev , u8 * data , int size )
{
struct rmi_data * hdata = hid_get_drvdata ( hdev ) ;
if ( ! test_bit ( RMI_READ_REQUEST_PENDING , & hdata - > flags ) ) {
2014-07-12 01:35:51 +04:00
hid_dbg ( hdev , " no read request pending \n " ) ;
2014-04-07 21:39:33 +04:00
return 0 ;
}
memcpy ( hdata - > readReport , data , size < hdata - > input_report_size ?
size : hdata - > input_report_size ) ;
set_bit ( RMI_READ_DATA_PENDING , & hdata - > flags ) ;
wake_up ( & hdata - > wait ) ;
return 1 ;
}
2014-09-11 05:02:37 +04:00
static int rmi_check_sanity ( struct hid_device * hdev , u8 * data , int size )
{
int valid_size = size ;
/*
* On the Dell XPS 13 9333 , the bus sometimes get confused and fills
* the report with a sentinel value " ff " . Synaptics told us that such
* behavior does not comes from the touchpad itself , so we filter out
* such reports here .
*/
while ( ( data [ valid_size - 1 ] = = 0xff ) & & valid_size > 0 )
valid_size - - ;
return valid_size ;
}
2014-04-07 21:39:33 +04:00
static int rmi_raw_event ( struct hid_device * hdev ,
struct hid_report * report , u8 * data , int size )
{
2017-10-18 04:37:36 +03:00
struct rmi_data * hdata = hid_get_drvdata ( hdev ) ;
if ( ! ( hdata - > device_flags & RMI_DEVICE ) )
return 0 ;
2014-09-11 05:02:37 +04:00
size = rmi_check_sanity ( hdev , data , size ) ;
if ( size < 2 )
return 0 ;
2014-04-07 21:39:33 +04:00
switch ( data [ 0 ] ) {
case RMI_READ_DATA_REPORT_ID :
return rmi_read_data_event ( hdev , data , size ) ;
case RMI_ATTN_REPORT_ID :
return rmi_input_event ( hdev , data , size ) ;
2014-12-20 01:45:41 +03:00
default :
return 1 ;
}
return 0 ;
}
static int rmi_event ( struct hid_device * hdev , struct hid_field * field ,
struct hid_usage * usage , __s32 value )
{
struct rmi_data * data = hid_get_drvdata ( hdev ) ;
if ( ( data - > device_flags & RMI_DEVICE ) & &
( field - > application = = HID_GD_POINTER | |
field - > application = = HID_GD_MOUSE ) ) {
2015-01-09 01:51:34 +03:00
if ( data - > device_flags & RMI_DEVICE_HAS_PHYS_BUTTONS ) {
if ( ( usage - > hid & HID_USAGE_PAGE ) = = HID_UP_BUTTON )
return 0 ;
if ( ( usage - > hid = = HID_GD_X | | usage - > hid = = HID_GD_Y )
& & ! value )
return 1 ;
}
2017-01-05 11:48:58 +03:00
schedule_work ( & data - > reset_work ) ;
2014-12-20 01:45:41 +03:00
return 1 ;
2014-04-07 21:39:33 +04:00
}
return 0 ;
}
2018-05-25 15:51:06 +03:00
static void rmi_report ( struct hid_device * hid , struct hid_report * report )
{
struct hid_field * field = report - > field [ 0 ] ;
if ( ! ( hid - > claimed & HID_CLAIMED_INPUT ) )
return ;
switch ( report - > id ) {
case RMI_READ_DATA_REPORT_ID :
/* fall-through */
case RMI_ATTN_REPORT_ID :
return ;
}
if ( field & & field - > hidinput & & field - > hidinput - > input )
input_sync ( field - > hidinput - > input ) ;
}
2014-06-11 23:03:18 +04:00
# ifdef CONFIG_PM
2015-07-07 02:48:31 +03:00
static int rmi_suspend ( struct hid_device * hdev , pm_message_t message )
{
2015-07-10 22:48:21 +03:00
struct rmi_data * data = hid_get_drvdata ( hdev ) ;
2017-01-05 11:48:58 +03:00
struct rmi_device * rmi_dev = data - > xport . rmi_dev ;
2015-07-07 02:48:31 +03:00
int ret ;
2016-01-27 02:40:36 +03:00
if ( ! ( data - > device_flags & RMI_DEVICE ) )
return 0 ;
2017-01-05 11:48:58 +03:00
ret = rmi_driver_suspend ( rmi_dev , false ) ;
2015-07-07 02:48:31 +03:00
if ( ret ) {
2017-01-05 11:48:58 +03:00
hid_warn ( hdev , " Failed to suspend device: %d \n " , ret ) ;
2015-07-07 02:48:31 +03:00
return ret ;
}
2017-01-05 11:48:58 +03:00
return 0 ;
2014-04-07 21:39:33 +04:00
}
static int rmi_post_resume ( struct hid_device * hdev )
{
2016-01-27 02:40:36 +03:00
struct rmi_data * data = hid_get_drvdata ( hdev ) ;
2017-01-05 11:48:58 +03:00
struct rmi_device * rmi_dev = data - > xport . rmi_dev ;
int ret ;
2016-01-27 02:40:36 +03:00
if ( ! ( data - > device_flags & RMI_DEVICE ) )
return 0 ;
2017-07-23 04:15:09 +03:00
/* Make sure the HID device is ready to receive events */
ret = hid_hw_open ( hdev ) ;
2017-01-05 11:48:58 +03:00
if ( ret )
2015-02-25 04:36:49 +03:00
return ret ;
2014-12-09 02:01:59 +03:00
2017-07-23 04:15:09 +03:00
ret = rmi_reset_attn_mode ( hdev ) ;
if ( ret )
goto out ;
2017-01-05 11:48:58 +03:00
ret = rmi_driver_resume ( rmi_dev , false ) ;
2014-04-07 21:39:33 +04:00
if ( ret ) {
2017-01-05 11:48:58 +03:00
hid_warn ( hdev , " Failed to resume device: %d \n " , ret ) ;
2017-07-23 04:15:09 +03:00
goto out ;
2014-04-07 21:39:33 +04:00
}
2017-07-23 04:15:09 +03:00
out :
hid_hw_close ( hdev ) ;
return ret ;
2014-04-07 21:39:33 +04:00
}
2017-01-05 11:48:58 +03:00
# endif /* CONFIG_PM */
2014-04-07 21:39:33 +04:00
2017-01-05 11:48:58 +03:00
static int rmi_hid_reset ( struct rmi_transport_dev * xport , u16 reset_addr )
2014-04-07 21:39:33 +04:00
{
2017-01-05 11:48:58 +03:00
struct rmi_data * data = container_of ( xport , struct rmi_data , xport ) ;
struct hid_device * hdev = data - > hdev ;
2014-04-07 21:39:33 +04:00
2017-01-05 11:48:58 +03:00
return rmi_reset_attn_mode ( hdev ) ;
2014-04-07 21:39:33 +04:00
}
2015-09-30 01:52:59 +03:00
static int rmi_input_configured ( struct hid_device * hdev , struct hid_input * hi )
2014-04-07 21:39:33 +04:00
{
struct rmi_data * data = hid_get_drvdata ( hdev ) ;
struct input_dev * input = hi - > input ;
2017-01-05 11:48:58 +03:00
int ret = 0 ;
if ( ! ( data - > device_flags & RMI_DEVICE ) )
return 0 ;
2014-04-07 21:39:33 +04:00
2017-01-05 11:48:58 +03:00
data - > xport . input = input ;
2014-04-07 21:39:33 +04:00
hid_dbg ( hdev , " Opening low level driver \n " ) ;
ret = hid_hw_open ( hdev ) ;
if ( ret )
2015-09-30 01:52:59 +03:00
return ret ;
2014-04-07 21:39:33 +04:00
/* Allow incoming hid reports */
hid_device_io_start ( hdev ) ;
ret = rmi_set_mode ( hdev , RMI_MODE_ATTN_REPORTS ) ;
if ( ret < 0 ) {
dev_err ( & hdev - > dev , " failed to set rmi mode \n " ) ;
goto exit ;
}
ret = rmi_set_page ( hdev , 0 ) ;
if ( ret < 0 ) {
dev_err ( & hdev - > dev , " failed to set page select to 0. \n " ) ;
goto exit ;
}
2017-01-05 11:48:58 +03:00
ret = rmi_register_transport_device ( & data - > xport ) ;
if ( ret < 0 ) {
dev_err ( & hdev - > dev , " failed to register transport driver \n " ) ;
2015-09-30 01:52:59 +03:00
goto exit ;
2014-04-07 21:39:33 +04:00
}
set_bit ( RMI_STARTED , & data - > flags ) ;
exit :
hid_device_io_stop ( hdev ) ;
hid_hw_close ( hdev ) ;
2015-09-30 01:52:59 +03:00
return ret ;
2014-04-07 21:39:33 +04:00
}
static int rmi_input_mapping ( struct hid_device * hdev ,
struct hid_input * hi , struct hid_field * field ,
struct hid_usage * usage , unsigned long * * bit , int * max )
{
2014-12-20 01:45:41 +03:00
struct rmi_data * data = hid_get_drvdata ( hdev ) ;
/*
* we want to make HID ignore the advertised HID collection
* for RMI deivces
*/
2015-01-09 01:51:34 +03:00
if ( data - > device_flags & RMI_DEVICE ) {
if ( ( data - > device_flags & RMI_DEVICE_HAS_PHYS_BUTTONS ) & &
( ( usage - > hid & HID_USAGE_PAGE ) = = HID_UP_BUTTON ) )
return 0 ;
2014-12-20 01:45:41 +03:00
return - 1 ;
2015-01-09 01:51:34 +03:00
}
2014-12-20 01:45:41 +03:00
return 0 ;
}
static int rmi_check_valid_report_id ( struct hid_device * hdev , unsigned type ,
unsigned id , struct hid_report * * report )
{
int i ;
* report = hdev - > report_enum [ type ] . report_id_hash [ id ] ;
if ( * report ) {
for ( i = 0 ; i < ( * report ) - > maxfield ; i + + ) {
unsigned app = ( * report ) - > field [ i ] - > application ;
if ( ( app & HID_USAGE_PAGE ) > = HID_UP_MSVENDOR )
return 1 ;
}
}
return 0 ;
2014-04-07 21:39:33 +04:00
}
2017-01-05 11:48:58 +03:00
static struct rmi_device_platform_data rmi_hid_pdata = {
. sensor_pdata = {
. sensor_type = rmi_sensor_touchpad ,
. axis_align . flip_y = true ,
. dribble = RMI_REG_STATE_ON ,
. palm_detect = RMI_REG_STATE_OFF ,
} ,
} ;
static const struct rmi_transport_ops hid_rmi_ops = {
. write_block = rmi_hid_write_block ,
. read_block = rmi_hid_read_block ,
. reset = rmi_hid_reset ,
} ;
static void rmi_irq_teardown ( void * data )
{
struct rmi_data * hdata = data ;
struct irq_domain * domain = hdata - > domain ;
if ( ! domain )
return ;
irq_dispose_mapping ( irq_find_mapping ( domain , 0 ) ) ;
irq_domain_remove ( domain ) ;
hdata - > domain = NULL ;
hdata - > rmi_irq = 0 ;
}
static int rmi_irq_map ( struct irq_domain * h , unsigned int virq ,
irq_hw_number_t hw_irq_num )
{
irq_set_chip_and_handler ( virq , & dummy_irq_chip , handle_simple_irq ) ;
return 0 ;
}
static const struct irq_domain_ops rmi_irq_ops = {
. map = rmi_irq_map ,
} ;
static int rmi_setup_irq_domain ( struct hid_device * hdev )
{
struct rmi_data * hdata = hid_get_drvdata ( hdev ) ;
int ret ;
hdata - > domain = irq_domain_create_linear ( hdev - > dev . fwnode , 1 ,
& rmi_irq_ops , hdata ) ;
if ( ! hdata - > domain )
return - ENOMEM ;
ret = devm_add_action_or_reset ( & hdev - > dev , & rmi_irq_teardown , hdata ) ;
if ( ret )
return ret ;
hdata - > rmi_irq = irq_create_mapping ( hdata - > domain , 0 ) ;
if ( hdata - > rmi_irq < = 0 ) {
hid_err ( hdev , " Can't allocate an IRQ \n " ) ;
return hdata - > rmi_irq < 0 ? hdata - > rmi_irq : - ENXIO ;
}
return 0 ;
}
2014-04-07 21:39:33 +04:00
static int rmi_probe ( struct hid_device * hdev , const struct hid_device_id * id )
{
struct rmi_data * data = NULL ;
int ret ;
size_t alloc_size ;
2014-07-18 03:14:44 +04:00
struct hid_report * input_report ;
struct hid_report * output_report ;
2014-12-20 01:45:41 +03:00
struct hid_report * feature_report ;
2014-04-07 21:39:33 +04:00
data = devm_kzalloc ( & hdev - > dev , sizeof ( struct rmi_data ) , GFP_KERNEL ) ;
if ( ! data )
return - ENOMEM ;
INIT_WORK ( & data - > reset_work , rmi_reset_work ) ;
data - > hdev = hdev ;
hid_set_drvdata ( hdev , data ) ;
hdev - > quirks | = HID_QUIRK_NO_INIT_REPORTS ;
2018-05-25 15:51:06 +03:00
hdev - > quirks | = HID_QUIRK_NO_INPUT_SYNC ;
2014-04-07 21:39:33 +04:00
ret = hid_parse ( hdev ) ;
if ( ret ) {
hid_err ( hdev , " parse failed \n " ) ;
return ret ;
}
2015-01-09 01:51:34 +03:00
if ( id - > driver_data )
data - > device_flags = id - > driver_data ;
2014-12-20 01:45:41 +03:00
/*
* Check for the RMI specific report ids . If they are misisng
* simply return and let the events be processed by hid - input
*/
if ( ! rmi_check_valid_report_id ( hdev , HID_FEATURE_REPORT ,
RMI_SET_RMI_MODE_REPORT_ID , & feature_report ) ) {
hid_dbg ( hdev , " device does not have set mode feature report \n " ) ;
goto start ;
}
if ( ! rmi_check_valid_report_id ( hdev , HID_INPUT_REPORT ,
RMI_ATTN_REPORT_ID , & input_report ) ) {
hid_dbg ( hdev , " device does not have attention input report \n " ) ;
goto start ;
2014-07-18 03:14:44 +04:00
}
2015-01-09 01:51:36 +03:00
data - > input_report_size = hid_report_len ( input_report ) ;
2014-07-18 03:14:44 +04:00
2014-12-20 01:45:41 +03:00
if ( ! rmi_check_valid_report_id ( hdev , HID_OUTPUT_REPORT ,
RMI_WRITE_REPORT_ID , & output_report ) ) {
hid_dbg ( hdev ,
" device does not have rmi write output report \n " ) ;
goto start ;
2014-07-18 03:14:44 +04:00
}
2015-01-09 01:51:36 +03:00
data - > output_report_size = hid_report_len ( output_report ) ;
2014-04-07 21:39:33 +04:00
2014-12-20 01:45:41 +03:00
data - > device_flags | = RMI_DEVICE ;
2014-04-07 21:39:33 +04:00
alloc_size = data - > output_report_size + data - > input_report_size ;
data - > writeReport = devm_kzalloc ( & hdev - > dev , alloc_size , GFP_KERNEL ) ;
if ( ! data - > writeReport ) {
2017-01-05 11:48:58 +03:00
hid_err ( hdev , " failed to allocate buffer for HID reports \n " ) ;
return - ENOMEM ;
2014-04-07 21:39:33 +04:00
}
data - > readReport = data - > writeReport + data - > output_report_size ;
init_waitqueue_head ( & data - > wait ) ;
mutex_init ( & data - > page_mutex ) ;
2017-01-05 11:48:58 +03:00
ret = rmi_setup_irq_domain ( hdev ) ;
if ( ret ) {
hid_err ( hdev , " failed to allocate IRQ domain \n " ) ;
return ret ;
}
if ( data - > device_flags & RMI_DEVICE_HAS_PHYS_BUTTONS )
rmi_hid_pdata . f30_data . disable = true ;
data - > xport . dev = hdev - > dev . parent ;
data - > xport . pdata = rmi_hid_pdata ;
data - > xport . pdata . irq = data - > rmi_irq ;
data - > xport . proto_name = " hid " ;
data - > xport . ops = & hid_rmi_ops ;
2014-12-20 01:45:41 +03:00
start :
2014-04-07 21:39:33 +04:00
ret = hid_hw_start ( hdev , HID_CONNECT_DEFAULT ) ;
if ( ret ) {
hid_err ( hdev , " hw start failed \n " ) ;
return ret ;
}
return 0 ;
}
static void rmi_remove ( struct hid_device * hdev )
{
struct rmi_data * hdata = hid_get_drvdata ( hdev ) ;
2017-10-18 04:37:36 +03:00
if ( hdata - > device_flags & RMI_DEVICE ) {
clear_bit ( RMI_STARTED , & hdata - > flags ) ;
cancel_work_sync ( & hdata - > reset_work ) ;
rmi_unregister_transport_device ( & hdata - > xport ) ;
}
2014-04-07 21:39:33 +04:00
hid_hw_stop ( hdev ) ;
}
static const struct hid_device_id rmi_id [ ] = {
2015-01-09 01:51:35 +03:00
{ HID_USB_DEVICE ( USB_VENDOR_ID_RAZER , USB_DEVICE_ID_RAZER_BLADE_14 ) ,
. driver_data = RMI_DEVICE_HAS_PHYS_BUTTONS } ,
2017-01-05 11:49:00 +03:00
{ HID_USB_DEVICE ( USB_VENDOR_ID_LENOVO , USB_DEVICE_ID_LENOVO_X1_COVER ) } ,
2017-11-18 04:21:30 +03:00
{ HID_USB_DEVICE ( USB_VENDOR_ID_PRIMAX , USB_DEVICE_ID_PRIMAX_REZEL ) } ,
2019-04-20 22:19:51 +03:00
{ HID_USB_DEVICE ( USB_VENDOR_ID_SYNAPTICS , USB_DEVICE_ID_SYNAPTICS_ACER_SWITCH5 ) ,
. driver_data = RMI_DEVICE_OUTPUT_SET_REPORT } ,
2014-05-21 19:15:56 +04:00
{ HID_DEVICE ( HID_BUS_ANY , HID_GROUP_RMI , HID_ANY_ID , HID_ANY_ID ) } ,
2014-04-07 21:39:33 +04:00
{ }
} ;
MODULE_DEVICE_TABLE ( hid , rmi_id ) ;
static struct hid_driver rmi_driver = {
. name = " hid-rmi " ,
. id_table = rmi_id ,
. probe = rmi_probe ,
. remove = rmi_remove ,
2014-12-20 01:45:41 +03:00
. event = rmi_event ,
2014-04-07 21:39:33 +04:00
. raw_event = rmi_raw_event ,
2018-05-25 15:51:06 +03:00
. report = rmi_report ,
2014-04-07 21:39:33 +04:00
. input_mapping = rmi_input_mapping ,
. input_configured = rmi_input_configured ,
# ifdef CONFIG_PM
2015-07-07 02:48:31 +03:00
. suspend = rmi_suspend ,
2014-04-07 21:39:33 +04:00
. resume = rmi_post_resume ,
2017-01-05 11:48:58 +03:00
. reset_resume = rmi_post_resume ,
2014-04-07 21:39:33 +04:00
# endif
} ;
module_hid_driver ( rmi_driver ) ;
MODULE_AUTHOR ( " Andrew Duggan <aduggan@synaptics.com> " ) ;
MODULE_DESCRIPTION ( " RMI HID driver " ) ;
MODULE_LICENSE ( " GPL " ) ;