linux/drivers/iio/chemical/sgp40.c

// SPDX-License-Identifier: GPL-2.0+
/*
 * sgp40.c - Support for Sensirion SGP40 Gas Sensor
 *
 * Copyright (C) 2021 Andreas Klinger <[email protected]>
 *
 * I2C slave address: 0x59
 *
 * Datasheet can be found here:
 * https://www.sensirion.com/file/datasheet_sgp40
 *
 * There are two functionalities supported:
 *
 * 1) read raw logarithmic resistance value from sensor
 *    --> useful to pass it to the algorithm of the sensor vendor for
 *    measuring deteriorations and improvements of air quality.
 *    It can be read from the attribute in_resistance_raw.
 *
 * 2) calculate an estimated absolute voc index (in_concentration_input)
 *    with 0 - 500 index points) for measuring the air quality.
 *    For this purpose the value of the resistance for which the voc index
 *    will be 250 can be set up using in_resistance_calibbias (default 30000).
 *
 *    The voc index is calculated as:
 *      x = (in_resistance_raw - in_resistance_calibbias) * 0.65
 *      in_concentration_input = 500 / (1 + e^x)
 *
 * Compensation values of relative humidity and temperature can be set up
 * by writing to the out values of temp and humidityrelative.
 */

#include <linux/delay.h>
#include <linux/crc8.h>
#include <linux/module.h>
#include <linux/mutex.h>
#include <linux/i2c.h>
#include <linux/iio/iio.h>

/*
 * floating point calculation of voc is done as integer
 * where numbers are multiplied by 1 << SGP40_CALC_POWER
 */
#define SGP40_CALC_POWER

#define SGP40_CRC8_POLYNOMIAL
#define SGP40_CRC8_INIT

DECLARE_CRC8_TABLE(sgp40_crc8_table);

struct sgp40_data {};

struct sgp40_tg_measure {} __packed;

struct sgp40_tg_result {} __packed;

static const struct iio_chan_spec sgp40_channels[] =;

/*
 * taylor approximation of e^x:
 * y = 1 + x + x^2 / 2 + x^3 / 6 + x^4 / 24 + ... + x^n / n!
 *
 * Because we are calculating x real value multiplied by 2^power we get
 * an additional 2^power^n to divide for every element. For a reasonable
 * precision this would overflow after a few iterations. Therefore we
 * divide the x^n part whenever its about to overflow (xmax).
 */

static u32 sgp40_exp(int exp, u32 power, u32 rounds)
{}

static int sgp40_calc_voc(struct sgp40_data *data, u16 resistance_raw, int *voc)
{}

static int sgp40_measure_resistance_raw(struct sgp40_data *data, u16 *resistance_raw)
{}

static int sgp40_read_raw(struct iio_dev *indio_dev,
			struct iio_chan_spec const *chan, int *val,
			int *val2, long mask)
{}

static int sgp40_write_raw(struct iio_dev *indio_dev,
			struct iio_chan_spec const *chan, int val,
			int val2, long mask)
{}

static const struct iio_info sgp40_info =;

static int sgp40_probe(struct i2c_client *client)
{}

static const struct i2c_device_id sgp40_id[] =;

MODULE_DEVICE_TABLE(i2c, sgp40_id);

static const struct of_device_id sgp40_dt_ids[] =;

MODULE_DEVICE_TABLE(of, sgp40_dt_ids);

static struct i2c_driver sgp40_driver =;
module_i2c_driver();

MODULE_AUTHOR();
MODULE_DESCRIPTION();
MODULE_LICENSE();