2021-01-29 12:31:58 +01:00
[![Arduino CI ](https://github.com/RobTillaart/MAX31855_RT/workflows/Arduino%20CI/badge.svg )](https://github.com/marketplace/actions/arduino_ci)
2021-12-09 20:42:42 +01:00
[![Arduino-lint ](https://github.com/RobTillaart/MAX31855_RT/actions/workflows/arduino-lint.yml/badge.svg )](https://github.com/RobTillaart/MAX31855_RT/actions/workflows/arduino-lint.yml)
[![JSON check ](https://github.com/RobTillaart/MAX31855_RT/actions/workflows/jsoncheck.yml/badge.svg )](https://github.com/RobTillaart/MAX31855_RT/actions/workflows/jsoncheck.yml)
2021-01-29 12:31:58 +01:00
[![License: MIT ](https://img.shields.io/badge/license-MIT-green.svg )](https://github.com/RobTillaart/MAX31855_RT/blob/master/LICENSE)
[![GitHub release ](https://img.shields.io/github/release/RobTillaart/MAX31855_RT.svg?maxAge=3600 )](https://github.com/RobTillaart/MAX31855_RT/releases)
2021-12-09 20:42:42 +01:00
2021-01-29 12:31:58 +01:00
# MAX31855_RT
2021-08-12 11:47:56 +02:00
Arduino library for MAX31855 chip for K type thermocouple.
2021-12-09 20:42:42 +01:00
The library has experimental support for other types of thermocouples E, J, N, R, S, T.
2021-01-29 12:31:58 +01:00
## Description
The MAX38155 is a chip to convert the reading of a K-type thermocouple to a temperature.
The working of thermocouples (TC) is based upon Seebeck effect.
Different TC's have a different Seebeck Coefficient (SC) expressed in µV/°C.
See http://www.analog.com/library/analogDialogue/archives/44-10/thermocouple.html
For every type of TC there exist an MAX31855 variant, this library is primary
developed for the K-type sensor. However it has experimental support for all
other types of TC's. See details below.
Library tested with breakout board
```
2022-11-16 15:46:22 +01:00
+---------+
Vin | o |
3V3 | o |
GND | o O | Thermocouple
D0 | o O | Thermocouple
CS | o |
CLK | o |
+---------+
2021-01-29 12:31:58 +01:00
```
## Hardware SPI vs software SPI
2021-08-12 11:47:56 +02:00
Default pin connections. ESP32 can overrule with **setGPIOpins()** .
2021-01-29 12:31:58 +01:00
2021-08-12 11:47:56 +02:00
| HW SPI | UNO | ESP32 VSPI | ESP32 HSPI | Notes
2021-12-09 20:42:42 +01:00
|:---------|:-----:|:-----------:|:-----------:|:----------|
| CLOCKPIN | 13 | 18 | 14 |
| MISO | 12 | 19 | 12 |
| MOSI | 11 | 23 | 13 | *not used...*
| SELECT | eg. 4 | 5 | 15 | *can be others too.*
2021-01-29 12:31:58 +01:00
2021-07-05 08:37:20 +02:00
Performance read() function, timing in us. (ESP32 @240MHz )
2021-08-12 11:47:56 +02:00
| mode | clock | timing UNO | timing ESP32 | Notes
|:-------|---------:|-----------:|-------------:|:----------|
| HW SPI | 32000000 | ni | ~15 | *less reliable*
| HW SPI | 16000000 | ~68 | ~16 |
| HW SPI | 4000000 | ~72 | ~23 |
| HW SPI | 1000000 | ~100 | ~51 |
| HW SPI | 500000 | ~128 | ~89 |
| SW SPI | bit bang | ~500 | ~17 (!) |
2021-01-29 12:31:58 +01:00
## Interface
2021-08-12 11:47:56 +02:00
### Constructor
2021-12-09 20:42:42 +01:00
- **MAX31855()** create object.
- **void begin(const uint8_t select)** set select pin => hardware SPI
- **void begin(const uint8_t sclk, const uint8_t select, const uint8_t miso)** set clock, select and miso pin => software SPI
2021-08-12 11:47:56 +02:00
### Hardware SPI
To be used only if one needs a specific speed.
2021-12-09 20:42:42 +01:00
- **void setSPIspeed(uint32_t speed)** set SPI transfer rate.
- **uint32_t getSPIspeed()** returns SPI transfer rate.
2021-12-10 11:12:41 +01:00
- **void setSWSPIdelay(uint16_t del = 0)** for tuning SW SPI signal quality. Del is the time in micros added per bit. Even numbers keep the duty cycle of the clock around 50%.
- **uint16_t getSWSPIdelay()** get set value in micros.
2021-08-12 11:47:56 +02:00
### ESP32 specific
- **void selectHSPI()** must be called before **begin()**
- **void selectVSPI()** must be called before **begin()**
- **bool usesHSPI()**
- **bool usesVSPI()**
- **void setGPIOpins(uint8_t clk, uint8_t miso, uint8_t mosi, uint8_t select)** to overrule ESP32 default hardware pins
### Reading
To make a temperature reading call **read()** .
2021-01-29 12:31:58 +01:00
It returns the status of the read which is a value between 0..7
The function **getStatus()** returns the same status value.
2021-08-12 11:47:56 +02:00
Table: values returned from **uint8_t read()** and **uint8_t getStatus()**
2021-01-29 12:31:58 +01:00
2021-07-05 08:37:20 +02:00
| value | Description | Action |
|:-----:|:--------------------------|:-------------|
| 0 | OK | |
| 1 | Thermocouple open circuit | check wiring |
| 2 | Thermocouple short to GND | check wiring |
| 4 | Thermocouple short to VCC | check wiring |
| 7 | Generic error | |
| 128 | No read done yet | check wiring |
| 129 | No communication | check wiring |
2021-01-29 12:31:58 +01:00
There are six functions to check the individual error conditions mentioned above.
These make it easier to check them.
2021-08-12 11:47:56 +02:00
- **bool openCircuit()**
- **bool shortToGND()**
- **bool shortToVCC()**
- **bool genericError()**
- **bool noRead()**
- **bool noCommunication()**
After a **uint8_t read()** you can get the temperature with **float getTemperature()**
and **float getInternal()** for the internal temperature of the chip / board itself.
Normally these are (almost) equal.
2021-01-29 12:31:58 +01:00
2021-08-12 11:47:56 +02:00
Repeated calls to **getTemperature()** will give the same value until a new **read()** .
The latter fetches a new value from the sensor. Note that if the **read()** fails
the value of **getTemperature()** can become incorrect. So it is important to check
the return value of **read()** .
2021-01-29 12:31:58 +01:00
2021-08-12 11:47:56 +02:00
### Offset
2021-01-29 12:31:58 +01:00
2021-07-05 08:37:20 +02:00
The library supports a fixed offset to calibrate the thermocouple.
2021-08-12 11:47:56 +02:00
For this the functions **float getOffset()** and **void setOffset(float offset)** are available.
This offset is "added" in the **getTemperature()** function.
Note the offset used is a float, so decimals can be used.
### Delta analysis
2021-01-29 12:31:58 +01:00
As the **tc** object holds its last known temperature it is easy to determine the delta
with the last known temperature, e.g. for trend analysis.
```cpp
float last = tc.getTemperature();
int state = tc.read();
if (state == STATUS_OK)
{
float new = tc.getTemperature();
float delta = new - last;
// process data
}
```
2021-08-12 11:47:56 +02:00
### Last time read
The **tc** object keeps track of the last time **read()** is called in the function **uint32_t lastRead()** .
2021-01-29 12:31:58 +01:00
The time is tracked in **millis()** . This makes it easy to read the sensor at certain intervals.
```cpp
if (millis() - tc.lastRead() >= interval)
{
int state = tc.read();
if (state == STATUS_OK)
{
float new = tc.getTemperature();
// process read value.
}
else
{
// handle error
}
}
```
2021-08-12 11:47:56 +02:00
### GetRawData
2021-01-29 12:31:58 +01:00
2021-08-12 11:47:56 +02:00
The function **uint32_t getRawData()** allows you to get all the 32 bits raw data from the board,
after the standard **uint8_t tc.read()** call.
2021-01-29 12:31:58 +01:00
Example code can be found in the examples folder.
```cpp
int state = thermocouple.read();
uint32_t value = thermocouple.getRawData(); // Read the raw Data value from the module
```
2021-08-12 11:47:56 +02:00
This allows one to compact the measurement e.g. for storage or sending over a network.
2021-01-29 12:31:58 +01:00
## Pull Up Resistor
To have proper working of the MAX31855 board, you need to add a pull-up resistor
(e.g. 4K7 - 1K depending on wirelength) between the MISO pin (from constructor call) and the
VCC (5Volt). This improves the signal quality and will allow you to detect if there is
proper communication with the board. WIthout pull-up one might get random noise that could
look like real data.
**Note:** the MISO pin can be different from each board, please refer to your board datasheet.
If the MAX31855 board is not connected **tc.read()** will return **STATUS_NO_COMMUNICATION** .
You can verify this by **tc.getRawData()** which will give 32 HIGH bits or 0xFFFFFFFF).
You can use a simple code to detect connection error board:
```cpp
uint8_t status = thermocouple.read();
if (status == STATUS_NO_COMMUNICATION)
{
Serial.println("NO COMMUNICATION");
}
```
or
```cpp
uint8_t status = thermocouple.read();
if (thermocouple.getRawData() == 0xFFFFFFFF)
{
Serial.println("NO COMMUNICATION");
}
```
## Operation
See examples
2021-12-09 20:42:42 +01:00
#### breaking change 0.4.0
In issue #21 it became clear that the code in the constructor is not always executed correctly.
Therefore this code + parameters is moved to the **Begin()** function.
2021-01-29 12:31:58 +01:00
## Experimental part (to be tested)
2021-08-12 11:47:56 +02:00
(to be tested)
2021-01-29 12:31:58 +01:00
**NOTE:**
The support for other thermocouples is experimental **use at your own risk** .
The MAX31855 is designed for K type sensors. It essentially measures a
voltage difference and converts this voltage using the Seebeck Coefficient (SC)
2021-07-05 08:37:20 +02:00
to the temperature. As the SC is linear in its nature it is possible
2021-01-29 12:31:58 +01:00
to replace the K-type TC with one of the other types of TC.
2021-07-05 08:37:20 +02:00
Datasheet Table 1, page 8 SC = Seebeck Coefficient
| Sensor type | SC in µV/°C | Temp Range in °C | Material |
|:-----------:|:------------|:-----------------|:--------------------------|
| E_TC | 76.373 | -270 to +1000 | Constantan Chromel |
| J_TC | 57.953 | -210 to +1200 | Constantan Iron |
| K_TC | 41.276 | -270 to +1372 | Alumel Chromel |
| N_TC | 36.256 | -270 to +1300 | Nisil Nicrosil |
| R_TC | 10.506 | -50 to +1768 | Platinum Platinum/Rhodium |
| S_TC | 9.587 | +50 to +1768 | Platinum Platinum/Rhodium |
| T_TC | 52.18 | -270 to +400 | Constantan Copper |
2021-01-29 12:31:58 +01:00
The core formula to calculate the temperature is (Datasheet page 8)
```
Vout = (41.276µV/°C) x (Temp_R - Temp_internal)
```
As we know the internal temperature and the returned temperature from the sensor
the library can calculate the Vout measured (as the chip assumes that a K-type
thermocouple is connected.
Having that Vout we can redo the math for the actual thermocouple type and
calculate the real temperature.
2021-08-12 11:47:56 +02:00
The library has two functions **setSeebeckCoefficient(float factor)** and
**float getSeebeckCoefficient()**
2021-01-29 12:31:58 +01:00
to get/set the Seebeck Coefficient (== thermocouple) to be used.
One can adjust the values to improve the accuracy of the temperature read.
2021-08-12 11:47:56 +02:00
The **float getTemperature()** has implemented this algorithm, however as long
2021-01-29 12:31:58 +01:00
as one does not set the Seebeck Coefficient it will use the K_TC as default.
2021-08-12 11:47:56 +02:00
2021-12-09 20:42:42 +01:00
## Future
2022-11-16 15:46:22 +01:00
#### must
#### should
2021-12-09 20:42:42 +01:00
- investigate other TC's
-
2022-11-16 15:46:22 +01:00
#### could