14193c0f1SVlad Dogaru /* 24193c0f1SVlad Dogaru * Copyright (c) 2014 Intel Corporation 34193c0f1SVlad Dogaru * 44193c0f1SVlad Dogaru * Driver for Semtech's SX9500 capacitive proximity/button solution. 54193c0f1SVlad Dogaru * Datasheet available at 64193c0f1SVlad Dogaru * <http://www.semtech.com/images/datasheet/sx9500.pdf>. 74193c0f1SVlad Dogaru * 84193c0f1SVlad Dogaru * This program is free software; you can redistribute it and/or modify it 94193c0f1SVlad Dogaru * under the terms of the GNU General Public License version 2 as published by 104193c0f1SVlad Dogaru * the Free Software Foundation. 114193c0f1SVlad Dogaru */ 124193c0f1SVlad Dogaru 134193c0f1SVlad Dogaru #include <linux/kernel.h> 144193c0f1SVlad Dogaru #include <linux/slab.h> 154193c0f1SVlad Dogaru #include <linux/module.h> 164193c0f1SVlad Dogaru #include <linux/i2c.h> 174193c0f1SVlad Dogaru #include <linux/irq.h> 184193c0f1SVlad Dogaru #include <linux/acpi.h> 194193c0f1SVlad Dogaru #include <linux/gpio/consumer.h> 204193c0f1SVlad Dogaru #include <linux/regmap.h> 217840ffeeSVlad Dogaru #include <linux/pm.h> 2259bd0427SVlad Dogaru #include <linux/delay.h> 234193c0f1SVlad Dogaru 244193c0f1SVlad Dogaru #include <linux/iio/iio.h> 254193c0f1SVlad Dogaru #include <linux/iio/buffer.h> 264193c0f1SVlad Dogaru #include <linux/iio/sysfs.h> 274193c0f1SVlad Dogaru #include <linux/iio/events.h> 284193c0f1SVlad Dogaru #include <linux/iio/trigger.h> 294193c0f1SVlad Dogaru #include <linux/iio/triggered_buffer.h> 304193c0f1SVlad Dogaru #include <linux/iio/trigger_consumer.h> 314193c0f1SVlad Dogaru 324193c0f1SVlad Dogaru #define SX9500_DRIVER_NAME "sx9500" 334193c0f1SVlad Dogaru #define SX9500_IRQ_NAME "sx9500_event" 3463de9f92SVlad Dogaru 354193c0f1SVlad Dogaru /* Register definitions. */ 364193c0f1SVlad Dogaru #define SX9500_REG_IRQ_SRC 0x00 374193c0f1SVlad Dogaru #define SX9500_REG_STAT 0x01 384193c0f1SVlad Dogaru #define SX9500_REG_IRQ_MSK 0x03 394193c0f1SVlad Dogaru 404193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL0 0x06 414193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL1 0x07 424193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL2 0x08 434193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL3 0x09 444193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL4 0x0a 454193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL5 0x0b 464193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL6 0x0c 474193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL7 0x0d 484193c0f1SVlad Dogaru #define SX9500_REG_PROX_CTRL8 0x0e 494193c0f1SVlad Dogaru 504193c0f1SVlad Dogaru #define SX9500_REG_SENSOR_SEL 0x20 514193c0f1SVlad Dogaru #define SX9500_REG_USE_MSB 0x21 524193c0f1SVlad Dogaru #define SX9500_REG_USE_LSB 0x22 534193c0f1SVlad Dogaru #define SX9500_REG_AVG_MSB 0x23 544193c0f1SVlad Dogaru #define SX9500_REG_AVG_LSB 0x24 554193c0f1SVlad Dogaru #define SX9500_REG_DIFF_MSB 0x25 564193c0f1SVlad Dogaru #define SX9500_REG_DIFF_LSB 0x26 574193c0f1SVlad Dogaru #define SX9500_REG_OFFSET_MSB 0x27 584193c0f1SVlad Dogaru #define SX9500_REG_OFFSET_LSB 0x28 594193c0f1SVlad Dogaru 604193c0f1SVlad Dogaru #define SX9500_REG_RESET 0x7f 614193c0f1SVlad Dogaru 624193c0f1SVlad Dogaru /* Write this to REG_RESET to do a soft reset. */ 634193c0f1SVlad Dogaru #define SX9500_SOFT_RESET 0xde 644193c0f1SVlad Dogaru 654193c0f1SVlad Dogaru #define SX9500_SCAN_PERIOD_MASK GENMASK(6, 4) 664193c0f1SVlad Dogaru #define SX9500_SCAN_PERIOD_SHIFT 4 674193c0f1SVlad Dogaru 684193c0f1SVlad Dogaru /* 694193c0f1SVlad Dogaru * These serve for identifying IRQ source in the IRQ_SRC register, and 704193c0f1SVlad Dogaru * also for masking the IRQs in the IRQ_MSK register. 714193c0f1SVlad Dogaru */ 724193c0f1SVlad Dogaru #define SX9500_CLOSE_IRQ BIT(6) 734193c0f1SVlad Dogaru #define SX9500_FAR_IRQ BIT(5) 744193c0f1SVlad Dogaru #define SX9500_CONVDONE_IRQ BIT(3) 754193c0f1SVlad Dogaru 764193c0f1SVlad Dogaru #define SX9500_PROXSTAT_SHIFT 4 7759bd0427SVlad Dogaru #define SX9500_COMPSTAT_MASK GENMASK(3, 0) 784193c0f1SVlad Dogaru 794193c0f1SVlad Dogaru #define SX9500_NUM_CHANNELS 4 8068958bd5SVlad Dogaru #define SX9500_CHAN_MASK GENMASK(SX9500_NUM_CHANNELS - 1, 0) 814193c0f1SVlad Dogaru 824193c0f1SVlad Dogaru struct sx9500_data { 834193c0f1SVlad Dogaru struct mutex mutex; 844193c0f1SVlad Dogaru struct i2c_client *client; 854193c0f1SVlad Dogaru struct iio_trigger *trig; 864193c0f1SVlad Dogaru struct regmap *regmap; 8745fd5f8eSVlad Dogaru struct gpio_desc *gpiod_rst; 884193c0f1SVlad Dogaru /* 894193c0f1SVlad Dogaru * Last reading of the proximity status for each channel. We 904193c0f1SVlad Dogaru * only send an event to user space when this changes. 914193c0f1SVlad Dogaru */ 924193c0f1SVlad Dogaru bool prox_stat[SX9500_NUM_CHANNELS]; 934193c0f1SVlad Dogaru bool event_enabled[SX9500_NUM_CHANNELS]; 944193c0f1SVlad Dogaru bool trigger_enabled; 954193c0f1SVlad Dogaru u16 *buffer; 967840ffeeSVlad Dogaru /* Remember enabled channels and sample rate during suspend. */ 977840ffeeSVlad Dogaru unsigned int suspend_ctrl0; 9859bd0427SVlad Dogaru struct completion completion; 9959bd0427SVlad Dogaru int data_rdy_users, close_far_users; 10059bd0427SVlad Dogaru int channel_users[SX9500_NUM_CHANNELS]; 1014193c0f1SVlad Dogaru }; 1024193c0f1SVlad Dogaru 1034193c0f1SVlad Dogaru static const struct iio_event_spec sx9500_events[] = { 1044193c0f1SVlad Dogaru { 1054193c0f1SVlad Dogaru .type = IIO_EV_TYPE_THRESH, 1064193c0f1SVlad Dogaru .dir = IIO_EV_DIR_EITHER, 1074193c0f1SVlad Dogaru .mask_separate = BIT(IIO_EV_INFO_ENABLE), 1084193c0f1SVlad Dogaru }, 1094193c0f1SVlad Dogaru }; 1104193c0f1SVlad Dogaru 1114193c0f1SVlad Dogaru #define SX9500_CHANNEL(idx) \ 1124193c0f1SVlad Dogaru { \ 1134193c0f1SVlad Dogaru .type = IIO_PROXIMITY, \ 1144193c0f1SVlad Dogaru .info_mask_separate = BIT(IIO_CHAN_INFO_RAW), \ 1154193c0f1SVlad Dogaru .info_mask_shared_by_all = BIT(IIO_CHAN_INFO_SAMP_FREQ), \ 1164193c0f1SVlad Dogaru .indexed = 1, \ 1174193c0f1SVlad Dogaru .channel = idx, \ 1184193c0f1SVlad Dogaru .event_spec = sx9500_events, \ 1194193c0f1SVlad Dogaru .num_event_specs = ARRAY_SIZE(sx9500_events), \ 1204193c0f1SVlad Dogaru .scan_index = idx, \ 1214193c0f1SVlad Dogaru .scan_type = { \ 1224193c0f1SVlad Dogaru .sign = 'u', \ 1234193c0f1SVlad Dogaru .realbits = 16, \ 1244193c0f1SVlad Dogaru .storagebits = 16, \ 1254193c0f1SVlad Dogaru .shift = 0, \ 1264193c0f1SVlad Dogaru }, \ 1274193c0f1SVlad Dogaru } 1284193c0f1SVlad Dogaru 1294193c0f1SVlad Dogaru static const struct iio_chan_spec sx9500_channels[] = { 1304193c0f1SVlad Dogaru SX9500_CHANNEL(0), 1314193c0f1SVlad Dogaru SX9500_CHANNEL(1), 1324193c0f1SVlad Dogaru SX9500_CHANNEL(2), 1334193c0f1SVlad Dogaru SX9500_CHANNEL(3), 1344193c0f1SVlad Dogaru IIO_CHAN_SOFT_TIMESTAMP(4), 1354193c0f1SVlad Dogaru }; 1364193c0f1SVlad Dogaru 1374193c0f1SVlad Dogaru static const struct { 1384193c0f1SVlad Dogaru int val; 1394193c0f1SVlad Dogaru int val2; 1404193c0f1SVlad Dogaru } sx9500_samp_freq_table[] = { 1414193c0f1SVlad Dogaru {33, 333333}, 1424193c0f1SVlad Dogaru {16, 666666}, 1434193c0f1SVlad Dogaru {11, 111111}, 1444193c0f1SVlad Dogaru {8, 333333}, 1454193c0f1SVlad Dogaru {6, 666666}, 1464193c0f1SVlad Dogaru {5, 0}, 1474193c0f1SVlad Dogaru {3, 333333}, 1484193c0f1SVlad Dogaru {2, 500000}, 1494193c0f1SVlad Dogaru }; 1504193c0f1SVlad Dogaru 15159bd0427SVlad Dogaru static const unsigned int sx9500_scan_period_table[] = { 15259bd0427SVlad Dogaru 30, 60, 90, 120, 150, 200, 300, 400, 15359bd0427SVlad Dogaru }; 15459bd0427SVlad Dogaru 1554193c0f1SVlad Dogaru static const struct regmap_range sx9500_writable_reg_ranges[] = { 1564193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_IRQ_MSK, SX9500_REG_IRQ_MSK), 1574193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_PROX_CTRL0, SX9500_REG_PROX_CTRL8), 1584193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_SENSOR_SEL, SX9500_REG_SENSOR_SEL), 1594193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_OFFSET_MSB, SX9500_REG_OFFSET_LSB), 1604193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_RESET, SX9500_REG_RESET), 1614193c0f1SVlad Dogaru }; 1624193c0f1SVlad Dogaru 1634193c0f1SVlad Dogaru static const struct regmap_access_table sx9500_writeable_regs = { 1644193c0f1SVlad Dogaru .yes_ranges = sx9500_writable_reg_ranges, 1654193c0f1SVlad Dogaru .n_yes_ranges = ARRAY_SIZE(sx9500_writable_reg_ranges), 1664193c0f1SVlad Dogaru }; 1674193c0f1SVlad Dogaru 1684193c0f1SVlad Dogaru /* 1694193c0f1SVlad Dogaru * All allocated registers are readable, so we just list unallocated 1704193c0f1SVlad Dogaru * ones. 1714193c0f1SVlad Dogaru */ 1724193c0f1SVlad Dogaru static const struct regmap_range sx9500_non_readable_reg_ranges[] = { 1734193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_STAT + 1, SX9500_REG_STAT + 1), 1744193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_IRQ_MSK + 1, SX9500_REG_PROX_CTRL0 - 1), 1754193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_PROX_CTRL8 + 1, SX9500_REG_SENSOR_SEL - 1), 1764193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_OFFSET_LSB + 1, SX9500_REG_RESET - 1), 1774193c0f1SVlad Dogaru }; 1784193c0f1SVlad Dogaru 1794193c0f1SVlad Dogaru static const struct regmap_access_table sx9500_readable_regs = { 1804193c0f1SVlad Dogaru .no_ranges = sx9500_non_readable_reg_ranges, 1814193c0f1SVlad Dogaru .n_no_ranges = ARRAY_SIZE(sx9500_non_readable_reg_ranges), 1824193c0f1SVlad Dogaru }; 1834193c0f1SVlad Dogaru 1844193c0f1SVlad Dogaru static const struct regmap_range sx9500_volatile_reg_ranges[] = { 1854193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_IRQ_SRC, SX9500_REG_STAT), 1864193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_USE_MSB, SX9500_REG_OFFSET_LSB), 1874193c0f1SVlad Dogaru regmap_reg_range(SX9500_REG_RESET, SX9500_REG_RESET), 1884193c0f1SVlad Dogaru }; 1894193c0f1SVlad Dogaru 1904193c0f1SVlad Dogaru static const struct regmap_access_table sx9500_volatile_regs = { 1914193c0f1SVlad Dogaru .yes_ranges = sx9500_volatile_reg_ranges, 1924193c0f1SVlad Dogaru .n_yes_ranges = ARRAY_SIZE(sx9500_volatile_reg_ranges), 1934193c0f1SVlad Dogaru }; 1944193c0f1SVlad Dogaru 1954193c0f1SVlad Dogaru static const struct regmap_config sx9500_regmap_config = { 1964193c0f1SVlad Dogaru .reg_bits = 8, 1974193c0f1SVlad Dogaru .val_bits = 8, 1984193c0f1SVlad Dogaru 1994193c0f1SVlad Dogaru .max_register = SX9500_REG_RESET, 2004193c0f1SVlad Dogaru .cache_type = REGCACHE_RBTREE, 2014193c0f1SVlad Dogaru 2024193c0f1SVlad Dogaru .wr_table = &sx9500_writeable_regs, 2034193c0f1SVlad Dogaru .rd_table = &sx9500_readable_regs, 2044193c0f1SVlad Dogaru .volatile_table = &sx9500_volatile_regs, 2054193c0f1SVlad Dogaru }; 2064193c0f1SVlad Dogaru 20759bd0427SVlad Dogaru static int sx9500_inc_users(struct sx9500_data *data, int *counter, 20859bd0427SVlad Dogaru unsigned int reg, unsigned int bitmask) 20959bd0427SVlad Dogaru { 21059bd0427SVlad Dogaru (*counter)++; 21159bd0427SVlad Dogaru if (*counter != 1) 21259bd0427SVlad Dogaru /* Bit is already active, nothing to do. */ 21359bd0427SVlad Dogaru return 0; 21459bd0427SVlad Dogaru 21559bd0427SVlad Dogaru return regmap_update_bits(data->regmap, reg, bitmask, bitmask); 21659bd0427SVlad Dogaru } 21759bd0427SVlad Dogaru 21859bd0427SVlad Dogaru static int sx9500_dec_users(struct sx9500_data *data, int *counter, 21959bd0427SVlad Dogaru unsigned int reg, unsigned int bitmask) 22059bd0427SVlad Dogaru { 22159bd0427SVlad Dogaru (*counter)--; 22259bd0427SVlad Dogaru if (*counter != 0) 22359bd0427SVlad Dogaru /* There are more users, do not deactivate. */ 22459bd0427SVlad Dogaru return 0; 22559bd0427SVlad Dogaru 22659bd0427SVlad Dogaru return regmap_update_bits(data->regmap, reg, bitmask, 0); 22759bd0427SVlad Dogaru } 22859bd0427SVlad Dogaru 22959bd0427SVlad Dogaru static int sx9500_inc_chan_users(struct sx9500_data *data, int chan) 23059bd0427SVlad Dogaru { 23159bd0427SVlad Dogaru return sx9500_inc_users(data, &data->channel_users[chan], 23259bd0427SVlad Dogaru SX9500_REG_PROX_CTRL0, BIT(chan)); 23359bd0427SVlad Dogaru } 23459bd0427SVlad Dogaru 23559bd0427SVlad Dogaru static int sx9500_dec_chan_users(struct sx9500_data *data, int chan) 23659bd0427SVlad Dogaru { 23759bd0427SVlad Dogaru return sx9500_dec_users(data, &data->channel_users[chan], 23859bd0427SVlad Dogaru SX9500_REG_PROX_CTRL0, BIT(chan)); 23959bd0427SVlad Dogaru } 24059bd0427SVlad Dogaru 24159bd0427SVlad Dogaru static int sx9500_inc_data_rdy_users(struct sx9500_data *data) 24259bd0427SVlad Dogaru { 24359bd0427SVlad Dogaru return sx9500_inc_users(data, &data->data_rdy_users, 24459bd0427SVlad Dogaru SX9500_REG_IRQ_MSK, SX9500_CONVDONE_IRQ); 24559bd0427SVlad Dogaru } 24659bd0427SVlad Dogaru 24759bd0427SVlad Dogaru static int sx9500_dec_data_rdy_users(struct sx9500_data *data) 24859bd0427SVlad Dogaru { 24959bd0427SVlad Dogaru return sx9500_dec_users(data, &data->data_rdy_users, 25059bd0427SVlad Dogaru SX9500_REG_IRQ_MSK, SX9500_CONVDONE_IRQ); 25159bd0427SVlad Dogaru } 25259bd0427SVlad Dogaru 25359bd0427SVlad Dogaru static int sx9500_inc_close_far_users(struct sx9500_data *data) 25459bd0427SVlad Dogaru { 25559bd0427SVlad Dogaru return sx9500_inc_users(data, &data->close_far_users, 25659bd0427SVlad Dogaru SX9500_REG_IRQ_MSK, 25759bd0427SVlad Dogaru SX9500_CLOSE_IRQ | SX9500_FAR_IRQ); 25859bd0427SVlad Dogaru } 25959bd0427SVlad Dogaru 26059bd0427SVlad Dogaru static int sx9500_dec_close_far_users(struct sx9500_data *data) 26159bd0427SVlad Dogaru { 26259bd0427SVlad Dogaru return sx9500_dec_users(data, &data->close_far_users, 26359bd0427SVlad Dogaru SX9500_REG_IRQ_MSK, 26459bd0427SVlad Dogaru SX9500_CLOSE_IRQ | SX9500_FAR_IRQ); 26559bd0427SVlad Dogaru } 26659bd0427SVlad Dogaru 26759bd0427SVlad Dogaru static int sx9500_read_prox_data(struct sx9500_data *data, 2684193c0f1SVlad Dogaru const struct iio_chan_spec *chan, 2694193c0f1SVlad Dogaru int *val) 2704193c0f1SVlad Dogaru { 2714193c0f1SVlad Dogaru int ret; 2724193c0f1SVlad Dogaru __be16 regval; 2734193c0f1SVlad Dogaru 2744193c0f1SVlad Dogaru ret = regmap_write(data->regmap, SX9500_REG_SENSOR_SEL, chan->channel); 2754193c0f1SVlad Dogaru if (ret < 0) 2764193c0f1SVlad Dogaru return ret; 2774193c0f1SVlad Dogaru 2784193c0f1SVlad Dogaru ret = regmap_bulk_read(data->regmap, SX9500_REG_USE_MSB, ®val, 2); 2794193c0f1SVlad Dogaru if (ret < 0) 2804193c0f1SVlad Dogaru return ret; 2814193c0f1SVlad Dogaru 282fd1883f0SDaniel Baluta *val = be16_to_cpu(regval); 2834193c0f1SVlad Dogaru 2844193c0f1SVlad Dogaru return IIO_VAL_INT; 2854193c0f1SVlad Dogaru } 2864193c0f1SVlad Dogaru 28759bd0427SVlad Dogaru /* 28859bd0427SVlad Dogaru * If we have no interrupt support, we have to wait for a scan period 28959bd0427SVlad Dogaru * after enabling a channel to get a result. 29059bd0427SVlad Dogaru */ 29159bd0427SVlad Dogaru static int sx9500_wait_for_sample(struct sx9500_data *data) 29259bd0427SVlad Dogaru { 29359bd0427SVlad Dogaru int ret; 29459bd0427SVlad Dogaru unsigned int val; 29559bd0427SVlad Dogaru 29659bd0427SVlad Dogaru ret = regmap_read(data->regmap, SX9500_REG_PROX_CTRL0, &val); 29759bd0427SVlad Dogaru if (ret < 0) 29859bd0427SVlad Dogaru return ret; 29959bd0427SVlad Dogaru 30059bd0427SVlad Dogaru val = (val & SX9500_SCAN_PERIOD_MASK) >> SX9500_SCAN_PERIOD_SHIFT; 30159bd0427SVlad Dogaru 30259bd0427SVlad Dogaru msleep(sx9500_scan_period_table[val]); 30359bd0427SVlad Dogaru 30459bd0427SVlad Dogaru return 0; 30559bd0427SVlad Dogaru } 30659bd0427SVlad Dogaru 30759bd0427SVlad Dogaru static int sx9500_read_proximity(struct sx9500_data *data, 30859bd0427SVlad Dogaru const struct iio_chan_spec *chan, 30959bd0427SVlad Dogaru int *val) 31059bd0427SVlad Dogaru { 31159bd0427SVlad Dogaru int ret; 31259bd0427SVlad Dogaru 31359bd0427SVlad Dogaru mutex_lock(&data->mutex); 31459bd0427SVlad Dogaru 31559bd0427SVlad Dogaru ret = sx9500_inc_chan_users(data, chan->channel); 31659bd0427SVlad Dogaru if (ret < 0) 31759bd0427SVlad Dogaru goto out; 31859bd0427SVlad Dogaru 31959bd0427SVlad Dogaru ret = sx9500_inc_data_rdy_users(data); 32059bd0427SVlad Dogaru if (ret < 0) 32159bd0427SVlad Dogaru goto out_dec_chan; 32259bd0427SVlad Dogaru 32359bd0427SVlad Dogaru mutex_unlock(&data->mutex); 32459bd0427SVlad Dogaru 32559bd0427SVlad Dogaru if (data->client->irq > 0) 32659bd0427SVlad Dogaru ret = wait_for_completion_interruptible(&data->completion); 32759bd0427SVlad Dogaru else 32859bd0427SVlad Dogaru ret = sx9500_wait_for_sample(data); 32959bd0427SVlad Dogaru 33059bd0427SVlad Dogaru mutex_lock(&data->mutex); 33159bd0427SVlad Dogaru 332657c7ff5SVlad Dogaru if (ret < 0) 333657c7ff5SVlad Dogaru goto out_dec_data_rdy; 334657c7ff5SVlad Dogaru 33559bd0427SVlad Dogaru ret = sx9500_read_prox_data(data, chan, val); 33659bd0427SVlad Dogaru if (ret < 0) 337657c7ff5SVlad Dogaru goto out_dec_data_rdy; 33859bd0427SVlad Dogaru 33959bd0427SVlad Dogaru ret = sx9500_dec_data_rdy_users(data); 34059bd0427SVlad Dogaru if (ret < 0) 341657c7ff5SVlad Dogaru goto out_dec_chan; 342657c7ff5SVlad Dogaru 343657c7ff5SVlad Dogaru ret = sx9500_dec_chan_users(data, chan->channel); 344657c7ff5SVlad Dogaru if (ret < 0) 34559bd0427SVlad Dogaru goto out; 34659bd0427SVlad Dogaru 34759bd0427SVlad Dogaru ret = IIO_VAL_INT; 34859bd0427SVlad Dogaru 34959bd0427SVlad Dogaru goto out; 35059bd0427SVlad Dogaru 351657c7ff5SVlad Dogaru out_dec_data_rdy: 352657c7ff5SVlad Dogaru sx9500_dec_data_rdy_users(data); 35359bd0427SVlad Dogaru out_dec_chan: 35459bd0427SVlad Dogaru sx9500_dec_chan_users(data, chan->channel); 35559bd0427SVlad Dogaru out: 35659bd0427SVlad Dogaru mutex_unlock(&data->mutex); 35759bd0427SVlad Dogaru reinit_completion(&data->completion); 35859bd0427SVlad Dogaru 35959bd0427SVlad Dogaru return ret; 36059bd0427SVlad Dogaru } 36159bd0427SVlad Dogaru 3624193c0f1SVlad Dogaru static int sx9500_read_samp_freq(struct sx9500_data *data, 3634193c0f1SVlad Dogaru int *val, int *val2) 3644193c0f1SVlad Dogaru { 3654193c0f1SVlad Dogaru int ret; 3664193c0f1SVlad Dogaru unsigned int regval; 3674193c0f1SVlad Dogaru 3684193c0f1SVlad Dogaru mutex_lock(&data->mutex); 3694193c0f1SVlad Dogaru ret = regmap_read(data->regmap, SX9500_REG_PROX_CTRL0, ®val); 3704193c0f1SVlad Dogaru mutex_unlock(&data->mutex); 3714193c0f1SVlad Dogaru 3724193c0f1SVlad Dogaru if (ret < 0) 3734193c0f1SVlad Dogaru return ret; 3744193c0f1SVlad Dogaru 3754193c0f1SVlad Dogaru regval = (regval & SX9500_SCAN_PERIOD_MASK) >> SX9500_SCAN_PERIOD_SHIFT; 3764193c0f1SVlad Dogaru *val = sx9500_samp_freq_table[regval].val; 3774193c0f1SVlad Dogaru *val2 = sx9500_samp_freq_table[regval].val2; 3784193c0f1SVlad Dogaru 3794193c0f1SVlad Dogaru return IIO_VAL_INT_PLUS_MICRO; 3804193c0f1SVlad Dogaru } 3814193c0f1SVlad Dogaru 3824193c0f1SVlad Dogaru static int sx9500_read_raw(struct iio_dev *indio_dev, 3834193c0f1SVlad Dogaru const struct iio_chan_spec *chan, 3844193c0f1SVlad Dogaru int *val, int *val2, long mask) 3854193c0f1SVlad Dogaru { 3864193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 3876b2e7589SAlison Schofield int ret; 3884193c0f1SVlad Dogaru 3894193c0f1SVlad Dogaru switch (chan->type) { 3904193c0f1SVlad Dogaru case IIO_PROXIMITY: 3914193c0f1SVlad Dogaru switch (mask) { 3924193c0f1SVlad Dogaru case IIO_CHAN_INFO_RAW: 3936b2e7589SAlison Schofield ret = iio_device_claim_direct_mode(indio_dev); 3946b2e7589SAlison Schofield if (ret) 3956b2e7589SAlison Schofield return ret; 3966b2e7589SAlison Schofield ret = sx9500_read_proximity(data, chan, val); 3976b2e7589SAlison Schofield iio_device_release_direct_mode(indio_dev); 3986b2e7589SAlison Schofield return ret; 3994193c0f1SVlad Dogaru case IIO_CHAN_INFO_SAMP_FREQ: 4004193c0f1SVlad Dogaru return sx9500_read_samp_freq(data, val, val2); 4014193c0f1SVlad Dogaru default: 4024193c0f1SVlad Dogaru return -EINVAL; 4034193c0f1SVlad Dogaru } 4044193c0f1SVlad Dogaru default: 4054193c0f1SVlad Dogaru return -EINVAL; 4064193c0f1SVlad Dogaru } 4074193c0f1SVlad Dogaru } 4084193c0f1SVlad Dogaru 4094193c0f1SVlad Dogaru static int sx9500_set_samp_freq(struct sx9500_data *data, 4104193c0f1SVlad Dogaru int val, int val2) 4114193c0f1SVlad Dogaru { 4124193c0f1SVlad Dogaru int i, ret; 4134193c0f1SVlad Dogaru 4144193c0f1SVlad Dogaru for (i = 0; i < ARRAY_SIZE(sx9500_samp_freq_table); i++) 4154193c0f1SVlad Dogaru if (val == sx9500_samp_freq_table[i].val && 4164193c0f1SVlad Dogaru val2 == sx9500_samp_freq_table[i].val2) 4174193c0f1SVlad Dogaru break; 4184193c0f1SVlad Dogaru 4194193c0f1SVlad Dogaru if (i == ARRAY_SIZE(sx9500_samp_freq_table)) 4204193c0f1SVlad Dogaru return -EINVAL; 4214193c0f1SVlad Dogaru 4224193c0f1SVlad Dogaru mutex_lock(&data->mutex); 4234193c0f1SVlad Dogaru 4244193c0f1SVlad Dogaru ret = regmap_update_bits(data->regmap, SX9500_REG_PROX_CTRL0, 4254193c0f1SVlad Dogaru SX9500_SCAN_PERIOD_MASK, 4264193c0f1SVlad Dogaru i << SX9500_SCAN_PERIOD_SHIFT); 4274193c0f1SVlad Dogaru 4284193c0f1SVlad Dogaru mutex_unlock(&data->mutex); 4294193c0f1SVlad Dogaru 4304193c0f1SVlad Dogaru return ret; 4314193c0f1SVlad Dogaru } 4324193c0f1SVlad Dogaru 4334193c0f1SVlad Dogaru static int sx9500_write_raw(struct iio_dev *indio_dev, 4344193c0f1SVlad Dogaru const struct iio_chan_spec *chan, 4354193c0f1SVlad Dogaru int val, int val2, long mask) 4364193c0f1SVlad Dogaru { 4374193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 4384193c0f1SVlad Dogaru 4394193c0f1SVlad Dogaru switch (chan->type) { 4404193c0f1SVlad Dogaru case IIO_PROXIMITY: 4414193c0f1SVlad Dogaru switch (mask) { 4424193c0f1SVlad Dogaru case IIO_CHAN_INFO_SAMP_FREQ: 4434193c0f1SVlad Dogaru return sx9500_set_samp_freq(data, val, val2); 4444193c0f1SVlad Dogaru default: 4454193c0f1SVlad Dogaru return -EINVAL; 4464193c0f1SVlad Dogaru } 4474193c0f1SVlad Dogaru default: 4484193c0f1SVlad Dogaru return -EINVAL; 4494193c0f1SVlad Dogaru } 4504193c0f1SVlad Dogaru } 4514193c0f1SVlad Dogaru 4524193c0f1SVlad Dogaru static irqreturn_t sx9500_irq_handler(int irq, void *private) 4534193c0f1SVlad Dogaru { 4544193c0f1SVlad Dogaru struct iio_dev *indio_dev = private; 4554193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 4564193c0f1SVlad Dogaru 4574193c0f1SVlad Dogaru if (data->trigger_enabled) 4584193c0f1SVlad Dogaru iio_trigger_poll(data->trig); 4594193c0f1SVlad Dogaru 4604193c0f1SVlad Dogaru /* 4614193c0f1SVlad Dogaru * Even if no event is enabled, we need to wake the thread to 4624193c0f1SVlad Dogaru * clear the interrupt state by reading SX9500_REG_IRQ_SRC. It 4634193c0f1SVlad Dogaru * is not possible to do that here because regmap_read takes a 4644193c0f1SVlad Dogaru * mutex. 4654193c0f1SVlad Dogaru */ 4664193c0f1SVlad Dogaru return IRQ_WAKE_THREAD; 4674193c0f1SVlad Dogaru } 4684193c0f1SVlad Dogaru 46959bd0427SVlad Dogaru static void sx9500_push_events(struct iio_dev *indio_dev) 4704193c0f1SVlad Dogaru { 4714193c0f1SVlad Dogaru int ret; 4724193c0f1SVlad Dogaru unsigned int val, chan; 47359bd0427SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 4744193c0f1SVlad Dogaru 4754193c0f1SVlad Dogaru ret = regmap_read(data->regmap, SX9500_REG_STAT, &val); 4764193c0f1SVlad Dogaru if (ret < 0) { 4774193c0f1SVlad Dogaru dev_err(&data->client->dev, "i2c transfer error in irq\n"); 47859bd0427SVlad Dogaru return; 4794193c0f1SVlad Dogaru } 4804193c0f1SVlad Dogaru 4814193c0f1SVlad Dogaru val >>= SX9500_PROXSTAT_SHIFT; 4824193c0f1SVlad Dogaru for (chan = 0; chan < SX9500_NUM_CHANNELS; chan++) { 4834193c0f1SVlad Dogaru int dir; 4844193c0f1SVlad Dogaru u64 ev; 4854193c0f1SVlad Dogaru bool new_prox = val & BIT(chan); 4864193c0f1SVlad Dogaru 4874193c0f1SVlad Dogaru if (!data->event_enabled[chan]) 4884193c0f1SVlad Dogaru continue; 4894193c0f1SVlad Dogaru if (new_prox == data->prox_stat[chan]) 4904193c0f1SVlad Dogaru /* No change on this channel. */ 4914193c0f1SVlad Dogaru continue; 4924193c0f1SVlad Dogaru 49359bd0427SVlad Dogaru dir = new_prox ? IIO_EV_DIR_FALLING : IIO_EV_DIR_RISING; 49459bd0427SVlad Dogaru ev = IIO_UNMOD_EVENT_CODE(IIO_PROXIMITY, chan, 49559bd0427SVlad Dogaru IIO_EV_TYPE_THRESH, dir); 496bc2b7dabSGregor Boirie iio_push_event(indio_dev, ev, iio_get_time_ns(indio_dev)); 4974193c0f1SVlad Dogaru data->prox_stat[chan] = new_prox; 4984193c0f1SVlad Dogaru } 49959bd0427SVlad Dogaru } 50059bd0427SVlad Dogaru 50159bd0427SVlad Dogaru static irqreturn_t sx9500_irq_thread_handler(int irq, void *private) 50259bd0427SVlad Dogaru { 50359bd0427SVlad Dogaru struct iio_dev *indio_dev = private; 50459bd0427SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 50559bd0427SVlad Dogaru int ret; 50659bd0427SVlad Dogaru unsigned int val; 50759bd0427SVlad Dogaru 50859bd0427SVlad Dogaru mutex_lock(&data->mutex); 50959bd0427SVlad Dogaru 51059bd0427SVlad Dogaru ret = regmap_read(data->regmap, SX9500_REG_IRQ_SRC, &val); 51159bd0427SVlad Dogaru if (ret < 0) { 51259bd0427SVlad Dogaru dev_err(&data->client->dev, "i2c transfer error in irq\n"); 51359bd0427SVlad Dogaru goto out; 51459bd0427SVlad Dogaru } 51559bd0427SVlad Dogaru 51659bd0427SVlad Dogaru if (val & (SX9500_CLOSE_IRQ | SX9500_FAR_IRQ)) 51759bd0427SVlad Dogaru sx9500_push_events(indio_dev); 51859bd0427SVlad Dogaru 51959bd0427SVlad Dogaru if (val & SX9500_CONVDONE_IRQ) 5208c11e161SDaniel Wagner complete(&data->completion); 5214193c0f1SVlad Dogaru 5224193c0f1SVlad Dogaru out: 5234193c0f1SVlad Dogaru mutex_unlock(&data->mutex); 5244193c0f1SVlad Dogaru 5254193c0f1SVlad Dogaru return IRQ_HANDLED; 5264193c0f1SVlad Dogaru } 5274193c0f1SVlad Dogaru 5284193c0f1SVlad Dogaru static int sx9500_read_event_config(struct iio_dev *indio_dev, 5294193c0f1SVlad Dogaru const struct iio_chan_spec *chan, 5304193c0f1SVlad Dogaru enum iio_event_type type, 5314193c0f1SVlad Dogaru enum iio_event_direction dir) 5324193c0f1SVlad Dogaru { 5334193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 5344193c0f1SVlad Dogaru 5354193c0f1SVlad Dogaru if (chan->type != IIO_PROXIMITY || type != IIO_EV_TYPE_THRESH || 5364193c0f1SVlad Dogaru dir != IIO_EV_DIR_EITHER) 5374193c0f1SVlad Dogaru return -EINVAL; 5384193c0f1SVlad Dogaru 5394193c0f1SVlad Dogaru return data->event_enabled[chan->channel]; 5404193c0f1SVlad Dogaru } 5414193c0f1SVlad Dogaru 5424193c0f1SVlad Dogaru static int sx9500_write_event_config(struct iio_dev *indio_dev, 5434193c0f1SVlad Dogaru const struct iio_chan_spec *chan, 5444193c0f1SVlad Dogaru enum iio_event_type type, 5454193c0f1SVlad Dogaru enum iio_event_direction dir, 5464193c0f1SVlad Dogaru int state) 5474193c0f1SVlad Dogaru { 5484193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 54959bd0427SVlad Dogaru int ret; 5504193c0f1SVlad Dogaru 5514193c0f1SVlad Dogaru if (chan->type != IIO_PROXIMITY || type != IIO_EV_TYPE_THRESH || 5524193c0f1SVlad Dogaru dir != IIO_EV_DIR_EITHER) 5534193c0f1SVlad Dogaru return -EINVAL; 5544193c0f1SVlad Dogaru 5554193c0f1SVlad Dogaru mutex_lock(&data->mutex); 5564193c0f1SVlad Dogaru 55759bd0427SVlad Dogaru if (state == 1) { 55859bd0427SVlad Dogaru ret = sx9500_inc_chan_users(data, chan->channel); 55959bd0427SVlad Dogaru if (ret < 0) 56059bd0427SVlad Dogaru goto out_unlock; 56159bd0427SVlad Dogaru ret = sx9500_inc_close_far_users(data); 56259bd0427SVlad Dogaru if (ret < 0) 56359bd0427SVlad Dogaru goto out_undo_chan; 56459bd0427SVlad Dogaru } else { 56559bd0427SVlad Dogaru ret = sx9500_dec_chan_users(data, chan->channel); 56659bd0427SVlad Dogaru if (ret < 0) 56759bd0427SVlad Dogaru goto out_unlock; 56859bd0427SVlad Dogaru ret = sx9500_dec_close_far_users(data); 56959bd0427SVlad Dogaru if (ret < 0) 57059bd0427SVlad Dogaru goto out_undo_chan; 5714193c0f1SVlad Dogaru } 5724193c0f1SVlad Dogaru 57359bd0427SVlad Dogaru data->event_enabled[chan->channel] = state; 57459bd0427SVlad Dogaru goto out_unlock; 57559bd0427SVlad Dogaru 57659bd0427SVlad Dogaru out_undo_chan: 57759bd0427SVlad Dogaru if (state == 1) 57859bd0427SVlad Dogaru sx9500_dec_chan_users(data, chan->channel); 5794193c0f1SVlad Dogaru else 58059bd0427SVlad Dogaru sx9500_inc_chan_users(data, chan->channel); 58159bd0427SVlad Dogaru out_unlock: 5824193c0f1SVlad Dogaru mutex_unlock(&data->mutex); 5834193c0f1SVlad Dogaru return ret; 5844193c0f1SVlad Dogaru } 5854193c0f1SVlad Dogaru 5864193c0f1SVlad Dogaru static int sx9500_update_scan_mode(struct iio_dev *indio_dev, 5874193c0f1SVlad Dogaru const unsigned long *scan_mask) 5884193c0f1SVlad Dogaru { 5894193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 5904193c0f1SVlad Dogaru 5914193c0f1SVlad Dogaru mutex_lock(&data->mutex); 5924193c0f1SVlad Dogaru kfree(data->buffer); 5934193c0f1SVlad Dogaru data->buffer = kzalloc(indio_dev->scan_bytes, GFP_KERNEL); 5944193c0f1SVlad Dogaru mutex_unlock(&data->mutex); 5954193c0f1SVlad Dogaru 5964193c0f1SVlad Dogaru if (data->buffer == NULL) 5974193c0f1SVlad Dogaru return -ENOMEM; 5984193c0f1SVlad Dogaru 5994193c0f1SVlad Dogaru return 0; 6004193c0f1SVlad Dogaru } 6014193c0f1SVlad Dogaru 6024193c0f1SVlad Dogaru static IIO_CONST_ATTR_SAMP_FREQ_AVAIL( 6034193c0f1SVlad Dogaru "2.500000 3.333333 5 6.666666 8.333333 11.111111 16.666666 33.333333"); 6044193c0f1SVlad Dogaru 6054193c0f1SVlad Dogaru static struct attribute *sx9500_attributes[] = { 6064193c0f1SVlad Dogaru &iio_const_attr_sampling_frequency_available.dev_attr.attr, 6074193c0f1SVlad Dogaru NULL, 6084193c0f1SVlad Dogaru }; 6094193c0f1SVlad Dogaru 6104193c0f1SVlad Dogaru static const struct attribute_group sx9500_attribute_group = { 6114193c0f1SVlad Dogaru .attrs = sx9500_attributes, 6124193c0f1SVlad Dogaru }; 6134193c0f1SVlad Dogaru 6144193c0f1SVlad Dogaru static const struct iio_info sx9500_info = { 6154193c0f1SVlad Dogaru .attrs = &sx9500_attribute_group, 6164193c0f1SVlad Dogaru .read_raw = &sx9500_read_raw, 6174193c0f1SVlad Dogaru .write_raw = &sx9500_write_raw, 6184193c0f1SVlad Dogaru .read_event_config = &sx9500_read_event_config, 6194193c0f1SVlad Dogaru .write_event_config = &sx9500_write_event_config, 6204193c0f1SVlad Dogaru .update_scan_mode = &sx9500_update_scan_mode, 6214193c0f1SVlad Dogaru }; 6224193c0f1SVlad Dogaru 6234193c0f1SVlad Dogaru static int sx9500_set_trigger_state(struct iio_trigger *trig, 6244193c0f1SVlad Dogaru bool state) 6254193c0f1SVlad Dogaru { 6264193c0f1SVlad Dogaru struct iio_dev *indio_dev = iio_trigger_get_drvdata(trig); 6274193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 6284193c0f1SVlad Dogaru int ret; 6294193c0f1SVlad Dogaru 6304193c0f1SVlad Dogaru mutex_lock(&data->mutex); 6314193c0f1SVlad Dogaru 63259bd0427SVlad Dogaru if (state) 63359bd0427SVlad Dogaru ret = sx9500_inc_data_rdy_users(data); 63459bd0427SVlad Dogaru else 63559bd0427SVlad Dogaru ret = sx9500_dec_data_rdy_users(data); 63659bd0427SVlad Dogaru if (ret < 0) 63759bd0427SVlad Dogaru goto out; 63859bd0427SVlad Dogaru 6394193c0f1SVlad Dogaru data->trigger_enabled = state; 6404193c0f1SVlad Dogaru 64159bd0427SVlad Dogaru out: 6424193c0f1SVlad Dogaru mutex_unlock(&data->mutex); 6434193c0f1SVlad Dogaru 6444193c0f1SVlad Dogaru return ret; 6454193c0f1SVlad Dogaru } 6464193c0f1SVlad Dogaru 6474193c0f1SVlad Dogaru static const struct iio_trigger_ops sx9500_trigger_ops = { 6484193c0f1SVlad Dogaru .set_trigger_state = sx9500_set_trigger_state, 6494193c0f1SVlad Dogaru }; 6504193c0f1SVlad Dogaru 6514193c0f1SVlad Dogaru static irqreturn_t sx9500_trigger_handler(int irq, void *private) 6524193c0f1SVlad Dogaru { 6534193c0f1SVlad Dogaru struct iio_poll_func *pf = private; 6544193c0f1SVlad Dogaru struct iio_dev *indio_dev = pf->indio_dev; 6554193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 6564193c0f1SVlad Dogaru int val, bit, ret, i = 0; 6574193c0f1SVlad Dogaru 6584193c0f1SVlad Dogaru mutex_lock(&data->mutex); 6594193c0f1SVlad Dogaru 66070dddeeeSOctavian Purdila for_each_set_bit(bit, indio_dev->active_scan_mask, 6614193c0f1SVlad Dogaru indio_dev->masklength) { 66259bd0427SVlad Dogaru ret = sx9500_read_prox_data(data, &indio_dev->channels[bit], 6634193c0f1SVlad Dogaru &val); 6644193c0f1SVlad Dogaru if (ret < 0) 6654193c0f1SVlad Dogaru goto out; 6664193c0f1SVlad Dogaru 6674193c0f1SVlad Dogaru data->buffer[i++] = val; 6684193c0f1SVlad Dogaru } 6694193c0f1SVlad Dogaru 6704193c0f1SVlad Dogaru iio_push_to_buffers_with_timestamp(indio_dev, data->buffer, 671bc2b7dabSGregor Boirie iio_get_time_ns(indio_dev)); 6724193c0f1SVlad Dogaru 6734193c0f1SVlad Dogaru out: 6744193c0f1SVlad Dogaru mutex_unlock(&data->mutex); 6754193c0f1SVlad Dogaru 6764193c0f1SVlad Dogaru iio_trigger_notify_done(indio_dev->trig); 6774193c0f1SVlad Dogaru 6784193c0f1SVlad Dogaru return IRQ_HANDLED; 6794193c0f1SVlad Dogaru } 6804193c0f1SVlad Dogaru 68159bd0427SVlad Dogaru static int sx9500_buffer_preenable(struct iio_dev *indio_dev) 68259bd0427SVlad Dogaru { 68359bd0427SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 684897993feSGeert Uytterhoeven int ret = 0, i; 68559bd0427SVlad Dogaru 68659bd0427SVlad Dogaru mutex_lock(&data->mutex); 68759bd0427SVlad Dogaru 68859bd0427SVlad Dogaru for (i = 0; i < SX9500_NUM_CHANNELS; i++) 68959bd0427SVlad Dogaru if (test_bit(i, indio_dev->active_scan_mask)) { 69059bd0427SVlad Dogaru ret = sx9500_inc_chan_users(data, i); 69159bd0427SVlad Dogaru if (ret) 69259bd0427SVlad Dogaru break; 69359bd0427SVlad Dogaru } 69459bd0427SVlad Dogaru 69559bd0427SVlad Dogaru if (ret) 69659bd0427SVlad Dogaru for (i = i - 1; i >= 0; i--) 69759bd0427SVlad Dogaru if (test_bit(i, indio_dev->active_scan_mask)) 69859bd0427SVlad Dogaru sx9500_dec_chan_users(data, i); 69959bd0427SVlad Dogaru 70059bd0427SVlad Dogaru mutex_unlock(&data->mutex); 70159bd0427SVlad Dogaru 70259bd0427SVlad Dogaru return ret; 70359bd0427SVlad Dogaru } 70459bd0427SVlad Dogaru 70559bd0427SVlad Dogaru static int sx9500_buffer_predisable(struct iio_dev *indio_dev) 70659bd0427SVlad Dogaru { 70759bd0427SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 708897993feSGeert Uytterhoeven int ret = 0, i; 70959bd0427SVlad Dogaru 71059bd0427SVlad Dogaru iio_triggered_buffer_predisable(indio_dev); 71159bd0427SVlad Dogaru 71259bd0427SVlad Dogaru mutex_lock(&data->mutex); 71359bd0427SVlad Dogaru 71459bd0427SVlad Dogaru for (i = 0; i < SX9500_NUM_CHANNELS; i++) 71559bd0427SVlad Dogaru if (test_bit(i, indio_dev->active_scan_mask)) { 71659bd0427SVlad Dogaru ret = sx9500_dec_chan_users(data, i); 71759bd0427SVlad Dogaru if (ret) 71859bd0427SVlad Dogaru break; 71959bd0427SVlad Dogaru } 72059bd0427SVlad Dogaru 72159bd0427SVlad Dogaru if (ret) 72259bd0427SVlad Dogaru for (i = i - 1; i >= 0; i--) 72359bd0427SVlad Dogaru if (test_bit(i, indio_dev->active_scan_mask)) 72459bd0427SVlad Dogaru sx9500_inc_chan_users(data, i); 72559bd0427SVlad Dogaru 72659bd0427SVlad Dogaru mutex_unlock(&data->mutex); 72759bd0427SVlad Dogaru 72859bd0427SVlad Dogaru return ret; 72959bd0427SVlad Dogaru } 73059bd0427SVlad Dogaru 73159bd0427SVlad Dogaru static const struct iio_buffer_setup_ops sx9500_buffer_setup_ops = { 73259bd0427SVlad Dogaru .preenable = sx9500_buffer_preenable, 73359bd0427SVlad Dogaru .postenable = iio_triggered_buffer_postenable, 73459bd0427SVlad Dogaru .predisable = sx9500_buffer_predisable, 73559bd0427SVlad Dogaru }; 73659bd0427SVlad Dogaru 7374193c0f1SVlad Dogaru struct sx9500_reg_default { 7384193c0f1SVlad Dogaru u8 reg; 7394193c0f1SVlad Dogaru u8 def; 7404193c0f1SVlad Dogaru }; 7414193c0f1SVlad Dogaru 7424193c0f1SVlad Dogaru static const struct sx9500_reg_default sx9500_default_regs[] = { 7434193c0f1SVlad Dogaru { 7444193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL1, 7454193c0f1SVlad Dogaru /* Shield enabled, small range. */ 7464193c0f1SVlad Dogaru .def = 0x43, 7474193c0f1SVlad Dogaru }, 7484193c0f1SVlad Dogaru { 7494193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL2, 7504193c0f1SVlad Dogaru /* x8 gain, 167kHz frequency, finest resolution. */ 7514193c0f1SVlad Dogaru .def = 0x77, 7524193c0f1SVlad Dogaru }, 7534193c0f1SVlad Dogaru { 7544193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL3, 7554193c0f1SVlad Dogaru /* Doze enabled, 2x scan period doze, no raw filter. */ 7564193c0f1SVlad Dogaru .def = 0x40, 7574193c0f1SVlad Dogaru }, 7584193c0f1SVlad Dogaru { 7594193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL4, 7604193c0f1SVlad Dogaru /* Average threshold. */ 7614193c0f1SVlad Dogaru .def = 0x30, 7624193c0f1SVlad Dogaru }, 7634193c0f1SVlad Dogaru { 7644193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL5, 7654193c0f1SVlad Dogaru /* 7664193c0f1SVlad Dogaru * Debouncer off, lowest average negative filter, 7674193c0f1SVlad Dogaru * highest average postive filter. 7684193c0f1SVlad Dogaru */ 7694193c0f1SVlad Dogaru .def = 0x0f, 7704193c0f1SVlad Dogaru }, 7714193c0f1SVlad Dogaru { 7724193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL6, 7734193c0f1SVlad Dogaru /* Proximity detection threshold: 280 */ 7744193c0f1SVlad Dogaru .def = 0x0e, 7754193c0f1SVlad Dogaru }, 7764193c0f1SVlad Dogaru { 7774193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL7, 7784193c0f1SVlad Dogaru /* 7794193c0f1SVlad Dogaru * No automatic compensation, compensate each pin 7804193c0f1SVlad Dogaru * independently, proximity hysteresis: 32, close 7814193c0f1SVlad Dogaru * debouncer off, far debouncer off. 7824193c0f1SVlad Dogaru */ 7834193c0f1SVlad Dogaru .def = 0x00, 7844193c0f1SVlad Dogaru }, 7854193c0f1SVlad Dogaru { 7864193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL8, 7874193c0f1SVlad Dogaru /* No stuck timeout, no periodic compensation. */ 7884193c0f1SVlad Dogaru .def = 0x00, 7894193c0f1SVlad Dogaru }, 7904193c0f1SVlad Dogaru { 7914193c0f1SVlad Dogaru .reg = SX9500_REG_PROX_CTRL0, 79259bd0427SVlad Dogaru /* Scan period: 30ms, all sensors disabled. */ 79359bd0427SVlad Dogaru .def = 0x00, 7944193c0f1SVlad Dogaru }, 7954193c0f1SVlad Dogaru }; 7964193c0f1SVlad Dogaru 79759bd0427SVlad Dogaru /* Activate all channels and perform an initial compensation. */ 79859bd0427SVlad Dogaru static int sx9500_init_compensation(struct iio_dev *indio_dev) 79959bd0427SVlad Dogaru { 80059bd0427SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 80159bd0427SVlad Dogaru int i, ret; 80259bd0427SVlad Dogaru unsigned int val; 80359bd0427SVlad Dogaru 80459bd0427SVlad Dogaru ret = regmap_update_bits(data->regmap, SX9500_REG_PROX_CTRL0, 80568958bd5SVlad Dogaru SX9500_CHAN_MASK, SX9500_CHAN_MASK); 80659bd0427SVlad Dogaru if (ret < 0) 80759bd0427SVlad Dogaru return ret; 80859bd0427SVlad Dogaru 80959bd0427SVlad Dogaru for (i = 10; i >= 0; i--) { 81059bd0427SVlad Dogaru usleep_range(10000, 20000); 81159bd0427SVlad Dogaru ret = regmap_read(data->regmap, SX9500_REG_STAT, &val); 81259bd0427SVlad Dogaru if (ret < 0) 81359bd0427SVlad Dogaru goto out; 81459bd0427SVlad Dogaru if (!(val & SX9500_COMPSTAT_MASK)) 81559bd0427SVlad Dogaru break; 81659bd0427SVlad Dogaru } 81759bd0427SVlad Dogaru 81859bd0427SVlad Dogaru if (i < 0) { 81959bd0427SVlad Dogaru dev_err(&data->client->dev, "initial compensation timed out"); 82059bd0427SVlad Dogaru ret = -ETIMEDOUT; 82159bd0427SVlad Dogaru } 82259bd0427SVlad Dogaru 82359bd0427SVlad Dogaru out: 82459bd0427SVlad Dogaru regmap_update_bits(data->regmap, SX9500_REG_PROX_CTRL0, 82568958bd5SVlad Dogaru SX9500_CHAN_MASK, 0); 82659bd0427SVlad Dogaru return ret; 82759bd0427SVlad Dogaru } 82859bd0427SVlad Dogaru 8294193c0f1SVlad Dogaru static int sx9500_init_device(struct iio_dev *indio_dev) 8304193c0f1SVlad Dogaru { 8314193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 8324193c0f1SVlad Dogaru int ret, i; 8334193c0f1SVlad Dogaru unsigned int val; 8344193c0f1SVlad Dogaru 83545fd5f8eSVlad Dogaru if (data->gpiod_rst) { 83645fd5f8eSVlad Dogaru gpiod_set_value_cansleep(data->gpiod_rst, 0); 83745fd5f8eSVlad Dogaru usleep_range(1000, 2000); 83845fd5f8eSVlad Dogaru gpiod_set_value_cansleep(data->gpiod_rst, 1); 83945fd5f8eSVlad Dogaru usleep_range(1000, 2000); 84045fd5f8eSVlad Dogaru } 84145fd5f8eSVlad Dogaru 8424193c0f1SVlad Dogaru ret = regmap_write(data->regmap, SX9500_REG_IRQ_MSK, 0); 8434193c0f1SVlad Dogaru if (ret < 0) 8444193c0f1SVlad Dogaru return ret; 8454193c0f1SVlad Dogaru 8464193c0f1SVlad Dogaru ret = regmap_write(data->regmap, SX9500_REG_RESET, 8474193c0f1SVlad Dogaru SX9500_SOFT_RESET); 8484193c0f1SVlad Dogaru if (ret < 0) 8494193c0f1SVlad Dogaru return ret; 8504193c0f1SVlad Dogaru 8514193c0f1SVlad Dogaru ret = regmap_read(data->regmap, SX9500_REG_IRQ_SRC, &val); 8524193c0f1SVlad Dogaru if (ret < 0) 8534193c0f1SVlad Dogaru return ret; 8544193c0f1SVlad Dogaru 8554193c0f1SVlad Dogaru for (i = 0; i < ARRAY_SIZE(sx9500_default_regs); i++) { 8564193c0f1SVlad Dogaru ret = regmap_write(data->regmap, 8574193c0f1SVlad Dogaru sx9500_default_regs[i].reg, 8584193c0f1SVlad Dogaru sx9500_default_regs[i].def); 8594193c0f1SVlad Dogaru if (ret < 0) 8604193c0f1SVlad Dogaru return ret; 8614193c0f1SVlad Dogaru } 8624193c0f1SVlad Dogaru 8631a30295aSJonathan Cameron return sx9500_init_compensation(indio_dev); 8644193c0f1SVlad Dogaru } 8654193c0f1SVlad Dogaru 866*b61d8e63SAndy Shevchenko static const struct acpi_gpio_params reset_gpios = { 0, 0, false }; 867*b61d8e63SAndy Shevchenko static const struct acpi_gpio_params interrupt_gpios = { 2, 0, false }; 868*b61d8e63SAndy Shevchenko 869*b61d8e63SAndy Shevchenko static const struct acpi_gpio_mapping acpi_sx9500_gpios[] = { 870*b61d8e63SAndy Shevchenko { "reset-gpios", &reset_gpios, 1 }, 871*b61d8e63SAndy Shevchenko /* 872*b61d8e63SAndy Shevchenko * Some platforms have a bug in ACPI GPIO description making IRQ 873*b61d8e63SAndy Shevchenko * GPIO to be output only. Ask the GPIO core to ignore this limit. 874*b61d8e63SAndy Shevchenko */ 875*b61d8e63SAndy Shevchenko { "interrupt-gpios", &interrupt_gpios, 1, ACPI_GPIO_QUIRK_NO_IO_RESTRICTION }, 876*b61d8e63SAndy Shevchenko { }, 877*b61d8e63SAndy Shevchenko }; 878*b61d8e63SAndy Shevchenko 879821ace29SVlad Dogaru static void sx9500_gpio_probe(struct i2c_client *client, 8804193c0f1SVlad Dogaru struct sx9500_data *data) 8814193c0f1SVlad Dogaru { 882e53111adSAndy Shevchenko struct gpio_desc *gpiod_int; 8834193c0f1SVlad Dogaru struct device *dev; 884*b61d8e63SAndy Shevchenko int ret; 8854193c0f1SVlad Dogaru 8864193c0f1SVlad Dogaru if (!client) 887821ace29SVlad Dogaru return; 8884193c0f1SVlad Dogaru 8894193c0f1SVlad Dogaru dev = &client->dev; 8904193c0f1SVlad Dogaru 891*b61d8e63SAndy Shevchenko ret = devm_acpi_dev_add_driver_gpios(dev, acpi_sx9500_gpios); 892*b61d8e63SAndy Shevchenko if (ret) 893*b61d8e63SAndy Shevchenko dev_dbg(dev, "Unable to add GPIO mapping table\n"); 894*b61d8e63SAndy Shevchenko 895e53111adSAndy Shevchenko if (client->irq <= 0) { 896*b61d8e63SAndy Shevchenko gpiod_int = devm_gpiod_get(dev, "interrupt", GPIOD_IN); 897e53111adSAndy Shevchenko if (IS_ERR(gpiod_int)) 898e53111adSAndy Shevchenko dev_err(dev, "gpio get irq failed\n"); 899e53111adSAndy Shevchenko else 900e53111adSAndy Shevchenko client->irq = gpiod_to_irq(gpiod_int); 901e53111adSAndy Shevchenko } 902e53111adSAndy Shevchenko 903*b61d8e63SAndy Shevchenko data->gpiod_rst = devm_gpiod_get(dev, "reset", GPIOD_OUT_HIGH); 90445fd5f8eSVlad Dogaru if (IS_ERR(data->gpiod_rst)) { 90545fd5f8eSVlad Dogaru dev_warn(dev, "gpio get reset pin failed\n"); 90645fd5f8eSVlad Dogaru data->gpiod_rst = NULL; 90745fd5f8eSVlad Dogaru } 9084193c0f1SVlad Dogaru } 9094193c0f1SVlad Dogaru 9104193c0f1SVlad Dogaru static int sx9500_probe(struct i2c_client *client, 9114193c0f1SVlad Dogaru const struct i2c_device_id *id) 9124193c0f1SVlad Dogaru { 9134193c0f1SVlad Dogaru int ret; 9144193c0f1SVlad Dogaru struct iio_dev *indio_dev; 9154193c0f1SVlad Dogaru struct sx9500_data *data; 9164193c0f1SVlad Dogaru 9174193c0f1SVlad Dogaru indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data)); 9184193c0f1SVlad Dogaru if (indio_dev == NULL) 9194193c0f1SVlad Dogaru return -ENOMEM; 9204193c0f1SVlad Dogaru 9214193c0f1SVlad Dogaru data = iio_priv(indio_dev); 9224193c0f1SVlad Dogaru data->client = client; 9234193c0f1SVlad Dogaru mutex_init(&data->mutex); 92459bd0427SVlad Dogaru init_completion(&data->completion); 9254193c0f1SVlad Dogaru data->trigger_enabled = false; 9264193c0f1SVlad Dogaru 9274193c0f1SVlad Dogaru data->regmap = devm_regmap_init_i2c(client, &sx9500_regmap_config); 9284193c0f1SVlad Dogaru if (IS_ERR(data->regmap)) 9294193c0f1SVlad Dogaru return PTR_ERR(data->regmap); 9304193c0f1SVlad Dogaru 9314193c0f1SVlad Dogaru indio_dev->dev.parent = &client->dev; 9324193c0f1SVlad Dogaru indio_dev->name = SX9500_DRIVER_NAME; 9334193c0f1SVlad Dogaru indio_dev->channels = sx9500_channels; 9344193c0f1SVlad Dogaru indio_dev->num_channels = ARRAY_SIZE(sx9500_channels); 9354193c0f1SVlad Dogaru indio_dev->info = &sx9500_info; 9364193c0f1SVlad Dogaru indio_dev->modes = INDIO_DIRECT_MODE; 9374193c0f1SVlad Dogaru i2c_set_clientdata(client, indio_dev); 9384193c0f1SVlad Dogaru 939821ace29SVlad Dogaru sx9500_gpio_probe(client, data); 9404193c0f1SVlad Dogaru 94145fd5f8eSVlad Dogaru ret = sx9500_init_device(indio_dev); 94245fd5f8eSVlad Dogaru if (ret < 0) 94345fd5f8eSVlad Dogaru return ret; 94445fd5f8eSVlad Dogaru 94559bd0427SVlad Dogaru if (client->irq <= 0) 94659bd0427SVlad Dogaru dev_warn(&client->dev, "no valid irq found\n"); 94759bd0427SVlad Dogaru else { 9484193c0f1SVlad Dogaru ret = devm_request_threaded_irq(&client->dev, client->irq, 9494193c0f1SVlad Dogaru sx9500_irq_handler, sx9500_irq_thread_handler, 9504193c0f1SVlad Dogaru IRQF_TRIGGER_FALLING | IRQF_ONESHOT, 9514193c0f1SVlad Dogaru SX9500_IRQ_NAME, indio_dev); 9524193c0f1SVlad Dogaru if (ret < 0) 9534193c0f1SVlad Dogaru return ret; 9544193c0f1SVlad Dogaru 9554193c0f1SVlad Dogaru data->trig = devm_iio_trigger_alloc(&client->dev, 9564193c0f1SVlad Dogaru "%s-dev%d", indio_dev->name, indio_dev->id); 9574193c0f1SVlad Dogaru if (!data->trig) 9584193c0f1SVlad Dogaru return -ENOMEM; 9594193c0f1SVlad Dogaru 9604193c0f1SVlad Dogaru data->trig->dev.parent = &client->dev; 9614193c0f1SVlad Dogaru data->trig->ops = &sx9500_trigger_ops; 9624193c0f1SVlad Dogaru iio_trigger_set_drvdata(data->trig, indio_dev); 9634193c0f1SVlad Dogaru 9644193c0f1SVlad Dogaru ret = iio_trigger_register(data->trig); 9654193c0f1SVlad Dogaru if (ret) 9664193c0f1SVlad Dogaru return ret; 9674193c0f1SVlad Dogaru } 9684193c0f1SVlad Dogaru 9694193c0f1SVlad Dogaru ret = iio_triggered_buffer_setup(indio_dev, NULL, 97059bd0427SVlad Dogaru sx9500_trigger_handler, 97159bd0427SVlad Dogaru &sx9500_buffer_setup_ops); 9724193c0f1SVlad Dogaru if (ret < 0) 9734193c0f1SVlad Dogaru goto out_trigger_unregister; 9744193c0f1SVlad Dogaru 9754193c0f1SVlad Dogaru ret = iio_device_register(indio_dev); 9764193c0f1SVlad Dogaru if (ret < 0) 9774193c0f1SVlad Dogaru goto out_buffer_cleanup; 9784193c0f1SVlad Dogaru 9794193c0f1SVlad Dogaru return 0; 9804193c0f1SVlad Dogaru 9814193c0f1SVlad Dogaru out_buffer_cleanup: 9824193c0f1SVlad Dogaru iio_triggered_buffer_cleanup(indio_dev); 9834193c0f1SVlad Dogaru out_trigger_unregister: 9844193c0f1SVlad Dogaru if (client->irq > 0) 9854193c0f1SVlad Dogaru iio_trigger_unregister(data->trig); 9864193c0f1SVlad Dogaru 9874193c0f1SVlad Dogaru return ret; 9884193c0f1SVlad Dogaru } 9894193c0f1SVlad Dogaru 9904193c0f1SVlad Dogaru static int sx9500_remove(struct i2c_client *client) 9914193c0f1SVlad Dogaru { 9924193c0f1SVlad Dogaru struct iio_dev *indio_dev = i2c_get_clientdata(client); 9934193c0f1SVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 9944193c0f1SVlad Dogaru 9954193c0f1SVlad Dogaru iio_device_unregister(indio_dev); 9964193c0f1SVlad Dogaru iio_triggered_buffer_cleanup(indio_dev); 9974193c0f1SVlad Dogaru if (client->irq > 0) 9984193c0f1SVlad Dogaru iio_trigger_unregister(data->trig); 9994193c0f1SVlad Dogaru kfree(data->buffer); 10004193c0f1SVlad Dogaru 10014193c0f1SVlad Dogaru return 0; 10024193c0f1SVlad Dogaru } 10034193c0f1SVlad Dogaru 10047840ffeeSVlad Dogaru #ifdef CONFIG_PM_SLEEP 10057840ffeeSVlad Dogaru static int sx9500_suspend(struct device *dev) 10067840ffeeSVlad Dogaru { 10077840ffeeSVlad Dogaru struct iio_dev *indio_dev = i2c_get_clientdata(to_i2c_client(dev)); 10087840ffeeSVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 10097840ffeeSVlad Dogaru int ret; 10107840ffeeSVlad Dogaru 10117840ffeeSVlad Dogaru mutex_lock(&data->mutex); 10127840ffeeSVlad Dogaru ret = regmap_read(data->regmap, SX9500_REG_PROX_CTRL0, 10137840ffeeSVlad Dogaru &data->suspend_ctrl0); 10147840ffeeSVlad Dogaru if (ret < 0) 10157840ffeeSVlad Dogaru goto out; 10167840ffeeSVlad Dogaru 10177840ffeeSVlad Dogaru /* 10187840ffeeSVlad Dogaru * Scan period doesn't matter because when all the sensors are 10197840ffeeSVlad Dogaru * deactivated the device is in sleep mode. 10207840ffeeSVlad Dogaru */ 10217840ffeeSVlad Dogaru ret = regmap_write(data->regmap, SX9500_REG_PROX_CTRL0, 0); 10227840ffeeSVlad Dogaru 10237840ffeeSVlad Dogaru out: 10247840ffeeSVlad Dogaru mutex_unlock(&data->mutex); 10257840ffeeSVlad Dogaru return ret; 10267840ffeeSVlad Dogaru } 10277840ffeeSVlad Dogaru 10287840ffeeSVlad Dogaru static int sx9500_resume(struct device *dev) 10297840ffeeSVlad Dogaru { 10307840ffeeSVlad Dogaru struct iio_dev *indio_dev = i2c_get_clientdata(to_i2c_client(dev)); 10317840ffeeSVlad Dogaru struct sx9500_data *data = iio_priv(indio_dev); 10327840ffeeSVlad Dogaru int ret; 10337840ffeeSVlad Dogaru 10347840ffeeSVlad Dogaru mutex_lock(&data->mutex); 10357840ffeeSVlad Dogaru ret = regmap_write(data->regmap, SX9500_REG_PROX_CTRL0, 10367840ffeeSVlad Dogaru data->suspend_ctrl0); 10377840ffeeSVlad Dogaru mutex_unlock(&data->mutex); 10387840ffeeSVlad Dogaru 10397840ffeeSVlad Dogaru return ret; 10407840ffeeSVlad Dogaru } 10417840ffeeSVlad Dogaru #endif /* CONFIG_PM_SLEEP */ 10427840ffeeSVlad Dogaru 10437840ffeeSVlad Dogaru static const struct dev_pm_ops sx9500_pm_ops = { 10447840ffeeSVlad Dogaru SET_SYSTEM_SLEEP_PM_OPS(sx9500_suspend, sx9500_resume) 10457840ffeeSVlad Dogaru }; 10467840ffeeSVlad Dogaru 10474193c0f1SVlad Dogaru static const struct acpi_device_id sx9500_acpi_match[] = { 10484193c0f1SVlad Dogaru {"SSX9500", 0}, 1049b8793555SAndy Shevchenko {"SASX9500", 0}, 10504193c0f1SVlad Dogaru { }, 10514193c0f1SVlad Dogaru }; 10524193c0f1SVlad Dogaru MODULE_DEVICE_TABLE(acpi, sx9500_acpi_match); 10534193c0f1SVlad Dogaru 1054a5c8b11aSChristoph Fritz static const struct of_device_id sx9500_of_match[] = { 1055a5c8b11aSChristoph Fritz { .compatible = "semtech,sx9500", }, 1056a5c8b11aSChristoph Fritz { } 1057a5c8b11aSChristoph Fritz }; 1058a5c8b11aSChristoph Fritz MODULE_DEVICE_TABLE(of, sx9500_of_match); 1059a5c8b11aSChristoph Fritz 10604193c0f1SVlad Dogaru static const struct i2c_device_id sx9500_id[] = { 10614193c0f1SVlad Dogaru {"sx9500", 0}, 1062a40c0ac1SVlad Dogaru { }, 10634193c0f1SVlad Dogaru }; 10644193c0f1SVlad Dogaru MODULE_DEVICE_TABLE(i2c, sx9500_id); 10654193c0f1SVlad Dogaru 10664193c0f1SVlad Dogaru static struct i2c_driver sx9500_driver = { 10674193c0f1SVlad Dogaru .driver = { 10684193c0f1SVlad Dogaru .name = SX9500_DRIVER_NAME, 10694193c0f1SVlad Dogaru .acpi_match_table = ACPI_PTR(sx9500_acpi_match), 1070a5c8b11aSChristoph Fritz .of_match_table = of_match_ptr(sx9500_of_match), 10717840ffeeSVlad Dogaru .pm = &sx9500_pm_ops, 10724193c0f1SVlad Dogaru }, 10734193c0f1SVlad Dogaru .probe = sx9500_probe, 10744193c0f1SVlad Dogaru .remove = sx9500_remove, 10754193c0f1SVlad Dogaru .id_table = sx9500_id, 10764193c0f1SVlad Dogaru }; 10774193c0f1SVlad Dogaru module_i2c_driver(sx9500_driver); 10784193c0f1SVlad Dogaru 10794193c0f1SVlad Dogaru MODULE_AUTHOR("Vlad Dogaru <vlad.dogaru@intel.com>"); 10804193c0f1SVlad Dogaru MODULE_DESCRIPTION("Driver for Semtech SX9500 proximity sensor"); 10814193c0f1SVlad Dogaru MODULE_LICENSE("GPL v2"); 1082