123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300 |
- /*
- * ROHM 1780GLI Ambient Light Sensor Driver
- *
- * Copyright (C) 2016 Linaro Ltd.
- * Author: Linus Walleij <linus.walleij@linaro.org>
- * Loosely based on the previous BH1780 ALS misc driver
- * Copyright (C) 2010 Texas Instruments
- * Author: Hemanth V <hemanthv@ti.com>
- */
- #include <linux/i2c.h>
- #include <linux/slab.h>
- #include <linux/platform_device.h>
- #include <linux/delay.h>
- #include <linux/module.h>
- #include <linux/of.h>
- #include <linux/pm_runtime.h>
- #include <linux/iio/iio.h>
- #include <linux/iio/sysfs.h>
- #include <linux/bitops.h>
- #define BH1780_CMD_BIT BIT(7)
- #define BH1780_REG_CONTROL 0x00
- #define BH1780_REG_PARTID 0x0A
- #define BH1780_REG_MANFID 0x0B
- #define BH1780_REG_DLOW 0x0C
- #define BH1780_REG_DHIGH 0x0D
- #define BH1780_REVMASK GENMASK(3,0)
- #define BH1780_POWMASK GENMASK(1,0)
- #define BH1780_POFF (0x0)
- #define BH1780_PON (0x3)
- /* power on settling time in ms */
- #define BH1780_PON_DELAY 2
- /* max time before value available in ms */
- #define BH1780_INTERVAL 250
- struct bh1780_data {
- struct i2c_client *client;
- };
- static int bh1780_write(struct bh1780_data *bh1780, u8 reg, u8 val)
- {
- int ret = i2c_smbus_write_byte_data(bh1780->client,
- BH1780_CMD_BIT | reg,
- val);
- if (ret < 0)
- dev_err(&bh1780->client->dev,
- "i2c_smbus_write_byte_data failed error "
- "%d, register %01x\n",
- ret, reg);
- return ret;
- }
- static int bh1780_read(struct bh1780_data *bh1780, u8 reg)
- {
- int ret = i2c_smbus_read_byte_data(bh1780->client,
- BH1780_CMD_BIT | reg);
- if (ret < 0)
- dev_err(&bh1780->client->dev,
- "i2c_smbus_read_byte_data failed error "
- "%d, register %01x\n",
- ret, reg);
- return ret;
- }
- static int bh1780_read_word(struct bh1780_data *bh1780, u8 reg)
- {
- int ret = i2c_smbus_read_word_data(bh1780->client,
- BH1780_CMD_BIT | reg);
- if (ret < 0)
- dev_err(&bh1780->client->dev,
- "i2c_smbus_read_word_data failed error "
- "%d, register %01x\n",
- ret, reg);
- return ret;
- }
- static int bh1780_debugfs_reg_access(struct iio_dev *indio_dev,
- unsigned int reg, unsigned int writeval,
- unsigned int *readval)
- {
- struct bh1780_data *bh1780 = iio_priv(indio_dev);
- int ret;
- if (!readval)
- return bh1780_write(bh1780, (u8)reg, (u8)writeval);
- ret = bh1780_read(bh1780, (u8)reg);
- if (ret < 0)
- return ret;
- *readval = ret;
- return 0;
- }
- static int bh1780_read_raw(struct iio_dev *indio_dev,
- struct iio_chan_spec const *chan,
- int *val, int *val2, long mask)
- {
- struct bh1780_data *bh1780 = iio_priv(indio_dev);
- int value;
- switch (mask) {
- case IIO_CHAN_INFO_RAW:
- switch (chan->type) {
- case IIO_LIGHT:
- pm_runtime_get_sync(&bh1780->client->dev);
- value = bh1780_read_word(bh1780, BH1780_REG_DLOW);
- if (value < 0)
- return value;
- pm_runtime_mark_last_busy(&bh1780->client->dev);
- pm_runtime_put_autosuspend(&bh1780->client->dev);
- *val = value;
- return IIO_VAL_INT;
- default:
- return -EINVAL;
- }
- case IIO_CHAN_INFO_INT_TIME:
- *val = 0;
- *val2 = BH1780_INTERVAL * 1000;
- return IIO_VAL_INT_PLUS_MICRO;
- default:
- return -EINVAL;
- }
- }
- static const struct iio_info bh1780_info = {
- .driver_module = THIS_MODULE,
- .read_raw = bh1780_read_raw,
- .debugfs_reg_access = bh1780_debugfs_reg_access,
- };
- static const struct iio_chan_spec bh1780_channels[] = {
- {
- .type = IIO_LIGHT,
- .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) |
- BIT(IIO_CHAN_INFO_INT_TIME)
- }
- };
- static int bh1780_probe(struct i2c_client *client,
- const struct i2c_device_id *id)
- {
- int ret;
- struct bh1780_data *bh1780;
- struct i2c_adapter *adapter = to_i2c_adapter(client->dev.parent);
- struct iio_dev *indio_dev;
- if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE))
- return -EIO;
- indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*bh1780));
- if (!indio_dev)
- return -ENOMEM;
- bh1780 = iio_priv(indio_dev);
- bh1780->client = client;
- i2c_set_clientdata(client, indio_dev);
- /* Power up the device */
- ret = bh1780_write(bh1780, BH1780_REG_CONTROL, BH1780_PON);
- if (ret < 0)
- return ret;
- msleep(BH1780_PON_DELAY);
- pm_runtime_get_noresume(&client->dev);
- pm_runtime_set_active(&client->dev);
- pm_runtime_enable(&client->dev);
- ret = bh1780_read(bh1780, BH1780_REG_PARTID);
- if (ret < 0)
- goto out_disable_pm;
- dev_info(&client->dev,
- "Ambient Light Sensor, Rev : %lu\n",
- (ret & BH1780_REVMASK));
- /*
- * As the device takes 250 ms to even come up with a fresh
- * measurement after power-on, do not shut it down unnecessarily.
- * Set autosuspend to a five seconds.
- */
- pm_runtime_set_autosuspend_delay(&client->dev, 5000);
- pm_runtime_use_autosuspend(&client->dev);
- pm_runtime_put(&client->dev);
- indio_dev->dev.parent = &client->dev;
- indio_dev->info = &bh1780_info;
- indio_dev->name = "bh1780";
- indio_dev->channels = bh1780_channels;
- indio_dev->num_channels = ARRAY_SIZE(bh1780_channels);
- indio_dev->modes = INDIO_DIRECT_MODE;
- ret = iio_device_register(indio_dev);
- if (ret)
- goto out_disable_pm;
- return 0;
- out_disable_pm:
- pm_runtime_put_noidle(&client->dev);
- pm_runtime_disable(&client->dev);
- return ret;
- }
- static int bh1780_remove(struct i2c_client *client)
- {
- struct iio_dev *indio_dev = i2c_get_clientdata(client);
- struct bh1780_data *bh1780 = iio_priv(indio_dev);
- int ret;
- iio_device_unregister(indio_dev);
- pm_runtime_get_sync(&client->dev);
- pm_runtime_put_noidle(&client->dev);
- pm_runtime_disable(&client->dev);
- ret = bh1780_write(bh1780, BH1780_REG_CONTROL, BH1780_POFF);
- if (ret < 0) {
- dev_err(&client->dev, "failed to power off\n");
- return ret;
- }
- return 0;
- }
- #ifdef CONFIG_PM
- static int bh1780_runtime_suspend(struct device *dev)
- {
- struct i2c_client *client = to_i2c_client(dev);
- struct iio_dev *indio_dev = i2c_get_clientdata(client);
- struct bh1780_data *bh1780 = iio_priv(indio_dev);
- int ret;
- ret = bh1780_write(bh1780, BH1780_REG_CONTROL, BH1780_POFF);
- if (ret < 0) {
- dev_err(dev, "failed to runtime suspend\n");
- return ret;
- }
- return 0;
- }
- static int bh1780_runtime_resume(struct device *dev)
- {
- struct i2c_client *client = to_i2c_client(dev);
- struct iio_dev *indio_dev = i2c_get_clientdata(client);
- struct bh1780_data *bh1780 = iio_priv(indio_dev);
- int ret;
- ret = bh1780_write(bh1780, BH1780_REG_CONTROL, BH1780_PON);
- if (ret < 0) {
- dev_err(dev, "failed to runtime resume\n");
- return ret;
- }
- /* Wait for power on, then for a value to be available */
- msleep(BH1780_PON_DELAY + BH1780_INTERVAL);
- return 0;
- }
- #endif /* CONFIG_PM */
- static const struct dev_pm_ops bh1780_dev_pm_ops = {
- SET_SYSTEM_SLEEP_PM_OPS(pm_runtime_force_suspend,
- pm_runtime_force_resume)
- SET_RUNTIME_PM_OPS(bh1780_runtime_suspend,
- bh1780_runtime_resume, NULL)
- };
- static const struct i2c_device_id bh1780_id[] = {
- { "bh1780", 0 },
- { },
- };
- MODULE_DEVICE_TABLE(i2c, bh1780_id);
- #ifdef CONFIG_OF
- static const struct of_device_id of_bh1780_match[] = {
- { .compatible = "rohm,bh1780gli", },
- {},
- };
- MODULE_DEVICE_TABLE(of, of_bh1780_match);
- #endif
- static struct i2c_driver bh1780_driver = {
- .probe = bh1780_probe,
- .remove = bh1780_remove,
- .id_table = bh1780_id,
- .driver = {
- .name = "bh1780",
- .pm = &bh1780_dev_pm_ops,
- .of_match_table = of_match_ptr(of_bh1780_match),
- },
- };
- module_i2c_driver(bh1780_driver);
- MODULE_DESCRIPTION("ROHM BH1780GLI Ambient Light Sensor Driver");
- MODULE_LICENSE("GPL");
- MODULE_AUTHOR("Linus Walleij <linus.walleij@linaro.org>");
|