-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSick_VBat.c
305 lines (255 loc) · 10 KB
/
Sick_VBat.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
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
/*
* Sick
* Compiler : Microchip xC16
* µC : 33FJ64MC804
* Mai 2015
* ____________ _ _
* |___ /| ___ \ | | | |
* / / | |_/ /___ | |__ ___ | |_
* / / | // _ \| '_ \ / _ \| __|
* / / | |\ \ (_) | |_) | (_) | |_
* /_/ |_| \_\___/|____/ \___/'\__|
* 7robot.fr
*/
/******************************************************************************/
/* Files to Include */
/******************************************************************************/
#if defined(__XC16__)
#include <xc.h>
#elif defined(__C30__)
#if defined(__dsPIC33E__)
#include <p33Exxxx.h>
#elif defined(__dsPIC33F__)
#include <p33Fxxxx.h>
#endif
#endif
#include <stdint.h> /* Includes uint16_t definition */
#include <stdio.h>
#include <stdbool.h> /* Includes true/false definition */
#include "main.h"
#include <timer.h>
/******************************************************************************/
/* Global Variable Declaration */
/******************************************************************************/
volatile int channel_conv = 0; // canal de conversion actuel
volatile uint8_t i_moy_sick = 0;
volatile uint16_t Value_Sick[NUMBER_OF_SICK][NUMBER_OF_POINTS_MOY_SICK]; // récupèrere la valeur de l'ADC du sick, puis fait une moyenne tournante
volatile uint32_t Sum_Value_Sick[NUMBER_OF_SICK];
volatile uint8_t Sick_Sector[NUMBER_OF_SICK] = {0}; // full of 0 // tout ou rien sur les seuils
volatile uint16_t Threshold[NUMBER_OF_SICK] = {DEFAULT_THRESHOLD, DEFAULT_THRESHOLD, DEFAULT_THRESHOLD, DEFAULT_THRESHOLD};
volatile unsigned int V_Bat = 0;
volatile uint16_t V_Bat_Tab[NUMBER_MEAS_VBAT];
volatile uint8_t debug_sick_on = 0;
/******************************************************************************/
/* User Functions */
/******************************************************************************/
void InitSick_VBat()
{
uint8_t i, j;
// si utilisation en ré-init : on tue l'IT d'abord...
CloseTimer5();
_AD1IF = 0; //Clear the interrupt flag
// reset des systemes de moyenne tournante
// chargement de valeurs grosses, histoire que ça déclenche pas lors de l'init
for (j = 0; j < NUMBER_OF_SICK; j++ ) {
Threshold[j] = DEFAULT_THRESHOLD;
Sum_Value_Sick[j] = 0;
Sick_Sector[j] = 1;
for (i = 0; i < NUMBER_OF_POINTS_MOY_SICK; i++){
Value_Sick[j][i] = 512;
Sum_Value_Sick[j] += 512;
}
}
channel_conv = 0;
i_moy_sick = 0;
for (j = 0; j < NUMBER_MEAS_VBAT; j++) {
V_Bat_Tab[j] = 40000;
}
//Configuration du convertisseur Analog to Digital (ADC) du dspic33f
//Cf page 286 dspic33f Data Sheet
//AD1CON1 Configuration
AD1CON1bits.ADON = 0; //Eteindre A/D converter pour la configuration
AD1CON1bits.FORM = 0; //Configure le format de la sortie de l'ADC ( 3=signed float, 2=unsigned float, 1=signed integer, 0=unsigned integer
AD1CON1bits.SSRC = 4; //Config de l'échantillonnage : Timer5
AD1CON1bits.SIMSAM = 0; //Simultaneously Sample CH0
AD1CON1bits.ASAM = 1; //Début d'échantillonnage (1=tout de suite 0=dès que AD1CON1bits.SAMP est activé)
AD1CON1bits.AD12B = 0; //Choix du type de converter (10 ou 12 bits) 0 = 10 bits , 1 = 12bits
//AD1CON2 Configuration
AD1CON2bits.ALTS = 0; //Always sampling on channel A
AD1CON2bits.CHPS = 0; //Select CH0
//AD1CON3 Configuration
AD1CON3bits.ADRC = 1; //Choix du type de clock interne (=1) ou externe (=0)
//Choix des références de tensions
// Choix du (+) de la mesure pour le channel CH0, commençant par le sick 1
AD1CHS0bits.CH0SA = AN_CH_SICK_1;
AD1CHS0bits.CH0NA = 0; // Choix du (-) de la mesure pour le channel CH0 (0 = Masse interne pic)
//Configuration des pins analogiques
//AD1PCFGL = 0xFFFF; //Met tous les ports AN en Digital Input
/* COM A ENLEVER SUR DSPIC AVEC 8 PINS ANALOGIQUES */
// sicks :
AD1PCFGLbits.PCFG4 = 0;
AD1PCFGLbits.PCFG5 = 0;
AD1PCFGLbits.PCFG6 = 0;
AD1PCFGLbits.PCFG7 = 0;
// Vbat
AD1PCFGLbits.PCFG0 = 0;
//Configuration du Timer 5, pour l'ADC
// OpenTimer5(T5_ON & T5_GATE_OFF & T5_PS_1_256 & T5_SOURCE_INT, 15625); from 2014
OpenTimer5(T5_ON & T5_GATE_OFF & T5_PS_1_8 & T5_SOURCE_INT, 3124 ); // for 2015
// FCY = 40Meg
// prescaller à 8 et comparaison à 3125 => 40M / (8*3125) = 1600 => 1600 départ de comparaison / sec
// on mets un tout petit peu moins pour que ça se désinchronise avec les IT de base
// avec 4 canaux, ça fait 400 detections / sec
// attention : (moyennage sur 16)
//Configuration des interuption
IFS0bits.AD1IF = 0; //Mise à 0 du flag d'interrupt de ADC1
IEC0bits.AD1IE = 1; //Enable les interruptions d'ADC1
IPC3bits.AD1IP = 2; //Et les priorités (ici prio = 2)
AD1CON1bits.SAMP = 0;
AD1CON1bits.ADON = 1; // Turn on the A/D converter
}
void OnSickThreshold(unsigned char id, unsigned int threshold_cons)
{
int i=0;
if(id == 255) {
for (i=0;i<NUMBER_OF_SICK;i++) {
Threshold[i] = threshold_cons;
}
} else if(id >= 0 && id < NUMBER_OF_SICK) {
Threshold[id] = threshold_cons;
}
}
void OnAskSick(unsigned char id){
// SendSickState(id,Old_Sector[id]);
}
/******************************************************************************/
/* Interrupt */
/******************************************************************************/
void __attribute__ ((interrupt, auto_psv)) _ADC1Interrupt(void)
{
static uint16_t timer_for_VBat = TIMER_FOR_VBAT;
static uint8_t i_V_Bat_tab = 0;
static uint8_t State = 0;
uint8_t i;
uint32_t val32;
static uint16_t i_debug_sick = 0;
uint16_t val16 = ADC1BUF0;// récupération valeur depuis ADC
if (channel_conv != NUMBER_OF_SICK) { // si on est pas sur V_Bat
Sum_Value_Sick[channel_conv] -= Value_Sick[channel_conv][i_moy_sick]; // enlève la valeur de X coups d'avant
Sum_Value_Sick[channel_conv] += val16; // ajoute la valeur de maintenant
Value_Sick[channel_conv][i_moy_sick] = val16; // enregistrement de la valeur lue
if (channel_conv == (NUMBER_OF_SICK - 1)) {
i_moy_sick++;
if (i_moy_sick == NUMBER_OF_POINTS_MOY_SICK) {
i_moy_sick = 0;
}
}
val16 = (uint16_t)(Sum_Value_Sick[channel_conv] >> 4); // récup de la somme et division par 16
if (Sick_Sector[channel_conv] == 0) { // si on considère pour l'instant qu'il y a un truc "pres"
if (val16 > (Threshold[channel_conv] + MARGIN_SICK)) { // si la valeur repasse au dessus de seuil + marge
Sick_Sector[channel_conv] = 1; // on repasse en zone "sûre"
}
}
else { // if old = 1 // si, pour l'instant, il n'y a pas de truc "pres"
if ( (val16 < (Threshold[channel_conv] - MARGIN_SICK)) && (val16 > SICK_LIMIT_MIN) ) { // si on vient de detecter un truc
Sick_Sector[channel_conv] = 0; // on passe en zone "pas sûre"
}
}
if (debug_sick_on) {
if (i_debug_sick == 3000) {
printf ("Sick 1 : %ld\n"
"Sick 2 : %ld\n"
"Sick 3 : %ld\n"
"Sick 4 : %ld;", (Sum_Value_Sick[0]), (Sum_Value_Sick[1]), (Sum_Value_Sick[2]), (Sum_Value_Sick[3]) );
i_debug_sick = 0;
} else {
i_debug_sick ++;
}
}
} else { // pour VBAT
// de 0 à 1023 = 0 => 3.3V
// avec pont div de 11
// => 35.44.... mV de baterie par point
// = * 9075 puis div par 512 soit >> 8
V_Bat_Tab[i_V_Bat_tab] = (uint16_t)(((9075 * ((uint32_t)(val16))) >> 8));
i_V_Bat_tab ++;
if (i_V_Bat_tab == NUMBER_MEAS_VBAT) {
i_V_Bat_tab = 0;
}
val32 = 0;
for (i = 0; i < NUMBER_MEAS_VBAT; i++) {
val32 += V_Bat_Tab[i];
}
val32 = val32 >> 3;
V_Bat = (uint16_t)(val32);
}
// Select next sensor
switch(State)
{
case 0:
_CH0SA = AN_CH_SICK_1;
channel_conv = 0;
State = 1;
break;
case 1:
_CH0SA = AN_CH_SICK_2;
channel_conv = 1;
State = 2;
break;
case 2:
_CH0SA = AN_CH_SICK_3;
channel_conv = 2;
State = 3;
break;
case 3:
_CH0SA = AN_CH_SICK_4;
channel_conv = 3;
timer_for_VBat --;
if (timer_for_VBat) {
State = 0;
} else {
State = 4;
timer_for_VBat = TIMER_FOR_VBAT;
}
break;
case 4 :
_CH0SA = AN_CH_V_BAT; // V BAT
channel_conv = 4;
State = 0;
break;
default :
State = 0;
break;
}
_AD1IF = 0; //Clear the interrupt flag
}
unsigned int Get_Sick(int Sick_Voulu)
{
Sick_Voulu --;
if (Sick_Voulu < NUMBER_OF_SICK) {
return (uint16_t)(Sum_Value_Sick[Sick_Voulu] >> 4);
} else {
return 0;
}
}
// return 1 si rien devant
// return 0 si detection
char Get_Sick_Sector (int Sick_Voulu)
{
Sick_Voulu --;
if (Sick_Voulu < NUMBER_OF_SICK) {
return Sick_Sector[Sick_Voulu];
} else {
return 0;
}
}
void Start_Stop_Debug_Sick(void)
{
if (debug_sick_on) {
debug_sick_on = 0;
} else {
debug_sick_on = 1;
}
}
unsigned int Get_VBat(void)
{ return V_Bat; }