¿Cómo creo una interrupción de temporizador con Arduino?

Estoy tratando de crear una interrupción de retardo de tiempo con Arduino. Me gustaría usar la función interrupts(), porque es una interrupción interna.

Ejemplo: Digamos que me gustaría hacer que una luz parpadee y se apague, con solo el tiempo de la interrupción.

Hay un código de muestra, pero usa interrupciones externas (attachInterrupt()). Me gustaría seguir usando las interrupciones internas.

Creo que el punto que también mostró Kortuk es que addedInterrupt es algo abstracto, no estás adjuntando ningún componente externo :)
Este artículo podría ayudarte. engblaze.com/…

Respuestas (4)

El blog de Noah Stahl tiene un ejemplo de parpadeo de un LED con Timer2 . Con eso y la hoja de datos, debería poder adaptarlo a cualquier interrupción que desee usar, es decir, la interrupción cuya función normal puede darse el lujo de renunciar o está dispuesto a modificar. Timer2 se usa generalmente para algunas funciones PWM.

Su ejemplo cita el ATmega2560; Puedo confirmar que también funciona con un ATmega328p. Busque en su sitio más ejemplos útiles de interrupción de Arduino.

Editar:

Aquí está mi versión ligeramente editada, principalmente en los comentarios, del código de Noah. Llame a Timer2init() desde la función de configuración() de Arduino después de inicializar cualquier estructura de datos o hardware relacionado, porque el tiempo y la interrupción comenzarán una vez que lo haga.

F/ex, lo usé para multiplexar una pantalla de 7 segmentos de 3 dígitos, así que antes de inicializar el temporizador, inicialicé los registros de E/S de la pantalla y borré los datos de la pantalla en el lugar donde el ISR los buscará.

Hay una tabla en los comentarios de algunos datos de tiempo útiles de la hoja de datos y mis propios cálculos como referencia para configurar otro esquema de tiempo.

La macro ISR() se ocupa de crear un código de entrada y salida de interrupción para una ISR en lugar de la entrada y salida de una función normal, y de vincularlo con el vector de interrupción adecuado. El resto de esa función es 1) el código que se ejecutará en cada interrupción y 2) el código de código para restablecer el temporizador para la próxima interrupción.

Tal como está escrito, esto debería caer en un boceto .pde o .ino (o un archivo .cpp, si usa eclipse, f/ex). El boceto debe #definir LEDPIN y setup() debe llamar a Timer2init(). La función de bucle puede estar vacía o no; el LED debería comenzar a parpadear en la descarga (bueno, literalmente, después de llamar a Timer2init()).

/*
 * From sample interrupt code published by Noah Stahl on his blog, at:
 * http://arduinomega.blogspot.com/p/arduino-code.html
 * 
 */


/*** FUNC

Name:           Timer2init

Function:       Init timer 2 to interrupt periodically. Call this from 
                the Arduino setup() function.

Description:    The pre-scaler and the timer count divide the timer-counter
                clock frequency to give a timer overflow interrupt rate:

                Interrupt rate =  16MHz / (prescaler * (255 - TCNT2))

        TCCR2B[b2:0]   Prescaler    Freq [KHz], Period [usec] after prescale
          0x0            (TC stopped)     0         0
          0x1                1        16000.        0.0625
          0x2                8         2000.        0.500
          0x3               32          500.        2.000
          0x4               64          250.        4.000
          0x5              128          125.        8.000
          0x6              256           62.5      16.000
          0x7             1024           15.625    64.000


Parameters: void

Returns:    void

FUNC ***/

void Timer2init() {

    // Setup Timer2 overflow to fire every 8ms (125Hz)
    //   period [sec] = (1 / f_clock [sec]) * prescale * (255-count)
    //                  (1/16000000)  * 1024 * (255-130) = .008 sec


    TCCR2B = 0x00;        // Disable Timer2 while we set it up

    TCNT2  = 130;         // Reset Timer Count  (255-130) = execute ev 125-th T/C clock
    TIFR2  = 0x00;        // Timer2 INT Flag Reg: Clear Timer Overflow Flag
    TIMSK2 = 0x01;        // Timer2 INT Reg: Timer2 Overflow Interrupt Enable
    TCCR2A = 0x00;        // Timer2 Control Reg A: Wave Gen Mode normal
    TCCR2B = 0x07;        // Timer2 Control Reg B: Timer Prescaler set to 1024
}



/*** FUNC

Name:       Timer2 ISR

Function:   Handles the Timer2-overflow interrupt

Description:    Maintains the 7-segment display

Parameters: void

Returns:    void

FUNC ***/

ISR(TIMER2_OVF_vect) {
    static unsigned int led_state = 0; // LED state

    led_state = !led_state;         // toggles the LED state
    digitalWrite(TOGGLE_PIN, led_state);

    TCNT2 = 130;     // reset timer ct to 130 out of 255
    TIFR2 = 0x00;    // timer2 int flag reg: clear timer overflow flag
};
(@Kortuk: El comentario al que se refiere fue mi observación de varios comentaristas aquí y no estaba dirigido a usted personalmente y era innecesario. Me disculpo y lo eliminé). Amplié mi respuesta como sugirió y espero que sea ahora no sólo demostrativo, sino también instructivo. Incluye comentarios que he escrito en el código para mi propio uso (lo que significa: si puedo entenderlos dentro de 6 meses, otra persona también podrá hacerlo), así como algunas instrucciones de "cómo usar" en el respuesta. Gracias por tus sugerencias.
Tenga en cuenta que las preescalas de 32 y 128 no están disponibles para timer0 y timer1 (al menos con atmega328).
Es bueno saberlo, gracias. Uso esto para Timer2 (hasta ahora) y es básicamente un complemento.

