Subversion Repositories svnkaklik

Rev

Rev 181 | Go to most recent revision | Details | Compare with Previous | Last modification | View Log

Rev Author Line No. Line
141 kakl 1
#include ".\camerus.h"
136 kakl 2
 
154 kakl 3
#USE FAST_IO (C)     // Brana C je ve FAST_IO modu, aby slo rychle cteni z kamery
136 kakl 4
 
180 kakl 5
// A/D vstupy
6
#define  DALKOMER 4  // AN4/RA5 - dalkomer na cihlu
7
#define  CERNA    3  // AN3/RA3
8
#define  CERVENA  2  // AN2/RA2 - cervene kroutitko
9
#define  ZELENA   1  // AN1/RA0 - zelene kroutitko
10
#define  MODRA    0  // AN0/RA1 - modre kroutitko
150 kakl 11
 
12
// I/O
154 kakl 13
#define HREF   PIN_C5      // Signal HREF z kamery (v H po celou dobu radku)
14
#define PIX    PIN_C6      // Vstup pro body z kamery (za trivstupim hradlem OR (dig. komparator))
176 kakl 15
#define SERVO  PIN_B4      // Vystup na servo (1 az 2ms po cca 20ms (synchronizovano snimkovym kmitoctem))
154 kakl 16
#define MOT_L  PIN_B5      // Smer otaceni leveho motoru; druhy pol je RC2
17
#define MOT_R  PIN_B6      // Smer otaceni praveho motoru; druhy pol je RC1
179 kakl 18
#define DATA   PIN_B2      // K modulu LEDbar data
19
#define CP     PIN_B1      // K modulu LEDbar hodiny
180 kakl 20
#define ODO    PIN_A4      // Ze snimace z odometrie z praveho kola
184 kakl 21
//#define CIHLA  PIN_B0      // Vstup INT, generuje preruseni pri prekazce
176 kakl 22
 
179 kakl 23
#define CASMIN 6           // Rozsah radku snimace
24
#define CASMAX 192
25
#define CASAVR ((CASMAX+CASMIN) / 2)
26
 
184 kakl 27
#byte INTCON      = 0x0B         // Interrupt configuration register
28
   #bit GIE       = INTCON.7
29
   #bit PEIE      = INTCON.6
30
   #bit TMR0IE    = INTCON.5
31
   #bit INT0IE    = INTCON.4
32
   #bit RBIE      = INTCON.3
33
   #bit TMR0IF    = INTCON.2
34
   #bit INT0IF    = INTCON.1
35
   #bit RBIF      = INTCON.0
36
 
37
enum stavy {start,rozjezd,jizda,cihla,cil};
38
stavy stav;    // Kde jsme na trati
39
int8 cas;      // Cas hrany bila/cerna v radce
40
 
41
void SetServo(int8 angle)
42
{
43
   int8 n, offset;
44
 
45
   for(n=0; n<20; n++)
46
   {
47
      set_adc_channel(MODRA);    // Kroutitko na vystredeni predniho kolecka
48
      Delay_ms(1);
49
      offset=read_adc();
50
 
51
      output_high(SERVO);        // Odvysilani impuzu 1 az 2ms pro servo
52
      delay_us(1000);
53
      delay_us(offset);
54
      delay_us(offset);
55
      delay_us(offset);
56
      delay_us(angle);
57
      delay_us(angle);
58
      output_low(SERVO);
59
      delay_ms(18);
60
   }   
61
}
62
 
