- Akku Tests und Balkonkraftwerk Speicher         
Seite 1 von 2 12 LetzteLetzte
Ergebnis 1 bis 10 von 33

Thema: Timer beim ATMega32 in C

Hybrid-Darstellung

Vorheriger Beitrag Vorheriger Beitrag   Nächster Beitrag Nächster Beitrag
  1. #1
    Neuer Benutzer Öfters hier
    Registriert seit
    05.06.2012
    Beiträge
    14
    hmm.. also wenn ich alles richtig kapiert habe, dann mal eine kleine nachfrage...

    TCCR1B |= (1<<CTC1)| (1<<CS02) | (1<<CS00);
    OCR1C = 15625;
    OCR1A = 0;
    TIMSK |= (1<<OCIE1A);
    wie ich den code verstehe:
    zeile eins ergibt den vorteiler mit wert 1024 (dann wird der interrupt ausgelößt).
    nach dem wert in zeile zwei (15625) wird wider bei null (wegen zeile drei) begonnen.
    zweile vier schaltet den interrupt wieder frei.

    berechnung:
    16Mhz / 1024 = 15625Hz
    1/15625Hz = 0,000064 Sek.

    funktion:
    der timer wir alle 0,000064 Sek. um den wert eins erhöht (+1).
    bei "OCR1C = 15625;" bedeutet das 0,000064 Sek. * 15625 = 1,00 Sek.
    der timer springt ab 15625 auf 0 und dann wird die interrupt routine ausgelößt und so geht es weiter und weiter und weiter....

    liege ich soweit richtig mit meiner interpretation?!
    und wie um gottes willen lege ich jetzt diese interrupt routine an?!

  2. #2
    Erfahrener Benutzer Robotik Einstein Avatar von wkrug
    Registriert seit
    17.08.2006
    Ort
    Dietfurt
    Beiträge
    2.214
    Also.
    In deiner ersten Zeile stellst Du den Prescaler ein und sagst dem Timer, das das TCNT Register bei einem Comparematch 0 werden soll.
    In der zweiten Zeile gibst Du an, das das Comparematchregister 1C einen Wert von 15625 haben soll.
    In der dritten Zeile weist Du dem OCR1A Register den Wert 0 zu, was aber eigentlich keinen Sinn macht.
    In der vierten Zeile gibst Du den Interrupt bei Comparematch des 1A Comparematch frei.

    Nimm Zeile 2 raus und weise in Zeile 3 dem OCR1A den Wert 15625 zu.

    Das Ganze funktioniert nun so 16MHz/1024/15625 = 1Hz. Das passt dann schon mal.

    Das anlegen eines Interrupts ist von deinem Compiler abhängig.
    Soweit ich weiß muss man bei AVR GCC die interrupt.h Library includen #include <avr/interrupt.h> und kann dann mit der Funktion.
    ISR (Vektorname)
    {
    }

    Nach der Initialisierung und vor der Hauptschleife deines Programmes musst Du mit sei(); noch global die Interrupts frei geben.
    Sonst wird niemals ein Interrupt ausgeführt.

    Den Funktionsaufruf der Interruptroutine starten.
    Da gibts aber so viele Tutorials dafür, guck da einfach mal nach.

    In diesem Interrupt würde ich nur ein Flag setzen ( Bit Variable ) und im Hauptprogramm dieses Flag wieder löschen, sobald es verarbeiet wurde.
    Dadurch bleibt die Interruptroutine sehr kurz, was ja auch erwünscht ist.

  3. #3
    Erfahrener Benutzer Robotik Einstein
    Registriert seit
    07.03.2011
    Beiträge
    1.899
    Zitat Zitat von wkrug Beitrag anzeigen
    In diesem Interrupt würde ich nur ein Flag setzen ( Bit Variable ) und im Hauptprogramm dieses Flag wieder löschen, sobald es verarbeiet wurde.
    Wozu dann überhaupt eine ISR? Dann kannst du ja gleich das Interrupt Status Bit in der Mainloop pollen.

    MfG Klebwax
    Strom fließt auch durch krumme Drähte !

  4. #4
    Neuer Benutzer Öfters hier
    Registriert seit
    05.06.2012
    Beiträge
    14
    auch nach vielen versuchen und änderungen will der code nicht. jetzt wird zu langsam gezählt

    #include <avr/interrupt.h>
    ...
    unsigned long int timerLauf= 0;
    ...
    int main (void)
    {
    TCCR1B |= (1<<CS02) | (1<<CS00);
    OCR1A = 15625;
    TIMSK |= (1<<OCIE1A);
    sei();
    ...
    }

    ISR (TIMER1_COMPA_vect)
    {
    timerLauf++;
    }
    was mache ich nur falsch?
    Geändert von tom77? (10.06.2012 um 15:37 Uhr)

  5. #5
    Erfahrener Benutzer Roboter-Spezialist Avatar von robo_tom_24
    Registriert seit
    04.02.2012
    Ort
    Burgenland, Österreich
    Beiträge
    485
    Hast du den Quarz richtig eingestellt?

    Setzt du den Timer auch wieder zurück nachdem der ISR aufgerufen wurde?
    Du musst noch ins Register TCCR1B das Bit CTC aktivieren -> Clear Timer on Compare Match -> Rücksetzen nach Abgleich

    Wahrscheinlich zählt der Timer 1 mal richtig und dann zählt er immer die vollen 65535 Schritte durch
    Also ca so:
    Timerwert: 0...........15625 *INTERRUPT* ..............65535 - 0 ........ 15625 *INTERRUPT* ...........65535 - 0 .........15625 *INTERRUPT*.... ->
    Zeiterlauf: 0s.................1s............................. ...................................4,2s........... ............................................8,4s.. ................. ->
    Variable:...0....................1................ .................................................. 2................................................. ...........3.................... ->

  6. #6
    Neuer Benutzer Öfters hier
    Registriert seit
    05.06.2012
    Beiträge
    14
    wenn ich das CTC aktiviere lässt sich der code nicht kompilieren

    TCCR1B |= (1<<CTC) | (1<<CS02) | (1<<CS00);
    error: 'CTC' undeclared (first use in this function)

    achja: quarz ist richtig eingestellt (getestet mit _delay_ms(xx) -> funktioniert exakt, auch gemessen mit dem oszi)

    Wie genau meinst du das?
    Setzt du den Timer auch wieder zurück nachdem der ISR aufgerufen wurde?
    Geändert von tom77? (10.06.2012 um 19:50 Uhr)

  7. #7
    Erfahrener Benutzer Roboter-Spezialist Avatar von robo_tom_24
    Registriert seit
    04.02.2012
    Ort
    Burgenland, Österreich
    Beiträge
    485
    Ok...
    CTC heißt das der Timer nach dem Match - also der Übereinstimmung - zurückgesetzt wird

    Das kannst du aber auch händisch machen inem du in deiner ISR den Timer wieder auf 0 zurück setzt:
    Code:
    TCNT1=0; //Timer-Wert-Register auf 0 setzen

  8. #8
    Erfahrener Benutzer Robotik Einstein Avatar von wkrug
    Registriert seit
    17.08.2006
    Ort
    Dietfurt
    Beiträge
    2.214
    Ohne CTC läuft der Timer nicht richtig.
    Das CTC bedeutet, das das TCNT1 Register bei einem Comparematch auf 0 gesetzt wird.
    Wenn das nicht eingestellt ist wird der Timer über den Overflow springen und dann erst wieder beim Zählerstand von 15625 wieder einen Interrupt auslösen.
    Das bedeutet der Interrupt wird nur alle ca 4 Sekunden angesprungen.
    Da gibt es mehrere Lösungen.
    Entweder du zählst bei einem aufgetretenen Comparematch Interrupt ( in diesem natürlich ) einfach die 15625 zum Comparematchwert dazu.
    OCR1A+=15625;
    Das bewirkt, das beim Zählerstand von 31250 wieder ein Interrupt ausgelöst wird.
    Keine Angst bei den Überläufen ( >65535 ) Werden die Überlaufenden Bits gekillt, weil das ja nur ein 16Bit Register ist.
    Oder du aktivierst das CTC Bit im Register, das anscheinend in der Library anders heisst, oder nicht definiert ist.
    Guck auch mal ins Datenblatt unter Timer 1 WGM10 bis WGM13 bzw. CTC1.

    Das mit dem TCNT1=0 würde ich nicht machen, weil dabei ein kompletter Zählersprung "vergessen" werden könnte und deine Uhr damit auf Dauer falsch geht.
    Das ist halt eine Sache wie lange deine Controller sich in anderen Interupts aufhält, wie lange diese Interrupts sind und welche Prescaler man benutzt.
    Geändert von wkrug (10.06.2012 um 22:13 Uhr)

  9. #9
    Hi
    du machst den Fehler das du Timer1 mit parameter von Timer0 lädst
    so gehts - nur Quartz anpassen
    #include <avr/io.h>
    #include <avr/interrupt.h>
    #define F_CPU 5120000
    #include <util/delay.h>
    #include <avr/pgmspace.h>
    #include <stdio.h>
    #include <inttypes.h>

    ISR(TIMER1_COMPA_vect)
    {
    PORTA ^= (1<<PA1);
    }

    int main(void)
    {
    TCCR1B |= (1<<WGM12)| (1<<CS12) | (1<<CS10);
    OCR1A = 5000;
    TIMSK |= (1 << TOIE1);
    TIMSK |= (1<<OCIE1A);
    sei();

    while(1);
    }

  10. #10
    Neuer Benutzer Öfters hier
    Registriert seit
    05.06.2012
    Beiträge
    14
    danke. inzwischen habe ich aber schon die lösung für mein problem gefunden.

