Contiki 2.5
tmr.c
1 /*
2  * Copyright (c) 2010, Mariano Alvira <mar@devl.org> and other contributors
3  * to the MC1322x project (http://mc1322x.devl.org)
4  * All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  * notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  * notice, this list of conditions and the following disclaimer in the
13  * documentation and/or other materials provided with the distribution.
14  * 3. Neither the name of the Institute nor the names of its contributors
15  * may be used to endorse or promote products derived from this software
16  * without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE INSTITUTE AND CONTRIBUTORS ``AS IS'' AND
19  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21  * ARE DISCLAIMED. IN NO EVENT SHALL THE INSTITUTE OR CONTRIBUTORS BE LIABLE
22  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28  * SUCH DAMAGE.
29  *
30  * This file is part of libmc1322x: see http://mc1322x.devl.org
31  * for details.
32  *
33  *
34  */
35 
36 #include <mc1322x.h>
37 #include <stdlib.h>
38 #include "tmr.h"
39 
40 /* Initialize timer. This just counts and interrupts, doesn't drive an output.
41  timer_num = 0, 1, 2, 3
42  rate = desired rate in Hz,
43  enable_int = whether to enable an interrupt on every cycle
44  Returns actual timer rate. */
45 uint32_t timer_setup_ex(int timer_num, uint32_t rate, int enable_int)
46 {
47  uint32_t actual_rate;
48  volatile struct TMR_struct *timer = TMR_ADDR(timer_num);
49  int log_divisor = 0;
50  uint32_t period;
51 
52  /* Turn timer off */
53  TMR0->ENBL &= ~(1 << timer_num);
54 
55  /* Calculate optimal rate */
56  for (log_divisor = 0; log_divisor < 8; log_divisor++)
57  {
58  int denom = (rate * (1 << log_divisor));
59  period = (REF_OSC + denom/2) / denom;
60  if (period <= 65535)
61  break;
62  }
63  if (log_divisor >= 8)
64  {
65  period = 65535;
66  log_divisor = 7;
67  }
68 
69  if (period < 2) period = 2;
70 
71  actual_rate = REF_OSC / (period * (1 << log_divisor));
72 
73  /* Set up timer */
74 
75  timer->LOAD = 0;
76  timer->CMPLD1 = (period - 1);
77  timer->COMP1 = timer->CMPLD1;
78  timer->CNTR = timer->LOAD;
79  timer->SCTRL = 0;
80  timer->CSCTRLbits = (struct TMR_CSCTRL) {
81  .CL1 = 0x01, // Reload COMP1 when COMP1 matches
82  };
83  timer->CTRLbits = (struct TMR_CTRL) {
84  .COUNT_MODE = 1, // Count rising edge of primary source
85  .PRIMARY_CNT_SOURCE = 8 + log_divisor, // Peripheral clock divided by (divisor)
86  .LENGTH = 1, // At compare, reset to LOAD
87  };
88 
89  TMR0->ENBL |= (1 << timer_num);
90 
91  if (enable_int) {
92  enable_irq(TMR);
93  timer->SCTRLbits.TCFIE = 1;
94  }
95 
96  return actual_rate;
97 }