/* * apds993x.c - Linux kernel modules for ambient light + proximity sensor * * Copyright (c) 2015, The Linux Foundation. All rights reserved. * Copyright (C) 2012 Lee Kai Koon * Copyright (C) 2012 Avago Technologies * Copyright (C) 2013 LGE Inc. * * 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. * * This program is distributed in the hope that 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. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #define APDS993X_HAL_USE_SYS_ENABLE #define APDS993X_DRV_NAME "apds993x" #define DRIVER_VERSION "1.0.0" #define ALS_POLLING_ENABLED #define APDS993X_PS_DETECTION_THRESHOLD 800 #define APDS993X_PS_HSYTERESIS_THRESHOLD 700 #define APDS993X_PS_PULSE_NUMBER 8 #define APDS993X_ALS_THRESHOLD_HSYTERESIS 20 /* % */ #define APDS993X_GA 48 /* 0.48 without glass window */ #define APDS993X_COE_B 223 /* 2.23 without glass window */ #define APDS993X_COE_C 70 /* 0.70 without glass window */ #define APDS993X_COE_D 142 /* 1.42 without glass window */ #define APDS993X_DF 52 #define ALS_MAX_RANGE 40000 #define APDS_CAL_SKIP_COUNT 5 #define APDS_MAX_CAL (10 + APDS_CAL_SKIP_COUNT) #define CAL_NUM 99 /* Change History * * 1.0.0 Fundamental Functions of APDS-993x * */ #define APDS993X_IOCTL_PS_ENABLE 1 #define APDS993X_IOCTL_PS_GET_ENABLE 2 #define APDS993X_IOCTL_PS_GET_PDATA 3 /* pdata */ #define APDS993X_IOCTL_ALS_ENABLE 4 #define APDS993X_IOCTL_ALS_GET_ENABLE 5 #define APDS993X_IOCTL_ALS_GET_CH0DATA 6 /* ch0data */ #define APDS993X_IOCTL_ALS_GET_CH1DATA 7 /* ch1data */ #define APDS993X_IOCTL_ALS_DELAY 8 /* * Defines */ #define APDS9930_ID 0x30 #define APDS9931_ID 0x39 #define APDS9900_ID 0x29 #define APDS9901_ID 0x20 #define APDS993X_ENABLE_REG 0x00 #define APDS993X_ATIME_REG 0x01 #define APDS993X_PTIME_REG 0x02 #define APDS993X_WTIME_REG 0x03 #define APDS993X_AILTL_REG 0x04 #define APDS993X_AILTH_REG 0x05 #define APDS993X_AIHTL_REG 0x06 #define APDS993X_AIHTH_REG 0x07 #define APDS993X_PILTL_REG 0x08 #define APDS993X_PILTH_REG 0x09 #define APDS993X_PIHTL_REG 0x0A #define APDS993X_PIHTH_REG 0x0B #define APDS993X_PERS_REG 0x0C #define APDS993X_CONFIG_REG 0x0D #define APDS993X_PPCOUNT_REG 0x0E #define APDS993X_CONTROL_REG 0x0F #define APDS993X_REV_REG 0x11 #define APDS993X_ID_REG 0x12 #define APDS993X_STATUS_REG 0x13 #define APDS993X_CH0DATAL_REG 0x14 #define APDS993X_CH0DATAH_REG 0x15 #define APDS993X_CH1DATAL_REG 0x16 #define APDS993X_CH1DATAH_REG 0x17 #define APDS993X_PDATAL_REG 0x18 #define APDS993X_PDATAH_REG 0x19 #define CMD_BYTE 0x80 #define CMD_WORD 0xA0 #define CMD_SPECIAL 0xE0 #define CMD_CLR_PS_INT 0xE5 #define CMD_CLR_ALS_INT 0xE6 #define CMD_CLR_PS_ALS_INT 0xE7 /* Register Value define : ATIME */ #define APDS993X_100MS_ADC_TIME 0xDB /* 100.64ms integration time */ #define APDS993X_50MS_ADC_TIME 0xED /* 51.68ms integration time */ #define APDS993X_27MS_ADC_TIME 0xF6 /* 27.2ms integration time */ /* Register Value define : PRXCNFG */ #define APDS993X_ALS_REDUCE 0x04 /* ALSREDUCE - ALS Gain reduced by 4x */ /* Register Value define : PERS */ #define APDS993X_PPERS_0 0x00 /* Every proximity ADC cycle */ #define APDS993X_PPERS_1 0x10 /* 1 consecutive proximity value out of range */ #define APDS993X_PPERS_2 0x20 /* 2 consecutive proximity value out of range */ #define APDS993X_PPERS_3 0x30 /* 3 consecutive proximity value out of range */ #define APDS993X_PPERS_4 0x40 /* 4 consecutive proximity value out of range */ #define APDS993X_PPERS_5 0x50 /* 5 consecutive proximity value out of range */ #define APDS993X_PPERS_6 0x60 /* 6 consecutive proximity value out of range */ #define APDS993X_PPERS_7 0x70 /* 7 consecutive proximity value out of range */ #define APDS993X_PPERS_8 0x80 /* 8 consecutive proximity value out of range */ #define APDS993X_PPERS_9 0x90 /* 9 consecutive proximity value out of range */ #define APDS993X_PPERS_10 0xA0 /* 10 consecutive proximity value out of range */ #define APDS993X_PPERS_11 0xB0 /* 11 consecutive proximity value out of range */ #define APDS993X_PPERS_12 0xC0 /* 12 consecutive proximity value out of range */ #define APDS993X_PPERS_13 0xD0 /* 13 consecutive proximity value out of range */ #define APDS993X_PPERS_14 0xE0 /* 14 consecutive proximity value out of range */ #define APDS993X_PPERS_15 0xF0 /* 15 consecutive proximity value out of range */ #define APDS993X_APERS_0 0x00 /* Every ADC cycle */ #define APDS993X_APERS_1 0x01 /* 1 consecutive proximity value out of range */ #define APDS993X_APERS_2 0x02 /* 2 consecutive proximity value out of range */ #define APDS993X_APERS_3 0x03 /* 3 consecutive proximity value out of range */ #define APDS993X_APERS_5 0x04 /* 5 consecutive proximity value out of range */ #define APDS993X_APERS_10 0x05 /* 10 consecutive proximity value out of range */ #define APDS993X_APERS_15 0x06 /* 15 consecutive proximity value out of range */ #define APDS993X_APERS_20 0x07 /* 20 consecutive proximity value out of range */ #define APDS993X_APERS_25 0x08 /* 25 consecutive proximity value out of range */ #define APDS993X_APERS_30 0x09 /* 30 consecutive proximity value out of range */ #define APDS993X_APERS_35 0x0A /* 35 consecutive proximity value out of range */ #define APDS993X_APERS_40 0x0B /* 40 consecutive proximity value out of range */ #define APDS993X_APERS_45 0x0C /* 45 consecutive proximity value out of range */ #define APDS993X_APERS_50 0x0D /* 50 consecutive proximity value out of range */ #define APDS993X_APERS_55 0x0E /* 55 consecutive proximity value out of range */ #define APDS993X_APERS_60 0x0F /* 60 consecutive proximity value out of range */ /* Register Value define : CONTROL */ #define APDS993X_AGAIN_1X 0x00 /* 1X ALS GAIN */ #define APDS993X_AGAIN_8X 0x01 /* 8X ALS GAIN */ #define APDS993X_AGAIN_16X 0x02 /* 16X ALS GAIN */ #define APDS993X_AGAIN_120X 0x03 /* 120X ALS GAIN */ #define APDS993X_PRX_IR_DIOD 0x20 /* Proximity uses CH1 diode */ #define APDS993X_PGAIN_1X 0x00 /* PS GAIN 1X */ #define APDS993X_PGAIN_2X 0x04 /* PS GAIN 2X */ #define APDS993X_PGAIN_4X 0x08 /* PS GAIN 4X */ #define APDS993X_PGAIN_8X 0x0C /* PS GAIN 8X */ #define APDS993X_PDRVIE_100MA 0x00 /* PS 100mA LED drive */ #define APDS993X_PDRVIE_50MA 0x40 /* PS 50mA LED drive */ #define APDS993X_PDRVIE_25MA 0x80 /* PS 25mA LED drive */ #define APDS993X_PDRVIE_12_5MA 0xC0 /* PS 12.5mA LED drive */ /*calibration*/ #define DEFAULT_CROSS_TALK 100 #define ADD_TO_CROSS_TALK 300 #define SUB_FROM_PS_THRESHOLD 100 /*PS tuning value*/ static int apds993x_ps_detection_threshold = 0; static int apds993x_ps_hsyteresis_threshold = 0; static int apds993x_ps_pulse_number = 0; static int apds993x_ps_pgain = 0; typedef enum { APDS993X_ALS_RES_10240 = 0, /* 27.2ms integration time */ APDS993X_ALS_RES_19456 = 1, /* 51.68ms integration time */ APDS993X_ALS_RES_37888 = 2 /* 100.64ms integration time */ } apds993x_als_res_e; typedef enum { APDS993X_ALS_GAIN_1X = 0, /* 1x AGAIN */ APDS993X_ALS_GAIN_8X = 1, /* 8x AGAIN */ APDS993X_ALS_GAIN_16X = 2, /* 16x AGAIN */ APDS993X_ALS_GAIN_120X = 3 /* 120x AGAIN */ } apds993x_als_gain_e; /* * Structs */ struct apds993x_data { struct i2c_client *client; struct mutex update_lock; struct mutex op_mutex; struct delayed_work dwork; /* for PS interrupt */ struct delayed_work als_dwork; /* for ALS polling */ struct input_dev *input_dev_als; struct input_dev *input_dev_ps; struct sensors_classdev als_cdev; struct sensors_classdev ps_cdev; /* pinctrl data*/ struct pinctrl *pinctrl; struct pinctrl_state *pin_default; struct pinctrl_state *pin_sleep; struct apds993x_platform_data *platform_data; int irq; /* regulator data */ bool power_on; struct regulator *vdd; struct regulator *vio; /* register configuration*/ unsigned int enable; unsigned int atime; unsigned int ptime; unsigned int wtime; unsigned int ailt; unsigned int aiht; unsigned int pilt; unsigned int piht; unsigned int pers; unsigned int config; unsigned int ppcount; unsigned int control; /* control flag from HAL */ unsigned int enable_ps_sensor; unsigned int enable_als_sensor; /* save sensor enabling state for resume */ unsigned int als_enable_state; /* PS parameters */ unsigned int ps_threshold; unsigned int ps_hysteresis_threshold; /* always lower than ps_threshold */ unsigned int ps_detection; /* 5 = near-to-far; 0 = far-to-near */ unsigned int ps_data; /* to store PS data */ /*calibration*/ unsigned int cross_talk; /* cross_talk value */ unsigned int avg_cross_talk; /* average cross_talk */ unsigned int ps_cal_result; /* result of calibration*/ int ps_cal_data; char calibrate_buf[CAL_NUM]; int ps_cal_params[3]; int pre_enable_ps; /* ALS parameters */ unsigned int als_threshold_l; /* low threshold */ unsigned int als_threshold_h; /* high threshold */ unsigned int als_data; /* to store ALS data */ int als_prev_lux; /* to store previous lux value */ unsigned int als_gain; /* needed for Lux calculation */ unsigned int als_poll_delay; /* needed for light sensor polling : micro-second (us) */ unsigned int als_atime_index; /* storage for als integratiion time */ unsigned int als_again_index; /* storage for als GAIN */ unsigned int als_reduce; /* flag indicate ALS 6x reduction */ }; static struct sensors_classdev sensors_light_cdev = { .name = "apds9930-light", .vendor = "avago", .version = 1, .handle = SENSORS_LIGHT_HANDLE, .type = SENSOR_TYPE_LIGHT, .max_range = "60000", .resolution = "0.0125", .sensor_power = "0.20", .min_delay = 30000, /* in microseconds */ .max_delay = 8393, .fifo_reserved_event_count = 0, .fifo_max_event_count = 0, .flags = 2, .enabled = 0, .delay_msec = 100, .sensors_enable = NULL, .sensors_poll_delay = NULL, .sensors_write_cal_params = NULL, .params = NULL, .sensors_calibrate = NULL, }; static struct sensors_classdev sensors_proximity_cdev = { .name = "apds9930-proximity", .vendor = "avago", .version = 1, .handle = SENSORS_PROXIMITY_HANDLE, .type = SENSOR_TYPE_PROXIMITY, .max_range = "5", .resolution = "5.0", .sensor_power = "3", .min_delay = 30000, /* in microseconds */ .max_delay = 8393, .fifo_reserved_event_count = 0, .fifo_max_event_count = 0, .flags = 3, .enabled = 0, .delay_msec = 100, .sensors_enable = NULL, .sensors_poll_delay = NULL, .sensors_write_cal_params = NULL, .params = NULL, .sensors_calibrate = NULL, }; /* * Global data */ static struct apds993x_data *pdev_data = NULL; /* global i2c_client to support ioctl */ static struct i2c_client *apds993x_i2c_client = NULL; static struct workqueue_struct *apds993x_workqueue = NULL; static unsigned char apds993x_als_atime_tb[] = { 0xF6, 0xED, 0xDB }; static unsigned short apds993x_als_integration_tb[] = {2720, 5168, 10064}; static unsigned short apds993x_als_res_tb[] = { 10240, 19456, 37888 }; static unsigned char apds993x_als_again_tb[] = { 1, 8, 16, 120 }; static unsigned char apds993x_als_again_bit_tb[] = { 0x00, 0x01, 0x02, 0x03 }; /*calibration*/ static int apds993x_cross_talk_val = 0; /* ALS tuning */ static int apds993x_ga = 0; static int apds993x_coe_b = 0; static int apds993x_coe_c = 0; static int apds993x_coe_d = 0; #ifdef ALS_POLLING_ENABLED static int apds993x_set_als_poll_delay(struct i2c_client *client, unsigned int val); #endif static int sensor_regulator_power_on(struct apds993x_data *data, bool on); static int apds993x_init_device(struct i2c_client *client); static int apds9930_ps_get_calibrate_data(struct apds993x_data *data); /* * Management functions */ static int apds993x_set_command(struct i2c_client *client, int command) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; int clearInt; if (command == 0) clearInt = CMD_CLR_PS_INT; else if (command == 1) clearInt = CMD_CLR_ALS_INT; else clearInt = CMD_CLR_PS_ALS_INT; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte(client, clearInt); mutex_unlock(&data->update_lock); return ret; } static int apds993x_set_enable(struct i2c_client *client, int enable) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_ENABLE_REG, enable); mutex_unlock(&data->update_lock); data->enable = enable; return ret; } static int apds993x_set_atime(struct i2c_client *client, int atime) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_ATIME_REG, atime); mutex_unlock(&data->update_lock); data->atime = atime; return ret; } static int apds993x_set_ptime(struct i2c_client *client, int ptime) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_PTIME_REG, ptime); mutex_unlock(&data->update_lock); data->ptime = ptime; return ret; } static int apds993x_set_wtime(struct i2c_client *client, int wtime) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_WTIME_REG, wtime); mutex_unlock(&data->update_lock); data->wtime = wtime; return ret; } static int apds993x_set_ailt(struct i2c_client *client, int threshold) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_AILTL_REG, threshold); mutex_unlock(&data->update_lock); data->ailt = threshold; return ret; } static int apds993x_set_aiht(struct i2c_client *client, int threshold) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_AIHTL_REG, threshold); mutex_unlock(&data->update_lock); data->aiht = threshold; return ret; } static int apds993x_set_pilt(struct i2c_client *client, int threshold) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PILTL_REG, threshold); mutex_unlock(&data->update_lock); data->pilt = threshold; return ret; } static int apds993x_set_piht(struct i2c_client *client, int threshold) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PIHTL_REG, threshold); mutex_unlock(&data->update_lock); data->piht = threshold; return ret; } static int apds993x_set_pers(struct i2c_client *client, int pers) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_PERS_REG, pers); mutex_unlock(&data->update_lock); data->pers = pers; return ret; } static int apds993x_set_config(struct i2c_client *client, int config) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_CONFIG_REG, config); mutex_unlock(&data->update_lock); data->config = config; return ret; } static int apds993x_set_ppcount(struct i2c_client *client, int ppcount) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_PPCOUNT_REG, ppcount); mutex_unlock(&data->update_lock); data->ppcount = ppcount; return ret; } static int apds993x_set_control(struct i2c_client *client, int control) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; mutex_lock(&data->update_lock); ret = i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_CONTROL_REG, control); mutex_unlock(&data->update_lock); data->control = control; return ret; } static void apds993x_report_ps_event(struct input_dev *ps_dev, const unsigned int dist) { ktime_t ts; ts = ktime_get(); input_event(ps_dev, EV_SYN, SYN_TIME_SEC, ktime_to_timespec(ts).tv_sec); input_event(ps_dev, EV_SYN, SYN_TIME_NSEC, ktime_to_timespec(ts).tv_nsec); input_report_abs(ps_dev, ABS_DISTANCE, dist); input_sync(ps_dev); } static void apds993x_report_als_event(struct input_dev *als_dev, const unsigned int lux) { ktime_t ts; ts = ktime_get(); input_event(als_dev, EV_SYN, SYN_TIME_SEC, ktime_to_timespec(ts).tv_sec); input_event(als_dev, EV_SYN, SYN_TIME_NSEC, ktime_to_timespec(ts).tv_nsec); input_report_abs(als_dev, ABS_MISC, lux); input_sync(als_dev); } /*calibration*/ void apds993x_swap(int *x, int *y) { int temp = *x; *x = *y; *y = temp; } static int apds993x_run_cross_talk_calibration(struct i2c_client *client) { struct apds993x_data *data = i2c_get_clientdata(client); unsigned int sum_of_pdata = 0; unsigned int temp_pdata[20]; unsigned int ArySize = 20; unsigned int cal_check_flag = 0; int i, j; #if defined(APDS993x_SENSOR_DEBUG) int status; int rdata; #endif pr_info("%s: START proximity sensor calibration\n", __func__); RECALIBRATION: apds993x_set_enable(client, 0x0D);/* Enable PS and Wait */ #if defined(APDS993x_SENSOR_DEBUG) mutex_lock(&data->update_lock); status = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_STATUS_REG); rdata = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_ENABLE_REG); mutex_unlock(&data->update_lock); pr_info("%s: APDS993x_ENABLE_REG=%2d APDS993x_STATUS_REG=%2d\n", __func__, rdata, status); #endif for (i = 0; i < 20; i++) { mdelay(6); mutex_lock(&data->update_lock); temp_pdata[i] = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_PDATAL_REG); mutex_unlock(&data->update_lock); } /* pdata sorting */ for (i = 0; i < ArySize - 1; i++) for (j = i+1; j < ArySize; j++) if (temp_pdata[i] > temp_pdata[j]) apds993x_swap(temp_pdata + i, temp_pdata + j); /* calculate the cross-talk using central 10 data */ for (i = 5; i < 15; i++) { pr_info("%s: temp_pdata = %d\n", __func__, temp_pdata[i]); sum_of_pdata = sum_of_pdata + temp_pdata[i]; } data->cross_talk = sum_of_pdata/10; pr_info("%s: sum_of_pdata = %d cross_talk = %d\n", __func__, sum_of_pdata, data->cross_talk); /* * this value is used at Hidden Menu to check * if the calibration is pass or fail */ data->avg_cross_talk = data->cross_talk; if (data->cross_talk > 720) { pr_warn("%s: invalid calibrated data\n", __func__); if (cal_check_flag == 0) { pr_info("%s: RECALIBRATION start\n", __func__); cal_check_flag = 1; goto RECALIBRATION; } else { pr_err("%s: CALIBRATION FAIL -> " "cross_talk is set to DEFAULT\n", __func__); data->cross_talk = DEFAULT_CROSS_TALK; apds993x_set_enable(client, 0x00); /* Power Off */ data->ps_cal_result = 0; /* 0:Fail, 1:Pass */ return -EINVAL; } } data->ps_threshold = ADD_TO_CROSS_TALK + data->cross_talk; data->ps_hysteresis_threshold = data->ps_threshold - SUB_FROM_PS_THRESHOLD; apds993x_set_enable(client, 0x00); /* Power Off */ data->ps_cal_result = 1; pr_info("%s: total_pdata = %d & cross_talk = %d\n", __func__, sum_of_pdata, data->cross_talk); pr_info("%s: FINISH proximity sensor calibration\n", __func__); /* Save the cross-talk to the non-volitile memory in the phone */ return data->cross_talk; } /* apply the Cross-talk value to threshold */ static void apds993x_set_ps_threshold_adding_cross_talk( struct i2c_client *client, int cal_data) { struct apds993x_data *data = i2c_get_clientdata(client); if (cal_data > 770) cal_data = 770; if (cal_data < 0) cal_data = 0; data->ps_threshold = apds993x_ps_detection_threshold + cal_data; data->ps_hysteresis_threshold = apds993x_ps_hsyteresis_threshold + cal_data; dev_dbg(&client->dev, "%s: configurations are set\n", __func__); } static int LuxCalculation(struct i2c_client *client, int ch0data, int ch1data) { struct apds993x_data *data = i2c_get_clientdata(client); int luxValue=0; int IAC1=0; int IAC2=0; int IAC=0; if (ch0data >= apds993x_als_res_tb[data->als_atime_index] || ch1data >= apds993x_als_res_tb[data->als_atime_index]) { luxValue = data->als_prev_lux; return luxValue; } /* re-adjust COE_B to avoid 2 decimal point */ IAC1 = (ch0data - (apds993x_coe_b * ch1data) / 100); /* re-adjust COE_C and COE_D to void 2 decimal point */ IAC2 = ((apds993x_coe_c * ch0data) / 100 - (apds993x_coe_d * ch1data) / 100); if (IAC1 > IAC2) IAC = IAC1; else if (IAC1 <= IAC2) IAC = IAC2; else IAC = 0; if (IAC1 < 0 && IAC2 < 0) { IAC = 0; /* cdata and irdata saturated */ return -1; /* don't report first, change gain may help */ } luxValue = ((IAC * apds993x_ga * APDS993X_DF) / 100) / ((apds993x_als_integration_tb[data->als_atime_index] / 100) * apds993x_als_again_tb[data->als_again_index]); return luxValue; } static void apds993x_change_ps_threshold(struct i2c_client *client) { struct apds993x_data *data = i2c_get_clientdata(client); data->ps_data = i2c_smbus_read_word_data( client, CMD_WORD|APDS993X_PDATAL_REG); if ((data->ps_data > data->pilt) && (data->ps_data >= data->piht)) { /* far-to-near detected */ data->ps_detection = 1; /* FAR-to-NEAR detection */ apds993x_report_ps_event(data->input_dev_ps, 0); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PILTL_REG, data->ps_hysteresis_threshold); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PIHTL_REG, 1023); data->pilt = data->ps_hysteresis_threshold; data->piht = 1023; pr_info("%s: far-to-near\n", __func__); } else if ((data->ps_data <= data->pilt) && (data->ps_data < data->piht)) { /* near-to-far detected */ data->ps_detection = 0; /* NEAR-to-FAR detection */ apds993x_report_ps_event(data->input_dev_ps, 1); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PILTL_REG, 0); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PIHTL_REG, data->ps_threshold); data->pilt = 0; data->piht = data->ps_threshold; pr_info("%s: near-to-far\n", __func__); } } static void apds993x_change_als_threshold(struct i2c_client *client) { struct apds993x_data *data = i2c_get_clientdata(client); int ch0data, ch1data, v; int luxValue=0; unsigned char change_again=0; unsigned char control_data=0; unsigned char lux_is_valid=1; ch0data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH0DATAL_REG); ch1data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH1DATAL_REG); luxValue = LuxCalculation(client, ch0data, ch1data); if (luxValue >= 0) { luxValue = (luxValue < ALS_MAX_RANGE) ? luxValue : ALS_MAX_RANGE; if (luxValue == data->als_prev_lux) lux_is_valid = 0; else data->als_prev_lux = luxValue; } else { /* don't report, the lux is invalid value */ lux_is_valid = 0; luxValue = data->als_prev_lux; /* report anyway since this is the lowest gain */ } /* pr_info("%s: lux=%d ch0data=%d ch1data=%d again=%d als_reduce=%d\n", __func__, luxValue, ch0data, ch1data, apds993x_als_again_tb[data->als_again_index], data->als_reduce); */ /* * check PS under sunlight * PS was previously in far-to-near condition */ v = 1024 * (256 - apds993x_als_atime_tb[data->als_atime_index]); v = (v * 75) / 100; if ((data->ps_detection == 1) && (ch0data > v)) { /* * need to inform input event as there will be no interrupt * from the PS */ /* NEAR-to-FAR detection */ apds993x_report_ps_event(data->input_dev_ps, 1); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PILTL_REG, 0); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PIHTL_REG, data->ps_threshold); data->pilt = 0; data->piht = data->ps_threshold; /* near-to-far detected */ data->ps_detection = 0; pr_info("%s: FAR\n", __func__); } if (lux_is_valid) /* report the lux level */ apds993x_report_als_event(data->input_dev_als, luxValue); data->als_data = ch0data; data->als_threshold_l = (data->als_data * (100 - APDS993X_ALS_THRESHOLD_HSYTERESIS)) / 100; data->als_threshold_h = (data->als_data * (100 + APDS993X_ALS_THRESHOLD_HSYTERESIS)) / 100; if (data->als_threshold_h >= apds993x_als_res_tb[data->als_atime_index]) { data->als_threshold_h = apds993x_als_res_tb[data->als_atime_index]; } if (data->als_data >= ((apds993x_als_res_tb[data->als_atime_index] * 90 ) / 100)) { /* lower AGAIN if possible */ if (data->als_again_index != APDS993X_ALS_GAIN_1X) { data->als_again_index--; change_again = 1; } } else if (data->als_data <= ((apds993x_als_res_tb[data->als_atime_index] * 10) / 100)) { /* increase AGAIN if possible */ if (data->als_again_index != APDS993X_ALS_GAIN_120X) { data->als_again_index++; change_again = 1; } } if (change_again) { control_data = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_CONTROL_REG); control_data = control_data & 0xFC; control_data = control_data | apds993x_als_again_bit_tb[data->als_again_index]; i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_CONTROL_REG, control_data); } i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_AILTL_REG, data->als_threshold_l); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_AIHTL_REG, data->als_threshold_h); } static void apds993x_reschedule_work(struct apds993x_data *data, unsigned long delay) { /* * If work is already scheduled then subsequent schedules will not * change the scheduled time that's why we have to cancel it first. */ cancel_delayed_work(&data->dwork); queue_delayed_work(apds993x_workqueue, &data->dwork, delay); } #ifdef ALS_POLLING_ENABLED /* ALS polling routine */ static void apds993x_als_polling_work_handler(struct work_struct *work) { struct apds993x_data *data = container_of(work, struct apds993x_data, als_dwork.work); struct i2c_client *client=data->client; int ch0data, ch1data, pdata, v; int luxValue=0; unsigned char change_again=0; unsigned char control_data=0; unsigned char lux_is_valid=1; ch0data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH0DATAL_REG); ch1data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH1DATAL_REG); pdata = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_PDATAL_REG); luxValue = LuxCalculation(client, ch0data, ch1data); if (luxValue >= 0) { luxValue = (luxValue < ALS_MAX_RANGE) ? luxValue : ALS_MAX_RANGE; if (luxValue == data->als_prev_lux) lux_is_valid = 0; else data->als_prev_lux = luxValue; } else { /* don't report, this is invalid lux value */ lux_is_valid = 0; luxValue = data->als_prev_lux; /* report anyway since this is the lowest gain */ } /* pr_info("%s: lux=%d ch0data=%d ch1data=%d pdata=%d delay=%d again=%d " "als_reduce=%d)\n", __func__, luxValue, ch0data, ch1data, pdata, data->als_poll_delay, apds993x_als_again_tb[data->als_again_index], data->als_reduce); */ /* * check PS under sunlight * PS was previously in far-to-near condition */ v = (75 * (1024 * (256 - data->atime))) / 100; if ((data->ps_detection == 1) && (ch0data > v)) { /* * need to inform input event as there will be no interrupt * from the PS */ /* NEAR-to-FAR detection */ apds993x_report_ps_event(data->input_dev_ps, 1); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PILTL_REG, 0); i2c_smbus_write_word_data(client, CMD_WORD|APDS993X_PIHTL_REG, data->ps_threshold); data->pilt = 0; data->piht = data->ps_threshold; data->ps_detection = 0; /* near-to-far detected */ pr_info("%s: FAR\n", __func__); } if (lux_is_valid) /* report the lux level */ apds993x_report_als_event(data->input_dev_als, luxValue); data->als_data = ch0data; if (data->als_data >= (apds993x_als_res_tb[data->als_atime_index]* 90) / 100) { /* lower AGAIN if possible */ if (data->als_again_index != APDS993X_ALS_GAIN_1X) { data->als_again_index--; change_again = 1; } } else if (data->als_data <= (apds993x_als_res_tb[data->als_atime_index] * 10) / 100) { /* increase AGAIN if possible */ if (data->als_again_index != APDS993X_ALS_GAIN_120X) { data->als_again_index++; change_again = 1; } } if (change_again) { control_data = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_CONTROL_REG); control_data = control_data & 0xFC; control_data = control_data | apds993x_als_again_bit_tb[data->als_again_index]; i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_CONTROL_REG, control_data); } /* restart timer */ queue_delayed_work(apds993x_workqueue, &data->als_dwork, msecs_to_jiffies(data->als_poll_delay)); } #endif /* ALS_POLLING_ENABLED */ /* PS interrupt routine */ static void apds993x_work_handler(struct work_struct *work) { struct apds993x_data *data = container_of(work, struct apds993x_data, dwork.work); struct i2c_client *client=data->client; int status; int ch0data; int enable; status = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_STATUS_REG); enable = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_ENABLE_REG); /* disable 993x's ADC first */ i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_ENABLE_REG, 1); pr_debug("%s: status = %x\n", __func__, status); if ((status & enable & 0x30) == 0x30) { /* both PS and ALS are interrupted */ apds993x_change_als_threshold(client); ch0data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH0DATAL_REG); if (ch0data < (75 * (1024 * (256 - data->atime))) / 100) { apds993x_change_ps_threshold(client); } else { if (data->ps_detection == 1) apds993x_change_ps_threshold(client); else pr_info("%s: background ambient noise\n", __func__); } /* 2 = CMD_CLR_PS_ALS_INT */ apds993x_set_command(client, 2); } else if ((status & enable & 0x20) == 0x20) { /* only PS is interrupted */ /* check if this is triggered by background ambient noise */ ch0data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH0DATAL_REG); if (ch0data < (75 * (apds993x_als_res_tb[data->als_atime_index])) / 100) { apds993x_change_ps_threshold(client); } else { if (data->ps_detection == 1) apds993x_change_ps_threshold(client); else pr_info("%s: background ambient noise\n", __func__); } /* 0 = CMD_CLR_PS_INT */ apds993x_set_command(client, 0); } else if ((status & enable & 0x10) == 0x10) { /* only ALS is interrupted */ apds993x_change_als_threshold(client); /* 1 = CMD_CLR_ALS_INT */ apds993x_set_command(client, 1); } i2c_smbus_write_byte_data(client, CMD_BYTE|APDS993X_ENABLE_REG, data->enable); } /* assume this is ISR */ static irqreturn_t apds993x_interrupt(int vec, void *info) { struct i2c_client *client=(struct i2c_client *)info; struct apds993x_data *data = i2c_get_clientdata(client); apds993x_reschedule_work(data, 0); return IRQ_HANDLED; } /* * IOCTL support */ static int apds993x_enable_als_sensor(struct i2c_client *client, int val) { struct apds993x_data *data = i2c_get_clientdata(client); struct apds993x_platform_data *pdata = data->platform_data; int rc; pr_debug("%s: val=%d\n", __func__, val); if ((val != 0) && (val != 1)) { pr_err("%s: invalid value (val = %d)\n", __func__, val); return -EINVAL; } mutex_lock(&data->op_mutex); if (val == 1) { /* turn on light sensor */ if ((data->enable_als_sensor == 0) && (data->enable_ps_sensor == 0)) { /* Power on and initalize the device */ if (pdata->power_on) pdata->power_on(true); rc = apds993x_init_device(client); if (rc) { dev_err(&client->dev, "Failed to init apds993x\n"); mutex_unlock(&data->op_mutex); return rc; } } if (data->enable_als_sensor == 0) { data->enable_als_sensor = 1; /* Power Off */ apds993x_set_enable(client,0); #ifdef ALS_POLLING_ENABLED if (data->enable_ps_sensor) { /* Enable PS with interrupt */ apds993x_set_enable(client, 0x27); } else { /* no interrupt*/ apds993x_set_enable(client, 0x03); } #else /* * force first ALS interrupt in order to * get environment reading */ apds993x_set_ailt( client, 0xFFFF); apds993x_set_aiht( client, 0); if (data->enable_ps_sensor) { /* Enable both ALS and PS with interrupt */ apds993x_set_enable(client, 0x37); } else { /* only enable light sensor with interrupt*/ apds993x_set_enable(client, 0x13); if (data->irq) enable_irq(data->irq); } #endif #ifdef ALS_POLLING_ENABLED /* * If work is already scheduled then subsequent * schedules will not change the scheduled time * that's why we have to cancel it first. */ cancel_delayed_work_sync(&data->als_dwork); queue_delayed_work(apds993x_workqueue, &data->als_dwork, msecs_to_jiffies(data->als_poll_delay)); #endif } } else { /* * turn off light sensor * what if the p sensor is active? */ data->enable_als_sensor = 0; if (data->enable_ps_sensor) { /* Power Off */ apds993x_set_enable(client,0); apds993x_set_piht(client, 0); apds993x_set_piht(client, apds993x_ps_detection_threshold); /* only enable prox sensor with interrupt */ apds993x_set_enable(client, 0x27); } else { apds993x_set_enable(client, 0); } #ifdef ALS_POLLING_ENABLED /* * If work is already scheduled then subsequent schedules * will not change the scheduled time that's why we have * to cancel it first. */ cancel_delayed_work_sync(&data->als_dwork); #endif } /* Vote off regulators if both light and prox sensor are off */ if ((data->enable_als_sensor == 0) && (data->enable_ps_sensor == 0) && (pdata->power_on)) pdata->power_on(false); mutex_unlock(&data->op_mutex); return 0; } #ifdef ALS_POLLING_ENABLED static int apds993x_set_als_poll_delay(struct i2c_client *client, unsigned int val) { struct apds993x_data *data = i2c_get_clientdata(client); int ret; int atime_index=0; pr_debug("%s: val=%d\n", __func__, val); mutex_lock(&data->op_mutex); /* minimum 30ms */ if (val < 30) val = 30; data->als_poll_delay = val; if (data->als_poll_delay >= 100) atime_index = APDS993X_ALS_RES_37888; else if (data->als_poll_delay >= 50) atime_index = APDS993X_ALS_RES_19456; else atime_index = APDS993X_ALS_RES_10240; ret = apds993x_set_atime(client, apds993x_als_atime_tb[atime_index]); if (ret >= 0) { data->als_atime_index = atime_index; pr_debug("poll delay %d, atime_index %d\n", data->als_poll_delay, data->als_atime_index); } else { mutex_unlock(&data->op_mutex); return ret; } if (data->enable_als_sensor) { mod_delayed_work(apds993x_workqueue, &data->als_dwork, msecs_to_jiffies(data->als_poll_delay)); } mutex_unlock(&data->op_mutex); return 0; } #endif static int apds993x_enable_ps_sensor(struct i2c_client *client, int val) { struct apds993x_data *data = i2c_get_clientdata(client); struct apds993x_platform_data *pdata = data->platform_data; int rc; pr_debug("%s: val=%d\n", __func__, val); if ((val != 0) && (val != 1)) { pr_err("%s: invalid value=%d\n", __func__, val); return -EINVAL; } if (val == 1) { /* turn on p sensor */ if ((data->enable_als_sensor == 0) && (data->enable_ps_sensor == 0)) { /* Power on and initalize the device */ if (pdata->power_on) pdata->power_on(true); rc = apds993x_init_device(client); if (rc) { dev_err(&client->dev, "Failed to init apds993x\n"); return rc; } } if (data->enable_ps_sensor==0) { data->enable_ps_sensor= 1; /* Power Off */ apds993x_set_enable(client,0); /* init threshold for proximity */ apds993x_set_pilt(client, apds993x_ps_detection_threshold); apds993x_set_piht(client, apds993x_ps_detection_threshold); /*calirbation*/ if (data->platform_data->default_cal) { apds993x_set_ps_threshold_adding_cross_talk( client, data->cross_talk); } if (data->enable_als_sensor==0) { /* only enable PS interrupt */ apds993x_set_enable(client, 0x27); if (data->irq) { enable_irq(data->irq); irq_set_irq_wake(client->irq, 1); } } else { #ifdef ALS_POLLING_ENABLED /* enable PS interrupt */ apds993x_set_enable(client, 0x27); if (data->irq) { enable_irq(data->irq); irq_set_irq_wake(client->irq, 1); } #else /* enable ALS and PS interrupt */ apds993x_set_enable(client, 0x37); irq_set_irq_wake(client->irq, 1); #endif } } } else { /* * turn off p sensor - kk 25 Apr 2011 * we can't turn off the entire sensor, * the light sensor may be needed by HAL */ data->enable_ps_sensor = 0; if (data->enable_als_sensor) { #ifdef ALS_POLLING_ENABLED /* no ALS interrupt */ if (data->irq) { irq_set_irq_wake(client->irq, 0); disable_irq(data->irq); } apds993x_set_enable(client, 0x03); /* * If work is already scheduled then subsequent * schedules will not change the scheduled time * that's why we have to cancel it first. */ cancel_delayed_work_sync(&data->als_dwork); /* 100ms */ queue_delayed_work(apds993x_workqueue, &data->als_dwork, msecs_to_jiffies(data->als_poll_delay)); #else /* reconfigute light sensor setting */ if (data->irq) irq_set_irq_wake(client->irq, 0); /* Power Off */ apds993x_set_enable(client,0); /* Force ALS interrupt */ apds993x_set_ailt( client, 0xFFFF); apds993x_set_aiht( client, 0); /* enable ALS interrupt */ apds993x_set_enable(client, 0x13); #endif } else { if (data->irq) { irq_set_irq_wake(client->irq, 0); disable_irq(data->irq); } apds993x_set_enable(client, 0); #ifdef ALS_POLLING_ENABLED /* * If work is already scheduled then subsequent * schedules will not change the scheduled time * that's why we have to cancel it first. */ cancel_delayed_work_sync(&data->als_dwork); #endif } } /* Vote off regulators if both light and prox sensor are off */ if ((data->enable_als_sensor == 0) && (data->enable_ps_sensor == 0) && (pdata->power_on)) pdata->power_on(false); return 0; } static int apds993x_ps_open(struct inode *inode, struct file *file) { return 0; } static int apds993x_ps_release(struct inode *inode, struct file *file) { return 0; } static long apds993x_ps_ioctl(struct file *file, unsigned int cmd, unsigned long arg) { struct apds993x_data *data; struct i2c_client *client; int enable; int ret = -1; if (arg == 0) return -EINVAL; if (apds993x_i2c_client == NULL) { pr_err("%s: i2c driver not installed\n", __func__); return -ENODEV; } client = apds993x_i2c_client; data = i2c_get_clientdata(apds993x_i2c_client); switch (cmd) { case APDS993X_IOCTL_PS_ENABLE: ret = copy_from_user(&enable, (void __user *)arg, sizeof(enable)); if (ret) { pr_err("%s: PS_ENABLE: copy_from_user failed\n", __func__); return -EFAULT; } ret = apds993x_enable_ps_sensor(client, enable); if (ret < 0) return ret; break; case APDS993X_IOCTL_PS_GET_ENABLE: ret = copy_to_user((void __user *)arg, &data->enable_ps_sensor, sizeof(data->enable_ps_sensor)); if (ret) { pr_err("%s: PS_GET_ENABLE: copy_to_user failed\n", __func__); return -EFAULT; } break; case APDS993X_IOCTL_PS_GET_PDATA: data->ps_data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_PDATAL_REG); ret = copy_to_user((void __user *)arg, &data->ps_data, sizeof(data->ps_data)); if (ret) { pr_err("%s: PS_GET_PDATA: copy_to_user failed\n", __func__); return -EFAULT; } break; default: pr_warn("%s: unknown ioctl (%d)\n", __func__, cmd); break; } return 0; } static int apds993x_als_open(struct inode *inode, struct file *file) { return 0; } static int apds993x_als_release(struct inode *inode, struct file *file) { return 0; } static long apds993x_als_ioctl(struct file *file, unsigned int cmd, unsigned long arg) { struct apds993x_data *data; struct i2c_client *client; int enable; int ret = -1; #ifdef ALS_POLLING_ENABLED unsigned int delay; #endif if (arg == 0) return -EINVAL; if (apds993x_i2c_client == NULL) { pr_err("%s: i2c driver not installed\n", __func__); return -ENODEV; } client = apds993x_i2c_client; data = i2c_get_clientdata(apds993x_i2c_client); switch (cmd) { case APDS993X_IOCTL_ALS_ENABLE: ret = copy_from_user(&enable, (void __user *)arg, sizeof(enable)); if (ret) { pr_err("%s: ALS_ENABLE: copy_from_user failed\n", __func__); return -EFAULT; } ret = apds993x_enable_als_sensor(client, enable); if (ret < 0) return ret; break; #ifdef ALS_POLLING_ENABLED case APDS993X_IOCTL_ALS_DELAY: ret = copy_from_user(&delay, (void __user *)arg, sizeof(delay)); if (ret) { pr_err("%s: ALS_DELAY: copy_to_user failed\n", __func__); return -EFAULT; } ret = apds993x_set_als_poll_delay (client, delay); if (ret < 0) return ret; break; #endif case APDS993X_IOCTL_ALS_GET_ENABLE: ret = copy_to_user((void __user *)arg, &data->enable_als_sensor, sizeof(data->enable_als_sensor)); if (ret) { pr_err("%s: ALS_GET_ENABLE: copy_to_user failed\n", __func__); return -EFAULT; } break; case APDS993X_IOCTL_ALS_GET_CH0DATA: data->als_data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH0DATAL_REG); ret = copy_to_user((void __user *)arg, &data->als_data, sizeof(data->als_data)); if (ret) { pr_err("%s: ALS_GET_CH0DATA: copy_to_user failed\n", __func__); return -EFAULT; } break; case APDS993X_IOCTL_ALS_GET_CH1DATA: data->als_data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH1DATAL_REG); ret = copy_to_user((void __user *)arg, &data->als_data, sizeof(data->als_data)); if (ret) { pr_err("%s: ALS_GET_CH1DATA: copy_to_user failed\n", __func__); return -EFAULT; } break; default: pr_warn("%s: unknown ioctl (%d)\n", __func__, cmd); break; } return 0; } /* * SysFS support */ static ssize_t apds993x_show_ch0data(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); int ch0data; mutex_lock(&data->update_lock); ch0data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH0DATAL_REG); mutex_unlock(&data->update_lock); return sprintf(buf, "%d\n", ch0data); } static DEVICE_ATTR(ch0data, S_IRUGO, apds993x_show_ch0data, NULL); static ssize_t apds993x_show_ch1data(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); int ch1data; mutex_lock(&data->update_lock); ch1data = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_CH1DATAL_REG); mutex_unlock(&data->update_lock); return sprintf(buf, "%d\n", ch1data); } static DEVICE_ATTR(ch1data, S_IRUGO, apds993x_show_ch1data, NULL); static ssize_t apds993x_show_pdata(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); int pdata; mutex_lock(&data->update_lock); pdata = i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_PDATAL_REG); pdata |= i2c_smbus_read_word_data(client, CMD_WORD|APDS993X_PDATAH_REG) << 8; mutex_unlock(&data->update_lock); return sprintf(buf, "%d\n", pdata); } static DEVICE_ATTR(pdata, S_IRUGO, apds993x_show_pdata, NULL); /*calibration sysfs*/ static ssize_t apds993x_show_status(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); int status; int rdata; mutex_lock(&data->update_lock); status = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_STATUS_REG); rdata = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_ENABLE_REG); mutex_unlock(&data->update_lock); pr_info("%s: APDS993x_ENABLE_REG=%2d APDS993x_STATUS_REG=%2d\n", __func__, rdata, status); return sprintf(buf, "%d\n", status); } static DEVICE_ATTR(status, S_IRUSR | S_IRGRP, apds993x_show_status, NULL); static ssize_t apds993x_show_ps_run_calibration(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); return sprintf(buf, "%d\n", data->avg_cross_talk); } static ssize_t apds993x_store_ps_run_calibration(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); int ret = 0; /* start calibration */ ret = apds993x_run_cross_talk_calibration(client); /* set threshold for near/far status */ data->ps_threshold = data->cross_talk + ADD_TO_CROSS_TALK; data->ps_hysteresis_threshold = data->ps_threshold - SUB_FROM_PS_THRESHOLD; pr_info("%s: [piht][pilt][c_t] = [%d][%d][%d]\n", __func__, data->ps_threshold, data->ps_hysteresis_threshold, data->cross_talk); if (ret < 0) return ret; return count; } static DEVICE_ATTR(ps_run_calibration, S_IWUSR | S_IWGRP | S_IRUGO, apds993x_show_ps_run_calibration, apds993x_store_ps_run_calibration); static ssize_t apds993x_show_ps_default_crosstalk(struct device *dev, struct device_attribute *attr, char *buf) { return sprintf(buf, "%d\n", DEFAULT_CROSS_TALK); } static ssize_t apds993x_store_ps_default_crosstalk(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); data->ps_threshold = DEFAULT_CROSS_TALK + ADD_TO_CROSS_TALK; data->ps_hysteresis_threshold = data->ps_threshold - SUB_FROM_PS_THRESHOLD; pr_info("%s: [piht][pilt][c_t] = [%d][%d][%d]\n", __func__, data->ps_threshold, data->ps_hysteresis_threshold, data->cross_talk); return count; } static DEVICE_ATTR(ps_default_crosstalk, S_IRUGO | S_IWUSR | S_IWGRP, apds993x_show_ps_default_crosstalk, apds993x_store_ps_default_crosstalk); /* for Calibration result */ static ssize_t apds993x_show_ps_cal_result(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); return sprintf(buf, "%d\n", data->ps_cal_result); } static DEVICE_ATTR(ps_cal_result, S_IRUGO, apds993x_show_ps_cal_result, NULL); /*calibration sysfs end*/ #ifdef APDS993X_HAL_USE_SYS_ENABLE static ssize_t apds993x_show_enable_ps_sensor(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); return sprintf(buf, "%d\n", data->enable_ps_sensor); } static ssize_t apds993x_store_enable_ps_sensor(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct i2c_client *client = to_i2c_client(dev); unsigned long val = simple_strtoul(buf, NULL, 10); pr_debug("%s: val=%ld\n", __func__, val); if (val != 0 && val != 1) { pr_err("%s: invalid value(%ld)\n", __func__, val); return -EINVAL; } apds993x_enable_ps_sensor(client, val); return count; } static DEVICE_ATTR(enable_ps_sensor, S_IWUSR | S_IWGRP | S_IRUGO, apds993x_show_enable_ps_sensor, apds993x_store_enable_ps_sensor); static ssize_t apds993x_show_enable_als_sensor(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); return sprintf(buf, "%d\n", data->enable_als_sensor); } static ssize_t apds993x_store_enable_als_sensor(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct i2c_client *client = to_i2c_client(dev); unsigned long val = simple_strtoul(buf, NULL, 10); pr_debug("%s: val=%ld\n", __func__, val); if (val != 0 && val != 1) { pr_err("%s: invalid value(%ld)\n", __func__, val); return -EINVAL; } apds993x_enable_als_sensor(client, val); return count; } static int apds993x_als_set_enable(struct sensors_classdev *sensors_cdev, unsigned int enable) { struct apds993x_data *data = container_of(sensors_cdev, struct apds993x_data, als_cdev); if ((enable != 0) && (enable != 1)) { pr_err("%s: invalid value(%d)\n", __func__, enable); return -EINVAL; } return apds993x_enable_als_sensor(data->client, enable); } static int apds993x_ps_set_enable(struct sensors_classdev *sensors_cdev, unsigned int enable) { struct apds993x_data *data = container_of(sensors_cdev, struct apds993x_data, ps_cdev); if ((enable != 0) && (enable != 1)) { pr_err("%s: invalid value(%d)\n", __func__, enable); return -EINVAL; } return apds993x_enable_ps_sensor(data->client, enable); } static int apds993x_ps_calibrate(struct sensors_classdev *sensors_cdev, int axis, int apply_now) { int i, arry = 0; int temp[3] = { 0 }; struct apds993x_data *data = container_of(sensors_cdev, struct apds993x_data, ps_cdev); data->pre_enable_ps = data->enable_ps_sensor; if (!data->enable_ps_sensor) apds993x_enable_ps_sensor(data->client, 1); for (i = 0; i < APDS_MAX_CAL; i++) { msleep(100); data->ps_cal_data = i2c_smbus_read_word_data( data->client, CMD_WORD|APDS993X_PDATAL_REG); if (i < APDS_CAL_SKIP_COUNT) continue; dev_dbg(&data->client->dev, "ps_cal data = %d\n", data->ps_cal_data); arry = arry + data->ps_cal_data; } arry = arry / (APDS_MAX_CAL - APDS_CAL_SKIP_COUNT); if (axis == AXIS_THRESHOLD_H) temp[0] = arry; else if (axis == AXIS_THRESHOLD_L) temp[1] = arry; else if (axis == AXIS_BIAS) temp[2] = arry; if (apply_now) { data->ps_cal_params[0] = temp[0]; data->ps_cal_params[1] = temp[1]; data->ps_cal_params[2] = temp[2]; apds9930_ps_get_calibrate_data(data); } memset(data->calibrate_buf, 0 , sizeof(data->calibrate_buf)); snprintf(data->calibrate_buf, sizeof(data->calibrate_buf), "%d,%d,%d", temp[0], temp[1], temp[2]); sensors_cdev->params = data->calibrate_buf; if (!data->pre_enable_ps) apds993x_enable_ps_sensor(data->client, 0); return 0; } static int apds993x_ps_write_calibrate(struct sensors_classdev *sensors_cdev, struct cal_result_t *cal_result) { struct apds993x_data *data = container_of(sensors_cdev, struct apds993x_data, ps_cdev); data->ps_cal_params[0] = cal_result->threshold_h; data->ps_cal_params[1] = cal_result->threshold_l; data->ps_cal_params[2] = cal_result->bias; apds9930_ps_get_calibrate_data(data); return 0; } static int apds9930_ps_get_calibrate_data(struct apds993x_data *data) { if (data->ps_cal_params[2]) { data->ps_hysteresis_threshold = apds993x_ps_hsyteresis_threshold + data->ps_cal_params[2]; data->ps_threshold = apds993x_ps_detection_threshold + data->ps_cal_params[2]; } else if (data->ps_cal_params[0] && data->ps_cal_params[1]) { apds993x_ps_detection_threshold = data->ps_cal_params[0]; data->ps_threshold = data->ps_cal_params[0]; apds993x_ps_hsyteresis_threshold = data->ps_cal_params[1]; data->ps_hysteresis_threshold = data->ps_cal_params[1]; } return 0; } static DEVICE_ATTR(enable_als_sensor, S_IWUSR | S_IWGRP | S_IRUGO, apds993x_show_enable_als_sensor, apds993x_store_enable_als_sensor); static ssize_t apds993x_show_als_poll_delay(struct device *dev, struct device_attribute *attr, char *buf) { struct i2c_client *client = to_i2c_client(dev); struct apds993x_data *data = i2c_get_clientdata(client); /* return in micro-second */ return snprintf(buf, PAGE_SIZE, "%d\n", data->als_poll_delay); } static ssize_t apds993x_store_als_poll_delay(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { #ifdef ALS_POLLING_ENABLED struct i2c_client *client = to_i2c_client(dev); unsigned long val = simple_strtoul(buf, NULL, 10); apds993x_set_als_poll_delay(client, val); #endif return count; } #ifdef ALS_POLLING_ENABLED static int apds993x_als_poll_delay(struct sensors_classdev *sensors_cdev, unsigned int delay_msec) { struct apds993x_data *data = container_of(sensors_cdev, struct apds993x_data, als_cdev); apds993x_set_als_poll_delay(data->client, delay_msec); return 0; } #else static int apds993x_als_poll_delay(struct sensors_classdev *sensors_cdev, unsigned int delay_msec) { return 0; } #endif static DEVICE_ATTR(als_poll_delay, S_IWUSR | S_IRUGO, apds993x_show_als_poll_delay, apds993x_store_als_poll_delay); #endif static struct attribute *apds993x_attributes[] = { &dev_attr_ch0data.attr, &dev_attr_ch1data.attr, &dev_attr_pdata.attr, #ifdef APDS993X_HAL_USE_SYS_ENABLE &dev_attr_enable_ps_sensor.attr, &dev_attr_enable_als_sensor.attr, &dev_attr_als_poll_delay.attr, #endif /*calibration*/ &dev_attr_status.attr, &dev_attr_ps_run_calibration.attr, &dev_attr_ps_default_crosstalk.attr, &dev_attr_ps_cal_result.attr, NULL }; static const struct attribute_group apds993x_attr_group = { .attrs = apds993x_attributes, }; static struct file_operations apds993x_ps_fops = { .owner = THIS_MODULE, .open = apds993x_ps_open, .release = apds993x_ps_release, .unlocked_ioctl = apds993x_ps_ioctl, }; static struct miscdevice apds993x_ps_device = { .minor = MISC_DYNAMIC_MINOR, .name = "apds993x_ps_dev", .fops = &apds993x_ps_fops, }; static struct file_operations apds993x_als_fops = { .owner = THIS_MODULE, .open = apds993x_als_open, .release = apds993x_als_release, .unlocked_ioctl = apds993x_als_ioctl, }; static struct miscdevice apds993x_als_device = { .minor = MISC_DYNAMIC_MINOR, .name = "apds993x_als_dev", .fops = &apds993x_als_fops, }; static int apds993x_check_chip_id(struct i2c_client *client) { int id; id = i2c_smbus_read_byte_data(client, CMD_BYTE|APDS993X_ID_REG); switch (id) { case APDS9931_ID: dev_dbg(&client->dev, "APDS9931\n"); break; case APDS9930_ID: dev_dbg(&client->dev, "APDS9930\n"); break; case APDS9900_ID: dev_dbg(&client->dev, "APDS9900\n"); break; case APDS9901_ID: dev_dbg(&client->dev, "APDS9931\n"); break; default: dev_err(&client->dev, "Neither APDS993x nor APDS990x\n"); return -ENODEV; } return 0; } /* * Initialization function */ static int apds993x_init_device(struct i2c_client *client) { struct apds993x_data *data = i2c_get_clientdata(client); int err; err = apds993x_set_enable(client, 0); if (err < 0) return err; /* 100.64ms ALS integration time */ err = apds993x_set_atime(client, apds993x_als_atime_tb[data->als_atime_index]); if (err < 0) return err; /* 2.72ms Prox integration time */ err = apds993x_set_ptime(client, 0xFF); if (err < 0) return err; /* 2.72ms Wait time */ err = apds993x_set_wtime(client, 0xFF); if (err < 0) return err; err = apds993x_set_ppcount(client, apds993x_ps_pulse_number); if (err < 0) return err; /* no long wait */ err = apds993x_set_config(client, 0); if (err < 0) return err; err = apds993x_set_control(client, APDS993X_PDRVIE_100MA | APDS993X_PRX_IR_DIOD | apds993x_ps_pgain | apds993x_als_again_bit_tb[data->als_again_index]); if (err < 0) return err; /* init threshold for proximity */ err = apds993x_set_pilt(client, 0); if (err < 0) return err; err = apds993x_set_piht(client, apds993x_ps_detection_threshold); if (err < 0) return err; /*calirbation*/ if (data->platform_data->default_cal) { apds993x_set_ps_threshold_adding_cross_talk(client, data->cross_talk); } data->ps_detection = 0; /* initial value = far*/ /* force first ALS interrupt to get the environment reading */ err = apds993x_set_ailt(client, 0xFFFF); if (err < 0) return err; err = apds993x_set_aiht(client, 0); if (err < 0) return err; /* 2 consecutive Interrupt persistence */ err = apds993x_set_pers(client, APDS993X_PPERS_2|APDS993X_APERS_2); if (err < 0) return err; /* sensor is in disabled mode but all the configurations are preset */ return 0; } static int apds993x_suspend(struct device *dev) { struct apds993x_data *data; struct apds993x_platform_data *pdata; int rc; data = dev_get_drvdata(dev); pdata = data->platform_data; /* * Save sensor state and disable them, * this is to ensure internal state flags are set correctly. * device will power off after both sensors are disabled. * P sensor will not be disabled because it is a wakeup sensor. */ data->als_enable_state = data->enable_als_sensor; if (data->als_enable_state) { rc = apds993x_enable_als_sensor(data->client, 0); if (rc) dev_err(&data->client->dev, "Disable light sensor fail! rc=%d\n", rc); } return 0; } static int apds993x_resume(struct device *dev) { struct apds993x_data *data; struct apds993x_platform_data *pdata; int rc; data = dev_get_drvdata(dev); pdata = data->platform_data; /* Resume L sensor state as P sensor does not disable */ if (data->als_enable_state) { rc = apds993x_enable_als_sensor(data->client, 1); if (rc) dev_err(&data->client->dev, "Disable light sensor fail! rc=%d\n", rc); } return 0; } static int sensor_regulator_configure(struct apds993x_data *data, bool on) { int rc; if (!on) { if (regulator_count_voltages(data->vdd) > 0) regulator_set_voltage(data->vdd, 0, APDS993X_VDD_MAX_UV); regulator_put(data->vdd); if (regulator_count_voltages(data->vio) > 0) regulator_set_voltage(data->vio, 0, APDS993X_VIO_MAX_UV); regulator_put(data->vio); } else { data->vdd = regulator_get(&data->client->dev, "vdd"); if (IS_ERR(data->vdd)) { rc = PTR_ERR(data->vdd); dev_err(&data->client->dev, "Regulator get failed vdd rc=%d\n", rc); return rc; } if (regulator_count_voltages(data->vdd) > 0) { rc = regulator_set_voltage(data->vdd, APDS993X_VDD_MIN_UV, APDS993X_VDD_MAX_UV); if (rc) { dev_err(&data->client->dev, "Regulator set failed vdd rc=%d\n", rc); goto reg_vdd_put; } } data->vio = regulator_get(&data->client->dev, "vio"); if (IS_ERR(data->vio)) { rc = PTR_ERR(data->vio); dev_err(&data->client->dev, "Regulator get failed vio rc=%d\n", rc); goto reg_vdd_set; } if (regulator_count_voltages(data->vio) > 0) { rc = regulator_set_voltage(data->vio, APDS993X_VIO_MIN_UV, APDS993X_VIO_MAX_UV); if (rc) { dev_err(&data->client->dev, "Regulator set failed vio rc=%d\n", rc); goto reg_vio_put; } } } return 0; reg_vio_put: regulator_put(data->vio); reg_vdd_set: if (regulator_count_voltages(data->vdd) > 0) regulator_set_voltage(data->vdd, 0, APDS993X_VDD_MAX_UV); reg_vdd_put: regulator_put(data->vdd); return rc; } static int sensor_regulator_power_on(struct apds993x_data *data, bool on) { int rc = 0; if (!on) { rc = regulator_disable(data->vdd); if (rc) { dev_err(&data->client->dev, "Regulator vdd disable failed rc=%d\n", rc); return rc; } rc = regulator_disable(data->vio); if (rc) { dev_err(&data->client->dev, "Regulator vio disable failed rc=%d\n", rc); rc = regulator_enable(data->vdd); dev_err(&data->client->dev, "Regulator vio re-enabled rc=%d\n", rc); /* * Successfully re-enable regulator. * Enter poweron delay and returns error. */ if (!rc) { rc = -EBUSY; goto enable_delay; } } return rc; } else { rc = regulator_enable(data->vdd); if (rc) { dev_err(&data->client->dev, "Regulator vdd enable failed rc=%d\n", rc); return rc; } rc = regulator_enable(data->vio); if (rc) { dev_err(&data->client->dev, "Regulator vio enable failed rc=%d\n", rc); regulator_disable(data->vdd); return rc; } } enable_delay: msleep(130); dev_dbg(&data->client->dev, "Sensor regulator power on =%d\n", on); return rc; } static int sensor_platform_hw_power_on(bool on) { struct apds993x_data *data; int err = 0; if (pdev_data == NULL) return -ENODEV; data = pdev_data; if (data->power_on != on) { if (!IS_ERR_OR_NULL(data->pinctrl)) { if (on) err = pinctrl_select_state(data->pinctrl, data->pin_default); else err = pinctrl_select_state(data->pinctrl, data->pin_sleep); if (err) dev_err(&data->client->dev, "Can't select pinctrl state\n"); } err = sensor_regulator_power_on(data, on); if (err) dev_err(&data->client->dev, "Can't configure regulator!\n"); else data->power_on = on; } return err; } static int sensor_platform_hw_init(void) { struct i2c_client *client; struct apds993x_data *data; int error; if (pdev_data == NULL) return -ENODEV; data = pdev_data; client = data->client; error = sensor_regulator_configure(data, true); if (error < 0) { dev_err(&client->dev, "unable to configure regulator\n"); return error; } if (gpio_is_valid(data->platform_data->irq_gpio)) { /* configure apds993x irq gpio */ error = gpio_request_one(data->platform_data->irq_gpio, GPIOF_DIR_IN, "apds993x_irq_gpio"); if (error) { dev_err(&client->dev, "unable to request gpio %d\n", data->platform_data->irq_gpio); } data->irq = client->irq = gpio_to_irq(data->platform_data->irq_gpio); } else { dev_err(&client->dev, "irq gpio not provided\n"); } return 0; } static void sensor_platform_hw_exit(void) { struct apds993x_data *data = pdev_data; if (data == NULL) return; sensor_regulator_configure(data, false); if (gpio_is_valid(data->platform_data->irq_gpio)) gpio_free(data->platform_data->irq_gpio); } static int apds993x_pinctrl_init(struct apds993x_data *data) { struct i2c_client *client = data->client; data->pinctrl = devm_pinctrl_get(&client->dev); if (IS_ERR_OR_NULL(data->pinctrl)) { dev_err(&client->dev, "Failed to get pinctrl\n"); return PTR_ERR(data->pinctrl); } data->pin_default = pinctrl_lookup_state(data->pinctrl, "default"); if (IS_ERR_OR_NULL(data->pin_default)) { dev_err(&client->dev, "Failed to look up default state\n"); return PTR_ERR(data->pin_default); } data->pin_sleep = pinctrl_lookup_state(data->pinctrl, "sleep"); if (IS_ERR_OR_NULL(data->pin_sleep)) { dev_err(&client->dev, "Failed to look up sleep state\n"); return PTR_ERR(data->pin_sleep); } return 0; } static int sensor_parse_dt(struct device *dev, struct apds993x_platform_data *pdata) { struct device_node *np = dev->of_node; unsigned int tmp; int rc = 0; /* set functions of platform data */ pdata->init = sensor_platform_hw_init; pdata->exit = sensor_platform_hw_exit; pdata->power_on = sensor_platform_hw_power_on; /* irq gpio */ rc = of_get_named_gpio_flags(dev->of_node, "avago,irq-gpio", 0, NULL); if (rc < 0) { dev_err(dev, "Unable to read irq gpio\n"); return rc; } pdata->irq_gpio = rc; /* ps tuning data*/ rc = of_property_read_u32(np, "avago,ps-threshold", &tmp); if (rc) { dev_err(dev, "Unable to read ps threshold\n"); return rc; } pdata->prox_threshold = tmp; rc = of_property_read_u32(np, "avago,ps-hysteresis-threshold", &tmp); if (rc) { dev_err(dev, "Unable to read ps hysteresis threshold\n"); return rc; } pdata->prox_hsyteresis_threshold = tmp; rc = of_property_read_u32(np, "avago,cross-talk", &tmp); if (rc) { dev_info(dev, "Unable to read cross_talk use default 100\n"); pdata->cross_talk = DEFAULT_CROSS_TALK; } else { pdata->cross_talk = tmp; } rc = of_property_read_u32(np, "avago,ps-pulse", &tmp); if (rc) { dev_err(dev, "Unable to read ps pulse\n"); return rc; } pdata->prox_pulse = tmp; rc = of_property_read_u32(np, "avago,ps-pgain", &tmp); if (rc) { dev_err(dev, "Unable to read ps pgain\n"); return rc; } pdata->prox_gain = tmp; /* ALS tuning value */ rc = of_property_read_u32(np, "avago,als-B", &tmp); if (rc) { dev_err(dev, "Unable to read apds993x coefficient b\n"); return rc; } pdata->als_B = tmp; rc = of_property_read_u32(np, "avago,als-C", &tmp); if (rc) { dev_err(dev, "Unable to read apds993x coefficient c\n"); return rc; } pdata->als_C = tmp; rc = of_property_read_u32(np, "avago,als-D", &tmp); if (rc) { dev_err(dev, "Unable to read apds993x coefficient d\n"); return rc; } pdata->als_D = tmp; rc = of_property_read_u32(np, "avago,ga-value", &tmp); if (rc) { dev_err(dev, "Unable to read gain value\n"); return rc; } pdata->ga_value = tmp; pdata->default_cal = of_property_read_bool(np, "avago,default-cal"); return 0; } /* * I2C init/probing/exit functions */ static struct i2c_driver apds993x_driver; static int apds993x_probe(struct i2c_client *client, const struct i2c_device_id *id) { struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent); struct apds993x_data *data; struct apds993x_platform_data *pdata; int err = 0; pr_debug("%s\n", __func__); if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE)) { err = -EIO; goto exit; } if (client->dev.of_node) { pdata = devm_kzalloc(&client->dev, sizeof(struct apds993x_platform_data), GFP_KERNEL); if (!pdata) { dev_err(&client->dev, "Failed to allocate memory\n"); return -ENOMEM; } client->dev.platform_data = pdata; err = sensor_parse_dt(&client->dev, pdata); if (err) { pr_err("%s: sensor_parse_dt() err\n", __func__); return err; } } else { pdata = client->dev.platform_data; if (!pdata) { dev_err(&client->dev, "No platform data\n"); return -ENODEV; } } /* Set the default parameters */ apds993x_ps_detection_threshold = pdata->prox_threshold; apds993x_ps_hsyteresis_threshold = pdata->prox_hsyteresis_threshold; apds993x_ps_pulse_number = pdata->prox_pulse; apds993x_ps_pgain = pdata->prox_gain; apds993x_coe_b = pdata->als_B; apds993x_coe_c = pdata->als_C; apds993x_coe_d = pdata->als_D; apds993x_ga = pdata->ga_value; data = kzalloc(sizeof(struct apds993x_data), GFP_KERNEL); if (!data) { dev_err(&client->dev, "Failed to allocate memory\n"); err = -ENOMEM; goto exit; } pdev_data = data; data->platform_data = pdata; data->client = client; apds993x_i2c_client = client; /* initialize pinctrl */ err = apds993x_pinctrl_init(data); if (err) { dev_err(&client->dev, "Can't initialize pinctrl\n"); goto exit_kfree; } err = pinctrl_select_state(data->pinctrl, data->pin_default); if (err) { dev_err(&client->dev, "Can't select pinctrl default state\n"); goto exit_kfree; } /* h/w initialization */ if (pdata->init) err = pdata->init(); if (pdata->power_on) err = pdata->power_on(true); i2c_set_clientdata(client, data); data->enable = 0; /* default mode is standard */ data->ps_threshold = apds993x_ps_detection_threshold; data->ps_hysteresis_threshold = apds993x_ps_hsyteresis_threshold; data->ps_detection = 0; /* default to no detection */ data->enable_als_sensor = 0; // default to 0 data->enable_ps_sensor = 0; // default to 0 data->als_poll_delay = 100; // default to 100ms data->als_atime_index = APDS993X_ALS_RES_37888; // 100ms ATIME data->als_again_index = APDS993X_ALS_GAIN_8X; // 8x AGAIN data->als_reduce = 0; // no ALS 6x reduction data->als_prev_lux = 0; /* calibration */ if (apds993x_cross_talk_val > 0 && apds993x_cross_talk_val < 1000) { data->cross_talk = apds993x_cross_talk_val; } else { /* * default value: Get the cross-talk value from the devicetree. * This value is saved during the cross-talk calibration */ data->cross_talk = pdata->cross_talk; } mutex_init(&data->update_lock); mutex_init(&data->op_mutex); INIT_DELAYED_WORK(&data->dwork, apds993x_work_handler); #ifdef ALS_POLLING_ENABLED INIT_DELAYED_WORK(&data->als_dwork, apds993x_als_polling_work_handler); #endif err = apds993x_check_chip_id(client); if (err) { dev_err(&client->dev, "Not a valid chip ID\n"); err = -ENODEV; goto exit_uninit; } /* Initialize the APDS993X chip */ err = apds993x_init_device(client); if (err) { pr_err("%s: Failed to init apds993x\n", __func__); goto exit_uninit; } if (data->irq) { err = request_irq(data->irq, apds993x_interrupt, IRQF_TRIGGER_FALLING, APDS993X_DRV_NAME, (void *)client); if (err < 0) { dev_err(&client->dev, "Could not allocate APDS993X_INT !\n"); goto exit_uninit; } disable_irq(data->irq); } /* Register to Input Device */ data->input_dev_als = devm_input_allocate_device(&client->dev); if (!data->input_dev_als) { err = -ENOMEM; pr_err("%s: Failed to allocate input device als\n", __func__); goto exit_free_irq; } data->input_dev_ps = devm_input_allocate_device(&client->dev); if (!data->input_dev_ps) { err = -ENOMEM; pr_err("%s: Failed to allocate input device ps\n", __func__); goto exit_free_dev_als; } set_bit(EV_ABS, data->input_dev_als->evbit); set_bit(EV_ABS, data->input_dev_ps->evbit); input_set_abs_params(data->input_dev_als, ABS_MISC, 0, 60000, 0, 0); input_set_abs_params(data->input_dev_ps, ABS_DISTANCE, 0, 1, 0, 0); data->input_dev_als->name = "light"; data->input_dev_ps->name = "proximity"; err = input_register_device(data->input_dev_als); if (err) { err = -ENOMEM; pr_err("%s: Unable to register input device als: %s\n", __func__, data->input_dev_als->name); goto exit_free_dev_ps; } err = input_register_device(data->input_dev_ps); if (err) { err = -ENOMEM; pr_err("%s: Unable to register input device ps: %s\n", __func__, data->input_dev_ps->name); goto exit_free_dev_ps; } /* Register sysfs hooks */ err = sysfs_create_group(&client->dev.kobj, &apds993x_attr_group); if (err) goto exit_free_dev_ps; /* Register for sensor ioctl */ err = misc_register(&apds993x_ps_device); if (err) { pr_err("%s: Unable to register ps ioctl: %d", __func__, err); goto exit_remove_sysfs_group; } err = misc_register(&apds993x_als_device); if (err) { pr_err("%s: Unable to register als ioctl: %d", __func__, err); goto exit_unregister_ps_ioctl; } /* Register to sensors class */ data->als_cdev = sensors_light_cdev; data->als_cdev.sensors_enable = apds993x_als_set_enable; data->als_cdev.sensors_poll_delay = apds993x_als_poll_delay; memset(&data->als_cdev.cal_result, 0, sizeof(data->als_cdev.cal_result)); data->ps_cdev = sensors_proximity_cdev; data->ps_cdev.sensors_enable = apds993x_ps_set_enable; data->ps_cdev.sensors_poll_delay = NULL; if (pdata->default_cal) { data->ps_cdev.sensors_calibrate = NULL; data->ps_cdev.sensors_write_cal_params = NULL; } else { data->ps_cdev.sensors_calibrate = apds993x_ps_calibrate; data->ps_cdev.sensors_write_cal_params = apds993x_ps_write_calibrate; } memset(&data->ps_cdev.cal_result, 0 , sizeof(data->ps_cdev.cal_result)); err = sensors_classdev_register(&client->dev, &data->als_cdev); if (err) { pr_err("%s: Unable to register to sensors class: %d\n", __func__, err); goto exit_unregister_als_ioctl; } err = sensors_classdev_register(&client->dev, &data->ps_cdev); if (err) { pr_err("%s: Unable to register to sensors class: %d\n", __func__, err); goto exit_unregister_als_class; } if (pdata->power_on) err = pdata->power_on(false); pr_info("%s: Support ver. %s enabled\n", __func__, DRIVER_VERSION); return 0; exit_unregister_als_class: sensors_classdev_unregister(&data->als_cdev); exit_unregister_als_ioctl: misc_deregister(&apds993x_als_device); exit_unregister_ps_ioctl: misc_deregister(&apds993x_ps_device); exit_remove_sysfs_group: sysfs_remove_group(&client->dev.kobj, &apds993x_attr_group); exit_free_dev_ps: exit_free_dev_als: exit_free_irq: free_irq(data->irq, client); exit_uninit: if (pdata->power_on) pdata->power_on(false); if (pdata->exit) pdata->exit(); exit_kfree: kfree(data); pdev_data = NULL; exit: return err; } static int apds993x_remove(struct i2c_client *client) { struct apds993x_data *data = i2c_get_clientdata(client); struct apds993x_platform_data *pdata = data->platform_data; /* Power down the device */ apds993x_set_enable(client, 0); misc_deregister(&apds993x_als_device); misc_deregister(&apds993x_ps_device); sysfs_remove_group(&client->dev.kobj, &apds993x_attr_group); free_irq(client->irq, data); if (pdata->power_on) pdata->power_on(false); if (pdata->exit) pdata->exit(); kfree(data); pdev_data = NULL; return 0; } static const struct i2c_device_id apds993x_id[] = { { "apds993x", 0 }, { } }; MODULE_DEVICE_TABLE(i2c, apds993x_id); static struct of_device_id apds993X_match_table[] = { { .compatible = "avago,apds9930",}, { .compatible = "avago,apds9900",}, { }, }; static const struct dev_pm_ops apds993x_pm_ops = { .suspend = apds993x_suspend, .resume = apds993x_resume, }; static struct i2c_driver apds993x_driver = { .driver = { .name = APDS993X_DRV_NAME, .owner = THIS_MODULE, .of_match_table = apds993X_match_table, .pm = &apds993x_pm_ops, }, .probe = apds993x_probe, .remove = apds993x_remove, .id_table = apds993x_id, }; static int __init apds993x_init(void) { apds993x_workqueue = create_freezable_workqueue("proximity_als"); if (!apds993x_workqueue) { pr_err("%s: out of memory\n", __func__); return -ENOMEM; } return i2c_add_driver(&apds993x_driver); } static void __exit apds993x_exit(void) { if (apds993x_workqueue) destroy_workqueue(apds993x_workqueue); i2c_del_driver(&apds993x_driver); } MODULE_AUTHOR("Lee Kai Koon "); MODULE_DESCRIPTION("APDS993X ambient light + proximity sensor driver"); MODULE_LICENSE("GPL"); MODULE_VERSION(DRIVER_VERSION); module_init(apds993x_init); module_exit(apds993x_exit);