2014-11-06 23:07:00 +00:00
/*
* KXCJK - 1013 3 - axis accelerometer driver
* Copyright ( c ) 2014 , Intel Corporation .
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License for
* more details .
*/
# include <linux/module.h>
# include <linux/i2c.h>
# include <linux/interrupt.h>
# include <linux/delay.h>
# include <linux/bitops.h>
# include <linux/slab.h>
# include <linux/string.h>
# include <linux/acpi.h>
2014-05-08 22:58:00 +01:00
# include <linux/pm.h>
# include <linux/pm_runtime.h>
2014-11-06 23:07:00 +00:00
# include <linux/iio/iio.h>
# include <linux/iio/sysfs.h>
# include <linux/iio/buffer.h>
# include <linux/iio/trigger.h>
2014-08-22 20:01:00 +01:00
# include <linux/iio/events.h>
2014-11-06 23:07:00 +00:00
# include <linux/iio/trigger_consumer.h>
# include <linux/iio/triggered_buffer.h>
# include <linux/iio/accel/kxcjk_1013.h>
# define KXCJK1013_DRV_NAME "kxcjk1013"
# define KXCJK1013_IRQ_NAME "kxcjk1013_event"
# define KXCJK1013_REG_XOUT_L 0x06
/*
* From low byte X axis register , all the other addresses of Y and Z can be
* obtained by just applying axis offset . The following axis defines are just
* provide clarity , but not used .
*/
# define KXCJK1013_REG_XOUT_H 0x07
# define KXCJK1013_REG_YOUT_L 0x08
# define KXCJK1013_REG_YOUT_H 0x09
# define KXCJK1013_REG_ZOUT_L 0x0A
# define KXCJK1013_REG_ZOUT_H 0x0B
# define KXCJK1013_REG_DCST_RESP 0x0C
# define KXCJK1013_REG_WHO_AM_I 0x0F
# define KXCJK1013_REG_INT_SRC1 0x16
# define KXCJK1013_REG_INT_SRC2 0x17
# define KXCJK1013_REG_STATUS_REG 0x18
# define KXCJK1013_REG_INT_REL 0x1A
# define KXCJK1013_REG_CTRL1 0x1B
# define KXCJK1013_REG_CTRL2 0x1D
# define KXCJK1013_REG_INT_CTRL1 0x1E
# define KXCJK1013_REG_INT_CTRL2 0x1F
# define KXCJK1013_REG_DATA_CTRL 0x21
# define KXCJK1013_REG_WAKE_TIMER 0x29
# define KXCJK1013_REG_SELF_TEST 0x3A
# define KXCJK1013_REG_WAKE_THRES 0x6A
# define KXCJK1013_REG_CTRL1_BIT_PC1 BIT(7)
# define KXCJK1013_REG_CTRL1_BIT_RES BIT(6)
# define KXCJK1013_REG_CTRL1_BIT_DRDY BIT(5)
# define KXCJK1013_REG_CTRL1_BIT_GSEL1 BIT(4)
# define KXCJK1013_REG_CTRL1_BIT_GSEL0 BIT(3)
# define KXCJK1013_REG_CTRL1_BIT_WUFE BIT(1)
# define KXCJK1013_REG_INT_REG1_BIT_IEA BIT(4)
# define KXCJK1013_REG_INT_REG1_BIT_IEN BIT(5)
# define KXCJK1013_DATA_MASK_12_BIT 0x0FFF
# define KXCJK1013_MAX_STARTUP_TIME_US 100000
2014-05-08 22:58:00 +01:00
# define KXCJK1013_SLEEP_DELAY_MS 2000
2014-08-22 20:01:00 +01:00
# define KXCJK1013_REG_INT_SRC2_BIT_ZP BIT(0)
# define KXCJK1013_REG_INT_SRC2_BIT_ZN BIT(1)
# define KXCJK1013_REG_INT_SRC2_BIT_YP BIT(2)
# define KXCJK1013_REG_INT_SRC2_BIT_YN BIT(3)
# define KXCJK1013_REG_INT_SRC2_BIT_XP BIT(4)
# define KXCJK1013_REG_INT_SRC2_BIT_XN BIT(5)
# define KXCJK1013_DEFAULT_WAKE_THRES 1
2014-03-09 08:33:00 +00:00
enum kx_chipset {
KXCJK1013 ,
KXCJ91008 ,
KXTJ21009 ,
KX_MAX_CHIPS /* this must be last */
} ;
2014-11-06 23:07:00 +00:00
struct kxcjk1013_data {
struct i2c_client * client ;
2014-08-22 20:01:00 +01:00
struct iio_trigger * dready_trig ;
struct iio_trigger * motion_trig ;
2014-11-06 23:07:00 +00:00
struct mutex mutex ;
s16 buffer [ 8 ] ;
u8 odr_bits ;
2014-05-08 22:58:00 +01:00
u8 range ;
2014-08-22 20:01:00 +01:00
int wake_thres ;
int wake_dur ;
2014-11-06 23:07:00 +00:00
bool active_high_intr ;
2014-08-22 20:01:00 +01:00
bool dready_trigger_on ;
int ev_enable_state ;
bool motion_trigger_on ;
int64_t timestamp ;
2014-03-09 08:33:00 +00:00
enum kx_chipset chipset ;
2014-05-11 22:09:00 +01:00
bool is_smo8500_device ;
2014-11-06 23:07:00 +00:00
} ;
enum kxcjk1013_axis {
AXIS_X ,
AXIS_Y ,
AXIS_Z ,
2016-03-24 11:29:30 +02:00
AXIS_MAX ,
2014-11-06 23:07:00 +00:00
} ;
enum kxcjk1013_mode {
STANDBY ,
OPERATION ,
} ;
2014-05-08 22:58:00 +01:00
enum kxcjk1013_range {
KXCJK1013_RANGE_2G ,
KXCJK1013_RANGE_4G ,
KXCJK1013_RANGE_8G ,
} ;
2014-11-06 23:07:00 +00:00
static const struct {
int val ;
int val2 ;
int odr_bits ;
} samp_freq_table [ ] = { { 0 , 781000 , 0x08 } , { 1 , 563000 , 0x09 } ,
2014-07-17 01:42:00 +01:00
{ 3 , 125000 , 0x0A } , { 6 , 250000 , 0x0B } , { 12 , 500000 , 0 } ,
2014-11-06 23:07:00 +00:00
{ 25 , 0 , 0x01 } , { 50 , 0 , 0x02 } , { 100 , 0 , 0x03 } ,
{ 200 , 0 , 0x04 } , { 400 , 0 , 0x05 } , { 800 , 0 , 0x06 } ,
{ 1600 , 0 , 0x07 } } ;
/* Refer to section 4 of the specification */
static const struct {
int odr_bits ;
int usec ;
2014-03-09 08:33:00 +00:00
} odr_start_up_times [ KX_MAX_CHIPS ] [ 12 ] = {
/* KXCJK-1013 */
{
{ 0x08 , 100000 } ,
{ 0x09 , 100000 } ,
{ 0x0A , 100000 } ,
{ 0x0B , 100000 } ,
{ 0 , 80000 } ,
{ 0x01 , 41000 } ,
{ 0x02 , 21000 } ,
{ 0x03 , 11000 } ,
{ 0x04 , 6400 } ,
{ 0x05 , 3900 } ,
{ 0x06 , 2700 } ,
{ 0x07 , 2100 } ,
} ,
/* KXCJ9-1008 */
{
{ 0x08 , 100000 } ,
{ 0x09 , 100000 } ,
{ 0x0A , 100000 } ,
{ 0x0B , 100000 } ,
{ 0 , 80000 } ,
{ 0x01 , 41000 } ,
{ 0x02 , 21000 } ,
{ 0x03 , 11000 } ,
{ 0x04 , 6400 } ,
{ 0x05 , 3900 } ,
{ 0x06 , 2700 } ,
{ 0x07 , 2100 } ,
} ,
/* KXCTJ2-1009 */
{
{ 0x08 , 1240000 } ,
{ 0x09 , 621000 } ,
{ 0x0A , 309000 } ,
{ 0x0B , 151000 } ,
{ 0 , 80000 } ,
{ 0x01 , 41000 } ,
{ 0x02 , 21000 } ,
{ 0x03 , 11000 } ,
{ 0x04 , 6000 } ,
{ 0x05 , 4000 } ,
{ 0x06 , 3000 } ,
{ 0x07 , 2000 } ,
} ,
} ;
2014-11-06 23:07:00 +00:00
2014-05-08 22:58:00 +01:00
static const struct {
u16 scale ;
u8 gsel_0 ;
u8 gsel_1 ;
} KXCJK1013_scale_table [ ] = { { 9582 , 0 , 0 } ,
{ 19163 , 1 , 0 } ,
{ 38326 , 0 , 1 } } ;
2014-08-22 20:01:00 +01:00
static const struct {
int val ;
int val2 ;
int odr_bits ;
} wake_odr_data_rate_table [ ] = { { 0 , 781000 , 0x00 } ,
{ 1 , 563000 , 0x01 } ,
{ 3 , 125000 , 0x02 } ,
{ 6 , 250000 , 0x03 } ,
{ 12 , 500000 , 0x04 } ,
{ 25 , 0 , 0x05 } ,
{ 50 , 0 , 0x06 } ,
{ 100 , 0 , 0x06 } ,
{ 200 , 0 , 0x06 } ,
{ 400 , 0 , 0x06 } ,
{ 800 , 0 , 0x06 } ,
{ 1600 , 0 , 0x06 } } ;
2014-11-06 23:07:00 +00:00
static int kxcjk1013_set_mode ( struct kxcjk1013_data * data ,
enum kxcjk1013_mode mode )
{
int ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_ctrl1 \n " ) ;
return ret ;
}
if ( mode = = STANDBY )
ret & = ~ KXCJK1013_REG_CTRL1_BIT_PC1 ;
else
ret | = KXCJK1013_REG_CTRL1_BIT_PC1 ;
ret = i2c_smbus_write_byte_data ( data - > client ,
KXCJK1013_REG_CTRL1 , ret ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_ctrl1 \n " ) ;
return ret ;
}
return 0 ;
}
2014-05-08 22:58:00 +01:00
static int kxcjk1013_get_mode ( struct kxcjk1013_data * data ,
enum kxcjk1013_mode * mode )
{
int ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_ctrl1 \n " ) ;
return ret ;
}
if ( ret & KXCJK1013_REG_CTRL1_BIT_PC1 )
* mode = OPERATION ;
else
* mode = STANDBY ;
return 0 ;
}
2014-05-08 22:58:00 +01:00
static int kxcjk1013_set_range ( struct kxcjk1013_data * data , int range_index )
{
int ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_ctrl1 \n " ) ;
return ret ;
}
2014-11-10 10:20:16 +02:00
ret & = ~ ( KXCJK1013_REG_CTRL1_BIT_GSEL0 |
KXCJK1013_REG_CTRL1_BIT_GSEL1 ) ;
2014-05-08 22:58:00 +01:00
ret | = ( KXCJK1013_scale_table [ range_index ] . gsel_0 < < 3 ) ;
ret | = ( KXCJK1013_scale_table [ range_index ] . gsel_1 < < 4 ) ;
ret = i2c_smbus_write_byte_data ( data - > client ,
KXCJK1013_REG_CTRL1 ,
ret ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_ctrl1 \n " ) ;
return ret ;
}
data - > range = range_index ;
return 0 ;
}
2014-11-06 23:07:00 +00:00
static int kxcjk1013_chip_init ( struct kxcjk1013_data * data )
{
int ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_WHO_AM_I ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading who_am_i \n " ) ;
return ret ;
}
dev_dbg ( & data - > client - > dev , " KXCJK1013 Chip Id %x \n " , ret ) ;
ret = kxcjk1013_set_mode ( data , STANDBY ) ;
if ( ret < 0 )
return ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_ctrl1 \n " ) ;
return ret ;
}
/* Set 12 bit mode */
ret | = KXCJK1013_REG_CTRL1_BIT_RES ;
ret = i2c_smbus_write_byte_data ( data - > client , KXCJK1013_REG_CTRL1 ,
ret ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_ctrl \n " ) ;
return ret ;
}
2014-05-08 22:58:00 +01:00
/* Setting range to 4G */
ret = kxcjk1013_set_range ( data , KXCJK1013_RANGE_4G ) ;
if ( ret < 0 )
return ret ;
2014-11-06 23:07:00 +00:00
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_DATA_CTRL ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_data_ctrl \n " ) ;
return ret ;
}
data - > odr_bits = ret ;
/* Set up INT polarity */
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_INT_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_int_ctrl1 \n " ) ;
return ret ;
}
if ( data - > active_high_intr )
ret | = KXCJK1013_REG_INT_REG1_BIT_IEA ;
else
ret & = ~ KXCJK1013_REG_INT_REG1_BIT_IEA ;
ret = i2c_smbus_write_byte_data ( data - > client , KXCJK1013_REG_INT_CTRL1 ,
ret ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_int_ctrl1 \n " ) ;
return ret ;
}
2014-05-08 22:58:00 +01:00
ret = kxcjk1013_set_mode ( data , OPERATION ) ;
if ( ret < 0 )
return ret ;
2014-08-22 20:01:00 +01:00
data - > wake_thres = KXCJK1013_DEFAULT_WAKE_THRES ;
2014-05-08 22:58:00 +01:00
return 0 ;
}
2014-12-04 01:08:13 +01:00
# ifdef CONFIG_PM
2014-05-08 22:58:00 +01:00
static int kxcjk1013_get_startup_times ( struct kxcjk1013_data * data )
{
int i ;
2014-03-09 08:33:00 +00:00
int idx = data - > chipset ;
2014-05-08 22:58:00 +01:00
2014-03-09 08:33:00 +00:00
for ( i = 0 ; i < ARRAY_SIZE ( odr_start_up_times [ idx ] ) ; + + i ) {
if ( odr_start_up_times [ idx ] [ i ] . odr_bits = = data - > odr_bits )
return odr_start_up_times [ idx ] [ i ] . usec ;
2014-05-08 22:58:00 +01:00
}
return KXCJK1013_MAX_STARTUP_TIME_US ;
}
2014-03-09 16:13:00 +00:00
# endif
2014-05-08 22:58:00 +01:00
static int kxcjk1013_set_power_state ( struct kxcjk1013_data * data , bool on )
{
2014-12-06 00:18:08 +02:00
# ifdef CONFIG_PM
2014-05-08 22:58:00 +01:00
int ret ;
if ( on )
ret = pm_runtime_get_sync ( & data - > client - > dev ) ;
else {
pm_runtime_mark_last_busy ( & data - > client - > dev ) ;
ret = pm_runtime_put_autosuspend ( & data - > client - > dev ) ;
}
if ( ret < 0 ) {
dev_err ( & data - > client - > dev ,
" Failed: kxcjk1013_set_power_state for %d \n " , on ) ;
2014-12-10 18:23:53 +02:00
if ( on )
pm_runtime_put_noidle ( & data - > client - > dev ) ;
2014-05-08 22:58:00 +01:00
return ret ;
}
2014-12-06 00:18:08 +02:00
# endif
2014-05-08 22:58:00 +01:00
2014-11-06 23:07:00 +00:00
return 0 ;
}
2014-08-22 20:01:00 +01:00
static int kxcjk1013_chip_update_thresholds ( struct kxcjk1013_data * data )
{
int ret ;
ret = i2c_smbus_write_byte_data ( data - > client ,
KXCJK1013_REG_WAKE_TIMER ,
data - > wake_dur ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev ,
" Error writing reg_wake_timer \n " ) ;
return ret ;
}
ret = i2c_smbus_write_byte_data ( data - > client ,
KXCJK1013_REG_WAKE_THRES ,
data - > wake_thres ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_wake_thres \n " ) ;
return ret ;
}
return 0 ;
}
static int kxcjk1013_setup_any_motion_interrupt ( struct kxcjk1013_data * data ,
bool status )
{
int ret ;
enum kxcjk1013_mode store_mode ;
ret = kxcjk1013_get_mode ( data , & store_mode ) ;
if ( ret < 0 )
return ret ;
/* This is requirement by spec to change state to STANDBY */
ret = kxcjk1013_set_mode ( data , STANDBY ) ;
if ( ret < 0 )
return ret ;
ret = kxcjk1013_chip_update_thresholds ( data ) ;
if ( ret < 0 )
return ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_INT_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_int_ctrl1 \n " ) ;
return ret ;
}
if ( status )
ret | = KXCJK1013_REG_INT_REG1_BIT_IEN ;
else
ret & = ~ KXCJK1013_REG_INT_REG1_BIT_IEN ;
ret = i2c_smbus_write_byte_data ( data - > client , KXCJK1013_REG_INT_CTRL1 ,
ret ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_int_ctrl1 \n " ) ;
return ret ;
}
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_ctrl1 \n " ) ;
return ret ;
}
if ( status )
ret | = KXCJK1013_REG_CTRL1_BIT_WUFE ;
else
ret & = ~ KXCJK1013_REG_CTRL1_BIT_WUFE ;
ret = i2c_smbus_write_byte_data ( data - > client ,
KXCJK1013_REG_CTRL1 , ret ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_ctrl1 \n " ) ;
return ret ;
}
if ( store_mode = = OPERATION ) {
ret = kxcjk1013_set_mode ( data , OPERATION ) ;
if ( ret < 0 )
return ret ;
}
return 0 ;
}
static int kxcjk1013_setup_new_data_interrupt ( struct kxcjk1013_data * data ,
bool status )
2014-11-06 23:07:00 +00:00
{
int ret ;
2014-05-08 22:58:00 +01:00
enum kxcjk1013_mode store_mode ;
ret = kxcjk1013_get_mode ( data , & store_mode ) ;
if ( ret < 0 )
return ret ;
2014-11-06 23:07:00 +00:00
/* This is requirement by spec to change state to STANDBY */
ret = kxcjk1013_set_mode ( data , STANDBY ) ;
if ( ret < 0 )
return ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_INT_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_int_ctrl1 \n " ) ;
return ret ;
}
if ( status )
ret | = KXCJK1013_REG_INT_REG1_BIT_IEN ;
else
ret & = ~ KXCJK1013_REG_INT_REG1_BIT_IEN ;
ret = i2c_smbus_write_byte_data ( data - > client , KXCJK1013_REG_INT_CTRL1 ,
ret ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_int_ctrl1 \n " ) ;
return ret ;
}
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_CTRL1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_ctrl1 \n " ) ;
return ret ;
}
if ( status )
ret | = KXCJK1013_REG_CTRL1_BIT_DRDY ;
else
ret & = ~ KXCJK1013_REG_CTRL1_BIT_DRDY ;
ret = i2c_smbus_write_byte_data ( data - > client ,
KXCJK1013_REG_CTRL1 , ret ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_ctrl1 \n " ) ;
return ret ;
}
2014-05-08 22:58:00 +01:00
if ( store_mode = = OPERATION ) {
ret = kxcjk1013_set_mode ( data , OPERATION ) ;
if ( ret < 0 )
return ret ;
}
return 0 ;
2014-11-06 23:07:00 +00:00
}
static int kxcjk1013_convert_freq_to_bit ( int val , int val2 )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( samp_freq_table ) ; + + i ) {
if ( samp_freq_table [ i ] . val = = val & &
samp_freq_table [ i ] . val2 = = val2 ) {
return samp_freq_table [ i ] . odr_bits ;
}
}
return - EINVAL ;
}
2014-08-22 20:01:00 +01:00
static int kxcjk1013_convert_wake_odr_to_bit ( int val , int val2 )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( wake_odr_data_rate_table ) ; + + i ) {
if ( wake_odr_data_rate_table [ i ] . val = = val & &
wake_odr_data_rate_table [ i ] . val2 = = val2 ) {
return wake_odr_data_rate_table [ i ] . odr_bits ;
}
}
return - EINVAL ;
}
2014-11-06 23:07:00 +00:00
static int kxcjk1013_set_odr ( struct kxcjk1013_data * data , int val , int val2 )
{
int ret ;
int odr_bits ;
2014-05-08 22:58:00 +01:00
enum kxcjk1013_mode store_mode ;
ret = kxcjk1013_get_mode ( data , & store_mode ) ;
if ( ret < 0 )
return ret ;
2014-11-06 23:07:00 +00:00
odr_bits = kxcjk1013_convert_freq_to_bit ( val , val2 ) ;
if ( odr_bits < 0 )
return odr_bits ;
/* To change ODR, the chip must be set to STANDBY as per spec */
ret = kxcjk1013_set_mode ( data , STANDBY ) ;
if ( ret < 0 )
return ret ;
ret = i2c_smbus_write_byte_data ( data - > client , KXCJK1013_REG_DATA_CTRL ,
odr_bits ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing data_ctrl \n " ) ;
return ret ;
}
data - > odr_bits = odr_bits ;
2014-08-22 20:01:00 +01:00
odr_bits = kxcjk1013_convert_wake_odr_to_bit ( val , val2 ) ;
if ( odr_bits < 0 )
return odr_bits ;
ret = i2c_smbus_write_byte_data ( data - > client , KXCJK1013_REG_CTRL2 ,
odr_bits ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error writing reg_ctrl2 \n " ) ;
return ret ;
}
2014-05-08 22:58:00 +01:00
if ( store_mode = = OPERATION ) {
2014-11-06 23:07:00 +00:00
ret = kxcjk1013_set_mode ( data , OPERATION ) ;
if ( ret < 0 )
return ret ;
}
return 0 ;
}
static int kxcjk1013_get_odr ( struct kxcjk1013_data * data , int * val , int * val2 )
{
int i ;
for ( i = 0 ; i < ARRAY_SIZE ( samp_freq_table ) ; + + i ) {
if ( samp_freq_table [ i ] . odr_bits = = data - > odr_bits ) {
* val = samp_freq_table [ i ] . val ;
* val2 = samp_freq_table [ i ] . val2 ;
return IIO_VAL_INT_PLUS_MICRO ;
}
}
return - EINVAL ;
}
static int kxcjk1013_get_acc_reg ( struct kxcjk1013_data * data , int axis )
{
u8 reg = KXCJK1013_REG_XOUT_L + axis * 2 ;
int ret ;
ret = i2c_smbus_read_word_data ( data - > client , reg ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev ,
" failed to read accel_%c registers \n " , ' x ' + axis ) ;
return ret ;
}
return ret ;
}
2014-05-08 22:58:00 +01:00
static int kxcjk1013_set_scale ( struct kxcjk1013_data * data , int val )
{
int ret , i ;
enum kxcjk1013_mode store_mode ;
for ( i = 0 ; i < ARRAY_SIZE ( KXCJK1013_scale_table ) ; + + i ) {
if ( KXCJK1013_scale_table [ i ] . scale = = val ) {
ret = kxcjk1013_get_mode ( data , & store_mode ) ;
if ( ret < 0 )
return ret ;
ret = kxcjk1013_set_mode ( data , STANDBY ) ;
if ( ret < 0 )
return ret ;
ret = kxcjk1013_set_range ( data , i ) ;
if ( ret < 0 )
return ret ;
if ( store_mode = = OPERATION ) {
ret = kxcjk1013_set_mode ( data , OPERATION ) ;
if ( ret )
return ret ;
}
return 0 ;
}
}
return - EINVAL ;
}
2014-11-06 23:07:00 +00:00
static int kxcjk1013_read_raw ( struct iio_dev * indio_dev ,
struct iio_chan_spec const * chan , int * val ,
int * val2 , long mask )
{
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
int ret ;
switch ( mask ) {
case IIO_CHAN_INFO_RAW :
mutex_lock ( & data - > mutex ) ;
if ( iio_buffer_enabled ( indio_dev ) )
ret = - EBUSY ;
else {
2014-05-08 22:58:00 +01:00
ret = kxcjk1013_set_power_state ( data , true ) ;
2014-06-16 20:00:00 +01:00
if ( ret < 0 ) {
mutex_unlock ( & data - > mutex ) ;
2014-11-06 23:07:00 +00:00
return ret ;
2014-06-16 20:00:00 +01:00
}
2014-11-06 23:07:00 +00:00
ret = kxcjk1013_get_acc_reg ( data , chan - > scan_index ) ;
2014-05-08 22:58:00 +01:00
if ( ret < 0 ) {
kxcjk1013_set_power_state ( data , false ) ;
mutex_unlock ( & data - > mutex ) ;
return ret ;
}
* val = sign_extend32 ( ret > > 4 , 11 ) ;
ret = kxcjk1013_set_power_state ( data , false ) ;
2014-11-06 23:07:00 +00:00
}
mutex_unlock ( & data - > mutex ) ;
if ( ret < 0 )
return ret ;
return IIO_VAL_INT ;
case IIO_CHAN_INFO_SCALE :
* val = 0 ;
2014-05-08 22:58:00 +01:00
* val2 = KXCJK1013_scale_table [ data - > range ] . scale ;
2014-11-06 23:07:00 +00:00
return IIO_VAL_INT_PLUS_MICRO ;
case IIO_CHAN_INFO_SAMP_FREQ :
mutex_lock ( & data - > mutex ) ;
ret = kxcjk1013_get_odr ( data , val , val2 ) ;
mutex_unlock ( & data - > mutex ) ;
return ret ;
default :
return - EINVAL ;
}
}
static int kxcjk1013_write_raw ( struct iio_dev * indio_dev ,
struct iio_chan_spec const * chan , int val ,
int val2 , long mask )
{
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
int ret ;
switch ( mask ) {
case IIO_CHAN_INFO_SAMP_FREQ :
mutex_lock ( & data - > mutex ) ;
ret = kxcjk1013_set_odr ( data , val , val2 ) ;
mutex_unlock ( & data - > mutex ) ;
break ;
2014-05-08 22:58:00 +01:00
case IIO_CHAN_INFO_SCALE :
if ( val )
return - EINVAL ;
mutex_lock ( & data - > mutex ) ;
ret = kxcjk1013_set_scale ( data , val2 ) ;
mutex_unlock ( & data - > mutex ) ;
break ;
2014-11-06 23:07:00 +00:00
default :
ret = - EINVAL ;
}
return ret ;
}
2014-08-22 20:01:00 +01:00
static int kxcjk1013_read_event ( struct iio_dev * indio_dev ,
const struct iio_chan_spec * chan ,
enum iio_event_type type ,
enum iio_event_direction dir ,
enum iio_event_info info ,
int * val , int * val2 )
{
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
* val2 = 0 ;
switch ( info ) {
case IIO_EV_INFO_VALUE :
* val = data - > wake_thres ;
break ;
case IIO_EV_INFO_PERIOD :
* val = data - > wake_dur ;
break ;
default :
return - EINVAL ;
}
return IIO_VAL_INT ;
}
static int kxcjk1013_write_event ( struct iio_dev * indio_dev ,
const struct iio_chan_spec * chan ,
enum iio_event_type type ,
enum iio_event_direction dir ,
enum iio_event_info info ,
int val , int val2 )
{
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
if ( data - > ev_enable_state )
return - EBUSY ;
switch ( info ) {
case IIO_EV_INFO_VALUE :
data - > wake_thres = val ;
break ;
case IIO_EV_INFO_PERIOD :
data - > wake_dur = val ;
break ;
default :
return - EINVAL ;
}
return 0 ;
}
static int kxcjk1013_read_event_config ( struct iio_dev * indio_dev ,
const struct iio_chan_spec * chan ,
enum iio_event_type type ,
enum iio_event_direction dir )
{
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
return data - > ev_enable_state ;
}
static int kxcjk1013_write_event_config ( struct iio_dev * indio_dev ,
const struct iio_chan_spec * chan ,
enum iio_event_type type ,
enum iio_event_direction dir ,
int state )
{
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
int ret ;
if ( state & & data - > ev_enable_state )
return 0 ;
mutex_lock ( & data - > mutex ) ;
if ( ! state & & data - > motion_trigger_on ) {
data - > ev_enable_state = 0 ;
mutex_unlock ( & data - > mutex ) ;
return 0 ;
}
/*
* We will expect the enable and disable to do operation in
* in reverse order . This will happen here anyway as our
* resume operation uses sync mode runtime pm calls , the
* suspend operation will be delayed by autosuspend delay
* So the disable operation will still happen in reverse of
* enable operation . When runtime pm is disabled the mode
* is always on so sequence doesn ' t matter
*/
ret = kxcjk1013_set_power_state ( data , state ) ;
if ( ret < 0 ) {
mutex_unlock ( & data - > mutex ) ;
return ret ;
}
ret = kxcjk1013_setup_any_motion_interrupt ( data , state ) ;
if ( ret < 0 ) {
2014-12-10 18:23:53 +02:00
kxcjk1013_set_power_state ( data , false ) ;
data - > ev_enable_state = 0 ;
2014-08-22 20:01:00 +01:00
mutex_unlock ( & data - > mutex ) ;
return ret ;
}
data - > ev_enable_state = state ;
mutex_unlock ( & data - > mutex ) ;
return 0 ;
}
2015-04-03 15:03:02 +03:00
static int kxcjk1013_buffer_preenable ( struct iio_dev * indio_dev )
2014-11-06 23:07:00 +00:00
{
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
2015-04-03 15:03:02 +03:00
return kxcjk1013_set_power_state ( data , true ) ;
}
2014-11-06 23:07:00 +00:00
2015-04-03 15:03:02 +03:00
static int kxcjk1013_buffer_postdisable ( struct iio_dev * indio_dev )
{
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
return kxcjk1013_set_power_state ( data , false ) ;
2014-11-06 23:07:00 +00:00
}
static IIO_CONST_ATTR_SAMP_FREQ_AVAIL (
" 0.781000 1.563000 3.125000 6.250000 12.500000 25 50 100 200 400 800 1600 " ) ;
2014-05-08 22:58:00 +01:00
static IIO_CONST_ATTR ( in_accel_scale_available , " 0.009582 0.019163 0.038326 " ) ;
2014-11-06 23:07:00 +00:00
static struct attribute * kxcjk1013_attributes [ ] = {
& iio_const_attr_sampling_frequency_available . dev_attr . attr ,
2014-05-08 22:58:00 +01:00
& iio_const_attr_in_accel_scale_available . dev_attr . attr ,
2014-11-06 23:07:00 +00:00
NULL ,
} ;
static const struct attribute_group kxcjk1013_attrs_group = {
. attrs = kxcjk1013_attributes ,
} ;
2014-08-22 20:01:00 +01:00
static const struct iio_event_spec kxcjk1013_event = {
. type = IIO_EV_TYPE_THRESH ,
2014-10-10 15:53:00 +01:00
. dir = IIO_EV_DIR_EITHER ,
2014-08-22 20:01:00 +01:00
. mask_separate = BIT ( IIO_EV_INFO_VALUE ) |
BIT ( IIO_EV_INFO_ENABLE ) |
BIT ( IIO_EV_INFO_PERIOD )
} ;
2014-11-06 23:07:00 +00:00
# define KXCJK1013_CHANNEL(_axis) { \
. type = IIO_ACCEL , \
. modified = 1 , \
. channel2 = IIO_MOD_ # # _axis , \
. info_mask_separate = BIT ( IIO_CHAN_INFO_RAW ) , \
. info_mask_shared_by_type = BIT ( IIO_CHAN_INFO_SCALE ) | \
BIT ( IIO_CHAN_INFO_SAMP_FREQ ) , \
. scan_index = AXIS_ # # _axis , \
. scan_type = { \
. sign = ' s ' , \
. realbits = 12 , \
. storagebits = 16 , \
. shift = 4 , \
2016-03-24 11:29:31 +02:00
. endianness = IIO_LE , \
2014-11-06 23:07:00 +00:00
} , \
2014-08-22 20:01:00 +01:00
. event_spec = & kxcjk1013_event , \
. num_event_specs = 1 \
2014-11-06 23:07:00 +00:00
}
static const struct iio_chan_spec kxcjk1013_channels [ ] = {
KXCJK1013_CHANNEL ( X ) ,
KXCJK1013_CHANNEL ( Y ) ,
KXCJK1013_CHANNEL ( Z ) ,
IIO_CHAN_SOFT_TIMESTAMP ( 3 ) ,
} ;
2015-04-03 15:03:02 +03:00
static const struct iio_buffer_setup_ops kxcjk1013_buffer_setup_ops = {
. preenable = kxcjk1013_buffer_preenable ,
. postenable = iio_triggered_buffer_postenable ,
. postdisable = kxcjk1013_buffer_postdisable ,
. predisable = iio_triggered_buffer_predisable ,
} ;
2014-11-06 23:07:00 +00:00
static const struct iio_info kxcjk1013_info = {
. attrs = & kxcjk1013_attrs_group ,
. read_raw = kxcjk1013_read_raw ,
. write_raw = kxcjk1013_write_raw ,
2014-08-22 20:01:00 +01:00
. read_event_value = kxcjk1013_read_event ,
. write_event_value = kxcjk1013_write_event ,
. write_event_config = kxcjk1013_write_event_config ,
. read_event_config = kxcjk1013_read_event_config ,
2014-11-06 23:07:00 +00:00
. driver_module = THIS_MODULE ,
} ;
2016-03-24 11:29:30 +02:00
static const unsigned long kxcjk1013_scan_masks [ ] = { 0x7 , 0 } ;
2014-11-06 23:07:00 +00:00
static irqreturn_t kxcjk1013_trigger_handler ( int irq , void * p )
{
struct iio_poll_func * pf = p ;
struct iio_dev * indio_dev = pf - > indio_dev ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
2016-03-24 11:29:31 +02:00
int ret ;
2014-11-06 23:07:00 +00:00
mutex_lock ( & data - > mutex ) ;
2016-03-24 11:29:31 +02:00
ret = i2c_smbus_read_i2c_block_data_or_emulated ( data - > client ,
KXCJK1013_REG_XOUT_L ,
AXIS_MAX * 2 ,
( u8 * ) data - > buffer ) ;
2014-11-06 23:07:00 +00:00
mutex_unlock ( & data - > mutex ) ;
2016-03-24 11:29:31 +02:00
if ( ret < 0 )
goto err ;
2014-11-06 23:07:00 +00:00
iio_push_to_buffers_with_timestamp ( indio_dev , data - > buffer ,
2014-08-22 20:01:00 +01:00
data - > timestamp ) ;
2014-11-06 23:07:00 +00:00
err :
iio_trigger_notify_done ( indio_dev - > trig ) ;
return IRQ_HANDLED ;
}
2014-07-17 01:42:00 +01:00
static int kxcjk1013_trig_try_reen ( struct iio_trigger * trig )
{
struct iio_dev * indio_dev = iio_trigger_get_drvdata ( trig ) ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
int ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_INT_REL ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_int_rel \n " ) ;
return ret ;
}
return 0 ;
}
2014-11-06 23:07:00 +00:00
static int kxcjk1013_data_rdy_trigger_set_state ( struct iio_trigger * trig ,
bool state )
{
struct iio_dev * indio_dev = iio_trigger_get_drvdata ( trig ) ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
2014-05-08 22:58:00 +01:00
int ret ;
2014-08-22 20:01:00 +01:00
mutex_lock ( & data - > mutex ) ;
if ( ! state & & data - > ev_enable_state & & data - > motion_trigger_on ) {
data - > motion_trigger_on = false ;
mutex_unlock ( & data - > mutex ) ;
2014-05-08 22:58:00 +01:00
return 0 ;
2014-08-22 20:01:00 +01:00
}
2014-11-06 23:07:00 +00:00
2014-08-22 20:01:00 +01:00
ret = kxcjk1013_set_power_state ( data , state ) ;
if ( ret < 0 ) {
mutex_unlock ( & data - > mutex ) ;
return ret ;
2014-11-06 23:07:00 +00:00
}
2014-08-22 20:01:00 +01:00
if ( data - > motion_trig = = trig )
ret = kxcjk1013_setup_any_motion_interrupt ( data , state ) ;
else
ret = kxcjk1013_setup_new_data_interrupt ( data , state ) ;
if ( ret < 0 ) {
2014-12-10 18:23:53 +02:00
kxcjk1013_set_power_state ( data , false ) ;
2014-08-22 20:01:00 +01:00
mutex_unlock ( & data - > mutex ) ;
return ret ;
}
if ( data - > motion_trig = = trig )
data - > motion_trigger_on = state ;
else
data - > dready_trigger_on = state ;
2014-11-06 23:07:00 +00:00
mutex_unlock ( & data - > mutex ) ;
return 0 ;
}
static const struct iio_trigger_ops kxcjk1013_trigger_ops = {
. set_trigger_state = kxcjk1013_data_rdy_trigger_set_state ,
2014-07-17 01:42:00 +01:00
. try_reenable = kxcjk1013_trig_try_reen ,
2014-11-06 23:07:00 +00:00
. owner = THIS_MODULE ,
} ;
2014-08-22 20:01:00 +01:00
static irqreturn_t kxcjk1013_event_handler ( int irq , void * private )
{
struct iio_dev * indio_dev = private ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
int ret ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_INT_SRC1 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " Error reading reg_int_src1 \n " ) ;
goto ack_intr ;
}
if ( ret & 0x02 ) {
ret = i2c_smbus_read_byte_data ( data - > client ,
KXCJK1013_REG_INT_SRC2 ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev ,
" Error reading reg_int_src2 \n " ) ;
goto ack_intr ;
}
if ( ret & KXCJK1013_REG_INT_SRC2_BIT_XN )
iio_push_event ( indio_dev ,
IIO_MOD_EVENT_CODE ( IIO_ACCEL ,
0 ,
IIO_MOD_X ,
IIO_EV_TYPE_THRESH ,
IIO_EV_DIR_FALLING ) ,
data - > timestamp ) ;
if ( ret & KXCJK1013_REG_INT_SRC2_BIT_XP )
iio_push_event ( indio_dev ,
IIO_MOD_EVENT_CODE ( IIO_ACCEL ,
0 ,
IIO_MOD_X ,
IIO_EV_TYPE_THRESH ,
IIO_EV_DIR_RISING ) ,
data - > timestamp ) ;
if ( ret & KXCJK1013_REG_INT_SRC2_BIT_YN )
iio_push_event ( indio_dev ,
IIO_MOD_EVENT_CODE ( IIO_ACCEL ,
0 ,
IIO_MOD_Y ,
IIO_EV_TYPE_THRESH ,
IIO_EV_DIR_FALLING ) ,
data - > timestamp ) ;
if ( ret & KXCJK1013_REG_INT_SRC2_BIT_YP )
iio_push_event ( indio_dev ,
IIO_MOD_EVENT_CODE ( IIO_ACCEL ,
0 ,
IIO_MOD_Y ,
IIO_EV_TYPE_THRESH ,
IIO_EV_DIR_RISING ) ,
data - > timestamp ) ;
if ( ret & KXCJK1013_REG_INT_SRC2_BIT_ZN )
iio_push_event ( indio_dev ,
IIO_MOD_EVENT_CODE ( IIO_ACCEL ,
0 ,
IIO_MOD_Z ,
IIO_EV_TYPE_THRESH ,
IIO_EV_DIR_FALLING ) ,
data - > timestamp ) ;
if ( ret & KXCJK1013_REG_INT_SRC2_BIT_ZP )
iio_push_event ( indio_dev ,
IIO_MOD_EVENT_CODE ( IIO_ACCEL ,
0 ,
IIO_MOD_Z ,
IIO_EV_TYPE_THRESH ,
IIO_EV_DIR_RISING ) ,
data - > timestamp ) ;
}
ack_intr :
if ( data - > dready_trigger_on )
return IRQ_HANDLED ;
ret = i2c_smbus_read_byte_data ( data - > client , KXCJK1013_REG_INT_REL ) ;
if ( ret < 0 )
dev_err ( & data - > client - > dev , " Error reading reg_int_rel \n " ) ;
return IRQ_HANDLED ;
}
static irqreturn_t kxcjk1013_data_rdy_trig_poll ( int irq , void * private )
{
struct iio_dev * indio_dev = private ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
2016-03-09 19:05:49 +01:00
data - > timestamp = iio_get_time_ns ( indio_dev ) ;
2014-08-22 20:01:00 +01:00
if ( data - > dready_trigger_on )
iio_trigger_poll ( data - > dready_trig ) ;
else if ( data - > motion_trigger_on )
iio_trigger_poll ( data - > motion_trig ) ;
if ( data - > ev_enable_state )
return IRQ_WAKE_THREAD ;
else
return IRQ_HANDLED ;
}
2014-03-09 08:33:00 +00:00
static const char * kxcjk1013_match_acpi_device ( struct device * dev ,
2014-05-11 22:09:00 +01:00
enum kx_chipset * chipset ,
bool * is_smo8500_device )
2014-11-06 23:07:00 +00:00
{
const struct acpi_device_id * id ;
2014-12-30 20:57:53 +02:00
2014-03-09 08:33:00 +00:00
id = acpi_match_device ( dev - > driver - > acpi_match_table , dev ) ;
if ( ! id )
return NULL ;
2015-05-04 11:13:04 +02:00
2014-05-11 22:09:00 +01:00
if ( strcmp ( id - > id , " SMO8500 " ) = = 0 )
* is_smo8500_device = true ;
2015-05-04 11:13:04 +02:00
2014-03-09 08:33:00 +00:00
* chipset = ( enum kx_chipset ) id - > driver_data ;
return dev_name ( dev ) ;
}
2014-11-06 23:07:00 +00:00
static int kxcjk1013_probe ( struct i2c_client * client ,
const struct i2c_device_id * id )
{
struct kxcjk1013_data * data ;
struct iio_dev * indio_dev ;
struct kxcjk_1013_platform_data * pdata ;
2014-03-09 08:33:00 +00:00
const char * name ;
2014-11-06 23:07:00 +00:00
int ret ;
indio_dev = devm_iio_device_alloc ( & client - > dev , sizeof ( * data ) ) ;
if ( ! indio_dev )
return - ENOMEM ;
data = iio_priv ( indio_dev ) ;
i2c_set_clientdata ( client , indio_dev ) ;
data - > client = client ;
pdata = dev_get_platdata ( & client - > dev ) ;
if ( pdata )
data - > active_high_intr = pdata - > active_high_intr ;
else
data - > active_high_intr = true ; /* default polarity */
2014-03-09 08:33:00 +00:00
if ( id ) {
data - > chipset = ( enum kx_chipset ) ( id - > driver_data ) ;
name = id - > name ;
} else if ( ACPI_HANDLE ( & client - > dev ) ) {
name = kxcjk1013_match_acpi_device ( & client - > dev ,
2014-05-11 22:09:00 +01:00
& data - > chipset ,
& data - > is_smo8500_device ) ;
2014-03-09 08:33:00 +00:00
} else
return - ENODEV ;
2014-11-06 23:07:00 +00:00
ret = kxcjk1013_chip_init ( data ) ;
if ( ret < 0 )
return ret ;
mutex_init ( & data - > mutex ) ;
indio_dev - > dev . parent = & client - > dev ;
indio_dev - > channels = kxcjk1013_channels ;
indio_dev - > num_channels = ARRAY_SIZE ( kxcjk1013_channels ) ;
2016-03-24 11:29:30 +02:00
indio_dev - > available_scan_masks = kxcjk1013_scan_masks ;
2014-03-09 08:33:00 +00:00
indio_dev - > name = name ;
2014-11-06 23:07:00 +00:00
indio_dev - > modes = INDIO_DIRECT_MODE ;
indio_dev - > info = & kxcjk1013_info ;
2015-09-23 12:02:01 +03:00
if ( client - > irq > 0 & & ! data - > is_smo8500_device ) {
2014-08-22 20:01:00 +01:00
ret = devm_request_threaded_irq ( & client - > dev , client - > irq ,
kxcjk1013_data_rdy_trig_poll ,
kxcjk1013_event_handler ,
IRQF_TRIGGER_RISING ,
KXCJK1013_IRQ_NAME ,
indio_dev ) ;
if ( ret )
2014-12-10 18:23:54 +02:00
goto err_poweroff ;
2014-11-06 23:07:00 +00:00
2014-08-22 20:01:00 +01:00
data - > dready_trig = devm_iio_trigger_alloc ( & client - > dev ,
" %s-dev%d " ,
indio_dev - > name ,
indio_dev - > id ) ;
2014-12-10 18:23:54 +02:00
if ( ! data - > dready_trig ) {
ret = - ENOMEM ;
goto err_poweroff ;
}
2014-11-06 23:07:00 +00:00
2014-08-22 20:01:00 +01:00
data - > motion_trig = devm_iio_trigger_alloc ( & client - > dev ,
" %s-any-motion-dev%d " ,
indio_dev - > name ,
indio_dev - > id ) ;
2014-12-10 18:23:54 +02:00
if ( ! data - > motion_trig ) {
ret = - ENOMEM ;
goto err_poweroff ;
}
2014-11-06 23:07:00 +00:00
2014-08-22 20:01:00 +01:00
data - > dready_trig - > dev . parent = & client - > dev ;
data - > dready_trig - > ops = & kxcjk1013_trigger_ops ;
iio_trigger_set_drvdata ( data - > dready_trig , indio_dev ) ;
indio_dev - > trig = data - > dready_trig ;
2014-07-17 01:42:00 +01:00
iio_trigger_get ( indio_dev - > trig ) ;
2014-08-22 20:01:00 +01:00
ret = iio_trigger_register ( data - > dready_trig ) ;
2014-11-06 23:07:00 +00:00
if ( ret )
2014-12-10 18:23:54 +02:00
goto err_poweroff ;
2014-08-22 20:01:00 +01:00
data - > motion_trig - > dev . parent = & client - > dev ;
data - > motion_trig - > ops = & kxcjk1013_trigger_ops ;
iio_trigger_set_drvdata ( data - > motion_trig , indio_dev ) ;
ret = iio_trigger_register ( data - > motion_trig ) ;
if ( ret ) {
data - > motion_trig = NULL ;
goto err_trigger_unregister ;
}
2015-04-03 15:03:02 +03:00
}
2014-11-06 23:07:00 +00:00
2015-04-03 15:03:02 +03:00
ret = iio_triggered_buffer_setup ( indio_dev ,
& iio_pollfunc_store_time ,
kxcjk1013_trigger_handler ,
& kxcjk1013_buffer_setup_ops ) ;
if ( ret < 0 ) {
dev_err ( & client - > dev , " iio triggered buffer setup failed \n " ) ;
goto err_trigger_unregister ;
2014-11-06 23:07:00 +00:00
}
2014-05-08 22:58:00 +01:00
ret = pm_runtime_set_active ( & client - > dev ) ;
if ( ret )
2015-11-05 16:25:29 +02:00
goto err_buffer_cleanup ;
2014-05-08 22:58:00 +01:00
pm_runtime_enable ( & client - > dev ) ;
pm_runtime_set_autosuspend_delay ( & client - > dev ,
KXCJK1013_SLEEP_DELAY_MS ) ;
pm_runtime_use_autosuspend ( & client - > dev ) ;
2015-11-05 16:25:29 +02:00
ret = iio_device_register ( indio_dev ) ;
if ( ret < 0 ) {
dev_err ( & client - > dev , " unable to register iio device \n " ) ;
goto err_buffer_cleanup ;
}
2014-11-06 23:07:00 +00:00
return 0 ;
err_buffer_cleanup :
2014-08-22 20:01:00 +01:00
if ( data - > dready_trig )
2014-11-06 23:07:00 +00:00
iio_triggered_buffer_cleanup ( indio_dev ) ;
err_trigger_unregister :
2014-08-22 20:01:00 +01:00
if ( data - > dready_trig )
iio_trigger_unregister ( data - > dready_trig ) ;
if ( data - > motion_trig )
iio_trigger_unregister ( data - > motion_trig ) ;
2014-12-10 18:23:54 +02:00
err_poweroff :
kxcjk1013_set_mode ( data , STANDBY ) ;
2014-11-06 23:07:00 +00:00
return ret ;
}
static int kxcjk1013_remove ( struct i2c_client * client )
{
struct iio_dev * indio_dev = i2c_get_clientdata ( client ) ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
2015-11-05 16:25:29 +02:00
iio_device_unregister ( indio_dev ) ;
2014-05-08 22:58:00 +01:00
pm_runtime_disable ( & client - > dev ) ;
pm_runtime_set_suspended ( & client - > dev ) ;
pm_runtime_put_noidle ( & client - > dev ) ;
2014-08-22 20:01:00 +01:00
if ( data - > dready_trig ) {
2014-11-06 23:07:00 +00:00
iio_triggered_buffer_cleanup ( indio_dev ) ;
2014-08-22 20:01:00 +01:00
iio_trigger_unregister ( data - > dready_trig ) ;
iio_trigger_unregister ( data - > motion_trig ) ;
2014-11-06 23:07:00 +00:00
}
mutex_lock ( & data - > mutex ) ;
kxcjk1013_set_mode ( data , STANDBY ) ;
mutex_unlock ( & data - > mutex ) ;
return 0 ;
}
# ifdef CONFIG_PM_SLEEP
static int kxcjk1013_suspend ( struct device * dev )
{
struct iio_dev * indio_dev = i2c_get_clientdata ( to_i2c_client ( dev ) ) ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
2014-05-08 22:58:00 +01:00
int ret ;
2014-11-06 23:07:00 +00:00
mutex_lock ( & data - > mutex ) ;
2014-05-08 22:58:00 +01:00
ret = kxcjk1013_set_mode ( data , STANDBY ) ;
2014-11-06 23:07:00 +00:00
mutex_unlock ( & data - > mutex ) ;
2014-05-08 22:58:00 +01:00
return ret ;
2014-11-06 23:07:00 +00:00
}
static int kxcjk1013_resume ( struct device * dev )
{
struct iio_dev * indio_dev = i2c_get_clientdata ( to_i2c_client ( dev ) ) ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
2014-05-08 22:58:00 +01:00
int ret = 0 ;
2014-11-06 23:07:00 +00:00
mutex_lock ( & data - > mutex ) ;
2014-12-06 00:18:07 +02:00
ret = kxcjk1013_set_mode ( data , OPERATION ) ;
2014-05-08 22:58:00 +01:00
mutex_unlock ( & data - > mutex ) ;
2014-11-06 23:07:00 +00:00
2014-05-08 22:58:00 +01:00
return ret ;
}
# endif
2014-11-06 23:07:00 +00:00
2014-12-04 01:08:13 +01:00
# ifdef CONFIG_PM
2014-05-08 22:58:00 +01:00
static int kxcjk1013_runtime_suspend ( struct device * dev )
{
struct iio_dev * indio_dev = i2c_get_clientdata ( to_i2c_client ( dev ) ) ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
2014-12-10 18:23:53 +02:00
int ret ;
2014-11-06 23:07:00 +00:00
2014-12-10 18:23:53 +02:00
ret = kxcjk1013_set_mode ( data , STANDBY ) ;
if ( ret < 0 ) {
dev_err ( & data - > client - > dev , " powering off device failed \n " ) ;
return - EAGAIN ;
}
return 0 ;
2014-11-06 23:07:00 +00:00
}
2014-05-08 22:58:00 +01:00
static int kxcjk1013_runtime_resume ( struct device * dev )
{
struct iio_dev * indio_dev = i2c_get_clientdata ( to_i2c_client ( dev ) ) ;
struct kxcjk1013_data * data = iio_priv ( indio_dev ) ;
int ret ;
int sleep_val ;
ret = kxcjk1013_set_mode ( data , OPERATION ) ;
if ( ret < 0 )
return ret ;
sleep_val = kxcjk1013_get_startup_times ( data ) ;
if ( sleep_val < 20000 )
usleep_range ( sleep_val , 20000 ) ;
else
msleep_interruptible ( sleep_val / 1000 ) ;
return 0 ;
}
2014-11-06 23:07:00 +00:00
# endif
2014-05-08 22:58:00 +01:00
static const struct dev_pm_ops kxcjk1013_pm_ops = {
SET_SYSTEM_SLEEP_PM_OPS ( kxcjk1013_suspend , kxcjk1013_resume )
SET_RUNTIME_PM_OPS ( kxcjk1013_runtime_suspend ,
kxcjk1013_runtime_resume , NULL )
} ;
2014-11-06 23:07:00 +00:00
static const struct acpi_device_id kx_acpi_match [ ] = {
2014-03-09 08:33:00 +00:00
{ " KXCJ1013 " , KXCJK1013 } ,
{ " KXCJ1008 " , KXCJ91008 } ,
2015-05-04 11:13:03 +02:00
{ " KXCJ9000 " , KXCJ91008 } ,
2016-07-17 10:15:15 +02:00
{ " KIOX000A " , KXCJ91008 } ,
2014-03-09 08:33:00 +00:00
{ " KXTJ1009 " , KXTJ21009 } ,
2014-05-11 22:09:00 +01:00
{ " SMO8500 " , KXCJ91008 } ,
2014-11-06 23:07:00 +00:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( acpi , kx_acpi_match ) ;
static const struct i2c_device_id kxcjk1013_id [ ] = {
2014-03-09 08:33:00 +00:00
{ " kxcjk1013 " , KXCJK1013 } ,
{ " kxcj91008 " , KXCJ91008 } ,
{ " kxtj21009 " , KXTJ21009 } ,
2014-05-11 22:09:00 +01:00
{ " SMO8500 " , KXCJ91008 } ,
2014-11-06 23:07:00 +00:00
{ }
} ;
MODULE_DEVICE_TABLE ( i2c , kxcjk1013_id ) ;
static struct i2c_driver kxcjk1013_driver = {
. driver = {
. name = KXCJK1013_DRV_NAME ,
. acpi_match_table = ACPI_PTR ( kx_acpi_match ) ,
2014-05-08 22:58:00 +01:00
. pm = & kxcjk1013_pm_ops ,
2014-11-06 23:07:00 +00:00
} ,
. probe = kxcjk1013_probe ,
. remove = kxcjk1013_remove ,
. id_table = kxcjk1013_id ,
} ;
module_i2c_driver ( kxcjk1013_driver ) ;
MODULE_AUTHOR ( " Srinivas Pandruvada <srinivas.pandruvada@linux.intel.com> " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;
MODULE_DESCRIPTION ( " KXCJK1013 accelerometer driver " ) ;