This shows you the differences between two versions of the page.
| Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
| et:avr:timers [2010/02/11 12:47] – raivo.sell | et:avr:timers [2020/07/20 12:00] (current) – external edit 127.0.0.1 | ||
|---|---|---|---|
| Line 1: | Line 1: | ||
| + | ====== Loendurid/ | ||
| + | Loendurid (inglise keeles // | ||
| + | | ||
| + | Atmeli xmega kontrollerite puhul tuleb jälgida, mis siini taimerit kasutatakse. AVR xmega taimerite registritesse kirjutamine toimib sarnaselt eelnevates peatükkides toodud näidetega. Režiimid on sarnased ning täpsem kirjeldus on toodud andmelehes. | ||
| + | |||
| + | ===== Loenduri normaalrežiim ===== | ||
| + | |||
| + | Normaalrežiimis ei täida loendur muud funktsiooni, | ||
| + | |||
| + | <box 100% round # | ||
| + | |||
| + | Vaja on 8 MHz taktsagedusel töötav ATmega128 10 ms (sagedus 100 Hz) ajavahemiku järel katkestust tekitama panna. Ülesandeks sobib 8-bitine loendur 0. | ||
| + | |||
| + | <code c> | ||
| + | #include < | ||
| + | |||
| + | ISR(TIMER0_OVF_vect) { | ||
| + | // Loendurile sellise väärtuse omistamine, | ||
| + | // et järgmine ületäitumine tekiks 10 ms pärast | ||
| + | // Valem: 256 - 8 MHz / 1024 / 100 Hz = 177,785 = ~178 | ||
| + | TCNT0 = 178; | ||
| + | } | ||
| + | |||
| + | int main(void) { | ||
| + | // Et esimene ületäitumise katkestus tekiks 10 ms pärast, | ||
| + | // tuleb ka siinkohal loendur algväärtustada | ||
| + | TCNT0 = 178; | ||
| + | // Sagedusjaguri teguriks 1024 | ||
| + | TCCR0 = 0x07; | ||
| + | // Loenduri täitumise katkestuse lubamine | ||
| + | TIMSK |= (1 << TOIE0); | ||
| + | |||
| + | // Globaalne katkestuste lubamine | ||
| + | sei(); | ||
| + | |||
| + | // Lõputu programmitsükkel | ||
| + | while (1) continue; | ||
| + | } | ||
| + | </ | ||
| + | |||
| + | Näites toodud loendurile omistatava väärtusega siiski täpselt 10 ms järel katkestust ei tekitata, sest vaja oleks loendurile omistada komakohaga väärtus, kuid see pole võimalik. Et täpset katkestuse intervalli saada, tuleb nii sagedusjaguri tegur kui ka loendurile täitumisel omistatav väärtus valida nii, et taktsagedus jaguks täpselt. Paraku pole see alati võimalik, ja eriti just 8-bitise loenduri puhul, sest selle väärtuste skaala on üsna väike. Täpsema ja suurema intervalli tekitamiseks saab kasutada 16-bitist loendurit. | ||
| + | </ | ||
| + | |||
| + | ==== Välise taktiga loendur ==== | ||
| + | |||
| + | Loenduri taktsignaalina saab kasutada ka mikrokontrollerivälist signaali (inglise keeles //external clock source//). Selleks on AVR mikrokontrolleril Tn viik, kus n tähistab loenduri numbrit. Välist taktsignaali ja polaarsust saab valida sagedusjaguri registriga. | ||
| + | |||
| + | ==== Sündmuste mõõtmine ==== | ||
| + | |||
| + | Kuna loendurid võimaldavad mõõta aega, on keerukamatel AVR mikrokontrolleritel võimalus riistvaraliselt mõõta ka aega, mil toimus mingi sündmus. Seda loenduri osa nimetatakse sündmuse püüdjaks (inglise keeles //input capture unit//). AVR-is on valida kahe sündmuse vahel: spetsiaalse sisendviigu või analoogkomparaatori võrdlustulemuse loogilise väärtuse muutus. Kui toimub valitud sündmus, kirjutatakse loenduri väärtus spetsiaalsesse registrisse, | ||
| + | |||
| + | <box 100% round # | ||
| + | Vaja on 8 MHz taktsagedusel töötava ATmega128-ga mõõta välise 122 Hz - 100 kHz loogilise nelinurksignaali sagedust 1 Hz täpsusega. Programm on tehtud 16-bitise loendur 1 sündmuste püüdjaga. | ||
| + | |||
| + | <code c> | ||
| + | #include < | ||
| + | |||
| + | unsigned long frequency; | ||
| + | |||
| + | // Sündmuse toimumise katkestus | ||
| + | ISR(TIMER1_CAPT_vect) { | ||
| + | // Loenduri nullimine | ||
| + | TCNT1 = 0; | ||
| + | |||
| + | // Tulemus on ainult siis arvestatav, kui | ||
| + | // loendur pole vahepeal üle täitunud | ||
| + | if (!(TIFR & (1 << TOV1))) | ||
| + | { | ||
| + | // Sageduse arvutamine perioodi pöördväärtusest. | ||
| + | frequency = (unsigned long)8000000 / | ||
| + | (unsigned long)ICR1; | ||
| + | } | ||
| + | else | ||
| + | { | ||
| + | // Sagedus on vähem kui 122 Hz | ||
| + | frequency = 0; | ||
| + | |||
| + | // Loenduri ületäitumise lipukese nullimine | ||
| + | TIFR &= ~(1 << TOV1); | ||
| + | } | ||
| + | } | ||
| + | |||
| + | int main(void) { | ||
| + | // Tõusva frondi registreerimine, | ||
| + | TCCR1B = (1 << ICES1) | (1 << CS10); | ||
| + | |||
| + | // Sündmuse toimumise katkestuse lubamine | ||
| + | TIMSK = (1 << TICIE1); | ||
| + | |||
| + | // Globaalne katkestuste lubamine | ||
| + | sei(); | ||
| + | |||
| + | // Lõputu programmitsükkel | ||
| + | while (1) continue; | ||
| + | } | ||
| + | </ | ||
| + | |||
| + | Programmis tekib välise signaali tõusva frondi ajal sündmuse katkestus. Katkestuse jooksul kontrollitakse, | ||
| + | </ | ||
| + | |||
| + | Sündmuste püüdmist ning nende aja registreerimist saab teha ka tarkvaraliselt. Saab kasutada väliseid või muid katkestusi ja nende tekkimise ajal lugeda loenduri väärtus. Riistvaraline sündmuste püüdmine on siiski mõeldud eeskätt programmist sõltumatuks töötamiseks ja suhteliselt lühiajaliste (või tihedate) sündmuste mõõtmiseks. | ||
| + | |||
| + | ===== Signaali genereerimine ===== | ||
| + | |||
| + | Keerukamate loenduritega saab peale signaali pikkuse mõõtmise ka signaali tekitada. Selleks on loenduril väärtuse võrdlemise üksus (inglise keeles //output compare unit//) ja võrdlustulemuse väljastusüksus (inglise keeles //compare match output unit//). Võrdlusüksusesse kuuluvad registrid sama bitilaiusega kui loendur ise ja nende registrite väärtusi võrreldakse loenduri väärtusega selle töö ajal. Hetkel, mil loenduri väärtus saab võrdseks võrdlusüksuse registri väärtusega, | ||
| + | |||
| + | Mõnedel signaali genereerimise režiimidel on määratav ka loenduri suurim väärtus - loenduri füüsiline suurus jääb küll samaks, kuid mängus on võrdlusregister, | ||
| + | |||
| + | Loendurid ja nende abil genereeritavate signaalide režiimid on ühed keerulisemad perifeeriamoodulid AVR-il. Kõigist neist kirjutamine läheks pikaks ja enamasti pole nende kasutamisel vaja kõike detailselt teada. Seetõttu on järgnevalt kirjeldatud vaid üht levinuimat PWM signaali robootikas. Ülejäänut saab juba AVR dokumentatsioonist järgi uurida. | ||
| + | |||
| + | ==== Pulsilaius-modulatsioon ==== | ||
| + | |||
| + | Pulsilaius-modulatsioon (inglise keeles //pulse width modulation//, | ||
| + | |||
| + | <box 100% round # | ||
| + | |||
| + | Vaja on 8 MHz taktsagedusel töötava ATmega128-ga genereerida kaks kiirusreguleeritavat servomootori signaali. Viiguga PB5 (OC1A) tuleb genereerida pulsipikkus 1 ms ja viiguga PB6 (OC1B) pulsipikkus 2 ms. | ||
| + | |||
| + | <code c> | ||
| + | #include < | ||
| + | |||
| + | int main(void) { | ||
| + | // Viigud väljundiks | ||
| + | DDRB |= (1 << PIN5) | (1 << PIN6); | ||
| + | |||
| + | // Väljundid A ja B võrdusmomendil madalaks, | ||
| + | // "Fast PWM" režiim, sagedusjagur 8 | ||
| + | TCCR1A = (1 << COM1A1) | (1 << COM1B1) | (1 << WGM11); | ||
| + | TCCR1B = (1 << WGM13) | (1 << WGM12) | (1 << CS11); | ||
| + | |||
| + | // Suurim loenduri väärtus. Valem: | ||
| + | // TOP = 8 MHz / 8 / 50 Hz | ||
| + | ICR1 = 20000; | ||
| + | |||
| + | // Esimese mootori poolperiood 1 ms, teisel 2 ms | ||
| + | OCR1A = 1000; | ||
| + | OCR1B = 2000; | ||
| + | |||
| + | // Lõputu programmitsükkel | ||
| + | while (1) continue; | ||
| + | } | ||
| + | </ | ||
| + | |||
| + | </ | ||