63
#int_EXT
64
EXT_isr()   // Preruseni od prekazky
65
{
66
   set_pwm1_duty(0);    // reverz (zabrzdi)
67
   set_pwm2_duty(0);
68
   output_high(MOT_L);
69
   output_high(MOT_R);
70
   delay_ms(400);
71
   output_low(MOT_L); // zastav
72
   output_low(MOT_R);
73
   if (stav==cihla) while(true); // Zastav na furt
74
   if(stav==jizda)
75
   {
76
      int n;
77
 
78
      SetServo(CASAVR-CASMIN);      
79
      set_pwm1_duty(40); // pomalu couvej
80
      set_pwm2_duty(40);
81
      output_high(MOT_L);
82
      output_high(MOT_R);
83
      n=0;
84
      while(true)
85
      {
86
         while(input(ODO));
87
         while(!input(ODO));
88
         n++;
89
         if(n==6) break;
90
      }
91
      set_pwm1_duty(255);    // reverz (zabrzdi)
92
      set_pwm2_duty(255);
93
      output_low(MOT_L);
94
      output_low(MOT_R);
95
      delay_ms(100);
96
      set_pwm1_duty(0);  // Zastav
97
      set_pwm2_duty(0);
98
      output_low(MOT_L);
99
      output_low(MOT_R);
100
      delay_ms(1000);
101
 
102
      SetServo((CASAVR-CASMIN)-20);   // doleva        
103
      set_pwm1_duty(150);  // vpred
104
      set_pwm2_duty(200);
105
      output_low(MOT_L);
106
      output_low(MOT_R);
107
      stav=cihla;
108
      n=0;
109
      while(true)
110
      {
111
         while(input(ODO));
112
         while(!input(ODO));
113
         n++;
114
         if(n==10) break;
115
      }
116
      set_pwm1_duty(0);    // reverz (zabrzdi)
117
      set_pwm2_duty(0);
118
      output_high(MOT_L);
119
      output_high(MOT_R);
120
 
121
      SetServo((CASAVR-CASMIN));   // rovne        
122
      set_pwm1_duty(140);  // vpred
123
      set_pwm2_duty(140);
124
      output_low(MOT_L);
125
      output_low(MOT_R);
126
      stav=cihla;
127
      n=0;
128
      while(true)
129
      {
130
         while(input(ODO));
131
         while(!input(ODO));
132
         n++;
133
         if(n==10) break;
134
      }
135
      set_pwm1_duty(0);    // reverz (zabrzdi)
136
      set_pwm2_duty(0);
137
      output_high(MOT_L);
138
      output_high(MOT_R);
139
      delay_ms(200);
140
      set_pwm1_duty(150);  // Zastav
141
      set_pwm2_duty(150);
142
      output_high(MOT_L);
143
      output_high(MOT_R);
144
      delay_ms(1000);
145
 
146
      SetServo(CASMIN);   // max. doleva      
147
      set_pwm1_duty(0);  // vzad
148
      set_pwm2_duty(20);
149
      output_low(MOT_L);
150
      output_high(MOT_R);
151
      stav=cihla;
152
      n=0;
153
      while(true)
154
      {
155
         while(input(ODO));
156
         while(!input(ODO));
157
         n++;
158
         if(n==10) break;
159
      }
160
      set_pwm1_duty(0);    // reverz (zabrzdi)
161
      set_pwm2_duty(255);
162
      output_low(MOT_L);
163
      output_low(MOT_R);
164
      delay_ms(100);
165
      set_pwm1_duty(0);  // Zastav
166
      set_pwm2_duty(0);
167
      output_low(MOT_L);
168
      output_low(MOT_R);
169
      delay_ms(500);
170
 
171
 
172
      SetServo((CASAVR-CASMIN)+5);   // mirne doprava      
173
      set_pwm1_duty(180);  // vpred
174
      set_pwm2_duty(180);
175
      output_low(MOT_L);
176
      output_low(MOT_R);
177
      stav=cihla;
178
      n=0;
179
      while(true)
180
      {
181
         while(input(ODO));
182
         while(!input(ODO));
183
         n++;
184
         if(n==10) break;
185
      }
186
      set_pwm1_duty(0);  // Zastav
187
      set_pwm2_duty(0);
188
      output_low(MOT_L);
189
      output_low(MOT_R);
190
      delay_ms(500);
191
 
192
 
193
      cas=CASMIN;  // Cara je vlevo
194
 
195
      stav=cihla;
196
   }
197
}
198
 
