-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathutils.c
executable file
·98 lines (84 loc) · 3.05 KB
/
utils.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
/*
* TongSheng TSDZ2 motor controller firmware/
*
* Copyright (C) Casainho, 2018.
*
* Released under the GPL License, Version 3
*/
#include <stdint.h>
#include <stdio.h>
#include "stm8s.h"
#include "utils.h"
int32_t map (int32_t x, int32_t in_min, int32_t in_max, int32_t out_min, int32_t out_max)
{
// if input is smaller/bigger than expected return the min/max out ranges value
if (x < in_min)
return out_min;
else if (x > in_max)
return out_max;
// map the input to the output range.
// round up if mapping bigger ranges to smaller ranges
else if ((in_max - in_min) > (out_max - out_min))
return (x - in_min) * (out_max - out_min + 1) / (in_max - in_min + 1) + out_min;
// round down if mapping smaller ranges to bigger ranges
else
return (x - in_min) * (out_max - out_min) / (in_max - in_min) + out_min;
}
int32_t map_inverse (int32_t x, int32_t in_min, int32_t in_max, int32_t out_min, int32_t out_max)
{
// if input is smaller/bigger than expected return the min/max out ranges value
if (x < in_min)
return out_min;
else if (x > in_max)
return out_max;
return (x - in_min) * (out_max - out_min) / (in_max - in_min) + out_min;
}
uint8_t ui8_min (uint8_t value_a, uint8_t value_b)
{
if (value_a < value_b) return value_a;
else return value_b;
}
uint8_t ui8_max (uint8_t value_a, uint8_t value_b)
{
if (value_a > value_b) return value_a;
else return value_b;
}
void pi_controller (struct_pi_controller_state *pi_controller)
{
int16_t i16_error;
int16_t i16_p_term;
int16_t i16_temp;
i16_error = pi_controller->ui8_target_value - pi_controller->ui8_current_value; // 255-0 or 0-255 --> [-255 ; 255]
i16_p_term = (i16_error * pi_controller->ui8_kp_dividend) >> pi_controller->ui8_kp_divisor;
pi_controller->i16_i_term += (i16_error * pi_controller->ui8_ki_dividend) >> pi_controller->ui8_ki_divisor;
if (pi_controller->i16_i_term > 255) { pi_controller->i16_i_term = 255; }
if (pi_controller->i16_i_term < 0) { pi_controller->i16_i_term = 0; }
i16_temp = i16_p_term + pi_controller->i16_i_term;
// limit to [0 ; 255] as duty_cycle that will be controlled can't have other values than that ones
if (i16_temp > 255) { i16_temp = 255; }
if (i16_temp < 0) { i16_temp = 0; }
pi_controller->ui8_controller_output_value = (uint8_t) i16_temp;
}
void pi_controller_reset (struct_pi_controller_state *pi_controller)
{
pi_controller->i16_i_term = 0;
}
// from here: https://github.com/FxDev/PetitModbus/blob/master/PetitModbus.c
/*
* Function Name : CRC16
* @param[in] : ui8_data - Data to Calculate CRC
* @param[in/out] : ui16_crc - Anlik CRC degeri
* @How to use : First initial data has to be 0xFFFF.
*/
void crc16(uint8_t ui8_data, uint16_t* ui16_crc)
{
unsigned int i;
*ui16_crc = *ui16_crc ^(uint16_t) ui8_data;
for (i = 8; i > 0; i--)
{
if (*ui16_crc & 0x0001)
*ui16_crc = (*ui16_crc >> 1) ^ 0xA001;
else
*ui16_crc >>= 1;
}
}