Subversion Repositories Projects

Rev

Rev 691 | Go to most recent revision | Only display areas with differences | Regard whitespace | Details | Blame | Last modification | View Log | RSS feed

Rev 691 Rev 759
1
#include <avr/boot.h>
1
#include <avr/boot.h>
2
 
2
 
3
#include <avr/io.h>
3
#include <avr/io.h>
4
#include <avr/interrupt.h>
4
#include <avr/interrupt.h>
5
 
5
 
6
#include "main.h"
6
#include "main.h"
7
#include "timer0.h"
7
#include "timer0.h"
8
#include "uart0.h"
8
#include "uart0.h"
9
#include "uart1.h"
9
#include "uart1.h"
10
#include "fat16.h"
10
#include "fat16.h"
11
#include "led.h"
11
#include "led.h"
12
#include "menu.h"
12
#include "menu.h"
13
#include "printf_P.h"
13
#include "printf_P.h"
14
#include "analog.h"
14
#include "analog.h"
15
#include "gps.h"
15
#include "gps.h"
16
#include "button.h"
16
#include "button.h"
17
#include "logging.h"
17
#include "logging.h"
18
#include "settings.h"
18
#include "settings.h"
19
 
19
 
20
#define FOLLOWME_INTERVAL 1000 // 1 second update
20
#define FOLLOWME_INTERVAL 1000 // 1 second update
21
#define CELLUNDERVOLTAGE 32 // lowest allowed voltage/cell; 32 = 3.2V
21
#define CELLUNDERVOLTAGE 32 // lowest allowed voltage/cell; 32 = 3.2V
22
 
22
 
23
#ifdef USE_FOLLOWME
23
#ifdef USE_FOLLOWME
24
int16_t UBat = 120;
24
int16_t UBat = 120;
25
int16_t Zellenzahl = 0;
25
int16_t Zellenzahl = 0;
26
int16_t PowerOn = 0;
26
int16_t PowerOn = 0;
27
int16_t i = 0;
27
int16_t i = 0;
28
int16_t delay = 0;
28
int16_t delay = 0;
29
#endif
29
#endif
30
 
30
 
31
uint16_t Error = 0;
31
uint16_t Error = 0;
32
SysState_t SysState = STATE_UNDEFINED;
32
SysState_t SysState = STATE_UNDEFINED;
33
 
33
 
