PDA

Archiv verlassen und diese Seite im Standarddesign anzeigen : Frequenzmessung mit einem Mega32



Kampi
27.05.2012, 14:58
Hallo Forum,

ich versuche gerade meinen Feuchtigkeitssensor (http://www.hoperf.com/upload/sensor/hh10d.pdf) mit meinem RN-Control auszuwerten.
Den Frequenzausgang des Sensors habe ich mit Pin D2 (INT0) verbunden.
Der Grundgedanke ist, dass ich bei einer steigenden Flanke in der ISR von INT0 eine Variable hochzähle und die dann 1x in der Sekunde auf ein Terminal ausgebe.
Damit bekomme ich ja dann die Impulse pro Sekunde sprich die Frequenz raus.
Jetzt verwende ich dieses Programm:



/*
* HH10D.c
*
* Created: 03.05.2012 20:46:19
* Author: Daniel
*/

#include <avr/io.h>
#include <avr/interrupt.h>
#include <string.h>
#include <util/delay.h>
#include <stdlib.h>

int lenght = 0x00;
char data[50];
volatile int Zaehler = 0x00;
int Frequenz = 0x00;

int main(void)
{
UART_Init();
Port_Init();
Timer1_Init();
sei();

while(1)
{

}
}

void UART_Init()
{
UCSRB = (1<<RXEN) | (1<<TXEN); // Rx und Tx aktivieren
UCSRC = (1<<URSEL)| (1<<UCSZ1) | (1<<UCSZ0); // 8 Bit Nachrichtenlänge einstellen


UBRRH = 0x00;
UBRRL = 0x33; // Baudrate auf 19200 festlegen
}

void Port_Init()
{
DDRD = 0x00;

MCUCR = (1<<ISC01); // Interrupt auf fallende Flanke stellen
GIMSK = (1<<INT0); // INT0 aktivieren
}

void Timer1_Init()
{
TCCR1B = (1<<CS12); // Prescaler 256
TCNT1H = 0x0B;
TCNT1L = 0xDB;
TIMSK |= (1<<TOIE1); // Timer1 Overflow Interrupt aktivieren
}


void Send_UART(char data[])
{
char Counter;

lenght = strlen(data);

while(Counter < lenght)
{
while (!(UCSRA & (1<<UDRE)));
UDR = data[Counter];
Counter++;
}

Counter = 0x00;
while (!(UCSRA & (1<<UDRE)));
UDR = 0x0A;
while (!(UCSRA & (1<<UDRE)));
UDR = 0x0D;
}

ISR(TIMER1_OVF_vect)
{
cli();
Frequenz = Zaehler;
Zaehler = 0x00;
TCNT1H = 0x0B;
TCNT1L = 0xDB;
sei();

itoa(Frequenz, data, 10);
Send_UART(data);
}


ISR(INT0_vect)
{
Zaehler++;
}


Allerdings stimmt die Frequenz nicht mit der Frequenz des Sensors überein (ich kontrolliere das per Oszi). Der Mikrocontroller spuckt mir immer einen Wert aus der 100-200 zu hoch ist.
Weiß einer woran das liegen könnte?

Danke für die Hilfe!

MagicWSmoke
27.05.2012, 16:34
Der Code ist recht ineffektiv, insbesondere wenn ein Input Capture zur Verfügung steht.
Counter ist lokal zu Send_UART, befindet sich damit auf dem Stack. Wenn die lokale Variable nicht initialisiert wird, ist ihr Startwert unbestimmt, sagt Dir aber der Compiler schon.
Den Wert nachträglich auf 0 zu setzen ist sinnlos, da Counter als Index auf den Ausgabestring dient, ist das eher Zufall wenn's klappt.

Dirk
27.05.2012, 16:54
Hallo Daniel,

vielleicht kannst du dir da (https://www.roboternetz.de/community/threads/47088-RP6Control-M32-Impulslängen-Messgerät) etwas abgucken?
(Ist zwar für den RP6, aber der hat auch den ATmega32...)

Kampi
27.05.2012, 16:59
Danke euch beiden.
Ich habe das gestern auch mal mit der Capture and Compare Unit probiert aber damit hat es leider gar nicht geklappt. Eigentlich dachte ich für sowas einfaches würde das ohne Probleme mit dem Timer klappen....aber naja :/
Der Compiler hat mir gar nichts zu der Variable Counter gesagt (und die Ausgabe per UART klappt auch ohne Probleme) aber trotzdem danke für den Hinweis :D
Den Link schaue ich mir mal an und gucke mal ob ich da was von entnehmen kann.

MagicWSmoke
27.05.2012, 19:21
vielleicht kannst du dir da (https://www.roboternetz.de/community/threads/47088-RP6Control-M32-Impulslängen-Messgerät (https://www.roboternetz.de/community/threads/47088-RP6Control-M32-Impulsl%C3%A4ngen-Messger%C3%A4t)) etwas abgucken?
Warum setzt Du eigentlich den Wert icrcnt_start.i16h in der ISR laufend zurück, ohne dass Du ihn jemals änderst ? Nicht dass es viel ausmachen würd', aber es ist sinnlos.

icrcnt_start.i16h = 0; // Reset upper 16 bits (start value)
Für die einmalige Initialisierung hätte Dir das gereicht:

static icrcounter_t icrcnt_start = {.i32=0};

Der Compiler hat mir gar nichts zu der Variable Counter gesagt (und die Ausgabe per UART klappt auch ohne Probleme) aber trotzdem danke für den Hinweis :-)

../hh10d_meas.c:48: warning: 'Counter' may be used uninitialized in this function
Wenn der Stackspeicher nicht stark benutzt wird, dann ist es möglich dass zufällig für sich diese lokale Variable immer wieder die gleiche Adresse im Stack benutzt wird, dann geht es.
Änderst sich die Stackbenutzung kann dort ein anderer Wert stehen, mit entsprechendem Ergebnis.
Grundsätzlich hat eine lokale Variable als verloren betrachtet zu werden, außer sie wird static deklariert, dann bleibt sie im Speicher erhalten.

Kampi
27.05.2012, 19:39
Warum setzt Du eigentlich den Wert icrcnt_start.i16h in der ISR laufend zurück, ohne dass Du ihn jemals änderst ? Nicht dass es viel ausmachen würd', aber es ist sinnlos.

icrcnt_start.i16h = 0; // Reset upper 16 bits (start value)
Für die einmalige Initialisierung hätte Dir das gereicht:

static icrcounter_t icrcnt_start = {.i32=0};


Wenn der Stackspeicher nicht stark benutzt wird, dann ist es möglich dass zufällig für sich diese lokale Variable immer wieder die gleiche Adresse im Stack benutzt wird, dann geht es.
Änderst sich die Stackbenutzung kann dort ein anderer Wert stehen, mit entsprechendem Ergebnis.
Grundsätzlich hat eine lokale Variable als verloren betrachtet zu werden, außer sie wird static deklariert, dann bleibt sie im Speicher erhalten.

Ok dank dir für den Hinweis :)
Ich programmiere nicht so viel und so oft in C, daher sind mir solche Feinheiten nicht ganz klar.

Kampi
28.05.2012, 10:04
Hallo Daniel,

vielleicht kannst du dir da (https://www.roboternetz.de/community/threads/47088-RP6Control-M32-Impulslängen-Messgerät (https://www.roboternetz.de/community/threads/47088-RP6Control-M32-Impulsl%C3%A4ngen-Messger%C3%A4t)) etwas abgucken?
(Ist zwar für den RP6, aber der hat auch den ATmega32...)

Hi,

nochmal kurz ne Frage.....
So richtig steige ich durch dein Programm (leider) nicht durch. Kannst du mir im groben Erklären wie das mit der Input-Capture Unit und deinem Programm funktioniert?
Dank dir!

MagicWSmoke
28.05.2012, 11:51
Ich empfand das verlinkte Programm auch ein wenig undurchsichtig, vor allem die Structs und Unions, kann man zwar so machen, muss man aber nicht. Auch ist, obwohl's tatsächlich nicht viel ausmacht, bei jedem Triggern des Input Capture eine Messung möglich. Ein Unterscheiden zwischen erster und zweiter Flanke ist nicht notwendig.
Die normale Vorgehensweise ist so, man nimmt sich 'ne unsigned 16Bit Variable und zieht in der ISR immer den alten Zählerstand vom Neuen ab. Dass dann auch z.B. 200 - 65000 richtig behandelt wird, wird durch Berücksichtigung der Overflows erreicht. Die Overflows werden mit der Zählerbreite multipliziert und zum Ergebnis addiert.
Wichtig ist dabei auf race-conditions aufgrund der Codelaufzeit zu achten, die im Bereich des Zählerüberlaufs vorkommen können, als auch das Ergebnis atomar zu übergeben, damit während der Ausgabe nichts vefälscht wird.