2007-11-29 02:44:05 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2007, Swedish Institute of Computer Science.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. Neither the name of the Institute nor the names of its contributors
|
|
|
|
* may be used to endorse or promote products derived from this software
|
|
|
|
* without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
|
|
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
|
|
|
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
|
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
|
|
|
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
|
|
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
|
|
|
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
|
|
* SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
* This file is part of the Contiki operating system.
|
|
|
|
*
|
2010-02-28 21:29:19 +00:00
|
|
|
* $Id: rtimer-arch.c,v 1.10 2010/02/28 21:29:19 dak664 Exp $
|
2007-11-29 02:44:05 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* \file
|
|
|
|
* AVR-specific rtimer code
|
2011-08-05 15:14:35 -04:00
|
|
|
* Defaults to Timer3 for those ATMEGAs that have it.
|
|
|
|
* If Timer3 not present Timer1 will be used.
|
2007-11-29 02:44:05 +00:00
|
|
|
* \author
|
|
|
|
* Fredrik Osterlind <fros@sics.se>
|
2007-12-11 17:21:14 +00:00
|
|
|
* Joakim Eriksson <joakime@sics.se>
|
2007-11-29 02:44:05 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* OBS: 8 seconds maximum time! */
|
|
|
|
|
|
|
|
#include <avr/io.h>
|
|
|
|
#include <avr/interrupt.h>
|
2010-02-18 17:21:44 +00:00
|
|
|
#include <stdio.h>
|
2007-11-29 02:44:05 +00:00
|
|
|
|
2009-09-07 12:02:58 +00:00
|
|
|
#include "sys/energest.h"
|
2007-11-29 02:44:05 +00:00
|
|
|
#include "sys/rtimer.h"
|
|
|
|
#include "rtimer-arch.h"
|
2008-11-29 15:55:02 +00:00
|
|
|
|
|
|
|
#if defined(__AVR_ATmega1281__) || defined(__AVR_ATmega1284P__)
|
2010-02-16 21:48:38 +00:00
|
|
|
#define ETIMSK TIMSK3
|
|
|
|
#define ETIFR TIFR3
|
|
|
|
#define TICIE3 ICIE3
|
2008-11-29 15:55:02 +00:00
|
|
|
|
2010-02-16 21:48:38 +00:00
|
|
|
//Has no 'C', so we just set it to B. The code doesn't really use C so this
|
2010-02-26 20:51:48 +00:00
|
|
|
//is safe to do but lets it compile. Probably should enable the warning if
|
|
|
|
//it is ever used on other platforms.
|
|
|
|
//#warning no OCIE3C in timer3 architecture, hopefully it won't be needed!
|
|
|
|
|
2010-02-16 21:48:38 +00:00
|
|
|
#define OCIE3C OCIE3B
|
|
|
|
#define OCF3C OCF3B
|
|
|
|
#endif
|
|
|
|
|
2011-02-07 13:46:34 -05:00
|
|
|
#if defined(__AVR_AT90USB1287__) || defined(__AVR_ATmega128RFA1__)
|
2010-02-16 21:48:38 +00:00
|
|
|
#define ETIMSK TIMSK3
|
|
|
|
#define ETIFR TIFR3
|
|
|
|
#define TICIE3 ICIE3
|
2008-11-29 15:55:02 +00:00
|
|
|
#endif
|
2011-08-03 11:18:55 -04:00
|
|
|
|
|
|
|
#if defined(__AVR_ATmega328P__) || defined(__AVR_ATmega644__)
|
|
|
|
#define TIMSK TIMSK1
|
|
|
|
#define TICIE1 ICIE1
|
|
|
|
#define TIFR TIFR1
|
|
|
|
#endif
|
|
|
|
|
2011-08-15 15:06:38 -04:00
|
|
|
/* Track flow through rtimer interrupts*/
|
|
|
|
#if DEBUGFLOWSIZE&&0
|
|
|
|
extern uint8_t debugflowsize,debugflow[DEBUGFLOWSIZE];
|
|
|
|
#define DEBUGFLOW(c) if (debugflowsize<(DEBUGFLOWSIZE-1)) debugflow[debugflowsize++]=c
|
|
|
|
#else
|
|
|
|
#define DEBUGFLOW(c)
|
|
|
|
#endif
|
|
|
|
|
2007-11-29 02:44:05 +00:00
|
|
|
/*---------------------------------------------------------------------------*/
|
2011-08-05 15:14:35 -04:00
|
|
|
#if defined(TCNT3) && RTIMER_ARCH_PRESCALER
|
2008-10-14 09:44:12 +00:00
|
|
|
ISR (TIMER3_COMPA_vect) {
|
2011-08-15 15:06:38 -04:00
|
|
|
DEBUGFLOW('/');
|
2007-11-29 02:44:05 +00:00
|
|
|
ENERGEST_ON(ENERGEST_TYPE_IRQ);
|
|
|
|
|
2011-08-13 11:35:03 -04:00
|
|
|
/* Disable rtimer interrupts */
|
2007-11-29 02:44:05 +00:00
|
|
|
ETIMSK &= ~((1 << OCIE3A) | (1 << OCIE3B) | (1 << TOIE3) |
|
|
|
|
(1 << TICIE3) | (1 << OCIE3C));
|
2010-02-26 20:51:48 +00:00
|
|
|
|
2011-08-13 11:35:03 -04:00
|
|
|
#if RTIMER_CONF_NESTED_INTERRUPTS
|
|
|
|
/* Enable nested interrupts. Allows radio interrupt during rtimer interrupt. */
|
|
|
|
/* All interrupts are enabled including recursive rtimer, so use with caution */
|
|
|
|
sei();
|
|
|
|
#endif
|
|
|
|
|
2007-11-29 02:44:05 +00:00
|
|
|
/* Call rtimer callback */
|
|
|
|
rtimer_run_next();
|
|
|
|
|
|
|
|
ENERGEST_OFF(ENERGEST_TYPE_IRQ);
|
2011-08-15 15:06:38 -04:00
|
|
|
DEBUGFLOW('\\');
|
2008-11-29 15:55:02 +00:00
|
|
|
}
|
|
|
|
|
2011-08-03 11:18:55 -04:00
|
|
|
#elif RTIMER_ARCH_PRESCALER
|
|
|
|
#warning "No Timer3 in rtimer-arch.c - using Timer1 instead"
|
|
|
|
ISR (TIMER1_COMPA_vect) {
|
2011-08-15 15:06:38 -04:00
|
|
|
DEBUGFLOW('/');
|
2011-08-03 11:18:55 -04:00
|
|
|
TIMSK &= ~((1<<TICIE1)|(1<<OCIE1A)|(1<<OCIE1B)|(1<<TOIE1));
|
|
|
|
|
|
|
|
rtimer_run_next();
|
2011-08-15 15:06:38 -04:00
|
|
|
DEBUGFLOW('\\');
|
2011-08-03 11:18:55 -04:00
|
|
|
}
|
2008-10-14 09:44:12 +00:00
|
|
|
|
2007-12-11 17:21:14 +00:00
|
|
|
#endif
|
2007-11-29 02:44:05 +00:00
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
rtimer_arch_init(void)
|
|
|
|
{
|
2011-08-05 15:14:35 -04:00
|
|
|
#if RTIMER_ARCH_PRESCALER
|
2007-11-29 02:44:05 +00:00
|
|
|
/* Disable interrupts (store old state) */
|
|
|
|
uint8_t sreg;
|
|
|
|
sreg = SREG;
|
|
|
|
cli ();
|
|
|
|
|
2007-12-11 17:21:14 +00:00
|
|
|
#ifdef TCNT3
|
2010-02-26 20:51:48 +00:00
|
|
|
/* Disable all timer functions */
|
2007-11-29 02:44:05 +00:00
|
|
|
ETIMSK &= ~((1 << OCIE3A) | (1 << OCIE3B) | (1 << TOIE3) |
|
|
|
|
(1 << TICIE3) | (1 << OCIE3C));
|
2010-02-26 20:51:48 +00:00
|
|
|
/* Write 1s to clear existing timer function flags */
|
2007-11-29 02:44:05 +00:00
|
|
|
ETIFR |= (1 << ICF3) | (1 << OCF3A) | (1 << OCF3B) | (1 << TOV3) |
|
|
|
|
(1 << OCF3C);
|
|
|
|
|
|
|
|
/* Default timer behaviour */
|
|
|
|
TCCR3A = 0;
|
|
|
|
TCCR3B = 0;
|
|
|
|
TCCR3C = 0;
|
|
|
|
|
|
|
|
/* Reset counter */
|
|
|
|
TCNT3 = 0;
|
|
|
|
|
2011-08-05 15:14:35 -04:00
|
|
|
#if RTIMER_ARCH_PRESCALER==1024
|
2008-11-29 15:55:02 +00:00
|
|
|
TCCR3B |= 5;
|
2011-08-05 15:14:35 -04:00
|
|
|
#elif RTIMER_ARCH_PRESCALER==256
|
|
|
|
TCCR3B |= 4;
|
|
|
|
#elif RTIMER_ARCH_PRESCALER==64
|
|
|
|
TCCR3B |= 3;
|
|
|
|
#elif RTIMER_ARCH_PRESCALER==8
|
|
|
|
TCCR3B |= 2;
|
|
|
|
#elif RTIMER_ARCH_PRESCALER==1
|
|
|
|
TCCR3B |= 1;
|
|
|
|
#else
|
|
|
|
#error Timer3 PRESCALER factor not supported.
|
|
|
|
#endif
|
2008-11-29 15:55:02 +00:00
|
|
|
|
2011-08-03 11:18:55 -04:00
|
|
|
#elif RTIMER_ARCH_PRESCALER
|
|
|
|
/* Leave timer1 alone if PRESCALER set to zero */
|
|
|
|
/* Obviously you can not then use rtimers */
|
2007-11-29 02:44:05 +00:00
|
|
|
|
2011-08-03 11:18:55 -04:00
|
|
|
TIMSK &= ~((1<<TICIE1)|(1<<OCIE1A)|(1<<OCIE1B)|(1<<TOIE1));
|
|
|
|
TIFR |= (1 << ICF1) | (1 << OCF1A) | (1 << OCF1B) | (1 << TOV1);
|
|
|
|
|
|
|
|
/* Default timer behaviour */
|
|
|
|
TCCR1A = 0;
|
|
|
|
TCCR1B = 0;
|
|
|
|
|
|
|
|
/* Reset counter */
|
|
|
|
TCNT1 = 0;
|
|
|
|
|
|
|
|
/* Start clock */
|
|
|
|
#if RTIMER_ARCH_PRESCALER==1024
|
|
|
|
TCCR1B |= 5;
|
|
|
|
#elif RTIMER_ARCH_PRESCALER==256
|
|
|
|
TCCR1B |= 4;
|
|
|
|
#elif RTIMER_ARCH_PRESCALER==64
|
|
|
|
TCCR1B |= 3;
|
|
|
|
#elif RTIMER_ARCH_PRESCALER==8
|
|
|
|
TCCR1B |= 2;
|
|
|
|
#elif RTIMER_ARCH_PRESCALER==1
|
|
|
|
TCCR1B |= 1;
|
|
|
|
#else
|
2011-08-05 15:14:35 -04:00
|
|
|
#error Timer1 PRESCALER factor not supported.
|
2007-12-11 17:21:14 +00:00
|
|
|
#endif
|
|
|
|
|
2011-08-03 11:18:55 -04:00
|
|
|
#endif /* TCNT3 */
|
|
|
|
|
2007-11-29 02:44:05 +00:00
|
|
|
/* Restore interrupt state */
|
|
|
|
SREG = sreg;
|
2011-08-05 15:14:35 -04:00
|
|
|
#endif /* RTIMER_ARCH_PRESCALER */
|
2007-11-29 02:44:05 +00:00
|
|
|
}
|
|
|
|
/*---------------------------------------------------------------------------*/
|
|
|
|
void
|
|
|
|
rtimer_arch_schedule(rtimer_clock_t t)
|
|
|
|
{
|
2011-08-05 15:14:35 -04:00
|
|
|
#if RTIMER_ARCH_PRESCALER
|
2007-11-29 02:44:05 +00:00
|
|
|
/* Disable interrupts (store old state) */
|
|
|
|
uint8_t sreg;
|
|
|
|
sreg = SREG;
|
|
|
|
cli ();
|
2011-08-15 15:06:38 -04:00
|
|
|
DEBUGFLOW(':');
|
2007-12-11 17:21:14 +00:00
|
|
|
#ifdef TCNT3
|
2007-11-29 02:44:05 +00:00
|
|
|
/* Set compare register */
|
|
|
|
OCR3A = t;
|
2010-02-26 20:51:48 +00:00
|
|
|
/* Write 1s to clear all timer function flags */
|
2007-11-29 02:44:05 +00:00
|
|
|
ETIFR |= (1 << ICF3) | (1 << OCF3A) | (1 << OCF3B) | (1 << TOV3) |
|
|
|
|
(1 << OCF3C);
|
2010-02-26 20:51:48 +00:00
|
|
|
/* Enable interrupt on OCR3A match */
|
2008-11-29 15:55:02 +00:00
|
|
|
ETIMSK |= (1 << OCIE3A);
|
|
|
|
|
2011-08-03 11:18:55 -04:00
|
|
|
#elif RTIMER_ARCH_PRESCALER
|
|
|
|
/* Set compare register */
|
|
|
|
OCR1A = t;
|
|
|
|
TIFR |= (1 << ICF1) | (1 << OCF1A) | (1 << OCF1B) | (1 << TOV1);
|
|
|
|
TIMSK |= (1 << OCIE1A);
|
2007-11-29 02:44:05 +00:00
|
|
|
|
2007-12-11 17:21:14 +00:00
|
|
|
#endif
|
|
|
|
|
2007-11-29 02:44:05 +00:00
|
|
|
/* Restore interrupt state */
|
|
|
|
SREG = sreg;
|
2011-08-05 15:14:35 -04:00
|
|
|
#endif /* RTIMER_ARCH_PRESCALER */
|
2007-11-29 02:44:05 +00:00
|
|
|
}
|