34
int main (void)
34
int main (void)
35
{
35
{
36
        static uint16_t FollowMe_Timer = 0;
36
        static uint16_t FollowMe_Timer = 0;
37
 
37
 
38
        // disable interrupts global
38
        // disable interrupts global
39
        //cli();
39
        //cli();
40
 
40
 
41
        // disable watchdog
41
        // disable watchdog
42
    MCUSR &=~(1<<WDRF);
42
    MCUSR &=~(1<<WDRF);
43
    WDTCSR |= (1<<WDCE)|(1<<WDE);
43
    WDTCSR |= (1<<WDCE)|(1<<WDE);
44
    WDTCSR = 0;
44
    WDTCSR = 0;
45
 
45
 
46
        // initalize modules
46
        // initalize modules
47
        LED_Init();
47
        LED_Init();
48
        LEDRED_ON;
48
        LEDRED_ON;
49
    TIMER0_Init();
49
    TIMER0_Init();
50
        USART0_Init();
50
        USART0_Init();
51
        UBX_Init();
51
        UBX_Init();
52
        USART1_Init();
52
        USART1_Init();
53
        ADC_Init();
53
        ADC_Init();
54
        Button_Init();
54
        Button_Init();
55
        // enable interrupts global
55
        // enable interrupts global
56
        sei();
56
        sei();
57
 
57
 
58
        // try to initialize the FAT 16 filesystem on the SD-Card
58
        // try to initialize the FAT 16 filesystem on the SD-Card
59
        Fat16_Init();
59
        Fat16_Init();
60
        // initialize the settings
60
        // initialize the settings
61
        Settings_Init();
61
        Settings_Init();
62
        // initialize logging (needs settings)
62
        // initialize logging (needs settings)
63
        Logging_Init();
63
        Logging_Init();
64
 
64
 
65
        LEDRED_OFF;
65
        LEDRED_OFF;
66
        LEDGRN_ON;
66
        LEDGRN_ON;
67
 
67
 
68
        #ifdef USE_SDLOGGER
68
        #ifdef USE_SDLOGGER
69
        printf("\r\n\r\nHW: SD-Logger");
69
        printf("\r\n\r\nHW: SD-Logger");
70
        #endif
70
        #endif
71
        #ifdef USE_FOLLOWME
71
        #ifdef USE_FOLLOWME
72
        printf("\r\n\r\nHW: Follow-Me");
72
        printf("\r\n\r\nHW: Follow-Me");
73
        #endif
73
        #endif
74
        printf("\r\nFollow Me\n\rSoftware:V%d.%d%c ",VERSION_MAJOR, VERSION_MINOR, VERSION_PATCH + 'a');
74
        printf("\r\nFollow Me\r\nSoftware:V%d.%d%c ",VERSION_MAJOR, VERSION_MINOR, VERSION_PATCH + 'a');
75
        printf("\r\n------------------------------");
75
        printf("\r\n------------------------------");
76
        printf("\r\n");
76
        printf("\r\n");
77
 
77
 
78
 
78
 
79
        //BeepTime = 2000;
79
        //BeepTime = 2000;
80
 
80
 
81
    LCD_Clear();
81
    LCD_Clear();
82
 
82
 
83
        FollowMe_Timer = SetDelay(FOLLOWME_INTERVAL);
83
        FollowMe_Timer = SetDelay(FOLLOWME_INTERVAL);
84
 
84
 
85
        while (1)
85
        while (1)
86
        {
86
        {
87
                // get gps data to update the follow me position
87
                // get gps data to update the follow me position
88
                GPS_Update();
88
                GPS_Update();
89
 
89
 
90
                // update logging
90
                // update logging
91
                Logging_Update();
91
                Logging_Update();
92
 
92
 
93
                // check for button action and change state resectively
93
                // check for button action and change state resectively
94
                if(GetButton())
94
                if(GetButton())
95
                {
95
                {
96
                        BeepTime = 200;
96
                        BeepTime = 200;
97
 
97
 
98
                        switch(SysState)
98
                        switch(SysState)
99
                        {
99
                        {
100
                                case STATE_IDLE:
100
                                case STATE_IDLE:
101
                                        if(!Error) SysState = STATE_SEND_FOLLOWME; // activate followme only of no error has occured
101
                                        if(!Error) SysState = STATE_SEND_FOLLOWME; // activate followme only of no error has occured
102
                                        break;
102
                                        break;
103
 
103
 
104
                                case STATE_SEND_FOLLOWME:
104
                                case STATE_SEND_FOLLOWME:
105
                                        SysState = STATE_IDLE;
105
                                        SysState = STATE_IDLE;
106
                                        break;
106
                                        break;
107
 
107
 
108
                                default:
108
                                default:
109
                                        SysState = STATE_IDLE;
109
                                        SysState = STATE_IDLE;
110
                                        break;
110
                                        break;
111
                        }
111
                        }
112
 
112
 
113
                }
113
                }
114
 
114
 
115
                // state machine
115
                // state machine
116
                DebugOut.Analog[9] = SysState;
116
                DebugOut.Analog[9] = SysState;
117
                switch(SysState)
117
                switch(SysState)
118
                {
118
                {
119
                        case STATE_SEND_FOLLOWME:
119
                        case STATE_SEND_FOLLOWME:
120
                                if(CheckDelay(FollowMe_Timer)) // time for next message?
120
                                if(CheckDelay(FollowMe_Timer)) // time for next message?
121
                                {
121
                                {
122
                                        if(FollowMe.Position.Status == NEWDATA)        // if new
122
                                        if(FollowMe.Position.Status == NEWDATA)        // if new
123
                                        {   // update remaining data
123
                                        {   // update remaining data
124
                                                FollowMe_Timer = SetDelay(FOLLOWME_INTERVAL);  // reset timer
124
                                                FollowMe_Timer = SetDelay(FOLLOWME_INTERVAL);  // reset timer
125
                                                FollowMe.Heading = -1;                  // invalid heading
125
                                                FollowMe.Heading = -1;                  // invalid heading
126
                                                FollowMe.ToleranceRadius = 1;   // 1 meter
126
                                                FollowMe.ToleranceRadius = 1;   // 1 meter
127
                                                FollowMe.HoldTime = 60;         // go home after 60s without any update
127
                                                FollowMe.HoldTime = 60;         // go home after 60s without any update
128
                                                FollowMe.Event_Flag = 0;        // no event
128
                                                FollowMe.Event_Flag = 0;        // no event
129
                                                FollowMe.Index = 1;             // 2st wp
129
                                                FollowMe.Index = 1;             // 2st wp
130
                                                FollowMe.reserve[0] = 0;                // reserve
130
                                                FollowMe.reserve[0] = 0;                // reserve
131
                                                FollowMe.reserve[1] = 0;                // reserve
131
                                                FollowMe.reserve[1] = 0;                // reserve
132
                                                FollowMe.reserve[2] = 0;                // reserve
132
                                                FollowMe.reserve[2] = 0;                // reserve
133
                                                FollowMe.reserve[3] = 0;                // reserve
133
                                                FollowMe.reserve[3] = 0;                // reserve
134
                                                Request_SendFollowMe = 1;       // triggers serial tranmission
134
                                                Request_SendFollowMe = 1;       // triggers serial tranmission
135
 
135
 
136
                                        }
136
                                        }
137
                                        else // now new position avalable (maybe bad gps signal condition)
137
                                        else // now new position avalable (maybe bad gps signal condition)
138
                                        {
138
                                        {
139
                                                FollowMe_Timer = SetDelay(FOLLOWME_INTERVAL/4);  // reset timer on higer frequency
139
                                                FollowMe_Timer = SetDelay(FOLLOWME_INTERVAL/4);  // reset timer on higer frequency
140
                                        }
140
                                        }
141
                                        LEDGRN_TOGGLE;                                          // indication of active follow me
141
                                        LEDGRN_TOGGLE;                                          // indication of active follow me
142
                                }
142
                                }
143
                                break;
143
                                break;
144
 
144
 
145
                        case STATE_IDLE:
145
                        case STATE_IDLE:
146
                                // do nothing
146
                                // do nothing
147
                                LEDGRN_ON;
147
                                LEDGRN_ON;
148
                                break;
148
                                break;
149
 
149
 
150
                        default:
150
                        default:
151
                                // triger to idle state
151
                                // triger to idle state
152
                                SysState = STATE_IDLE;
152
                                SysState = STATE_IDLE;
153
                                break;
153
                                break;
154
 
154
 
155
                }
155
                }
156
 
156
 
157
 
157
 
158
                // restart ADConversion if ready
158
                // restart ADConversion if ready
159
                if(ADReady)
159
                if(ADReady)
160
                {
160
                {
161
                        DebugOut.Analog[0] = Adc0;
161
                        DebugOut.Analog[0] = Adc0;
162
                        DebugOut.Analog[1] = Adc1;
162
                        DebugOut.Analog[1] = Adc1;
163
                        DebugOut.Analog[2] = Adc2;
163
                        DebugOut.Analog[2] = Adc2;
164
                        DebugOut.Analog[3] = Adc3;
164
                        DebugOut.Analog[3] = Adc3;
165
                        DebugOut.Analog[4] = Adc4;
165
                        DebugOut.Analog[4] = Adc4;
166
                        DebugOut.Analog[5] = Adc5;
166
                        DebugOut.Analog[5] = Adc5;
167
                        DebugOut.Analog[6] = Adc6;
167
                        DebugOut.Analog[6] = Adc6;
168
                        DebugOut.Analog[7] = Adc7;
168
                        DebugOut.Analog[7] = Adc7;
169
 
169
 
170
                        #ifdef USE_FOLLOWME
170
                        #ifdef USE_FOLLOWME
171
                        // AVcc = 5V --> 5V = 1024 counts
171
                        // AVcc = 5V --> 5V = 1024 counts
172
                        // the voltage at the voltage divider reference point is 0.8V less that the UBat
172
                        // the voltage at the voltage divider reference point is 0.8V less that the UBat
173
                        // because of the silicon diode inbetween.
173
                        // because of the silicon diode inbetween.
174
                        // voltage divider R2=10K, R3=3K9
174
                        // voltage divider R2=10K, R3=3K9
175
                        // UAdc4 = R3/(R3+R2)*UBat= 3.9/(3.9+10)*UBat = UBat/3.564
175
                        // UAdc4 = R3/(R3+R2)*UBat= 3.9/(3.9+10)*UBat = UBat/3.564
176
                        UBat = (3 * UBat + (64 * Adc4) / 368) / 4;
176
                        UBat = (3 * UBat + (64 * Adc4) / 368) / 4;
177
                        DebugOut.Analog[8] = UBat;
177
                        DebugOut.Analog[8] = UBat;
178
 
178
 
179
                        // check for zellenzahl
179
                        // check for zellenzahl
180
                        if(PowerOn < 100)
180
                        if(PowerOn < 100)
181
                        {
181
                        {
182
                                if(UBat<=84) Zellenzahl = 2;
182
                                if(UBat<=84) Zellenzahl = 2;
183
                                else Zellenzahl = 3;
183
                                else Zellenzahl = 3;
184
                                PowerOn++;
184
                                PowerOn++;
185
                        }
185
                        }
186
                        DebugOut.Analog[16] = Zellenzahl;
186
                        DebugOut.Analog[16] = Zellenzahl;
187
                        DebugOut.Analog[17] = PowerOn;
187
                        DebugOut.Analog[17] = PowerOn;
188
 
188
 
189
                        //show recognised Zellenzahl to user
189
                        //show recognised Zellenzahl to user
190
                        if(i < Zellenzahl && PowerOn >= 100 && BeepTime == 0 && delay > 1000)
190
                        if(i < Zellenzahl && PowerOn >= 100 && BeepTime == 0 && delay > 1000)
191
                        {
191
                        {
192
                                BeepTime = 100;
192
                                BeepTime = 100;
193
                                i++;
193
                                i++;
194
                                delay = 0;
194
                                delay = 0;
195
                        }
195
                        }
196
                        if(delay < 1500) delay++;
196
                        if(delay < 1500) delay++;
197
 
197
 
198
                        // monitor battery undervoltage [...||(UBat<74) as temporary workaround to protect 2s lipo packs]
198
                        // monitor battery undervoltage [...||(UBat<74) as temporary workaround to protect 2s lipo packs]
199
                        if(((UBat < Zellenzahl * CELLUNDERVOLTAGE)||(UBat < 74)) && (PowerOn >= 100))
199
                        if(((UBat < Zellenzahl * CELLUNDERVOLTAGE)||(UBat < 74)) && (PowerOn >= 100))
200
                        {   // sound for low battery
200
                        {   // sound for low battery
201
                                BeepModulation = 0x0300;
201
                                BeepModulation = 0x0300;
202
                                if(!BeepTime)
202
                                if(!BeepTime)
203
                                {
203
                                {
204
                                        BeepTime = 6000; // 0.6 seconds
204
                                        BeepTime = 6000; // 0.6 seconds
205
                                }
205
                                }
206
                                Error |= ERROR_LOW_BAT;
206
                                Error |= ERROR_LOW_BAT;
207
                        }
207
                        }
208
                        else
208
                        else
209
                        {
209
                        {
210
                                Error &= ~ERROR_LOW_BAT;
210
                                Error &= ~ERROR_LOW_BAT;
211
                        }
211
                        }
212
                        #endif
212
                        #endif
213
                        ADReady = 0;
213
                        ADReady = 0;
214
                        ADC_Enable(); // restart ad conversion sequence
214
                        ADC_Enable(); // restart ad conversion sequence
215
                }
215
                }
216
 
216
 
217
                // serial communication
217
                // serial communication
218
                USART0_ProcessRxData();
218
                USART0_ProcessRxData();
219
                USART0_TransmitTxData();
219
                USART0_TransmitTxData();
220
 
220
 
221
                // indicate error, blinking code tbd.
221
                // indicate error, blinking code tbd.
222
                if(Error)       LEDRED_ON;
222
                if(Error)       LEDRED_ON;
223
                else            LEDRED_OFF;
223
                else            LEDRED_OFF;
224
 
224
 
225
    }
225
    }
226
        return (1);
226
        return (1);
227
}
227
}
228
 
228
 
229
 
229