Skip to content

Reading Motor Temperature Data

Reading Raw Temperature Data

Each temperature sensor outputs an integer raw value whose range depends on the type of resistance thermometer mounted on the motor. Currently, we can configure 2 types of sensors: PT100 and PT1000. For PT100, raw values typically range from 0 to 3500. Instead, for PT1000, the range is between 0 and 8000.

Although it is very unlikely to read negative temperatures in Celsius degrees during normal working conditions, we do accept negative values as they are specifically employed to notify the user of error conditions. Finally, we have that the higher the temperature the higher the raw output value.

Reading compensated motor temperature data

Regarding the correlation between raw temperature data and Celsius degree values here follows a brief description. First of all, in order to reduce the effort of the end-user and at the same time remain compliant with the rest of the firmware, we have decided to work with values in Celsius degree on embObjMotionControl and, instead, use the raw value in the firmware part. However, the configuration files are always filled with values in Celsius degrees; likewise, the values printed on the YARP port are in the same format. By contrast, on the board level, we use raw values. All the applied conversions strictly depend on the physical components involved, including the voltage divider, the resistance thermometer as well as the ADC stage of the TDB board. One important note to highlight is that even if we know that the conversion between raw and Celsius degree values is not exactly linear, we have decided to linearize the conversion by accepting an error of around ±1.5 ℃ over 100 ℃. Anyways, if you are interested in the specific insight regarding the conversion you can check this repository, which has been mainly used for two scopes:

  • to check the validity of the linearization done and understand the magnitude of the error inside the working conditions, which are typically between 20 and 80 ℃.
  • to evaluate the exact raw values needed for setting the defined error values to be used in the firmware, so that we can have an accurate conversion (related to our formula) on the high level. (See error values for more details).

Data conversion from raw value to Celsius degree value

For converting a value from raw to Celsius degree the following steps are applied:

Evaluate a first temporary value:

    double tmp = temperature * ((_resolution_pga) / (_pga_gain * _vcc * (_resolution_tdb + 1)));

where:
- _resolution_pga: resolution of the Programmable Gain Amplifier (PGA) mounted on the ADC, which is set to 2.048 [V]
- _resolution_tdb: resolution of the TDB ADC, which is set to 32767 since we are working at 16bit and we are saving the value to a signed int
- _vcc: incoming voltage in the voltage divider, which for us is set to 5V
- _pga_gain: gain for the PGA, which we have set to 2

The use the calculated temporary value for evaluate the denominator of the final operation:

    double den = _ptc_gradient * (_r_2 - _r_2*tmp - _r_3*tmp);

where:
- _ptc_gradient: slope of the PTXXX line
- _r_2: one of the two upper resistors in the voltage divider bridge, which for us is equal to 4700 [Ohm]
- _r_3: lower resistor of the bridge, which it should be set equal to the value, in Ohm, of the PTXXX at zero degree. Therefore, if we are using the PT100 it should be equal to 100

Finally, the last calculation that outputs the value of the got temperature in Celsius Degree is:

    double res = (tmp * (_first_res_coeff) / den) + ((_second_res_coeff) / den);

where:
- _ptc_offset: is the offset of the PTXXX line, thus it is 1000 in the case of PT1000 and 100 in the other case.
- r1: the other of the two upper resistors in the voltage divider bridge, which for us is equal to 4700 [Ohm]
- _first_res_coeff  = _r_1*_r_2 + _r_1*_r_3 + _ptc_offset*_r_2 + _ptc_offset*_r_3
- _second_res_coeff = _r_3*_r_1 - _r_2*_ptc_offset

It should be noted that those offsets are used to have the PTXXX line passes through zero at 0 Celsius Degree.

Data conversion from Celsius degree value to raw value

Similarly, for converting from Celsius degree to raw value:

    double tmp = (( (_ptc_offset + _ptc_gradient * temperature) / ((double)_r_1 + (_ptc_offset + _ptc_gradient * temperature))) - _half_bridge_resistor_coeff) * (double)_vcc;
    double res = (_resolution_tdb + 1) * ((_pga_gain * tmp) / _resolution_pga);

where:
- temperature: temperature value passed in Celsius degree
- _ptc_gradient: slope of the PTXXX line
- _ptc_offset: offset of the PTXXX line, thus it is 1000 in the case of PT1000 and 100 in the other case.
- _half_bridge_resistor_coeff = (double)_r_3 / (double)(_r_2 + _r_3) and it is the coefficient of half of the voltage divider
- vcc: incoming voltage in the voltage divider, which for us is set to 5V
- _resolution_pga: resolution of the PGA on the TDB, which is set to 2.048 [V]
- _resolution_tdb: resolution of the TDB ADC, which is set to 32767 since it works at 16bit and we are saving the value to a signed int

Table of typical temperature values

Look up this table containing a good range of temperatures to better interpret the raw temperature values reported within the yarprobotinterface logs. In particular, our robots can mount either PT100 or PT1000 temperature sensors, thus find below the correlation between raw and Celsius values for both sensors:

Temperature PT100 [Celsius] Temperature PT100 [Raw] Temperature PT1000 [Celsius] Temperature PT1000 [Raw]
-100 -1269 -100 -9559
-90 -1141 -90 -8541
-80 -1013 -80 -7538
-70 -885 -70 -6549
-60 -758 -60 -5573
-50 -631 -50 -4612
-40 -504 -40 -3664
-30 -378 -30 -2729
-20 -251 -20 -1807
-10 -125 -10 -897
0 0 0 0
5 64 5 444
10 125 10 885
15 189 15 1323
20 251 20 1758
25 315 25 2191
30 377 30 2620
35 438 35 3047
40 503 40 3471
45 564 45 3892
50 626 50 4311
55 690 55 4726
60 752 60 5139
65 814 65 5549
70 875 70 5957
75 937 75 6362
80 999 80 6765
85 1061 85 7164
90 1126 90 7562
95 1188 95 7956
100 1246 100 8349
105 1308 105 8739
110 1370 110 9126
120 1494 120 9893
130 1619 130 10651
140 1740 140 11400
150 1865 150 12139
160 1987 160 12870
170 2109 170 13591
180 2230 180 14304
190 2353 190 15008
200 2475 200 15704