|
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269 |
-
-
- #include <avr/interrupt.h>
- #include "wiring.h"
- #include "core_pins.h"
- #include "pins_arduino.h"
-
- static uint8_t timer_acquired = 0;
-
- static uint8_t *tone1_reg = (uint8_t *)0;
- static uint8_t tone1_mask = 0;
- static uint16_t tone1_inc = 0;
- static uint32_t tone1_count = 0;
-
- static uint8_t *tone2_reg = (uint8_t *)0;
- static uint8_t tone2_mask = 0;
- static uint16_t tone2_inc = 0;
- static uint32_t tone2_count = 0;
-
- static uint8_t *tone3_reg = (uint8_t *)0;
- static uint8_t tone3_mask = 0;
- static uint16_t tone3_inc = 0;
- static uint32_t tone3_count = 0;
-
- #define MAX_FREQ (F_CPU / 16 / 25)
- #define MIN_FREQ (F_CPU / 16 / 65535 + 1)
-
-
-
- #define PIN_REG_AND_MASK_LOOKUP(pin, reg, mask) \
- asm volatile( \
- "lsl %2" "\n\t" \
- "add %A3, %2" "\n\t" \
- "adc %B3, __zero_reg__" "\n\n" \
- "lpm %1, Z+" "\n\t" \
- "lpm %A0, Z" "\n\t" \
- "ldi %B0, 0" "\n" \
- : "=z" (reg), "=r" (mask), "+r" (pin) \
- : "z" (digital_pin_table_PGM), "2" (pin))
-
- #if defined(__AVR_ATmega32U4__)
-
- #define TONE_USE_TIMER3
- #elif defined(__AVR_AT90USB1286__)
-
- #define TONE_USE_TIMER3
- #elif defined(__AVR_AT90USB162__)
- #define TONE_USE_TIMER1
- #elif defined(__AVR_AT90USB646__)
-
- #define TONE_USE_TIMER3
- #endif
-
- #ifdef TONE_USE_TIMER3
- #define TIMSKx TIMSK3
- #define OCIExA OCIE3A
- #define OCIExB OCIE3B
- #define OCIExC OCIE3C
- #define TCCRxA TCCR3A
- #define WGMx0 WGM30
- #define TCCRxB TCCR3B
- #define CSx1 CS31
- #define TCNTx TCNT3
- #define OCRxA OCR3A
- #define OCRxB OCR3B
- #define OCRxC OCR3C
- #define TIFRx TIFR3
- #define OCFxA OCF3A
- #define OCFxB OCF3B
- #define OCFxC OCF3C
- #define VECTxA TIMER3_COMPA_vect
- #define VECTxB TIMER3_COMPB_vect
- #define VECTxC TIMER3_COMPC_vect
- #endif
- #ifdef TONE_USE_TIMER1
- #define TIMSKx TIMSK1
- #define OCIExA OCIE1A
- #define OCIExB OCIE1B
- #define OCIExC OCIE1C
- #define TCCRxA TCCR1A
- #define WGMx0 WGM10
- #define TCCRxB TCCR1B
- #define CSx1 CS11
- #define TCNTx TCNT1
- #define OCRxA OCR1A
- #define OCRxB OCR1B
- #define OCRxC OCR1C
- #define TIFRx TIFR1
- #define OCFxA OCF1A
- #define OCFxB OCF1B
- #define OCFxC OCF1C
- #define VECTxA TIMER1_COMPA_vect
- #define VECTxB TIMER1_COMPB_vect
- #define VECTxC TIMER1_COMPC_vect
- #endif
-
-
- void tone(uint8_t pin, uint16_t frequency, uint32_t duration)
- {
- uint8_t *reg;
- uint8_t mask;
- uint16_t inc;
- uint32_t count;
-
- if (pin >= CORE_NUM_TOTAL_PINS) return;
- if (frequency < MIN_FREQ) {
- frequency = MIN_FREQ;
- } else if (frequency > MAX_FREQ) {
- frequency = MAX_FREQ;
- }
- inc = (F_CPU / 16 + frequency / 2) / frequency;
- if (duration) {
- count = duration * frequency / 500;
- } else {
- count = 0;
- }
- if (!timer_acquired) {
- TIMSKx = 0;
- TCCRxA = 0;
- TCCRxB = (1<<CSx1);
- timer_acquired = 1;
- }
- PIN_REG_AND_MASK_LOOKUP(pin, reg, mask);
-
- if (!tone1_mask || (tone1_mask == mask && tone1_reg == reg)) {
- TIMSKx &= ~(1<<OCIExA);
- tone1_reg = reg;
- tone1_mask = mask;
- tone1_count = count;
- tone1_inc = inc;
- cli();
- *(reg + 2) &= ~mask;
- *(reg + 1) |= mask;
- OCRxA = TCNTx + inc;
- TIFRx |= (1<<OCFxA);
- sei();
- TIMSKx |= (1<<OCIExA);
- return;
- }
- if (!tone2_mask || (tone2_mask == mask && tone2_reg == reg)) {
- TIMSKx &= ~(1<<OCIExB);
- tone2_reg = reg;
- tone2_mask = mask;
- tone2_count = count;
- tone2_inc = inc;
- cli();
- *(reg + 2) &= ~mask;
- *(reg + 1) |= mask;
- OCRxB = TCNTx + inc;
- TIFRx |= (1<<OCFxB);
- sei();
- TIMSKx |= (1<<OCIExB);
- return;
- }
- if (!tone3_mask || (tone3_mask == mask && tone3_reg == reg)) {
- TIMSKx &= ~(1<<OCIExC);
- tone3_reg = reg;
- tone3_mask = mask;
- tone3_count = count;
- tone3_inc = inc;
- cli();
- *(reg + 2) &= ~mask;
- *(reg + 1) |= mask;
- OCRxC = TCNTx + inc;
- TIFRx |= (1<<OCFxC);
- sei();
- TIMSKx |= (1<<OCIExC);
- return;
- }
- }
-
- void noTone(uint8_t pin)
- {
- uint8_t *reg;
- uint8_t mask;
-
- if (pin >= CORE_NUM_TOTAL_PINS) return;
- PIN_REG_AND_MASK_LOOKUP(pin, reg, mask);
-
- if (tone1_mask == mask && tone1_reg == reg) {
- TIMSKx &= ~(1<<OCIExA);
- tone1_mask = 0;
- } else if (tone2_mask == mask && tone2_reg == reg) {
- TIMSKx &= ~(1<<OCIExB);
- tone2_mask = 0;
- } else if (tone3_mask == mask && tone3_reg == reg) {
- TIMSKx &= ~(1<<OCIExC);
- tone3_mask = 0;
- }
- if (!tone1_mask && !tone2_mask && !tone3_mask) {
- TCCRxA = (1<<WGMx0);
- timer_acquired = 0;
- }
- }
-
-
- ISR(VECTxA)
- {
- OCRxA += tone1_inc;
- *(tone1_reg) = tone1_mask;
- if (tone1_count > 0) {
- if ((--tone1_count) == 0) {
- *(tone1_reg + 2) &= ~tone1_mask;
- TIMSKx &= ~(1<<OCIExA);
- tone1_mask = 0;
- if (!tone2_mask && !tone3_mask) {
- TCCRxA = (1<<WGMx0);
- timer_acquired = 0;
- }
- }
- }
- }
-
- ISR(VECTxB)
- {
- OCRxB += tone2_inc;
- *(tone2_reg) = tone2_mask;
- if (tone2_count > 0) {
- if ((--tone2_count) == 0) {
- *(tone2_reg + 2) &= ~tone2_mask;
- TIMSKx &= ~(1<<OCIExB);
- tone2_mask = 0;
- if (!tone1_mask && !tone3_mask) {
- TCCRxA = (1<<WGMx0);
- timer_acquired = 0;
- }
- }
- }
- }
-
- ISR(VECTxC)
- {
- OCRxC += tone3_inc;
- *(tone3_reg) = tone3_mask;
- if (tone3_count > 0) {
- if ((--tone3_count) == 0) {
- *(tone3_reg + 2) &= ~tone3_mask;
- TIMSKx &= ~(1<<OCIExC);
- tone3_mask = 0;
- if (!tone1_mask && !tone2_mask) {
- TCCRxA = (1<<WGMx0);
- timer_acquired = 0;
- }
- }
- }
- }
-
-
|