179 kakl 199
// Zobrazeni jednoho byte na modulu LEDbar
176 kakl 200
void disp(int8 x)
201
{
202
   int n;
203
 
204
   x=~x;
205
   for(n=0;n<=7;n++)
206
   {
207
      if (x & 1 == 1) output_high(DATA); else output_low(DATA);
208
      output_high(CP);
209
      x>>=1;
210
      output_low(CP);
211
   }
212
}
213
 
179 kakl 214
// Blikani LEDbarem ve stilu Night Rider
184 kakl 215
void NightRider(int8 x)
176 kakl 216
{
217
   int n,i,j;
218
 
219
   for(j=0;j<x;j++)
220
   {
221
      i=0x01;
222
      for(n=0;n<7;n++)
223
      {
224
         disp(i);
225
         rotate_left(&i, 1);
184 kakl 226
         delay_ms(40);
176 kakl 227
      }
228
      for(n=0;n<7;n++)
229
      {
230
         disp(i);
231
         rotate_right(&i, 1);
184 kakl 232
         delay_ms(40);
176 kakl 233
      }
234
   }
235
   disp(i);
184 kakl 236
   delay_ms(40);
176 kakl 237
   i=0;
238
   disp(i);
239
}
240
 
184 kakl 241
 
136 kakl 242
void main()
243
{
154 kakl 244
   int8 offset;   // Promena pro ulozeni ovsetu
245
   int8 rr;       // Promenna na ulozeni Rozumne rychlost
246
   int8 r1;       // Rychlost motoru 1
247
   int8 r2;       // Rychlost motoru 2
180 kakl 248
   int16 trasa;   // Pocitadlo ujete vzdalenosti
136 kakl 249
 
154 kakl 250
   setup_adc_ports(ALL_ANALOG);              // Zapnuti A/D prevodniku pro cteni kroutitek
136 kakl 251
   setup_adc(ADC_CLOCK_INTERNAL);
154 kakl 252
   setup_timer_0(RTCC_INTERNAL|RTCC_DIV_1);  // Casovac pro mereni casu hrany W/B v radce
136 kakl 253
   setup_timer_1(T1_DISABLED);
154 kakl 254
   setup_timer_2(T2_DIV_BY_16,255,1);        // Casovac PWM motoru
255
   setup_ccp1(CCP_PWM); // RC1               // PWM pro motory
136 kakl 256
   setup_ccp2(CCP_PWM); // RC2
176 kakl 257
   setup_comparator(NC_NC_NC_NC);
136 kakl 258
   setup_vref(FALSE);
259
 
154 kakl 260
   set_tris_c(0b11111000);     // Nastaveni vstup/vystup pro branu C, protoze se nedela automaticky
176 kakl 261
 
153 kakl 262
   set_pwm1_duty(0);   // Zastav motory
263
   set_pwm2_duty(0);
264
   output_low(MOT_L);
265
   output_low(MOT_R);
266
 
176 kakl 267
   NightRider(1);    // Aby se poznalo, ze byl RESET
268
                     // Musi se pockat, nez se rozjede kamera, nez se do ni zacnou posilat prikazy
136 kakl 269
 
176 kakl 270
   //... Nastaveni kamery ...
154 kakl 271
   i2c_start();      // Soft RESET kamery
272
   i2c_write(0xC0);        // Pro single slave musi mit vsechny zapisy adresu C0h
273
   i2c_write(0x12);        // Adresa registru COMH
274
   i2c_write(0x80 | 0x24); // Zapis ridiciho slova
136 kakl 275
   i2c_stop();
276
 
277
   i2c_start();      // BW
278
   i2c_write(0xC0);
279
   i2c_write(0x28);
280
   i2c_write(0b01000001);
281
   i2c_stop();
282
 
176 kakl 283
/*
136 kakl 284
   i2c_start();      // Contrast
285
   i2c_write(0xC0);
286
   i2c_write(0x05);
176 kakl 287
   i2c_write(0xA0);  // 48h
136 kakl 288
   i2c_stop();
181 kakl 289
 
136 kakl 290
   i2c_start();      // Brightness
291
   i2c_write(0xC0);
292
   i2c_write(0x06);
181 kakl 293
   i2c_write(0x0);  // 80h //9Ah
136 kakl 294
   i2c_stop();
295
 
296
   i2c_start();      // Band Filter
297
   i2c_write(0xC0);
298
   i2c_write(0x2D);
299
   i2c_write(0x04 | 0x03);
300
   i2c_stop();
301
*/
181 kakl 302
 
303
   i2c_start();      // Fame Rate
304
   i2c_write(0xC0);
305
   i2c_write(0x2B);
306
   i2c_write(0x00);  // cca 17ms (puvodni hodnota 5Eh = 20ms)
307
   i2c_stop();
308
 
136 kakl 309
   i2c_start();      // VSTRT
310
   i2c_write(0xC0);
311
   i2c_write(0x19);
181 kakl 312
   i2c_write(118);   // prostredni radka
136 kakl 313
   i2c_stop();
314
 
315
   i2c_start();      // VEND
316
   i2c_write(0xC0);
317
   i2c_write(0x1A);
181 kakl 318
   i2c_write(118);
136 kakl 319
   i2c_stop();
320
 
179 kakl 321
   NightRider(1);    // Musi se dat cas kamere na AGC a AEC
152 kakl 322
 
181 kakl 323
{ // Kalibrace kamery
324
   int8 t1,t2;
184 kakl 325
 
181 kakl 326
   for(offset=0;offset<=255;offset+=0x04) // Cita porad dokola
327
   {
328
      i2c_start();      // Brightness
329
      i2c_write(0xC0);
330
      i2c_write(0x06);
331
      i2c_write(offset);  // 80h default
332
      i2c_stop();
333
      disp(offset);
184 kakl 334
      delay_ms(50);
181 kakl 335
 
336
      t1=0;
337
      t2=0;
338
      while(!input(HREF));    // Cekej nez se zacnou posilat pixely z radky
184 kakl 339
      delay_ms(5);
181 kakl 340
      while(!input(HREF));    // Cekej nez se zacnou posilat pixely z radky
341
      set_timer0(0);          // Vynuluj pocitadlo casu
342
      if(!input(PIX)) continue;
343
      while(input(PIX));
344
      t1=get_timer0();    // Precti cas z citace casu hrany
184 kakl 345
      set_timer0(0);          // Vynuluj pocitadlo casu
181 kakl 346
      while(!input(PIX));
184 kakl 347
      t2=get_timer0();
348
 
181 kakl 349
      if((t1>60) && (t1<140) && (t2>5) && (t2<=10)) break;
184 kakl 350
 
181 kakl 351
      delay_ms(2);
352
   }
353
}
354
 
355
delay_ms(1000);
356
 
184 kakl 357
   cas=CASAVR-CASMIN;  // Inicializace promenych, aby neslo servo za roh a aby se to rozjelo jeste dneska
180 kakl 358
   stav=start;
359
   trasa=0;
181 kakl 360
 
154 kakl 361
   // ... Hlavni smycka ...
136 kakl 362
   while(true)
363
   {
179 kakl 364
      int8 pom;
180 kakl 365
      int8 n;
181 kakl 366
 
179 kakl 367
      pom=0;
154 kakl 368
      while(!input(HREF));    // Cekej nez se zacnou posilat pixely z radky
369
      set_timer0(0);          // Vynuluj pocitadlo casu
370
      while(input(HREF))      // Po dobu vysilani radky cekej na hranu W/B
149 kakl 371
      {
179 kakl 372
// !!!!Dodelat rozpoznani cerne cary napric pro zastaveni
184 kakl 373
         if(!input(PIX))   // Pokud se Xx za sebou precetla CERNA
152 kakl 374
         if(!input(PIX))
181 kakl 375
//         if(!input(PIX))
152 kakl 376
         {
176 kakl 377
            pom=get_timer0();    // Precti cas z citace casu hrany
152 kakl 378
            break;
379
         };
149 kakl 380
      };
179 kakl 381
      while(input(HREF));      // Pockej na shozeni signalu HREF
150 kakl 382
 
179 kakl 383
      if((pom<CASMAX) && (pom>CASMIN)) cas=pom;  // Orizni konce radku
384
      // Na konci obrazovaho radku to blbne. Jednak chyba od apertury
385
      // a vubec to nejak na kraji nefunguje.
386
 
154 kakl 387
      set_adc_channel(MODRA);    // Kroutitko na vystredeni predniho kolecka
150 kakl 388
      Delay_ms(1);
389
      offset=read_adc();
176 kakl 390
 
154 kakl 391
      output_high(SERVO);        // Odvysilani impuzu 1 az 2ms pro servo
152 kakl 392
      delay_us(1000);
150 kakl 393
      delay_us(offset);
152 kakl 394
      delay_us(offset);
184 kakl 395
      delay_us(offset);
149 kakl 396
      delay_us(cas);
397
      delay_us(cas);
152 kakl 398
      output_low(SERVO);
154 kakl 399
 
179 kakl 400
      set_adc_channel(ZELENA);   // Kroutitko pro vykon motoru
153 kakl 401
      Delay_ms(1);
179 kakl 402
      rr=read_adc()>>2; //!!! // 0-31
176 kakl 403
 
179 kakl 404
      // Elektronicky diferencial
180 kakl 405
      if(cas<CASAVR) {r1=cas-CASMIN; r2=CASAVR-CASMIN;}; // Normovani vystupni hodnoty radkoveho snimace
179 kakl 406
      if(cas==CASAVR) {r1=cas-CASMIN; r2=cas-CASMIN;};   // pro rizeni rychlosti motoru
407
      if(cas>CASAVR) {r1=CASAVR-CASMIN; r2=CASMAX-cas;}; // Rozsah 1 az 92
180 kakl 408
 
179 kakl 409
      if (r1>(CASAVR-CASMIN-rr)) r1=(r1<<1)+rr-(CASAVR-CASMIN);     // Neco jako nasobeni
180 kakl 410
      if (r2>(CASAVR-CASMIN-rr)) r2=(r2<<1)+rr-(CASAVR-CASMIN);
176 kakl 411
 
179 kakl 412
//      r1<<=1;     // Rychlost je dvojnasobna
413
//      r2<<=1;     // Rozsah 2 az 184
154 kakl 414
 
184 kakl 415
      if ((stav==jizda)||(stav==cihla))      // Jizda
154 kakl 416
      {
179 kakl 417
         set_pwm1_duty(r1);
418
         set_pwm2_duty(r2);
180 kakl 419
         trasa++;
153 kakl 420
      }
421
      else
422
      {
154 kakl 423
         set_pwm1_duty(0);       // Zastaveni
153 kakl 424
         set_pwm2_duty(0);
425
      }
180 kakl 426
 
427
      set_adc_channel(DALKOMER);   // Prepni A/D prevodnik na detektor cihly
428
      Delay_ms(1);
184 kakl 429
 
430
      if((stav==jizda)&&(trasa>50)) // musi to alespon 1s jet
180 kakl 431
      {
184 kakl 432
         ext_int_edge(H_TO_L);         // Nastav podminky preruseni od cihly
433
         INT0IF=0;                     // Zruseni predesle udalosti od startera
434
         enable_interrupts(INT_EXT);
435
         enable_interrupts(GLOBAL);
436
      };
437
 
438
      if(stav==start)
439
      if(read_adc()<128)
440
      {
441
         disp(0x55);
442
         while(read_adc()<128); // Cekej, dokud starter neda ruku pryc
443
         set_pwm1_duty(255);
444
         set_pwm2_duty(255);
445
         disp(0xAA);
446
         delay_ms(200);
447
         stav=jizda;
180 kakl 448
      }
449
 
184 kakl 450
      pom=0x80;
181 kakl 451
      for(n=CASMAX/8; n<cas; n+=CASMAX/8)
452
      {
453
         pom>>=1;
454
      }
455
      disp(pom);
184 kakl 456
 
181 kakl 457
   output_high(PIN_C0);
458
   delay_ms(1);
459
   output_low(PIN_C0);
460
 
136 kakl 461
   };
462
}