18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * Support for Vishay VCNL3020 proximity sensor on i2c bus. 48c2ecf20Sopenharmony_ci * Based on Vishay VCNL4000 driver code. 58c2ecf20Sopenharmony_ci * 68c2ecf20Sopenharmony_ci * TODO: interrupts. 78c2ecf20Sopenharmony_ci */ 88c2ecf20Sopenharmony_ci 98c2ecf20Sopenharmony_ci#include <linux/module.h> 108c2ecf20Sopenharmony_ci#include <linux/i2c.h> 118c2ecf20Sopenharmony_ci#include <linux/err.h> 128c2ecf20Sopenharmony_ci#include <linux/delay.h> 138c2ecf20Sopenharmony_ci#include <linux/regmap.h> 148c2ecf20Sopenharmony_ci 158c2ecf20Sopenharmony_ci#include <linux/iio/iio.h> 168c2ecf20Sopenharmony_ci#include <linux/iio/sysfs.h> 178c2ecf20Sopenharmony_ci 188c2ecf20Sopenharmony_ci#define VCNL3020_PROD_ID 0x21 198c2ecf20Sopenharmony_ci 208c2ecf20Sopenharmony_ci#define VCNL_COMMAND 0x80 /* Command register */ 218c2ecf20Sopenharmony_ci#define VCNL_PROD_REV 0x81 /* Product ID and Revision ID */ 228c2ecf20Sopenharmony_ci#define VCNL_PROXIMITY_RATE 0x82 /* Rate of Proximity Measurement */ 238c2ecf20Sopenharmony_ci#define VCNL_LED_CURRENT 0x83 /* IR LED current for proximity mode */ 248c2ecf20Sopenharmony_ci#define VCNL_PS_RESULT_HI 0x87 /* Proximity result register, MSB */ 258c2ecf20Sopenharmony_ci#define VCNL_PS_RESULT_LO 0x88 /* Proximity result register, LSB */ 268c2ecf20Sopenharmony_ci#define VCNL_PS_ICR 0x89 /* Interrupt Control Register */ 278c2ecf20Sopenharmony_ci#define VCNL_PS_LO_THR_HI 0x8a /* High byte of low threshold value */ 288c2ecf20Sopenharmony_ci#define VCNL_PS_LO_THR_LO 0x8b /* Low byte of low threshold value */ 298c2ecf20Sopenharmony_ci#define VCNL_PS_HI_THR_HI 0x8c /* High byte of high threshold value */ 308c2ecf20Sopenharmony_ci#define VCNL_PS_HI_THR_LO 0x8d /* Low byte of high threshold value */ 318c2ecf20Sopenharmony_ci#define VCNL_ISR 0x8e /* Interrupt Status Register */ 328c2ecf20Sopenharmony_ci#define VCNL_PS_MOD_ADJ 0x8f /* Proximity Modulator Timing Adjustment */ 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci/* Bit masks for COMMAND register */ 358c2ecf20Sopenharmony_ci#define VCNL_PS_RDY BIT(5) /* proximity data ready? */ 368c2ecf20Sopenharmony_ci#define VCNL_PS_OD BIT(3) /* start on-demand proximity 378c2ecf20Sopenharmony_ci * measurement 388c2ecf20Sopenharmony_ci */ 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_ci#define VCNL_ON_DEMAND_TIMEOUT_US 100000 418c2ecf20Sopenharmony_ci#define VCNL_POLL_US 20000 428c2ecf20Sopenharmony_ci 438c2ecf20Sopenharmony_ci/** 448c2ecf20Sopenharmony_ci * struct vcnl3020_data - vcnl3020 specific data. 458c2ecf20Sopenharmony_ci * @regmap: device register map. 468c2ecf20Sopenharmony_ci * @dev: vcnl3020 device. 478c2ecf20Sopenharmony_ci * @rev: revision id. 488c2ecf20Sopenharmony_ci * @lock: lock for protecting access to device hardware registers. 498c2ecf20Sopenharmony_ci */ 508c2ecf20Sopenharmony_cistruct vcnl3020_data { 518c2ecf20Sopenharmony_ci struct regmap *regmap; 528c2ecf20Sopenharmony_ci struct device *dev; 538c2ecf20Sopenharmony_ci u8 rev; 548c2ecf20Sopenharmony_ci struct mutex lock; 558c2ecf20Sopenharmony_ci}; 568c2ecf20Sopenharmony_ci 578c2ecf20Sopenharmony_ci/** 588c2ecf20Sopenharmony_ci * struct vcnl3020_property - vcnl3020 property. 598c2ecf20Sopenharmony_ci * @name: property name. 608c2ecf20Sopenharmony_ci * @reg: i2c register offset. 618c2ecf20Sopenharmony_ci * @conversion_func: conversion function. 628c2ecf20Sopenharmony_ci */ 638c2ecf20Sopenharmony_cistruct vcnl3020_property { 648c2ecf20Sopenharmony_ci const char *name; 658c2ecf20Sopenharmony_ci u32 reg; 668c2ecf20Sopenharmony_ci u32 (*conversion_func)(u32 *val); 678c2ecf20Sopenharmony_ci}; 688c2ecf20Sopenharmony_ci 698c2ecf20Sopenharmony_cistatic u32 microamp_to_reg(u32 *val) 708c2ecf20Sopenharmony_ci{ 718c2ecf20Sopenharmony_ci /* 728c2ecf20Sopenharmony_ci * An example of conversion from uA to reg val: 738c2ecf20Sopenharmony_ci * 200000 uA == 200 mA == 20 748c2ecf20Sopenharmony_ci */ 758c2ecf20Sopenharmony_ci return *val /= 10000; 768c2ecf20Sopenharmony_ci}; 778c2ecf20Sopenharmony_ci 788c2ecf20Sopenharmony_cistatic struct vcnl3020_property vcnl3020_led_current_property = { 798c2ecf20Sopenharmony_ci .name = "vishay,led-current-microamp", 808c2ecf20Sopenharmony_ci .reg = VCNL_LED_CURRENT, 818c2ecf20Sopenharmony_ci .conversion_func = microamp_to_reg, 828c2ecf20Sopenharmony_ci}; 838c2ecf20Sopenharmony_ci 848c2ecf20Sopenharmony_cistatic int vcnl3020_get_and_apply_property(struct vcnl3020_data *data, 858c2ecf20Sopenharmony_ci struct vcnl3020_property prop) 868c2ecf20Sopenharmony_ci{ 878c2ecf20Sopenharmony_ci int rc; 888c2ecf20Sopenharmony_ci u32 val; 898c2ecf20Sopenharmony_ci 908c2ecf20Sopenharmony_ci rc = device_property_read_u32(data->dev, prop.name, &val); 918c2ecf20Sopenharmony_ci if (rc) 928c2ecf20Sopenharmony_ci return 0; 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci if (prop.conversion_func) 958c2ecf20Sopenharmony_ci prop.conversion_func(&val); 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ci rc = regmap_write(data->regmap, prop.reg, val); 988c2ecf20Sopenharmony_ci if (rc) { 998c2ecf20Sopenharmony_ci dev_err(data->dev, "Error (%d) setting property (%s)\n", 1008c2ecf20Sopenharmony_ci rc, prop.name); 1018c2ecf20Sopenharmony_ci } 1028c2ecf20Sopenharmony_ci 1038c2ecf20Sopenharmony_ci return rc; 1048c2ecf20Sopenharmony_ci} 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_cistatic int vcnl3020_init(struct vcnl3020_data *data) 1078c2ecf20Sopenharmony_ci{ 1088c2ecf20Sopenharmony_ci int rc; 1098c2ecf20Sopenharmony_ci unsigned int reg; 1108c2ecf20Sopenharmony_ci 1118c2ecf20Sopenharmony_ci rc = regmap_read(data->regmap, VCNL_PROD_REV, ®); 1128c2ecf20Sopenharmony_ci if (rc) { 1138c2ecf20Sopenharmony_ci dev_err(data->dev, 1148c2ecf20Sopenharmony_ci "Error (%d) reading product revision\n", rc); 1158c2ecf20Sopenharmony_ci return rc; 1168c2ecf20Sopenharmony_ci } 1178c2ecf20Sopenharmony_ci 1188c2ecf20Sopenharmony_ci if (reg != VCNL3020_PROD_ID) { 1198c2ecf20Sopenharmony_ci dev_err(data->dev, 1208c2ecf20Sopenharmony_ci "Product id (%x) did not match vcnl3020 (%x)\n", reg, 1218c2ecf20Sopenharmony_ci VCNL3020_PROD_ID); 1228c2ecf20Sopenharmony_ci return -ENODEV; 1238c2ecf20Sopenharmony_ci } 1248c2ecf20Sopenharmony_ci 1258c2ecf20Sopenharmony_ci data->rev = reg; 1268c2ecf20Sopenharmony_ci mutex_init(&data->lock); 1278c2ecf20Sopenharmony_ci 1288c2ecf20Sopenharmony_ci return vcnl3020_get_and_apply_property(data, 1298c2ecf20Sopenharmony_ci vcnl3020_led_current_property); 1308c2ecf20Sopenharmony_ci}; 1318c2ecf20Sopenharmony_ci 1328c2ecf20Sopenharmony_cistatic int vcnl3020_measure_proximity(struct vcnl3020_data *data, int *val) 1338c2ecf20Sopenharmony_ci{ 1348c2ecf20Sopenharmony_ci int rc; 1358c2ecf20Sopenharmony_ci unsigned int reg; 1368c2ecf20Sopenharmony_ci __be16 res; 1378c2ecf20Sopenharmony_ci 1388c2ecf20Sopenharmony_ci mutex_lock(&data->lock); 1398c2ecf20Sopenharmony_ci 1408c2ecf20Sopenharmony_ci rc = regmap_write(data->regmap, VCNL_COMMAND, VCNL_PS_OD); 1418c2ecf20Sopenharmony_ci if (rc) 1428c2ecf20Sopenharmony_ci goto err_unlock; 1438c2ecf20Sopenharmony_ci 1448c2ecf20Sopenharmony_ci /* wait for data to become ready */ 1458c2ecf20Sopenharmony_ci rc = regmap_read_poll_timeout(data->regmap, VCNL_COMMAND, reg, 1468c2ecf20Sopenharmony_ci reg & VCNL_PS_RDY, VCNL_POLL_US, 1478c2ecf20Sopenharmony_ci VCNL_ON_DEMAND_TIMEOUT_US); 1488c2ecf20Sopenharmony_ci if (rc) { 1498c2ecf20Sopenharmony_ci dev_err(data->dev, 1508c2ecf20Sopenharmony_ci "Error (%d) reading vcnl3020 command register\n", rc); 1518c2ecf20Sopenharmony_ci goto err_unlock; 1528c2ecf20Sopenharmony_ci } 1538c2ecf20Sopenharmony_ci 1548c2ecf20Sopenharmony_ci /* high & low result bytes read */ 1558c2ecf20Sopenharmony_ci rc = regmap_bulk_read(data->regmap, VCNL_PS_RESULT_HI, &res, 1568c2ecf20Sopenharmony_ci sizeof(res)); 1578c2ecf20Sopenharmony_ci if (rc) 1588c2ecf20Sopenharmony_ci goto err_unlock; 1598c2ecf20Sopenharmony_ci 1608c2ecf20Sopenharmony_ci *val = be16_to_cpu(res); 1618c2ecf20Sopenharmony_ci 1628c2ecf20Sopenharmony_cierr_unlock: 1638c2ecf20Sopenharmony_ci mutex_unlock(&data->lock); 1648c2ecf20Sopenharmony_ci 1658c2ecf20Sopenharmony_ci return rc; 1668c2ecf20Sopenharmony_ci} 1678c2ecf20Sopenharmony_ci 1688c2ecf20Sopenharmony_cistatic const struct iio_chan_spec vcnl3020_channels[] = { 1698c2ecf20Sopenharmony_ci { 1708c2ecf20Sopenharmony_ci .type = IIO_PROXIMITY, 1718c2ecf20Sopenharmony_ci .info_mask_separate = BIT(IIO_CHAN_INFO_RAW), 1728c2ecf20Sopenharmony_ci }, 1738c2ecf20Sopenharmony_ci}; 1748c2ecf20Sopenharmony_ci 1758c2ecf20Sopenharmony_cistatic int vcnl3020_read_raw(struct iio_dev *indio_dev, 1768c2ecf20Sopenharmony_ci struct iio_chan_spec const *chan, int *val, 1778c2ecf20Sopenharmony_ci int *val2, long mask) 1788c2ecf20Sopenharmony_ci{ 1798c2ecf20Sopenharmony_ci int rc; 1808c2ecf20Sopenharmony_ci struct vcnl3020_data *data = iio_priv(indio_dev); 1818c2ecf20Sopenharmony_ci 1828c2ecf20Sopenharmony_ci switch (mask) { 1838c2ecf20Sopenharmony_ci case IIO_CHAN_INFO_RAW: 1848c2ecf20Sopenharmony_ci rc = vcnl3020_measure_proximity(data, val); 1858c2ecf20Sopenharmony_ci if (rc) 1868c2ecf20Sopenharmony_ci return rc; 1878c2ecf20Sopenharmony_ci return IIO_VAL_INT; 1888c2ecf20Sopenharmony_ci default: 1898c2ecf20Sopenharmony_ci return -EINVAL; 1908c2ecf20Sopenharmony_ci } 1918c2ecf20Sopenharmony_ci} 1928c2ecf20Sopenharmony_ci 1938c2ecf20Sopenharmony_cistatic const struct iio_info vcnl3020_info = { 1948c2ecf20Sopenharmony_ci .read_raw = vcnl3020_read_raw, 1958c2ecf20Sopenharmony_ci}; 1968c2ecf20Sopenharmony_ci 1978c2ecf20Sopenharmony_cistatic const struct regmap_config vcnl3020_regmap_config = { 1988c2ecf20Sopenharmony_ci .reg_bits = 8, 1998c2ecf20Sopenharmony_ci .val_bits = 8, 2008c2ecf20Sopenharmony_ci .max_register = VCNL_PS_MOD_ADJ, 2018c2ecf20Sopenharmony_ci}; 2028c2ecf20Sopenharmony_ci 2038c2ecf20Sopenharmony_cistatic int vcnl3020_probe(struct i2c_client *client) 2048c2ecf20Sopenharmony_ci{ 2058c2ecf20Sopenharmony_ci struct vcnl3020_data *data; 2068c2ecf20Sopenharmony_ci struct iio_dev *indio_dev; 2078c2ecf20Sopenharmony_ci struct regmap *regmap; 2088c2ecf20Sopenharmony_ci int rc; 2098c2ecf20Sopenharmony_ci 2108c2ecf20Sopenharmony_ci regmap = devm_regmap_init_i2c(client, &vcnl3020_regmap_config); 2118c2ecf20Sopenharmony_ci if (IS_ERR(regmap)) { 2128c2ecf20Sopenharmony_ci dev_err(&client->dev, "regmap_init failed\n"); 2138c2ecf20Sopenharmony_ci return PTR_ERR(regmap); 2148c2ecf20Sopenharmony_ci } 2158c2ecf20Sopenharmony_ci 2168c2ecf20Sopenharmony_ci indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data)); 2178c2ecf20Sopenharmony_ci if (!indio_dev) 2188c2ecf20Sopenharmony_ci return -ENOMEM; 2198c2ecf20Sopenharmony_ci 2208c2ecf20Sopenharmony_ci data = iio_priv(indio_dev); 2218c2ecf20Sopenharmony_ci i2c_set_clientdata(client, indio_dev); 2228c2ecf20Sopenharmony_ci data->regmap = regmap; 2238c2ecf20Sopenharmony_ci data->dev = &client->dev; 2248c2ecf20Sopenharmony_ci 2258c2ecf20Sopenharmony_ci rc = vcnl3020_init(data); 2268c2ecf20Sopenharmony_ci if (rc) 2278c2ecf20Sopenharmony_ci return rc; 2288c2ecf20Sopenharmony_ci 2298c2ecf20Sopenharmony_ci indio_dev->info = &vcnl3020_info; 2308c2ecf20Sopenharmony_ci indio_dev->channels = vcnl3020_channels; 2318c2ecf20Sopenharmony_ci indio_dev->num_channels = ARRAY_SIZE(vcnl3020_channels); 2328c2ecf20Sopenharmony_ci indio_dev->name = "vcnl3020"; 2338c2ecf20Sopenharmony_ci indio_dev->modes = INDIO_DIRECT_MODE; 2348c2ecf20Sopenharmony_ci 2358c2ecf20Sopenharmony_ci return devm_iio_device_register(&client->dev, indio_dev); 2368c2ecf20Sopenharmony_ci} 2378c2ecf20Sopenharmony_ci 2388c2ecf20Sopenharmony_cistatic const struct of_device_id vcnl3020_of_match[] = { 2398c2ecf20Sopenharmony_ci { 2408c2ecf20Sopenharmony_ci .compatible = "vishay,vcnl3020", 2418c2ecf20Sopenharmony_ci }, 2428c2ecf20Sopenharmony_ci {} 2438c2ecf20Sopenharmony_ci}; 2448c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, vcnl3020_of_match); 2458c2ecf20Sopenharmony_ci 2468c2ecf20Sopenharmony_cistatic struct i2c_driver vcnl3020_driver = { 2478c2ecf20Sopenharmony_ci .driver = { 2488c2ecf20Sopenharmony_ci .name = "vcnl3020", 2498c2ecf20Sopenharmony_ci .of_match_table = vcnl3020_of_match, 2508c2ecf20Sopenharmony_ci }, 2518c2ecf20Sopenharmony_ci .probe_new = vcnl3020_probe, 2528c2ecf20Sopenharmony_ci}; 2538c2ecf20Sopenharmony_cimodule_i2c_driver(vcnl3020_driver); 2548c2ecf20Sopenharmony_ci 2558c2ecf20Sopenharmony_ciMODULE_AUTHOR("Ivan Mikhaylov <i.mikhaylov@yadro.com>"); 2568c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Vishay VCNL3020 proximity sensor driver"); 2578c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 258