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

Annotation of /trunk/Embedded/main.c

Parent Directory Parent Directory | Revision Log Revision Log


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

  ViewVC Help
Powered by ViewVC 1.1.20