/[H9]/trunk/Embedded/main.c
ViewVC logotype

Annotation of /trunk/Embedded/main.c

Parent Directory Parent Directory | Revision Log Revision Log


Revision 165 - (hide annotations) (download)
Wed Dec 5 16:00:03 2007 UTC (16 years, 6 months ago) by hedin
File MIME type: text/plain
File size: 11788 byte(s)
Embedded backup
1 hedin 82 #include <pic18.h>
2     #include <stdio.h>
3 hedin 148 #include <htc.h>
4 hedin 82 #include <string.h>
5 hedin 160 #include <stdlib.h>
6 hedin 148
7 hedin 82 #include "lcd.h"
8 hedin 137 #include "Delay.h"
9 hedin 109 // Delay.h is included inside lcd.c
10 hedin 82
11 hedin 109 #define LCD_LENGTH 16
12     #define LCD_ROWS 2
13 hedin 150 #define BUFFER 128
14 hedin 148 #define PWRFAIL RB1
15     #define FIREDET RB2
16     #define FEEDING RB3
17     #define EMPTYTANK RB4
18 hedin 109
19    
20 hedin 137 unsigned char global_Pot_Hi, global_Pot_Lo;
21 hedin 150 unsigned char global_serial_send[BUFFER], global_serial_recieve_buffer[BUFFER];
22 hedin 148 bit global_recieve_done = 0, global_interval_changed = 0;
23     unsigned int global_serial_byte_counter = 0, global_sms_counter = 1, global_time_counter = 0;
24     unsigned int global_emergency_counter = 600, global_time_interval = 3600;
25     unsigned char global_temp = 0;
26 hedin 137
27 hedin 165 unsigned char global_temp_update_display = 0;
28    
29 hedin 160 unsigned char global_message_buffer1[BUFFER];
30     unsigned char global_message_buffer2[BUFFER];
31     unsigned char global_message_buffer_length1;
32     unsigned char global_message_buffer_length2;
33 hedin 155
34 hedin 151 unsigned char global_lcd_buf[16];
35    
36 hedin 150 unsigned short global_imei_tversum;
37    
38 hedin 160 unsigned char global_cell_nr[15] = ""; // = "21681784";
39 hedin 150 bit global_modem_init = 0;
40 hedin 160 bit global_has_imei = 0;
41 hedin 148
42 hedin 155 unsigned char global_sms_recieve_number[3];
43    
44 hedin 148 __EEPROM_DATA( 60, 0, 1, 8, '2', '1', '6', '8');
45     __EEPROM_DATA( '1', '7', '8', '4',0,0,0,0);
46    
47 hedin 137 void serial_recieved(void);
48 hedin 139 void serial_send(void);
49     void update_lcd(void);
50 hedin 148 void convertTemp(void);
51     void timer1_interrupt(void);
52 hedin 150 void on_recieve(void);
53     void on_initial_recieve(void);
54 hedin 155 void sms_recieved(void);
55 hedin 163 void send_sms(const unsigned char* payload);
56 hedin 137
57 hedin 150
58     void reset_recieve_buffer(void)
59 hedin 109 {
60 hedin 155 global_recieve_done = 0;
61    
62 hedin 160 global_message_buffer_length1 = 0;
63     //global_message_buffer1[0] = 0;
64     global_message_buffer_length2 = 0;
65     //global_message_buffer2[0] = 0;
66 hedin 155
67 hedin 160 memset(global_message_buffer1, 0, BUFFER);
68     memset(global_message_buffer2, 0, BUFFER);
69 hedin 150 }
70    
71 hedin 160 void pic18_io_init(void)
72     {
73     TRISA0 = 1; // analog input
74     TRISA5 = 0; // Output
75     TRISB1 = 1; // TRISB1-4 Digital input
76     TRISB2 = 1;
77     TRISB3 = 1;
78     TRISB4 = 1;
79     }
80 hedin 150
81 hedin 160
82    
83 hedin 150 void ad_init(void) // Nicked from H7
84     {
85 hedin 109 // AD Conversion clock
86     ADCS0 = 0;
87     ADCS1 = 0;
88     ADCS2 = 0;
89 hedin 82
90 hedin 109 //Select AN0/RA0 for AD source
91     // In this (000) setup, it's only RA0/AN0 that does ad convertion.
92     CHS0=0;
93     CHS1=0;
94     CHS2=0;
95    
96     //Only AN0 is selected for AD and with Vdd/Vss as limits
97     PCFG0=0;
98     PCFG1=1;
99     PCFG2=1;
100     PCFG3=1;
101    
102 hedin 148 //Result is right justified
103     ADFM=1;
104 hedin 109
105     //Fire up for A/D converter module
106     ADON=1;
107     }
108    
109 hedin 82 void rs232_init(void)
110     {
111     SPEN = 0; // Serial Port Enable Bit... 0 = disabled
112 hedin 137 TRISC6 = 0;
113 hedin 82 TRISC7 = 1;
114 hedin 137
115     SPBRG = 207; // 1200 baud rate... 25 = 9600
116     // x = (Fosc / (16*[baud rate]) )-1
117     TXSTA = 0x24; // Enables BRGH and TXEN inthe TXSTA register
118     RCSTA = 0x90; // 0x90 enables SPEN and CREN in the RCSTA register
119 hedin 82 }
120    
121     void interrupt_init(void)
122     {
123     // Assumes that all interrupts default is 0
124     PEIE = 1;
125     GIE = 1;
126 hedin 137 RCIE = 1; // Recieve interrupt enable.
127     IPEN = 0; // Nfo interrupt priority
128     TXIE = 0; // Serial interrupt enabled
129 hedin 148 TMR1IE = 1; // Enables timer 1
130 hedin 82 }
131    
132 hedin 148 void timer_init(void)
133     {
134     TMR1CS = 1; //use external clock
135    
136     T1CKPS1 = 1; //1:8 prescale
137     T1CKPS0 = 1;
138    
139     TMR1H = 0xEF;
140     TMR1L = 0xFF;
141    
142     T1OSCEN = 1; //enable oscillator circuit
143     RD16 = 0; //normal 8 bit writes
144     TMR1ON = 1;
145 hedin 82 }
146    
147 hedin 137 void sms_init(void)
148 hedin 82 {
149 hedin 148 int i;
150 hedin 160 char buf[2];
151 hedin 148
152 hedin 160
153 hedin 150 reset_recieve_buffer();
154 hedin 148 sprintf(global_serial_send,"at+cgsn\r");
155     serial_send();
156     DelaySek(1);
157 hedin 160 //while(!global_recieve_done) ;
158    
159     while (global_has_imei == 0)
160     {
161     if (strstr(global_message_buffer1,"OK") != 0)
162     {
163     global_imei_tversum = 0;
164     for (i=0; i<15; ++i)
165     {
166     buf[0] = global_message_buffer2[i];
167     buf[1] = 0;
168     global_imei_tversum += atoi(buf);
169     //global_imei_tversum += (global_message_buffer2[i] - '0');
170     }
171    
172     global_has_imei = 1;
173     }
174     }
175 hedin 148
176    
177 hedin 139 sprintf(global_serial_send,"%s", "at+cpin=8043\r");
178     serial_send();
179    
180 hedin 150 while(global_modem_init == 0)
181     {
182     on_initial_recieve();
183     }
184 hedin 160 DelayMs(5);
185    
186     reset_recieve_buffer();
187     sprintf(global_serial_send, "at+cmgd=1,4\r");
188     serial_send();
189     while ( strstr(global_message_buffer1,"OK") == 0) ;
190 hedin 82 }
191    
192 hedin 148
193 hedin 82 void interrupt interrupt_handler(void)
194     {
195 hedin 109 // Finds out what interrupt have been trigged, and starts the respective function.
196 hedin 137 if(RCIF == 1) // Serial recieve interrupt
197 hedin 82 {
198 hedin 137 serial_recieved();
199 hedin 82 RCIF = 0;
200     }
201 hedin 148
202     if(TMR1IF == 1) // timer1 interrupt trigger.
203     {
204     timer1_interrupt();
205     TMR1IF = 0;
206     }
207 hedin 82 }
208 hedin 109
209 hedin 137
210     void serial_send(void)
211     {
212     int i;
213 hedin 148 char data_byte;
214 hedin 150 for(i = 0; i < BUFFER; i++)
215 hedin 109 {
216 hedin 148 data_byte = global_serial_send[i];
217     if( data_byte == '\r')
218 hedin 150 i = (BUFFER - 1);
219 hedin 148 TXREG = data_byte;
220 hedin 139 while(TRMT==0) ;
221 hedin 148 DelayMs(10);
222 hedin 109 }
223 hedin 163 DelayMs(150);
224 hedin 160 global_serial_send[0] = 0;
225 hedin 163 DelayMs(150);
226 hedin 137 }
227    
228 hedin 160 ///////////////////////////////////////////////////////////////////////////////////////////////////
229     // Takes the recieving data and fills it in a buffer, when we meet a '\r' we fill the data from,
230     // global_serial_recieve_buffer, to global_message_buffer1, and sets the flag global_recieve_done.
231     void serial_recieved(void)
232 hedin 137 {
233 hedin 162 char data_byte;
234 hedin 137
235 hedin 148 data_byte = RCREG;
236 hedin 137
237 hedin 160 if (data_byte == '\n') // Cant be bothered to do anyting if the byte is a '\n'.
238 hedin 150 return;
239 hedin 151
240 hedin 160 if (global_serial_byte_counter == 0 && data_byte == '\r') // don't care about '\r', if it's the first byte we recieve.
241 hedin 151 return;
242 hedin 150
243 hedin 160
244 hedin 162 if ( global_serial_byte_counter < BUFFER) //Prevent buffer overrun
245     global_serial_recieve_buffer[ global_serial_byte_counter++ ] = data_byte; // fills the data_byte into our buffer.
246 hedin 160
247    
248 hedin 165
249 hedin 160 if (data_byte == '\r') // when we meet a '\r', the transmission is done, and we fill the constxt of
250     // global_message_buffer1 into global_message_buffer2 our main buffer into
251     // global_message_buffer2, and the same with our recieve buffer, global_serial_recieve_buffer
252     // into global_message_buffer1.
253 hedin 137 {
254 hedin 160 global_recieve_done = 1; // indicates the recieve transmission is done.
255 hedin 155 global_serial_recieve_buffer[global_serial_byte_counter] = 0; //zero terminate
256    
257 hedin 160 // global_message_buffer1 -> global_message_buffer2
258     strcpy(global_message_buffer2, global_message_buffer1);
259     global_message_buffer_length2 = global_message_buffer_length1;
260    
261     // global_serial_recieve_buffer -> global_message_buffer1
262     strcpy(global_message_buffer1, global_serial_recieve_buffer);
263     global_message_buffer_length1 = global_serial_byte_counter;
264    
265    
266     global_serial_byte_counter = 0;
267 hedin 137 }
268     }
269 hedin 148
270     void timer1_interrupt(void)
271     {
272     TMR1H = 0xEF;
273     TMR1L = 0xFF;
274     global_time_counter++;
275     global_emergency_counter++;
276     RA1 = !RA1;
277     }
278    
279 hedin 139 void update_lcd(void)
280     {
281 hedin 165 if(global_temp_update_display != global_time_counter)
282     {
283     lcd_clear();
284     lcd_goto(0x00);
285 hedin 151
286 hedin 165 sprintf(global_lcd_buf, "Temp: %3d", global_temp);
287     lcd_puts(global_lcd_buf);
288     global_temp_update_display = global_time_counter;
289     }
290 hedin 148
291     }
292    
293     void send_update(void)
294     {
295 hedin 160 char update[40];
296     sprintf(update, "%d:%d:%d:%d:%d:%d", global_sms_counter, global_temp, FIREDET, EMPTYTANK, FEEDING, PWRFAIL);
297     send_sms(update);
298    
299     global_sms_counter++;
300     }
301    
302 hedin 163 void send_sms(const unsigned char* payload)
303 hedin 160 {
304     sprintf(global_serial_send, "at+cmgs=\"%s\"\r", global_cell_nr);
305 hedin 148 serial_send();
306 hedin 160 sprintf(global_serial_send, "%s%c", payload, 0x1A);
307 hedin 148 serial_send();
308 hedin 163 DelayMs(150);
309 hedin 160
310 hedin 148 }
311    
312     void convertTemp()
313     {
314     short adVal;
315     adVal = (global_Pot_Hi << 8) | global_Pot_Lo;
316     if( adVal >=840 )
317     global_temp = 100;
318     else
319 hedin 160 global_temp = (unsigned char) (adVal / 8.3886);
320 hedin 148 }
321    
322    
323    
324     void eeprom_writer(void)
325     {
326     char len,i;
327    
328 hedin 160 len = strlen(global_cell_nr);
329 hedin 148 eeprom_write(0, (global_time_interval/60));
330     eeprom_write(1, global_sms_counter>>8);
331     eeprom_write(2, global_sms_counter);
332     eeprom_write(3, len);
333    
334     for (i=0; i<len; ++i)
335 hedin 139 {
336 hedin 160 eeprom_write(i+4, global_cell_nr[i] );
337 hedin 139 }
338     }
339 hedin 148
340     void eeprom_reader(void)
341     {
342     char len,i;
343 hedin 109
344 hedin 148 global_time_interval = eeprom_read(0);
345     global_time_interval *= 60;
346     global_sms_counter = (eeprom_read(1)<<8) | eeprom_read(2);
347     len = eeprom_read(3);
348    
349     for (i=0; i<len; ++i)
350     {
351 hedin 160 global_cell_nr[i] = eeprom_read(i+4);
352 hedin 148 }
353    
354 hedin 160 global_cell_nr[i] = 0; //zero terminated!
355 hedin 148 }
356    
357 hedin 150 void on_initial_recieve(void)
358     {
359     char imei[16];
360     char* ptr;
361     char i;
362 hedin 160 char buf[2];
363 hedin 150
364     if (strstr(global_serial_recieve_buffer,"+WIND: 11") != 0)
365     {
366     global_modem_init = 1;
367    
368     ptr = strstr(global_serial_recieve_buffer,"cgsn");
369     ptr +=4;
370     strncpy(imei, ptr,15);
371     imei[15] = 0;
372    
373     reset_recieve_buffer();
374     }
375     }
376    
377 hedin 148 void on_recieve(void)
378     {
379 hedin 160 char tmp[3];
380 hedin 151 char* ptr;
381     tmp[0]=0;
382    
383 hedin 160 if (global_recieve_done == 0 || global_message_buffer_length1 == 0)
384 hedin 150 return;
385 hedin 151
386 hedin 160 if (strstr(global_message_buffer1,"CMTI") != 0) // here we handles a incomming SMS
387     {
388     ptr = strstr(global_message_buffer1,","); // finds the point just before the nr. of the SMS.
389     strcat(tmp,ptr+1); // puts that number in tmp
390     global_sms_recieve_number[0] = 0; // wanna be sure that we write the new number from global_sms_recieve_number[0]
391     strcat(global_sms_recieve_number, tmp); // puts the sms number into the global variable.
392     sms_recieved();
393     }
394     reset_recieve_buffer();
395     }
396    
397     void sms_recieved(void)
398     {
399     char buf[4];
400     char i,imei;
401     char pos;
402 hedin 151
403 hedin 160 sprintf(global_serial_send, "AT+CMGR=%s\r", global_sms_recieve_number); // formates the variable that sends commands to the SMS modem.
404     serial_send(); // Sends the command.
405    
406     while(strstr(global_message_buffer1, "OK") == 0) // stays here until we recieve a "OK" from the modem.
407     DelayMs(1);
408    
409     DelayUs(10);
410    
411     for (i=0; global_message_buffer2[i] != ':' && global_message_buffer2[i] != 0; ++i)
412 hedin 151 {
413 hedin 160 buf[i] = global_message_buffer2[i];
414 hedin 150 }
415 hedin 160
416     buf[i] = 0;
417     imei = atoi(buf);
418    
419     if (imei == global_imei_tversum)
420 hedin 155 {
421 hedin 160 i++; //spring over ':'
422     pos = 0;
423     for ( ; global_message_buffer2[i] != ':'; ++i, ++pos)
424     {
425     global_cell_nr[pos] = global_message_buffer2[i];
426     }
427     global_cell_nr[pos] = 0; //zero terminator
428    
429     i++; //spring over ':'
430     pos=0;
431     for ( ; global_message_buffer2[i] ; ++i,++pos)
432     {
433     buf[pos] = global_message_buffer2[i];
434     }
435     buf[pos]=0;
436    
437     global_time_interval = atoi(buf);
438     global_time_interval *= 60;
439     eeprom_writer(); // writes the new cell nr. and time interval to the eeprom.
440     send_sms("conf ok");
441 hedin 155 }
442 hedin 148
443    
444 hedin 155 }
445    
446 hedin 160
447    
448 hedin 82 void main()
449     {
450 hedin 148 ////////////////////
451     // Running Init's //
452 hedin 109
453     // Running init for various components.
454 hedin 148 pic18_io_init();
455 hedin 160 RA5 = 1; // Indicates that the board is running inits.
456    
457 hedin 148 rs232_init();
458 hedin 109 ad_init();
459     lcd_init(0);
460 hedin 137 interrupt_init();
461 hedin 139 sms_init();
462 hedin 148 //eeprom_init();
463     timer_init();
464     eeprom_reader();
465     ///////////////
466     // Main loop //
467 hedin 150
468 hedin 151 DelayMs(50);
469     reset_recieve_buffer();
470 hedin 150
471 hedin 160 RA5 = 0; // Inits are done, and RA1 will now work as a error notifier.
472    
473 hedin 109 while(1)
474 hedin 100 {
475 hedin 148 // If there happends to be a critical state on the system, we send a sms.
476     if( (global_temp >= 90 || PWRFAIL == 1 || FIREDET == 0 || FEEDING == 1 || EMPTYTANK == 1) && global_emergency_counter >= 600 )
477     {
478     send_update();
479     global_emergency_counter = 0;
480     }
481     // Every X sec. a status sms is send.
482     if(global_time_counter >= 3600)
483     {
484     send_update();
485     global_time_counter = 0;
486     }
487     // To avoid buffer overrun.
488     if( global_emergency_counter > 7200 )
489     global_emergency_counter = 600;
490    
491     // Checks if there has been recieved a config sms.
492     if(global_interval_changed )
493     {
494     eeprom_writer();
495     global_interval_changed = 0;
496     }
497 hedin 109 // Checking if A/D convertion is done, and save the data in global_Pot_??
498     if(GODONE==0)
499     {
500     global_Pot_Hi = ADRESH;
501     global_Pot_Lo = ADRESL;
502 hedin 148 convertTemp();
503 hedin 165 update_lcd();
504 hedin 112 GODONE = 1;
505 hedin 109 }
506 hedin 151 // Handels the recieve sms'es.
507     on_recieve();
508 hedin 100 }
509 hedin 82 }

  ViewVC Help
Powered by ViewVC 1.1.20