-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathclock16.h
62 lines (46 loc) · 1.64 KB
/
clock16.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
// -*- mode: c++; indent-tabs-mode: nil; -*-
#ifndef ARDUINO_MINUS_MINUS_CLOCK16_H
#define ARDUINO_MINUS_MINUS_CLOCK16_H
#ifdef ARDUINO_MINUS_MINUS_CLOCK32_H
#error "Only one clock resolution can be used"
#endif
/* This file should be included - once - in the main C++ file of the
application.
Other headers should - if at all possible - not include this file and
use a template instead.
*/
/** This is a Clock with 16bit clock resolution.
The value from Clock16::millis() will wrap around after about 65 seconds.
The recommended way to use a clock value in user code is:
typename Clock16::time_res_t now = Arduino16::millis();
Motivation: With Clock16 instead of Clock32, Lars has seen a code size
reduction of 238 bytes with avr-gcc 4.6.1.
*/
// Define this for a slower clock (for low power modes). Note that you must
// also set the prescaler, for now, unless the default of 64 is used.
#ifndef CLOCK16_PRESCALE
# define CLOCK16_PRESCALE 64
#endif
typedef _Clock<uint16_t,Timer0> Clock16;
Clock16 clock;
/*
* Implementation of the clock ISR for 16 bits resolution
*/
ISR(TIMER0_OVF_vect)
{
// copy these to local variables so they can be stored in registers
// (volatile variables must be read from memory on every access)
typename Clock16::time_res_t m = Clock16::timer_millis;
uint16_t f = Clock16::timer_fract;
m += (CLOCK16_PRESCALE * (256 / (F_CPU / 1000000))) / 1000;
f += (CLOCK16_PRESCALE * (256 / (F_CPU / 1000000))) % 1000;
if (f >= 1000)
{
f -= 1000;
m += 1;
}
Clock16::timer_fract = f;
Clock16::timer_millis = m;
Clock16::timer_overflow_count++;
}
#endif