Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Tomasz Duszynski | 1077 | 99.35% | 2 | 40.00% |
Jonathan Cameron | 5 | 0.46% | 1 | 20.00% |
Uwe Kleine-König | 1 | 0.09% | 1 | 20.00% |
Jonathan Neuschäfer | 1 | 0.09% | 1 | 20.00% |
Total | 1084 | 5 |
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259
// SPDX-License-Identifier: GPL-2.0 /* * Sensirion SPS30 particulate matter sensor i2c driver * * Copyright (c) 2020 Tomasz Duszynski <tomasz.duszynski@octakon.com> * * I2C slave address: 0x69 */ #include <asm/unaligned.h> #include <linux/crc8.h> #include <linux/delay.h> #include <linux/device.h> #include <linux/errno.h> #include <linux/i2c.h> #include <linux/mod_devicetable.h> #include <linux/module.h> #include <linux/types.h> #include "sps30.h" #define SPS30_I2C_CRC8_POLYNOMIAL 0x31 /* max number of bytes needed to store PM measurements or serial string */ #define SPS30_I2C_MAX_BUF_SIZE 48 DECLARE_CRC8_TABLE(sps30_i2c_crc8_table); #define SPS30_I2C_START_MEAS 0x0010 #define SPS30_I2C_STOP_MEAS 0x0104 #define SPS30_I2C_READ_MEAS 0x0300 #define SPS30_I2C_MEAS_READY 0x0202 #define SPS30_I2C_RESET 0xd304 #define SPS30_I2C_CLEAN_FAN 0x5607 #define SPS30_I2C_PERIOD 0x8004 #define SPS30_I2C_READ_SERIAL 0xd033 #define SPS30_I2C_READ_VERSION 0xd100 static int sps30_i2c_xfer(struct sps30_state *state, unsigned char *txbuf, size_t txsize, unsigned char *rxbuf, size_t rxsize) { struct i2c_client *client = to_i2c_client(state->dev); int ret; /* * Sensor does not support repeated start so instead of * sending two i2c messages in a row we just send one by one. */ ret = i2c_master_send(client, txbuf, txsize); if (ret < 0) return ret; if (ret != txsize) return -EIO; if (!rxsize) return 0; ret = i2c_master_recv(client, rxbuf, rxsize); if (ret < 0) return ret; if (ret != rxsize) return -EIO; return 0; } static int sps30_i2c_command(struct sps30_state *state, u16 cmd, void *arg, size_t arg_size, void *rsp, size_t rsp_size) { /* * Internally sensor stores measurements in a following manner: * * PM1: upper two bytes, crc8, lower two bytes, crc8 * PM2P5: upper two bytes, crc8, lower two bytes, crc8 * PM4: upper two bytes, crc8, lower two bytes, crc8 * PM10: upper two bytes, crc8, lower two bytes, crc8 * * What follows next are number concentration measurements and * typical particle size measurement which we omit. */ unsigned char buf[SPS30_I2C_MAX_BUF_SIZE]; unsigned char *tmp; unsigned char crc; size_t i; int ret; put_unaligned_be16(cmd, buf); i = 2; if (rsp) { /* each two bytes are followed by a crc8 */ rsp_size += rsp_size / 2; } else { tmp = arg; while (arg_size) { buf[i] = *tmp++; buf[i + 1] = *tmp++; buf[i + 2] = crc8(sps30_i2c_crc8_table, buf + i, 2, CRC8_INIT_VALUE); arg_size -= 2; i += 3; } } ret = sps30_i2c_xfer(state, buf, i, buf, rsp_size); if (ret) return ret; /* validate received data and strip off crc bytes */ tmp = rsp; for (i = 0; i < rsp_size; i += 3) { crc = crc8(sps30_i2c_crc8_table, buf + i, 2, CRC8_INIT_VALUE); if (crc != buf[i + 2]) { dev_err(state->dev, "data integrity check failed\n"); return -EIO; } *tmp++ = buf[i]; *tmp++ = buf[i + 1]; } return 0; } static int sps30_i2c_start_meas(struct sps30_state *state) { /* request BE IEEE754 formatted data */ unsigned char buf[] = { 0x03, 0x00 }; return sps30_i2c_command(state, SPS30_I2C_START_MEAS, buf, sizeof(buf), NULL, 0); } static int sps30_i2c_stop_meas(struct sps30_state *state) { return sps30_i2c_command(state, SPS30_I2C_STOP_MEAS, NULL, 0, NULL, 0); } static int sps30_i2c_reset(struct sps30_state *state) { int ret; ret = sps30_i2c_command(state, SPS30_I2C_RESET, NULL, 0, NULL, 0); msleep(500); /* * Power-on-reset causes sensor to produce some glitch on i2c bus and * some controllers end up in error state. Recover simply by placing * some data on the bus, for example STOP_MEAS command, which * is NOP in this case. */ sps30_i2c_stop_meas(state); return ret; } static bool sps30_i2c_meas_ready(struct sps30_state *state) { unsigned char buf[2]; int ret; ret = sps30_i2c_command(state, SPS30_I2C_MEAS_READY, NULL, 0, buf, sizeof(buf)); if (ret) return false; return buf[1]; } static int sps30_i2c_read_meas(struct sps30_state *state, __be32 *meas, size_t num) { /* measurements are ready within a second */ if (msleep_interruptible(1000)) return -EINTR; if (!sps30_i2c_meas_ready(state)) return -ETIMEDOUT; return sps30_i2c_command(state, SPS30_I2C_READ_MEAS, NULL, 0, meas, sizeof(num) * num); } static int sps30_i2c_clean_fan(struct sps30_state *state) { return sps30_i2c_command(state, SPS30_I2C_CLEAN_FAN, NULL, 0, NULL, 0); } static int sps30_i2c_read_cleaning_period(struct sps30_state *state, __be32 *period) { return sps30_i2c_command(state, SPS30_I2C_PERIOD, NULL, 0, period, sizeof(*period)); } static int sps30_i2c_write_cleaning_period(struct sps30_state *state, __be32 period) { return sps30_i2c_command(state, SPS30_I2C_PERIOD, &period, sizeof(period), NULL, 0); } static int sps30_i2c_show_info(struct sps30_state *state) { /* extra nul just in case */ unsigned char buf[32 + 1] = { 0x00 }; int ret; ret = sps30_i2c_command(state, SPS30_I2C_READ_SERIAL, NULL, 0, buf, sizeof(buf) - 1); if (ret) return ret; dev_info(state->dev, "serial number: %s\n", buf); ret = sps30_i2c_command(state, SPS30_I2C_READ_VERSION, NULL, 0, buf, 2); if (ret) return ret; dev_info(state->dev, "fw version: %u.%u\n", buf[0], buf[1]); return 0; } static const struct sps30_ops sps30_i2c_ops = { .start_meas = sps30_i2c_start_meas, .stop_meas = sps30_i2c_stop_meas, .read_meas = sps30_i2c_read_meas, .reset = sps30_i2c_reset, .clean_fan = sps30_i2c_clean_fan, .read_cleaning_period = sps30_i2c_read_cleaning_period, .write_cleaning_period = sps30_i2c_write_cleaning_period, .show_info = sps30_i2c_show_info, }; static int sps30_i2c_probe(struct i2c_client *client) { if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) return -EOPNOTSUPP; crc8_populate_msb(sps30_i2c_crc8_table, SPS30_I2C_CRC8_POLYNOMIAL); return sps30_probe(&client->dev, client->name, NULL, &sps30_i2c_ops); } static const struct i2c_device_id sps30_i2c_id[] = { { "sps30" }, { } }; MODULE_DEVICE_TABLE(i2c, sps30_i2c_id); static const struct of_device_id sps30_i2c_of_match[] = { { .compatible = "sensirion,sps30" }, { } }; MODULE_DEVICE_TABLE(of, sps30_i2c_of_match); static struct i2c_driver sps30_i2c_driver = { .driver = { .name = KBUILD_MODNAME, .of_match_table = sps30_i2c_of_match, }, .id_table = sps30_i2c_id, .probe = sps30_i2c_probe, }; module_i2c_driver(sps30_i2c_driver); MODULE_AUTHOR("Tomasz Duszynski <tomasz.duszynski@octakon.com>"); MODULE_DESCRIPTION("Sensirion SPS30 particulate matter sensor i2c driver"); MODULE_LICENSE("GPL v2"); MODULE_IMPORT_NS(IIO_SPS30);
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1