Измерение частоты, используя используя input capture unit. Непонятный глюк.
- Войдите на сайт для отправки комментариев
Здравствуйте, понадобилось измерять частоту внешнего сигнала. Диапазон от 10Гц до 10кГц с точностью 1 Гц. Сигнал подается на пин D8 arduino uno (порт B.0 AtMega328). На другой пин сигнал подать нет возможности. На просторах сети нашел код, который хорошо подошел. http://www.gammon.com.au/forum/?id=11504&reply=12#reply12
Но, заметил странности, периодически значение частоты, которое выводится в COM порт имеет значение в два раза больше или в два раза меньше чем реальное значение частоты на пине D8. Сначала грешил на железо, думал помехи. Но в симуляторе Proteus ситуация такая-же. Ниже привожу скетч и скриншот, на котором видны эти выбросы. Используется симулятор Proteus программа для визуализации данных с COM порта Serial-Oscilloscope-v1.5
Может как то можно оптимизировать этот код? Или у кого-нибудь есть пример измерения частоты, используя Input Capture Unit
#include <Arduino.h>
// Frequency timer using input capture unit
// Author: Nick Gammon
// Date: 31 August 2013
// Input: Pin D8
volatile boolean first;
volatile boolean triggered;
volatile unsigned long overflowCount;
volatile unsigned long startTime;
volatile unsigned long finishTime;
// timer overflows (every 65536 counts)
ISR (TIMER1_OVF_vect)
{
overflowCount++;
} // end of TIMER1_OVF_vect
ISR (TIMER1_CAPT_vect)
{
// grab counter value before it changes any more
unsigned int timer1CounterValue;
timer1CounterValue = ICR1; // see datasheet, page 117 (accessing 16-bit registers)
unsigned long overflowCopy = overflowCount;
// if just missed an overflow
if ((TIFR1 & bit (TOV1)) && timer1CounterValue < 0x7FFF)
overflowCopy++;
// wait until we noticed last one
if (triggered)
return;
if (first)
{
startTime = (overflowCopy << 16) + timer1CounterValue;
first = false;
return;
}
finishTime = (overflowCopy << 16) + timer1CounterValue;
triggered = true;
TIMSK1 = 0; // no more interrupts for now
} // end of TIMER1_CAPT_vect
void prepareForInterrupts ()
{
noInterrupts (); // protected code
first = true;
triggered = false; // re-arm for next time
// reset Timer 1
TCCR1A = 0;
TCCR1B = 0;
TIFR1 = bit (ICF1) | bit (TOV1); // clear flags so we don't get a bogus interrupt
TCNT1 = 0; // Counter to zero
overflowCount = 0; // Therefore no overflows yet
// Timer 1 - counts clock pulses
TIMSK1 = bit (TOIE1) | bit (ICIE1); // interrupt on Timer 1 overflow and input capture
// start Timer 1, no prescaler
TCCR1B = bit (CS10) | bit (ICES1); // plus Input Capture Edge Select (rising on D8)
interrupts ();
} // end of prepareForInterrupts
void setup ()
{
Serial.begin(19200);
// set up for interrupts
prepareForInterrupts ();
} // end of setup
void loop ()
{
// wait till we have a reading
if (!triggered)
return;
// period is elapsed time
unsigned long elapsedTime = finishTime - startTime;
// frequency is inverse of period, adjusted for clock period
float freq = F_CPU / float (elapsedTime); // each tick is 62.5 nS at 16 MHz
Serial.println(freq,0);
prepareForInterrupts ();
} // end of loop
Вот скриншот https://dl.dropboxusercontent.com/u/37695080/screen.jpg