Forum: Compiler & IDEs PWM - allgemein


von Gast (Gast)


Angehängte Dateien:

Lesenswert?

Hallo zusammen,

Ist es möglich eine veränderbare PWM, parallel zur Main, aufzubauen? 
Wenn ja wie?
In der Main sollen andere Aufgaben erledigt werden.

Folgender Problemfall.
Es soll eine Temperatur, in einem gewissen Bereich 10°C +/-1°C, konstant 
gehalten werden. Durch einen Sensor wird die akt Temp aufgenommen und 
durch einen Lüfter, bei dem die Drehzahl mit ner PWM geregelt werden 
soll, die Temp konstant gehalten werden. Die Zuluft wird von einer 
Kühlung bereitgestellt.

Im WWW gibt es mehrere Ansetze wie man das Problem anpacken kann.
Ich würde es gerne so gestallten dass während in der Main Aufgaben 
erledigt werden, wie z.B. Ausgabe auf nem LCD, Tempmessung, Parameter 
setzen ....,
der Lüfter eigenständig und parallel über die PWM gehalten werden soll. 
Aus der Main soll lediglich der Vergleichswert für die Modulation 
übermittelt werden.

Ich verwende dafür einen Atmega48.
In der Zwischenzeit mache ich einen Beispielcode und stelle es hier 
rein.

Vielen Dank für die Hilfe

von Johannes M. (johnny-m)


Lesenswert?

AVR-GCC-Tutorial, speziell mal die Abschnitte zum Thema Interrupts 
lesen...

von Falk B. (falk)


Lesenswert?

Siehe Multitasking

von Gast (Gast)


Lesenswert?

So, ich habe mal das ganze auf ein Steckbrett aufgebaut. Statt dem 
Lüfter habe ich grad ne LED angeschlossen.

Durch 4 Taster an PC0-PC3, soll in 25% Schritten die Leuchtstärke an der 
LED (PB1) gewählt werden.
Nach einem Tastendruck, soll die gewählte Leuchtstärke z.B. 25% gesetzt 
werden, mit dem Wert aus der entsprechenden "IF-Schleife" in die 
PWM_init(), Vergleichswert setzten und wieder zurück zur Main.
Währenddessen, soll die PWM-LED (PB1) mit dem übermittelten Wert 
dauerleuchten.
Ist keine Taste gedrückt worden oder nach jedem Tastendruck, soll eine 
Kontroll-LED (PB5) ob man sich wieder in der MAIN befindet, blinken.
Soviel zu meiner Überlegung.
Nun zur Problembeschreibung:
Nach dem entsprechenden Tastendruck b.B. PC1, wird zwar die LED an PB2 
gesetzt aber der komischerweise der 100% PWM-Wertübermittelt
Kurz gesagt die PWM-LED leuchtet mit voller Leuchtstärke.
Was mache ich falsch???

Hier ist mein Code dafür.
1
#define F_CPU 8000000UL // 8 MHz
2
3
#include <stdio.h> 
4
#include <avr/io.h>           
5
#include <util/delay.h>
6
#include <avr/interrupt.h>
7
8
void PWM_init(int uewert);
9
int wert;
10
11
int main()
12
{
13
  
14
    PORTC |= (1<<PC0) | (1<<PC1) | (1<<PC2) | (1<<PC3);         // PORTC = PULL-UP
15
16
  while(1) 
17
  {
18
    DDRB |= (1<<PB5) | (1<<PB4) | (1<<PB3);
19
    DDRC &= ~(1<<PC0) | (1<<PC1) | (1<<PC2) | (1<<PC3);        // PORTC = Eingang
20
21
    if (!(PINC & (1<<PC0)))      // 0% Modulation
22
    {
23
      wert=0;
24
      PWM_init(wert);
25
    }
26
    if (!(PINC & (1<<PC1)))      // 25%
27
    {
28
      wert=341;
29
      PWM_init(wert);
30
      PORTB |=(1<<PB3);          // LED setzen
31
    }
32
    if (!(PINC & (1<<PC2)))      // 50%
33
    {
34
      wert=512;
35
      PWM_init(wert);
36
      PORTB |=(1<<PB4);          // LED setzen
37
    }
38
    if (!(PINC & (1<<PC3)))      // 100%
39
    {
40
      wert=1023;
41
      PWM_init(wert);
42
    }
43
    
44
        
45
// --------------  falls keinen Taster gedrückt ---------------------------
46
47
    PORTB |=(1<<PB5);          // LED setzen
48
    _delay_ms(25);          // Warte
49
    PORTB &= ~(1<<PB5);       // LED löschen
50
    _delay_ms(25);          // Warte    
51
  }
52
  return 0;
53
}

