linux/drivers/hwmon/aht10.c
<<
>>
Prefs
   1// SPDX-License-Identifier: GPL-2.0-only
   2
   3/*
   4 * aht10.c - Linux hwmon driver for AHT10 Temperature and Humidity sensor
   5 * Copyright (C) 2020 Johannes Cornelis Draaijer
   6 */
   7
   8#include <linux/delay.h>
   9#include <linux/hwmon.h>
  10#include <linux/i2c.h>
  11#include <linux/ktime.h>
  12#include <linux/module.h>
  13
  14#define AHT10_MEAS_SIZE         6
  15
  16/*
  17 * Poll intervals (in milliseconds)
  18 */
  19#define AHT10_DEFAULT_MIN_POLL_INTERVAL 2000
  20#define AHT10_MIN_POLL_INTERVAL         2000
  21
  22/*
  23 * I2C command delays (in microseconds)
  24 */
  25#define AHT10_MEAS_DELAY        80000
  26#define AHT10_CMD_DELAY         350000
  27#define AHT10_DELAY_EXTRA       100000
  28
  29/*
  30 * Command bytes
  31 */
  32#define AHT10_CMD_INIT  0b11100001
  33#define AHT10_CMD_MEAS  0b10101100
  34#define AHT10_CMD_RST   0b10111010
  35
  36/*
  37 * Flags in the answer byte/command
  38 */
  39#define AHT10_CAL_ENABLED       BIT(3)
  40#define AHT10_BUSY              BIT(7)
  41#define AHT10_MODE_NOR          (BIT(5) | BIT(6))
  42#define AHT10_MODE_CYC          BIT(5)
  43#define AHT10_MODE_CMD          BIT(6)
  44
  45#define AHT10_MAX_POLL_INTERVAL_LEN     30
  46
  47/**
  48 *   struct aht10_data - All the data required to operate an AHT10 chip
  49 *   @client: the i2c client associated with the AHT10
  50 *   @lock: a mutex that is used to prevent parallel access to the
  51 *          i2c client
  52 *   @min_poll_interval: the minimum poll interval
  53 *                   While the poll rate limit is not 100% necessary,
  54 *                   the datasheet recommends that a measurement
  55 *                   is not performed too often to prevent
  56 *                   the chip from warming up due to the heat it generates.
  57 *                   If it's unwanted, it can be ignored setting it to
  58 *                   it to 0. Default value is 2000 ms
  59 *   @previous_poll_time: the previous time that the AHT10
  60 *                        was polled
  61 *   @temperature: the latest temperature value received from
  62 *                 the AHT10
  63 *   @humidity: the latest humidity value received from the
  64 *              AHT10
  65 */
  66
  67struct aht10_data {
  68        struct i2c_client *client;
  69        /*
  70         * Prevent simultaneous access to the i2c
  71         * client and previous_poll_time
  72         */
  73        struct mutex lock;
  74        ktime_t min_poll_interval;
  75        ktime_t previous_poll_time;
  76        int temperature;
  77        int humidity;
  78};
  79
  80/**
  81 * aht10_init() - Initialize an AHT10 chip
  82 * @client: the i2c client associated with the AHT10
  83 * @data: the data associated with this AHT10 chip
  84 * Return: 0 if succesfull, 1 if not
  85 */
  86static int aht10_init(struct aht10_data *data)
  87{
  88        const u8 cmd_init[] = {AHT10_CMD_INIT, AHT10_CAL_ENABLED | AHT10_MODE_CYC,
  89                               0x00};
  90        int res;
  91        u8 status;
  92        struct i2c_client *client = data->client;
  93
  94        res = i2c_master_send(client, cmd_init, 3);
  95        if (res < 0)
  96                return res;
  97
  98        usleep_range(AHT10_CMD_DELAY, AHT10_CMD_DELAY +
  99                     AHT10_DELAY_EXTRA);
 100
 101        res = i2c_master_recv(client, &status, 1);
 102        if (res != 1)
 103                return -ENODATA;
 104
 105        if (status & AHT10_BUSY)
 106                return -EBUSY;
 107
 108        return 0;
 109}
 110
 111/**
 112 * aht10_polltime_expired() - check if the minimum poll interval has
 113 *                                  expired
 114 * @data: the data containing the time to compare
 115 * Return: 1 if the minimum poll interval has expired, 0 if not
 116 */
 117static int aht10_polltime_expired(struct aht10_data *data)
 118{
 119        ktime_t current_time = ktime_get_boottime();
 120        ktime_t difference = ktime_sub(current_time, data->previous_poll_time);
 121
 122        return ktime_after(difference, data->min_poll_interval);
 123}
 124
 125/**
 126 * aht10_read_values() - read and parse the raw data from the AHT10
 127 * @aht10_data: the struct aht10_data to use for the lock
 128 * Return: 0 if succesfull, 1 if not
 129 */
 130static int aht10_read_values(struct aht10_data *data)
 131{
 132        const u8 cmd_meas[] = {AHT10_CMD_MEAS, 0x33, 0x00};
 133        u32 temp, hum;
 134        int res;
 135        u8 raw_data[AHT10_MEAS_SIZE];
 136        struct i2c_client *client = data->client;
 137
 138        mutex_lock(&data->lock);
 139        if (aht10_polltime_expired(data)) {
 140                res = i2c_master_send(client, cmd_meas, sizeof(cmd_meas));
 141                if (res < 0) {
 142                        mutex_unlock(&data->lock);
 143                        return res;
 144                }
 145
 146                usleep_range(AHT10_MEAS_DELAY,
 147                             AHT10_MEAS_DELAY + AHT10_DELAY_EXTRA);
 148
 149                res = i2c_master_recv(client, raw_data, AHT10_MEAS_SIZE);
 150                if (res != AHT10_MEAS_SIZE) {
 151                        mutex_unlock(&data->lock);
 152                        if (res >= 0)
 153                                return -ENODATA;
 154                        else
 155                                return res;
 156                }
 157
 158                hum =   ((u32)raw_data[1] << 12u) |
 159                        ((u32)raw_data[2] << 4u) |
 160                        ((raw_data[3] & 0xF0u) >> 4u);
 161
 162                temp =  ((u32)(raw_data[3] & 0x0Fu) << 16u) |
 163                        ((u32)raw_data[4] << 8u) |
 164                        raw_data[5];
 165
 166                temp = ((temp * 625) >> 15u) * 10;
 167                hum = ((hum * 625) >> 16u) * 10;
 168
 169                data->temperature = (int)temp - 50000;
 170                data->humidity = hum;
 171                data->previous_poll_time = ktime_get_boottime();
 172        }
 173        mutex_unlock(&data->lock);
 174        return 0;
 175}
 176
 177/**
 178 * aht10_interval_write() - store the given minimum poll interval.
 179 * Return: 0 on success, -EINVAL if a value lower than the
 180 *         AHT10_MIN_POLL_INTERVAL is given
 181 */
 182static ssize_t aht10_interval_write(struct aht10_data *data,
 183                                    long val)
 184{
 185        data->min_poll_interval = ms_to_ktime(clamp_val(val, 2000, LONG_MAX));
 186        return 0;
 187}
 188
 189/**
 190 * aht10_interval_read() - read the minimum poll interval
 191 *                            in milliseconds
 192 */
 193static ssize_t aht10_interval_read(struct aht10_data *data,
 194                                   long *val)
 195{
 196        *val = ktime_to_ms(data->min_poll_interval);
 197        return 0;
 198}
 199
 200/**
 201 * aht10_temperature1_read() - read the temperature in millidegrees
 202 */
 203static int aht10_temperature1_read(struct aht10_data *data, long *val)
 204{
 205        int res;
 206
 207        res = aht10_read_values(data);
 208        if (res < 0)
 209                return res;
 210
 211        *val = data->temperature;
 212        return 0;
 213}
 214
 215/**
 216 * aht10_humidity1_read() - read the relative humidity in millipercent
 217 */
 218static int aht10_humidity1_read(struct aht10_data *data, long *val)
 219{
 220        int res;
 221
 222        res = aht10_read_values(data);
 223        if (res < 0)
 224                return res;
 225
 226        *val = data->humidity;
 227        return 0;
 228}
 229
 230static umode_t aht10_hwmon_visible(const void *data, enum hwmon_sensor_types type,
 231                                   u32 attr, int channel)
 232{
 233        switch (type) {
 234        case hwmon_temp:
 235        case hwmon_humidity:
 236                return 0444;
 237        case hwmon_chip:
 238                return 0644;
 239        default:
 240                return 0;
 241        }
 242}
 243
 244static int aht10_hwmon_read(struct device *dev, enum hwmon_sensor_types type,
 245                            u32 attr, int channel, long *val)
 246{
 247        struct aht10_data *data = dev_get_drvdata(dev);
 248
 249        switch (type) {
 250        case hwmon_temp:
 251                return aht10_temperature1_read(data, val);
 252        case hwmon_humidity:
 253                return aht10_humidity1_read(data, val);
 254        case hwmon_chip:
 255                return aht10_interval_read(data, val);
 256        default:
 257                return -EOPNOTSUPP;
 258        }
 259}
 260
 261static int aht10_hwmon_write(struct device *dev, enum hwmon_sensor_types type,
 262                             u32 attr, int channel, long val)
 263{
 264        struct aht10_data *data = dev_get_drvdata(dev);
 265
 266        switch (type) {
 267        case hwmon_chip:
 268                return aht10_interval_write(data, val);
 269        default:
 270                return -EOPNOTSUPP;
 271        }
 272}
 273
 274static const struct hwmon_channel_info *aht10_info[] = {
 275        HWMON_CHANNEL_INFO(chip, HWMON_C_UPDATE_INTERVAL),
 276        HWMON_CHANNEL_INFO(temp, HWMON_T_INPUT),
 277        HWMON_CHANNEL_INFO(humidity, HWMON_H_INPUT),
 278        NULL,
 279};
 280
 281static const struct hwmon_ops aht10_hwmon_ops = {
 282        .is_visible = aht10_hwmon_visible,
 283        .read = aht10_hwmon_read,
 284        .write = aht10_hwmon_write,
 285};
 286
 287static const struct hwmon_chip_info aht10_chip_info = {
 288        .ops = &aht10_hwmon_ops,
 289        .info = aht10_info,
 290};
 291
 292static int aht10_probe(struct i2c_client *client,
 293                       const struct i2c_device_id *aht10_id)
 294{
 295        struct device *device = &client->dev;
 296        struct device *hwmon_dev;
 297        struct aht10_data *data;
 298        int res;
 299
 300        if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C))
 301                return -ENOENT;
 302
 303        data = devm_kzalloc(device, sizeof(*data), GFP_KERNEL);
 304        if (!data)
 305                return -ENOMEM;
 306
 307        data->min_poll_interval = ms_to_ktime(AHT10_DEFAULT_MIN_POLL_INTERVAL);
 308        data->client = client;
 309
 310        mutex_init(&data->lock);
 311
 312        res = aht10_init(data);
 313        if (res < 0)
 314                return res;
 315
 316        res = aht10_read_values(data);
 317        if (res < 0)
 318                return res;
 319
 320        hwmon_dev = devm_hwmon_device_register_with_info(device,
 321                                                         client->name,
 322                                                         data,
 323                                                         &aht10_chip_info,
 324                                                         NULL);
 325
 326        return PTR_ERR_OR_ZERO(hwmon_dev);
 327}
 328
 329static const struct i2c_device_id aht10_id[] = {
 330        { "aht10", 0 },
 331        { },
 332};
 333MODULE_DEVICE_TABLE(i2c, aht10_id);
 334
 335static struct i2c_driver aht10_driver = {
 336        .driver = {
 337                .name = "aht10",
 338        },
 339        .probe      = aht10_probe,
 340        .id_table   = aht10_id,
 341};
 342
 343module_i2c_driver(aht10_driver);
 344
 345MODULE_AUTHOR("Johannes Cornelis Draaijer <jcdra1@gmail.com>");
 346MODULE_DESCRIPTION("AHT10 Temperature and Humidity sensor driver");
 347MODULE_VERSION("1.0");
 348MODULE_LICENSE("GPL v2");
 349