Seite 1 von 2 12 LetzteLetzte

Ähnliche Themen

  1. Timer beim atmega8
    Von MrTaco im Forum C - Programmierung (GCC u.a.)
    Antworten: 1
    Letzter Beitrag: 19.05.2010, 15:28
  2. ausschaltverzögerung beim einschalten (timer)
    Von Warloxx im Forum Elektronik
    Antworten: 7
    Letzter Beitrag: 03.09.2009, 19:55
  3. Timer => Fehlermeldungen beim compilieren
    Von The Man im Forum Assembler-Programmierung
    Antworten: 5
    Letzter Beitrag: 14.02.2007, 11:07
  4. Timer/Counter0 beim Butterfly
    Von Heiko R. im Forum Basic-Programmierung (Bascom-Compiler)
    Antworten: 3
    Letzter Beitrag: 09.11.2005, 18:55
  5. Timer beim Mega16
    Von maw_hn im Forum C - Programmierung (GCC u.a.)
    Antworten: 3
    Letzter Beitrag: 22.08.2005, 10:03

Stichworte

Berechtigungen

  • Neue Themen erstellen: Nein
  • Themen beantworten: Nein
  • Anhänge hochladen: Nein
  • Beiträge bearbeiten: Nein
  •  

fchao-Sinus-Wechselrichter AliExpress