Prepared a library fro the sps30 dust sensor. The readout reads some values but the sensor we have is likely damaged.
This commit is contained in:
parent
00811846d5
commit
a1d3b42d53
78
fw/Core/Inc/sps30.h
Normal file
78
fw/Core/Inc/sps30.h
Normal file
@ -0,0 +1,78 @@
|
|||||||
|
/*
|
||||||
|
* sps30.h
|
||||||
|
*
|
||||||
|
* Created on: Jul 18, 2021
|
||||||
|
* Author: mrs
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef INC_SPS30_H_
|
||||||
|
#define INC_SPS30_H_
|
||||||
|
|
||||||
|
#include "stdint.h"
|
||||||
|
#include "stm32l0xx_ll_i2c.h"
|
||||||
|
#include "stm32l0xx_ll_utils.h"
|
||||||
|
#include "i2c.h"
|
||||||
|
#include "crc8.h"
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Defines & macros
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define SPS30_I2C_ADDRESS 0x69
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Return values
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define SPS30_OK 0
|
||||||
|
#define SPS30_ERROR -1 // generic error
|
||||||
|
#define SPS30_CRC8_ERROR -2 // checksum failed
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Data types
|
||||||
|
*/
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
SPS30_START_MEASUREMENT = 0x0010,
|
||||||
|
SPS30_STOP_MEASUREMENT = 0x0104,
|
||||||
|
SPS30_READ_DATA_READY_FLAG = 0x0202,
|
||||||
|
SPS30_READ_MEASURED_VALUES = 0x0300,
|
||||||
|
SPS30_SLEEP = 0x1001,
|
||||||
|
SPS30_WAKE_UP = 0x1103,
|
||||||
|
SPS30_START_FAN_CLEANING = 0x5607,
|
||||||
|
SPS30_READ_AUTO_CLEANING_INTERVAL = 0x8004,
|
||||||
|
SPS30_WRITE_AUTO_CLEANING_INTERVAL = 0x8004,
|
||||||
|
SPS30_READ_PRODUCT_TYPE = 0xD002,
|
||||||
|
SPS30_READ_SERIAL_NUMBER = 0xD033,
|
||||||
|
SPS30_READ_VERSION = 0xD100,
|
||||||
|
SPS30_READ_DEVICE_STATUS_REGISTER = 0xD206,
|
||||||
|
SPS30_CLEAR_DEVICE_STATUS_REGISTER = 0xD210,
|
||||||
|
SPS30_RESET = 0xD304
|
||||||
|
|
||||||
|
} sps30_cmd_t;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
SPS30_FLOAT_FORMAT = 0x03,
|
||||||
|
SPS30_UINT16_FORMAT = 0x05
|
||||||
|
} sps30_data_format;
|
||||||
|
|
||||||
|
int8_t sps30_send_cmd(sps30_cmd_t cmd);
|
||||||
|
|
||||||
|
int8_t sps30_start_measurement( void );
|
||||||
|
int8_t sps30_stop_measurement( void );
|
||||||
|
int8_t sps30_read_measured_values(uint16_t *measured_values, uint8_t measured_values_len);
|
||||||
|
|
||||||
|
int8_t sps30_sleep( void );
|
||||||
|
int8_t sps30_wake_up( void );
|
||||||
|
|
||||||
|
int8_t sps30_start_fan_cleaning( void );
|
||||||
|
|
||||||
|
int8_t sps30_reset( void );
|
||||||
|
|
||||||
|
int8_t sps30_read_status_register ( void );
|
||||||
|
|
||||||
|
int8_t sps30_read_firmware_version ( uint8_t * fw_ver_hi, uint8_t * fw_ver_lo );
|
||||||
|
|
||||||
|
static uint8_t calculate_crc(uint8_t data[2]);
|
||||||
|
|
||||||
|
#endif /* INC_SPS30_H_ */
|
@ -111,22 +111,41 @@ int main(void)
|
|||||||
//scd4x_perform_factory_reset();
|
//scd4x_perform_factory_reset();
|
||||||
//LL_mDelay(2000);
|
//LL_mDelay(2000);
|
||||||
scd4x_start_periodic_measurement();
|
scd4x_start_periodic_measurement();
|
||||||
|
|
||||||
|
uint8_t sps30_fw_v_hi, sps30_fw_v_lo;
|
||||||
|
|
||||||
|
sps30_read_firmware_version(&sps30_fw_v_hi, &sps30_fw_v_lo);
|
||||||
|
|
||||||
|
sps30_reset();
|
||||||
|
sps30_start_measurement();
|
||||||
LL_mDelay(2000);
|
LL_mDelay(2000);
|
||||||
int CO2, T_SCD4x, RH_SCD4x;
|
int CO2, T_SCD4x, RH_SCD4x;
|
||||||
int T_SHT4x, RH_SHT4x;
|
int T_SHT4x, RH_SHT4x;
|
||||||
|
uint16_t sps30_measured_data[10];
|
||||||
|
|
||||||
|
/*
|
||||||
|
sps30_start_fan_cleaning();
|
||||||
|
LL_mDelay(15000);
|
||||||
|
*/
|
||||||
while (1)
|
while (1)
|
||||||
{
|
{
|
||||||
/* RS485 test */
|
/* RS485 test */
|
||||||
LL_LPUART_SetDESignalPolarity(LPUART1, 1);
|
/*LL_LPUART_SetDESignalPolarity(LPUART1, 1);
|
||||||
LL_LPUART_TransmitData8(LPUART1, uart_data_dummy);
|
LL_LPUART_TransmitData8(LPUART1, uart_data_dummy);
|
||||||
uart_data_dummy++;
|
uart_data_dummy++;*/
|
||||||
|
|
||||||
/* SHT41 measurement */
|
/* SHT41 measurement */
|
||||||
sht4x_measure(&T_SHT4x, &RH_SHT4x);
|
sht4x_measure(&T_SHT4x, &RH_SHT4x);
|
||||||
|
LL_mDelay(10);
|
||||||
|
|
||||||
/* SCD4x measurement */
|
/* SCD4x measurement */
|
||||||
scd4x_read_measurement(&CO2, &T_SCD4x, &RH_SCD4x);
|
scd4x_read_measurement(&CO2, &T_SCD4x, &RH_SCD4x);
|
||||||
|
LL_mDelay(10);
|
||||||
|
|
||||||
|
/* SPS30 measurement*/
|
||||||
|
sps30_read_measured_values(sps30_measured_data, 10);
|
||||||
|
|
||||||
|
/* SLEEP */
|
||||||
LL_mDelay(1000);
|
LL_mDelay(1000);
|
||||||
/* USER CODE END WHILE */
|
/* USER CODE END WHILE */
|
||||||
/* USER CODE BEGIN 3 */
|
/* USER CODE BEGIN 3 */
|
||||||
|
193
fw/Core/Src/sps30.c
Normal file
193
fw/Core/Src/sps30.c
Normal file
@ -0,0 +1,193 @@
|
|||||||
|
/*
|
||||||
|
* sps30.c
|
||||||
|
*
|
||||||
|
* Created on: Jul 18, 2021
|
||||||
|
* Author: david
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "sps30.h"
|
||||||
|
|
||||||
|
int8_t sps30_send_cmd(sps30_cmd_t cmd)
|
||||||
|
{
|
||||||
|
uint8_t buffer[32];
|
||||||
|
uint8_t result;
|
||||||
|
|
||||||
|
// start measurement
|
||||||
|
buffer[0] = cmd >> 8;
|
||||||
|
buffer[1] = cmd & 0x00ff;
|
||||||
|
result = i2c_transmit(SPS30_I2C_ADDRESS<<1, buffer, 2);
|
||||||
|
// TODO: Proc to vraci NACK? Vyresit.
|
||||||
|
if (result != I2C_OK) {
|
||||||
|
return SPS30_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
return SPS30_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
int8_t sps30_start_measurement( void )
|
||||||
|
{
|
||||||
|
uint8_t i2c_tx_buffer[5];
|
||||||
|
uint8_t data_for_crc = {SPS30_UINT16_FORMAT, 0x00};
|
||||||
|
|
||||||
|
uint8_t result;
|
||||||
|
|
||||||
|
i2c_tx_buffer[0] = SPS30_START_MEASUREMENT >> 8;
|
||||||
|
i2c_tx_buffer[1] = SPS30_START_MEASUREMENT & 0x00ff;
|
||||||
|
i2c_tx_buffer[2] = SPS30_UINT16_FORMAT;
|
||||||
|
i2c_tx_buffer[3] = 0x00;
|
||||||
|
i2c_tx_buffer[4] = calculate_crc(data_for_crc);
|
||||||
|
|
||||||
|
result = i2c_transmit(SPS30_I2C_ADDRESS<<1, i2c_tx_buffer, 5);
|
||||||
|
|
||||||
|
// TODO: Proc to vraci NACK? Vyresit.
|
||||||
|
if (result != I2C_OK) {
|
||||||
|
return SPS30_ERROR;
|
||||||
|
}
|
||||||
|
return SPS30_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
int8_t sps30_stop_measurement( void )
|
||||||
|
{
|
||||||
|
return sps30_send_cmd(SPS30_STOP_MEASUREMENT);
|
||||||
|
}
|
||||||
|
|
||||||
|
int8_t sps30_read_measured_values(uint16_t *measured_values, uint8_t measured_values_len)
|
||||||
|
{
|
||||||
|
|
||||||
|
if (measured_values_len != 10)
|
||||||
|
{
|
||||||
|
return -5;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t i2c_tx_buffer[2];
|
||||||
|
uint8_t i2c_rx_buffer[30];
|
||||||
|
|
||||||
|
uint8_t result;
|
||||||
|
|
||||||
|
// start measurement
|
||||||
|
i2c_tx_buffer[0] = SPS30_READ_MEASURED_VALUES >> 8;
|
||||||
|
i2c_tx_buffer[1] = SPS30_READ_MEASURED_VALUES & 0x00ff;
|
||||||
|
result = i2c_transmit(SPS30_I2C_ADDRESS<<1, i2c_tx_buffer, 2);
|
||||||
|
|
||||||
|
// TODO: Proc to vraci NACK? Vyresit.
|
||||||
|
/*if (result != I2C_OK) {
|
||||||
|
return SPS30_ERROR;
|
||||||
|
}
|
||||||
|
return SPS30_OK;*/
|
||||||
|
|
||||||
|
LL_mDelay(1); // 10 ms should be enough
|
||||||
|
// read out
|
||||||
|
result = i2c_receive(SPS30_I2C_ADDRESS<<1, i2c_rx_buffer, 30);
|
||||||
|
if (result != I2C_OK)
|
||||||
|
{
|
||||||
|
return SPS30_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t checksums[10];
|
||||||
|
|
||||||
|
uint8_t j = 0;
|
||||||
|
for (uint8_t i = 0; i < 10; i++)
|
||||||
|
{
|
||||||
|
|
||||||
|
measured_values[i] = (i2c_rx_buffer[j++] << 8) + i2c_rx_buffer[j++];
|
||||||
|
checksums[i] = i2c_rx_buffer[j++];
|
||||||
|
}
|
||||||
|
|
||||||
|
return SPS30_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
int8_t sps30_sleep( void )
|
||||||
|
{
|
||||||
|
return sps30_send_cmd(SPS30_SLEEP);
|
||||||
|
}
|
||||||
|
|
||||||
|
int8_t sps30_wake_up( void )
|
||||||
|
{
|
||||||
|
return sps30_send_cmd(SPS30_WAKE_UP);
|
||||||
|
return sps30_send_cmd(SPS30_WAKE_UP);
|
||||||
|
}
|
||||||
|
|
||||||
|
int8_t sps30_start_fan_cleaning( void )
|
||||||
|
{
|
||||||
|
return sps30_send_cmd(SPS30_START_FAN_CLEANING);
|
||||||
|
}
|
||||||
|
|
||||||
|
int8_t sps30_reset( void )
|
||||||
|
{
|
||||||
|
return sps30_send_cmd(SPS30_RESET);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int8_t sps30_read_status_register ( void )
|
||||||
|
{
|
||||||
|
uint8_t i2c_tx_buffer[2];
|
||||||
|
uint8_t i2c_rx_buffer[6];
|
||||||
|
|
||||||
|
uint8_t result;
|
||||||
|
|
||||||
|
// start measurement
|
||||||
|
i2c_tx_buffer[0] = SPS30_READ_DEVICE_STATUS_REGISTER >> 8;
|
||||||
|
i2c_tx_buffer[1] = SPS30_READ_DEVICE_STATUS_REGISTER & 0x00ff;
|
||||||
|
result = i2c_transmit(SPS30_I2C_ADDRESS<<1, i2c_tx_buffer, 2);
|
||||||
|
|
||||||
|
// TODO: Proc to vraci NACK? Vyresit.
|
||||||
|
/*if (result != I2C_OK) {
|
||||||
|
return SPS30_ERROR;
|
||||||
|
}
|
||||||
|
return SPS30_OK;*/
|
||||||
|
|
||||||
|
LL_mDelay(1); // 10 ms should be enough
|
||||||
|
// read out
|
||||||
|
result = i2c_receive(SPS30_I2C_ADDRESS<<1, i2c_rx_buffer, 6);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int8_t sps30_read_firmware_version ( uint8_t * fw_ver_hi, uint8_t * fw_ver_lo )
|
||||||
|
{
|
||||||
|
uint8_t i2c_tx_buffer[2];
|
||||||
|
uint8_t i2c_rx_buffer[3];
|
||||||
|
|
||||||
|
uint8_t result;
|
||||||
|
|
||||||
|
// start measurement
|
||||||
|
i2c_tx_buffer[0] = SPS30_READ_VERSION >> 8;
|
||||||
|
i2c_tx_buffer[1] = SPS30_READ_VERSION & 0x00ff;
|
||||||
|
result = i2c_transmit(SPS30_I2C_ADDRESS<<1, i2c_tx_buffer, 2);
|
||||||
|
|
||||||
|
// TODO: Proc to vraci NACK? Vyresit.
|
||||||
|
/*if (result != I2C_OK) {
|
||||||
|
return SPS30_ERROR;
|
||||||
|
}
|
||||||
|
return SPS30_OK;*/
|
||||||
|
|
||||||
|
LL_mDelay(1); // 10 ms should be enough
|
||||||
|
// read out
|
||||||
|
result = i2c_receive(SPS30_I2C_ADDRESS<<1, i2c_rx_buffer, 3);
|
||||||
|
/*if (result != I2C_OK)
|
||||||
|
{
|
||||||
|
return SPS30_ERROR;
|
||||||
|
}*/
|
||||||
|
|
||||||
|
*fw_ver_hi = i2c_rx_buffer[0];
|
||||||
|
*fw_ver_lo = i2c_rx_buffer[1];
|
||||||
|
|
||||||
|
return SPS30_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
static uint8_t calculate_crc(uint8_t data[2])
|
||||||
|
{
|
||||||
|
uint8_t crc = 0xFF;
|
||||||
|
for(uint8_t i = 0; i < 2; i++) {
|
||||||
|
crc ^= data[i];
|
||||||
|
for(uint8_t bit = 8; bit > 0; --bit) {
|
||||||
|
if(crc & 0x80) {
|
||||||
|
crc = (crc << 1) ^ 0x31u;
|
||||||
|
} else {
|
||||||
|
crc = (crc << 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return crc;
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user