I have grabbed the code for PI2, instead of using a PIC12F1840 I changed the code so it runs on arduino (atmega328 in my case). There are quite a bit of differences from the PIC code:
Timer0 needs to be disabled, or it will mess up your timings. This means you can not use any of the delay functions, nor can you use serial output. Digital outputs used are D13 for the txpulse, D12 for the sample pulse and D11 for the PWM audio. The delay pot is read at analog input 5 (A5). I have added comments in the code to clarify things.
One thing that puzzles me... using Timer1Delay, the smallest possible delay is 15 us. This is much larger than when using the PIC. I don't know the reason for this. Maybe I can find a solution for this one day.
//------------------------------------------------------------------------------
// Arduino PI Firmware
//------------------------------------------------------------------------------
// This project uses the Arduino Uno
// Clock is run off the internal 16MHz oscillator
// Each instruction takes 1 clock cycles so time resolution is 0.0625us
// Timer2 controls the main loop, has 8 bits plus prescaler.
// Timer1 controls delays, has 16 bits plus prescaler.
//--- Defines ---
#define TxPulse 13
#define SamplePulse 12
#define PWMPulse 11
#define DelayPot A5
#define period 47 // 255-208 -> 1664us (~600 Hz)
#define TxPulseWidth 63935 // 65535 - 1600 -> 100 uS
#define SampleWidth 65295 // 65535 - 240 -> 15 uS
#define AudioDelay 65135 // 65535 - 400 -> 25 uS
#define AudioWidth 62335 // 65535 - 3200 -> 200 uS
//--- Variables ---
unsigned int Sample1Delay = 65535-240; // 15us
//------------------------------------------------------------------------------
// Initialization routines
// Initialize the I/O pins and set some flags
//
void Init (void)
{
pinMode (TxPulse, OUTPUT);
pinMode (SamplePulse, OUTPUT);
pinMode (PWMPulse, OUTPUT);
TCCR0A = 0; // disable timer0
TCCR0B = 0;
}
// Timer2 is 8 bits with Div128, so delay is (255-n)*128/Fosc
// For Fosc = 16MHz Timer2 has 8us of resolution
void InitTimer2 (void)
{
TCCR2A = 0;
TCCR2B = 0;
bitSet (TIMSK2, TOIE2); // Enable Timer2 interrupt
TCCR2B = 0x05; // Prescaler = /128 (8 us resolution)
}
// Timer1 is 16 bits and clocked at 16MHz; use PS=1 for 0.0625us resolution
//
void InitTimer1 (void)
{
TCCR1A = 0;
TCCR1B = 0;
bitClear (TIMSK1, TOIE1); // Don't allow an interrupt
TCNT1 = 0; // Clear the timer
TCCR1B = 0x01; // Prescaler = /1 (62.5ns resolution)
}
void setup ()
{
Init ();
InitTimer2 ();
InitTimer1 ();
TCNT2 = 200; // Set the timer
}
//------------------------------------------------------------------------------
// Processing routines
// Delay (in us) = 65535-n
// Because of the fudge factor, ~15us is the minimum delay.
//
void Timer1Delay (unsigned int n)
{
n += 235; // Adjust for fixed error (approx 15us)
TCNT1 = n; // Load the counter
bitSet (TIFR1, TOV1); // Clear the flag
while (!(TIFR1 & (1<<TOV1))) {;} // Wait for a flag
}
// Read the pulse delay pot
//
void ReadDelayPot (void)
{
static int value;
value = analogRead(DelayPot); // Wait for the ADC to finish
Sample1Delay = 65535 - 240 - value; // 15us min, 64us max
}
// ISR() is the interrupt service routine for Timer2. The main loop is
// triggered off this interrupt and must be completed before the next
// interrupt.
ISR (TIMER2_OVF_vect)
{
TCNT2 = period; // Reset the timer
digitalWrite (TxPulse, HIGH);
Timer1Delay (TxPulseWidth); // Transmit pulse
digitalWrite (TxPulse, LOW);
Timer1Delay (Sample1Delay); // TX-to-sample pulse delay
digitalWrite (SamplePulse, HIGH);
Timer1Delay (SampleWidth); // Sample pulse
digitalWrite (SamplePulse, LOW);
Timer1Delay (AudioDelay); // Delay for audio
digitalWrite (PWMPulse, HIGH);
Timer1Delay(AudioWidth); // Speaker pulse
digitalWrite (PWMPulse, LOW);
ReadDelayPot();
// The remaining routines are where extra processing gets done. It is critical
// that all processing is complete before the next TCNT0 interrupt occurs, which
// depends on the TX pulse rate, TX pulse width, and sampling time. If the pulse
// rate = 600Hz, pulse width = 100us, and max sampling time = 35us then the
// processing time available is 1667us - 100us - 35us = 1532us. With a 16MHz
// clock we have an instruction cycle of 0.0625us, so there is time for 24512 code
// instructions, including calls and returns.
}
void loop ()
{
}
- Each instruction takes 1 clock cycles. Time resolution is 0.0625us.
- Timer2 controls the main loop (8 bits).
- Timer1 controls delays (16 bits).
Timer0 needs to be disabled, or it will mess up your timings. This means you can not use any of the delay functions, nor can you use serial output. Digital outputs used are D13 for the txpulse, D12 for the sample pulse and D11 for the PWM audio. The delay pot is read at analog input 5 (A5). I have added comments in the code to clarify things.
One thing that puzzles me... using Timer1Delay, the smallest possible delay is 15 us. This is much larger than when using the PIC. I don't know the reason for this. Maybe I can find a solution for this one day.
//------------------------------------------------------------------------------
// Arduino PI Firmware
//------------------------------------------------------------------------------
// This project uses the Arduino Uno
// Clock is run off the internal 16MHz oscillator
// Each instruction takes 1 clock cycles so time resolution is 0.0625us
// Timer2 controls the main loop, has 8 bits plus prescaler.
// Timer1 controls delays, has 16 bits plus prescaler.
//--- Defines ---
#define TxPulse 13
#define SamplePulse 12
#define PWMPulse 11
#define DelayPot A5
#define period 47 // 255-208 -> 1664us (~600 Hz)
#define TxPulseWidth 63935 // 65535 - 1600 -> 100 uS
#define SampleWidth 65295 // 65535 - 240 -> 15 uS
#define AudioDelay 65135 // 65535 - 400 -> 25 uS
#define AudioWidth 62335 // 65535 - 3200 -> 200 uS
//--- Variables ---
unsigned int Sample1Delay = 65535-240; // 15us
//------------------------------------------------------------------------------
// Initialization routines
// Initialize the I/O pins and set some flags
//
void Init (void)
{
pinMode (TxPulse, OUTPUT);
pinMode (SamplePulse, OUTPUT);
pinMode (PWMPulse, OUTPUT);
TCCR0A = 0; // disable timer0
TCCR0B = 0;
}
// Timer2 is 8 bits with Div128, so delay is (255-n)*128/Fosc
// For Fosc = 16MHz Timer2 has 8us of resolution
void InitTimer2 (void)
{
TCCR2A = 0;
TCCR2B = 0;
bitSet (TIMSK2, TOIE2); // Enable Timer2 interrupt
TCCR2B = 0x05; // Prescaler = /128 (8 us resolution)
}
// Timer1 is 16 bits and clocked at 16MHz; use PS=1 for 0.0625us resolution
//
void InitTimer1 (void)
{
TCCR1A = 0;
TCCR1B = 0;
bitClear (TIMSK1, TOIE1); // Don't allow an interrupt
TCNT1 = 0; // Clear the timer
TCCR1B = 0x01; // Prescaler = /1 (62.5ns resolution)
}
void setup ()
{
Init ();
InitTimer2 ();
InitTimer1 ();
TCNT2 = 200; // Set the timer
}
//------------------------------------------------------------------------------
// Processing routines
// Delay (in us) = 65535-n
// Because of the fudge factor, ~15us is the minimum delay.
//
void Timer1Delay (unsigned int n)
{
n += 235; // Adjust for fixed error (approx 15us)
TCNT1 = n; // Load the counter
bitSet (TIFR1, TOV1); // Clear the flag
while (!(TIFR1 & (1<<TOV1))) {;} // Wait for a flag
}
// Read the pulse delay pot
//
void ReadDelayPot (void)
{
static int value;
value = analogRead(DelayPot); // Wait for the ADC to finish
Sample1Delay = 65535 - 240 - value; // 15us min, 64us max
}
// ISR() is the interrupt service routine for Timer2. The main loop is
// triggered off this interrupt and must be completed before the next
// interrupt.
ISR (TIMER2_OVF_vect)
{
TCNT2 = period; // Reset the timer
digitalWrite (TxPulse, HIGH);
Timer1Delay (TxPulseWidth); // Transmit pulse
digitalWrite (TxPulse, LOW);
Timer1Delay (Sample1Delay); // TX-to-sample pulse delay
digitalWrite (SamplePulse, HIGH);
Timer1Delay (SampleWidth); // Sample pulse
digitalWrite (SamplePulse, LOW);
Timer1Delay (AudioDelay); // Delay for audio
digitalWrite (PWMPulse, HIGH);
Timer1Delay(AudioWidth); // Speaker pulse
digitalWrite (PWMPulse, LOW);
ReadDelayPot();
// The remaining routines are where extra processing gets done. It is critical
// that all processing is complete before the next TCNT0 interrupt occurs, which
// depends on the TX pulse rate, TX pulse width, and sampling time. If the pulse
// rate = 600Hz, pulse width = 100us, and max sampling time = 35us then the
// processing time available is 1667us - 100us - 35us = 1532us. With a 16MHz
// clock we have an instruction cycle of 0.0625us, so there is time for 24512 code
// instructions, including calls and returns.
}
void loop ()
{
}
Comment