La función attachInterrupt() en realidad está adjuntando una interrupción a un cambio de estado externo en un pin, no tiene otras opciones.

En la misma página , las opciones de modo se enumeran como:

El modo define cuándo debe activarse la interrupción. Cuatro constantes están predefinidas como valores válidos:

  • BAJO para activar la interrupción siempre que el pin esté bajo,
  • CAMBIAR para activar la interrupción cada vez que el pin cambie de valor
  • RISING para disparar cuando el pin va de bajo a alto,
  • FALLING para cuando el pin va de mayor a menor.

Lamento ser el portador de malas noticias, esa es una de las primeras cosas que busqué también.

Creo que quiere decir que quiere usar un temporizador interno, en lugar de un dispositivo externo... pero no conozco muy bien a Arduino, así que no puedo decir si es posible.
@clabacchio, digo que la única opción es usar un disparador externo, no hay una función de temporizador interno.
Ah, bien :) pero al menos, ¿las placas Arduino tienen algún temporizador?
Sí, así es como logran cosas como la demora.
@Kortuk, tal vez este sea un problema para comprender el significado que está tratando de transmitir, sin embargo, el AVR uC en Arduino, es decir, ATmega328 tiene un oscilador incorporado de 8MHz, por lo que no necesita un disparador externo. El reloj del sistema, es el que alimenta los 3 temporizadores diferentes (Timer0, Timer1, Timer2), y multiplicadores individuales. Si bien no he analizado esto en detalle recientemente, pero por lo que recuerdo, Arduino usa uno de ellos para las funciones de temporización/retraso y otro para PWM, dejando un temporizador sin usar. El usuario puede conectarse al gratuito (que es un temporizador de 8 bits), o incluso conectarse en cadena con otros temporizadores.
@ icarus74, eso suena razonable, estaba respondiendo que arduino no tiene una función integrada que pueda encontrar para permitir el uso de una interrupción basada en su temporizador. Declaro específicamente que no puede usar el temporizador para una interrupción ya que el paquete arduino está configurado. Para mí, tendría sentido que un usuario pudiera configurar su propia interrupción escribiendo una biblioteca, solo explique lo que sé que está disponible.
@ icarus74 ATMega328 realmente tiene 3 temporizadores (uno es 16b y dos son 8b) pero Arduino los usa todos. Uno se usa para funciones como delay() y millis() y los tres se usan para PWM (puede encontrar más información en la función 'init()', archivo 'wiring.c' en Arduino IDE).

Este artículo sobre PWM aclarará muchas de sus dudas sobre el uso de temporizadores Arduino. Hay dos temporizadores de 8 bits y un temporizador de 16 bits en Arduino. No hay una API de alto nivel para conectar la función ISR directamente a los temporizadores, que se envía con el SDK de Arduino (es decir, como una biblioteca estándar), sino un método de nivel algo más bajo para configurar registros de funciones especiales y aritmética de bits. operaciones sobre ellos. Sin embargo, hay una biblioteca aportada por el usuario llamada Timer one .

En realidad, hay varias combinaciones de temporizador diferentes posibles dependiendo de a qué Arduino se haga referencia. La respuesta es engañosa.
@SeeminglySo, ¿quieres dar más detalles? Si está hablando del hardware Arduino, tenga en cuenta que la respuesta está en el contexto de la pregunta y también en el momento en que se hace la pregunta.
El Arduino Mega (basado en ATmega1280) se lanzó el 26 de marzo de 2009 y el Mega 2560 (ATmega2560) se lanzó el 24 de septiembre de 2010, ambos mucho antes de que se hiciera esta pregunta. Ambos microcontroladores tienen más del temporizador/contador de 2x 8 bits y 1x 16 bits especificado en la respuesta.
La mayoría de las interacciones que he visto hasta ahora tienen una referencia inequívoca a Arduino para referirse a Duemilanove o Uno, es decir, las placas basadas en la serie 328. Otros tableros siempre han sido calificados explícitamente por la serie uP no. o Mega, Nano, Micro, etc. De todos modos, humildemente aceptaré la corrección. En este contexto, es mejor una aclaración.

Arduino está usando los tres temporizadores en ATMega328. Timer1(16 bits) se usa para funciones como delay()y millis()para la salida PWM en los pines 5 y 6. Los otros dos temporizadores Timer0se Timer2usan para la salida PWM en los pines 3, 9, 10, 11.

Entonces, no hay una función Arduino para la interrupción del temporizador. Pero, hay una manera. Puede usar este código para habilitar la interrupción del temporizador en Timer2:

ISR(TIMER2_OVF_vect) {
  // Interrupt routine.
}

void setup() {
  // Enable Timer2 interrupt.
  TIMSK2 = (0<<OCIE2A) | (1<<TOIE2);
}

void loop() {
  // Your main loop.
}

Escribí este código sin probarlo, así que es posible que haya cometido un error. En ese caso, consulte la hoja de datos, p.156 .

Si desea cambiar la frecuencia del temporizador (prescaler), simplemente cambie el registro TCCR2A. Para obtener más información, consulte la hoja de datos de la página 153. ¡Pero si cambia la frecuencia del temporizador, también cambia la frecuencia de la señal PWM en dos pines de salida!

AFAIK en ATmega328 Timer0y Timer2son de 8 bits y solo Timer1son de 16 bits.
No esto no es correcto. Es Timer0, no Timer1, el que se usa para delay() y millis() y para la salida PWM en los pines 5 y 6. Timer0 es un temporizador de 8 bits. Véase, por ejemplo , Temporizadores e interrupciones de Arduino .