Torrentula
28.12.2011, 13:40
Hallo RNler!
Da ich mal wieder Plane einen SF05 einzusetzen habe ich ein Programm zum auslesen geschrieben, da das Programm aus einem anderen Thread nicht funktionierte (immer mal wieder Ergebnis 0).
Das Programm soll wie folgt funktionieren:
1. Messung wird per 12µs-Impuls an SRF05 ausgelöst
2. Externer Interrupt 0 wird auf steigende Flanke konfiguriert
3. INT0 wird zum ersten mal ausgelöst und startet den Timer, welcher selber alle 10µs einen Interrupt auslöst; gleichzeitig wird INT0 auf fallende Flanke konfiguriert
4. INT0 wird nun bei fallender Flanke ausgelöst und stoppt den Timer --> variable measurement_complete wird nun auf 1 gesetzt
5. Die Dauer des Impulses in µs wird durch 58 geteilt um Entfernung in cm zu erhalten und die Entfernung wird per RS232 an den PC gesendet
Ich erhalte jedoch immer eine 0 im Terminal. Wenn ich den Timer auf 10Hz umgestellt habe, haben sich einfach alle Werte aufaddiert.
Hier mein Code:
#ifndef F_CPU
#define F_CPU 16000000UL
#endif
#include <avr/io.h>
#include <avr/interrupt.h>
#include <util/delay.h>
#include <stdlib.h>
#include "standards.h" // hier sind init_USART() und sendchar() definiert
#define SRFout PD7
#define SRFin PD2
volatile unsigned int microseconds = 0; // um die Pulslänge zu speichern
volatile uint8_t INT0_interrupt = 0; // wird benutzt um zwischen Pulbeginn und Pulsende zu entscheiden
volatile uint8_t measurement_complete = 0; // wird in der hauptschleife gepollt um Abschluss der Messung festzustellen
int main(void)
{
init_USART();
// Timer einstellen
TCCR0A = (1<<WGM01); // CTC Mode of Timer 0
// ((16000000 / 1024) / 100) -1 = 155
OCR0A = 155; // 155 steps = interrupt ca. alle 10µs
TIMSK0 |= (1<<OCIE0A); // Enable compare interrupt
//~~~~~~~~~~~~~~~~~
EIMSK |= (1<<INT0); // External interrupt mask register INT0 aktivieren
DDRD |= (1<<SRFout); // Auslösepin auf Ausgang
DDRD &= ~(1<<SRFin); // INT0 pin muss auch auf input stehen
// Variablen
uint16_t distance = 0;
unsigned char buffer[10];
while(1)
{
// Auslösen der Messung:
PORTD |= (1<<SRFout);
_delay_us(12); // 12µs Auslöse-Signal
PORTD &= ~(1<<SRFout);
// external interrupt 0 bei steigender flanke auslösen:
EICRA = (ISC01) | (1<<ISC00);
sei(); // alle interrupts aktivieren
while(measurement_complete != 1){
// auf Abschluss der Messung warten
}
measurement_complete = 0;
distance = microseconds / 58;
itoa(distance, buffer, 10);
sendUSART(buffer);
sendchar('\n');
distance = 0; // distance wieder zurücksetzen, sonst werden Messungen aufaddiert
microseconds = 0; // microseconds ebenfalls zurücksetzen
_delay_ms(1000);
}
}
ISR(INT0_vect){ // external interrupt 0
if(INT0_interrupt == 0){ // nur wenn noch nicht bei steigender Flanke ausgelöst wurde
TCCR0B |= (1<<CS02) | (1<<CS00); // Timer starten mit Prescaler 1024
// umschalten interrupt bei fallender Flanke:
EICRA = (1<<ISC01); // EICRA = External Interrup Control Register A
INT0_interrupt = 1; // da nun bei steigender Flanke bereits getriggert wurde
}
else{ // wenn bereits ein INT0 interrupt erfolgte
TCCR0B &= ~((1<<CS02) | (1<<CS00)); // Timer stoppen
cli(); // alle interrupts sperren
INT0_interrupt = 0; // Variable zurücksetzen
measurement_complete = 1; // Variable um in der Hauptschleife den Abschluss der Messung festzustellen
}
}
//gibt uns einen Interrupt ca. alle 10µs
ISR(TIMER0_COMPA_vect){
microseconds += 10;
}
MfG
Torrentula
Da ich mal wieder Plane einen SF05 einzusetzen habe ich ein Programm zum auslesen geschrieben, da das Programm aus einem anderen Thread nicht funktionierte (immer mal wieder Ergebnis 0).
Das Programm soll wie folgt funktionieren:
1. Messung wird per 12µs-Impuls an SRF05 ausgelöst
2. Externer Interrupt 0 wird auf steigende Flanke konfiguriert
3. INT0 wird zum ersten mal ausgelöst und startet den Timer, welcher selber alle 10µs einen Interrupt auslöst; gleichzeitig wird INT0 auf fallende Flanke konfiguriert
4. INT0 wird nun bei fallender Flanke ausgelöst und stoppt den Timer --> variable measurement_complete wird nun auf 1 gesetzt
5. Die Dauer des Impulses in µs wird durch 58 geteilt um Entfernung in cm zu erhalten und die Entfernung wird per RS232 an den PC gesendet
Ich erhalte jedoch immer eine 0 im Terminal. Wenn ich den Timer auf 10Hz umgestellt habe, haben sich einfach alle Werte aufaddiert.
Hier mein Code:
#ifndef F_CPU
#define F_CPU 16000000UL
#endif
#include <avr/io.h>
#include <avr/interrupt.h>
#include <util/delay.h>
#include <stdlib.h>
#include "standards.h" // hier sind init_USART() und sendchar() definiert
#define SRFout PD7
#define SRFin PD2
volatile unsigned int microseconds = 0; // um die Pulslänge zu speichern
volatile uint8_t INT0_interrupt = 0; // wird benutzt um zwischen Pulbeginn und Pulsende zu entscheiden
volatile uint8_t measurement_complete = 0; // wird in der hauptschleife gepollt um Abschluss der Messung festzustellen
int main(void)
{
init_USART();
// Timer einstellen
TCCR0A = (1<<WGM01); // CTC Mode of Timer 0
// ((16000000 / 1024) / 100) -1 = 155
OCR0A = 155; // 155 steps = interrupt ca. alle 10µs
TIMSK0 |= (1<<OCIE0A); // Enable compare interrupt
//~~~~~~~~~~~~~~~~~
EIMSK |= (1<<INT0); // External interrupt mask register INT0 aktivieren
DDRD |= (1<<SRFout); // Auslösepin auf Ausgang
DDRD &= ~(1<<SRFin); // INT0 pin muss auch auf input stehen
// Variablen
uint16_t distance = 0;
unsigned char buffer[10];
while(1)
{
// Auslösen der Messung:
PORTD |= (1<<SRFout);
_delay_us(12); // 12µs Auslöse-Signal
PORTD &= ~(1<<SRFout);
// external interrupt 0 bei steigender flanke auslösen:
EICRA = (ISC01) | (1<<ISC00);
sei(); // alle interrupts aktivieren
while(measurement_complete != 1){
// auf Abschluss der Messung warten
}
measurement_complete = 0;
distance = microseconds / 58;
itoa(distance, buffer, 10);
sendUSART(buffer);
sendchar('\n');
distance = 0; // distance wieder zurücksetzen, sonst werden Messungen aufaddiert
microseconds = 0; // microseconds ebenfalls zurücksetzen
_delay_ms(1000);
}
}
ISR(INT0_vect){ // external interrupt 0
if(INT0_interrupt == 0){ // nur wenn noch nicht bei steigender Flanke ausgelöst wurde
TCCR0B |= (1<<CS02) | (1<<CS00); // Timer starten mit Prescaler 1024
// umschalten interrupt bei fallender Flanke:
EICRA = (1<<ISC01); // EICRA = External Interrup Control Register A
INT0_interrupt = 1; // da nun bei steigender Flanke bereits getriggert wurde
}
else{ // wenn bereits ein INT0 interrupt erfolgte
TCCR0B &= ~((1<<CS02) | (1<<CS00)); // Timer stoppen
cli(); // alle interrupts sperren
INT0_interrupt = 0; // Variable zurücksetzen
measurement_complete = 1; // Variable um in der Hauptschleife den Abschluss der Messung festzustellen
}
}
//gibt uns einen Interrupt ca. alle 10µs
ISR(TIMER0_COMPA_vect){
microseconds += 10;
}
MfG
Torrentula