und der Code der PWM_init().
1
#define F_CPU 8000000UL // 8 MHz
2
3
#include <stdio.h> 
4
#include <avr/io.h>           
5
#include <util/delay.h>
6
#include <avr/interrupt.h>
7
8
9
void PWM_init(int vergleichswert)
10
{
11
  DDRB = (1 << PB1);          // PORTB als Ausgang   
12
  ICR1 =(uint16_t)250;
13
  OCR1A = vergleichswert;          // Vergleichswert aus Main
14
15
  //sei();
16
  TCCR1A = (1<<COM1A1) | (1 << WGM11) | (1 << WGM10); // non-inverting mode fuer Pin OC1A und Fast PWM Mode 10Bit
17
  TCCR1B = (1 << WGM13) | (1 << WGM12) | (1<<CS11);  // Fast PWM Mode und Prescaler: 1/8    
18
  
19
    
20
//return vergleichswert;
21
}

von Stefan E. (sternst)


Lesenswert?

Du benutzt Timer-Mode 15, dabei legt OCR1A den Top-Wert fest, nicht den 
Vergleichswert. Aber du setzt auch ICR1, wolltest du also eigentlich 
Mode 14 haben? Aber wenn ja, welchen Sinn macht es dann, den 
Vergleichswert größer zu machen, als den Top-Wert?

von Gast (Gast)


Lesenswert?

...jetzt mal vorerst eine Verständnisfrage.

In welchem Register ( TCCR1A oder TCCR1B ) wird der Modus eingestellt 
und wie verwendet man genau die verschiedenen Modi. Ausserdem gibt es 
Für beide Register WGM-Einstellungen?
Aus dem Datenblatt (Seite 127) kann ich zwar den Top-Wert lesen, aber 
ich bin mir nicht immer sicher welchen man für was verwendet.
Z.B. Fast PWM ist auf 10 Modi aufgeteilt!!

Vielen Dank

von Oliver (Gast)


Lesenswert?

>In welchem Register ( TCCR1A oder TCCR1B ) wird der Modus eingestellt
>Ausserdem gibt es Für beide Register WGM-Einstellungen?

Eben. Der Modus wird mit daher in beiden Register eingestellt, durch 
setzten aller WGM-Bits auf den gewünschten Wert.

>Aus dem Datenblatt (Seite 127) kann ich zwar den Top-Wert lesen, aber
>ich bin mir nicht immer sicher welchen man für was verwendet.

Das steht in der Beschreibung zu den einzelnen Modi. Wenn es zu einem 
Mosdus mehrere Variante gibt, unterschieden die sich dann entweder durch 
die verwendeten Register (z.B. enthält mal OCRx den Wert für TOP, mal 
ICRx), oder durch verschieden, fest vorgegeben TOP-Werte. Bei Fast-PWM 
z.B. 0x00FF, 0x01FF, usw.

Oliver

von Oliver (Gast)


Lesenswert?

Nachtrag:
>Aus dem Datenblatt (Seite 127)
>...
>Z.B. Fast PWM ist auf 10 Modi aufgeteilt!!

Eigentlich hat der Timer nur vier Betriebsarten: Normal, CTC, Fast PWM 
und Phase Correct PWM. Welche dieser Varianten jetzt für deine 
Aufgabenstellung die richtige ist, musst du selbst entscheiden.

Das Datenblatt, welches ich vom Mega48 habe, zeigt auf Seite 131 in 
Tabelle 13-4 genau fünf Fast-PWM-Varianten. FAST-PWM-8bit, -9-bit, und 
-10-bit, mit festen TOP Werten, sowie jeweils eine mit einstellbarem 
TOP-Wert in ICR1 bzw. OCR1A.

Dazu kommen dann noch die COM1Ax und COM1Bx-Bits in TCCR1A, über die du 
einstellen kannst, wann und wie die Pins OCR1A und OCR1B jeweils ihren 
Zustand ändern.

Ähnlich ist es bei den anderen Timer-Betriebsarten.


Oliver

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.