Forum: Mikrocontroller und Digitale Elektronik ATtiny25, watchdog, power-down, PWM


von Gast (Gast)


Lesenswert?

Ich möchte für Batteriebetrieb den AVR alle Sekunde per Watchdog 
Interrupt aus dem power-down Modus holen.

Power-down habe ich so verstanden, dass alle Clocks bis auf den 128 kHz 
Watchdog-Timer gestoppt werden, Ports disconnected..etc.

Im Zusammenspiel mit PWM ergibt sich mit diesem Programm:
1
// fuses: factory
2
// F_CPU: 1e6
3
4
#include <stdint.h>
5
#include <avr/io.h>
6
#include <avr/interrupt.h>
7
#include <avr/wdt.h>
8
#include <avr/power.h>
9
#include <avr/sleep.h>
10
#include <util/delay.h>
11
12
ISR(WDT_vect){
13
  WDTCR |= (1<<WDIE);
14
}
15
16
int __attribute__((OS_main)) main(void){
17
  PORTB = 0;
18
  DDRB = (1<<PB3) | (1<<PB4) | (1<<PB1);
19
  ACSR  = (1<<ACD);
20
  PRR = (1<<PRUSI);
21
  MCUCR = (1<<SE) | (1<<SM1);
22
  TCCR1 = (1<<PWM1A) | (1<<COM1A1) | (1<<CS12) | (1<<CS11);
23
  //~ OCR1A = 128;
24
  OCR1A = 255;
25
  WDTCR = (1<<WDIE) | (1<<WDP2) | (1<<WDP1);
26
  _delay_ms(533);
27
  sei();
28
  while (1){
29
    PORTB ^= (1<<PB3);
30
    sleep_cpu();
31
  }
32
}

folgendes:
Mit OCR1A = 128; erscheint für das delay die PWM an PB1, PB1 geht aus, 
PB3 geht an, und dann blinkt PB3 im Sekundentakt.

Mit OCR1A = 255; bleibt PB1 die ganze Zeit an, hinzu kommt das 1 s - 
Blinken von PB3.

D.h. wohl dass meine Annahme, dass alles abgeschaltet wird anscheinend 
falsch ist und ich das manuell machen muss?

Was ist die Ursache der unterschiedlichen Wirkung der OCR1A Werte?

von Gast (Gast)


Lesenswert?

Man kann das //~ WDTCR |= (1<<WDIE); in der ISR sparen, da es nicht 
automatisch gelöscht wird, wenn der Watchdog nur mit WDIE gestartet 
wird.

von Gast (Gast)


Lesenswert?

_delay_ms(533); hat bei 1 MHz auf 10tel Sekunden umgeschaltet. Das 
Verhalten bleibt aber auch mit _delay_ms(255); gleich.
Wenn das delay auskommentiert wird, schaltet die PWM sofort ab.

von Gast (Gast)


Lesenswert?

Es wird an der Synchronisation von OCRxx liegen ("temporary register", 
"latch"). OCRxx vor dem Starten des Timers behebt zumindest das.
1
// fuses: factory
2
// F_CPU: 1e6
3
4
#include <stdint.h>
5
#include <avr/io.h>
6
#include <avr/interrupt.h>
7
#include <avr/wdt.h>
8
#include <avr/power.h>
9
#include <avr/sleep.h>
10
#include <util/delay.h>
11
12
ISR(WDT_vect){
13
}
14
15
int main(void){
16
  PORTB = 0;
17
  DDRB = (1<<PB3) | (1<<PB4) | (1<<PB1);
18
  
19
  ACSR  = (1<<ACD);
20
  PRR = (1<<PRUSI);
21
  
22
  MCUCR = (1<<SE) 
23
        | (1<<SM1)
24
        //~ | (1<<SM0)
25
        ;
26
  
27
  OCR1A = 128;
28
  TCCR1 = (1<<PWM1A) | (1<<COM1A1) | (1<<CS12) | (1<<CS11);
29
  
30
  WDTCR = (1<<WDIE) | (1<<WDP2) | (1<<WDP1);
31
  
32
  sei();
33
  
34
  while (1)
35
  {
36
    PORTB ^= (1<<PB3);
37
    sleep_cpu();
38
  }
39
}

Durch den power-down wird bei OCRA0 > 0 allerdings der Pin nach dem 
Aufwachen auf 1 gesetzt, die PWM ist nicht mehr aktiv.
Wo steht etwas zu diesem Verhalten (ausser, dass power down alle clocks 
stoppt) und wie geht man damit um?

von Gast (Gast)


Lesenswert?

Vorher PWM auf 0, nachher den Timer wieder starten..

Bitte melde dich an um einen Beitrag zu schreiben. Anmeldung ist kostenlos und dauert nur eine Minute.
Bestehender Account
Schon ein Account bei Google/GoogleMail? Keine Anmeldung erforderlich!
Mit Google-Account einloggen
Noch kein Account? Hier anmelden.