-
Notifications
You must be signed in to change notification settings - Fork 72
/
Copy pathrmp_test_msp430g2553.h
247 lines (217 loc) · 8.6 KB
/
rmp_test_msp430g2553.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
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
/******************************************************************************
Filename : rmp_test_msp430g2553.h
Author : pry
Date : 22/07/2017
Licence : The Unlicense; see LICENSE for details.
Description : The battery-powered testbench for MSP430G2553. This test header
is only used in junction with the tickless benchmark to show
how RMP tickless mode fits the battery-powered devices. Best used
with MSP-EXP430G2ET for ease of real-time power minitoring.
Possible execution paths for the hooks:
1. Tim - Tim
2. Tim - Ctx
2. Tim - Dly - Ctx
3. Dly - Ctx
5. Ctx - Ctx
TICC V21.6.1LTS -O4 -mf5 (OS timer enabled)
AP : 5mA - 10 days AAx2
LPM3 : 140uA - 350 days AAx2
LPM4 : 77uA - 650 days AAx2
******************************************************************************/
/* Include *******************************************************************/
#include "rmp.h"
/* End Include ***************************************************************/
/* Define ********************************************************************/
#define RMP_TICK_1SEC (512U)
#define RMP_BUSY_LOOP (256U)
/* End Define ****************************************************************/
/* Global ********************************************************************/
rmp_ptr_t Stack_1[48];
rmp_ptr_t Stack_2[48];
rmp_ptr_t Slice_Last;
void RMP_MSP430_Tim_Off(void);
void RMP_MSP430_Tim_Set(rmp_ptr_t Slice);
rmp_ptr_t RMP_MSP430_Tim_Get(void);
void _RMP_MSP430_Tickless_Handler(void);
void RMP_Tim_Hook(rmp_ptr_t Slice);
void RMP_Sched_Hook(void);
void RMP_Dly_Hook(rmp_ptr_t Slice);
void RMP_Start_Hook(void);
void RMP_Init_Idle(void);
/* End Global ****************************************************************/
/* Function:RMP_MSP430_Tim_Off ************************************************
Description : Turn off the timer altogether.
Input : None.
Output : None.
Return : None.
******************************************************************************/
void RMP_MSP430_Tim_Off(void)
{
/* Just clear everything */
TA0CTL=0U;
TA0CCTL0=0U;
TA0R=0U;
Slice_Last=0U;
}
/* End Function:RMP_MSP430_Tim_Off *******************************************/
/* Function:RMP_MSP430_Tim_Set ************************************************
Description : Reprogram the timer for MSP430 series.
TA0 clock = LFXT1/64 = 512Hz, which allows the MSP430 to hold up
ticks for more than two minutes. More aggressive configurations
however, can stretch this to hours or even days.
Input : rmp_ptr_t Slice - The tick value to set.
Output : None.
Return : None.
******************************************************************************/
void RMP_MSP430_Tim_Set(rmp_ptr_t Slice)
{
/* Stop the timer first, and clear all pending interrupts */
TA0CTL=0U;
TA0CCTL0=0U;
/* Clear timer */
Slice_Last=Slice;
TA0R=0U;
/* Program and reenable timer if we do have slices */
if(Slice!=0U)
{
TA0CCR0=Slice;
TA0CTL=TASSEL_1|ID_3|MC_1;
TA0CCTL0=CCIE;
}
/* Trigger an timer interrupt immediately if we don't have slices */
else
{
TA0CCTL0=CCIE|CCIFG;
}
}
/* End Function:RMP_MSP430_Tim_Set *******************************************/
/* Function:RMP_MSP430_Tim_Get ************************************************
Description : Get the elapsed time. The difficulty with MSP430 is that
1. it lacks one-shot mode, and 2. it lacks countdown mode, so
we need some tricks here.
Input : None.
Output : None.
Return : rmp_ptr_t - The current counter value.
******************************************************************************/
rmp_ptr_t RMP_MSP430_Tim_Get(void)
{
/* Stop the timer then read - this is okay because the timer will always
* be reprogrammed in the scheduler hook eventually when this gets called,
* so there's no need to run the timer after this read has been done. */
TA0CTL=0U;
/* An interrupt pending, all time must have been expended */
if((TA0CCTL0&CCIFG)!=0U)
return Slice_Last;
/* Only expended partially, return whatever that has passed */
return TA0R;
}
/* End Function:RMP_MSP430_Tim_Get *******************************************/
/* Function:_RMP_MSP430_Tickless_Handler **************************************
Description : Tickless interrupt routine for MSP430.
Input : None
Output : None.
Return : None.
******************************************************************************/
void _RMP_MSP430_Tickless_Handler(void)
{
RMP_MSP430_TIM_CLR();
/* All slices must have been expended for timer interrupt to be triggered */
_RMP_Tim_Handler(Slice_Last);
}
/* End Function:_RMP_MSP430_Tickless_Handler *********************************/
/* Function:RMP_Tim_Hook ******************************************************
Description : RMP timer hook.
Input : rmp_ptr_t Slice - Number of slices passed.
Output : None.
Return : None.
******************************************************************************/
void RMP_Tim_Hook(rmp_ptr_t Slice)
{
/* If we have a pending context switch, just turn off timer because the
* context switch hook will help us to turn it back on very soon. */
if(RMP_Sched_Pend!=0U)
{
RMP_MSP430_Tim_Off();
return;
}
/* Program a shot into the future - the MSP430 timer can do RMP_SLICE_MAX in
* a single delay so we just directly pass the slices until future to it.*/
RMP_MSP430_Tim_Set(_RMP_Tim_Future());
}
/* End Function:RMP_Tim_Hook *************************************************/
/* Function:RMP_Sched_Hook ****************************************************
Description : RMP scheduler hook.
Input : None.
Output : None.
Return : None.
******************************************************************************/
void RMP_Sched_Hook(void)
{
/* See how many slices is left, and register whatever has passed */
_RMP_Tim_Elapse(RMP_MSP430_Tim_Get());
/* Program a shot into the future - the MSP430 timer can do RMP_SLICE_MAX in
* a single delay so we just directly pass the slices until future to it.*/
RMP_MSP430_Tim_Set(_RMP_Tim_Future());
}
/* End Function:RMP_Sched_Hook ***********************************************/
/* Function:RMP_Dly_Hook ******************************************************
Description : RMP timer hook.
Input : rmp_ptr_t Slice - Number of slices to delay.
Output : None.
Return : None.
******************************************************************************/
void RMP_Dly_Hook(rmp_ptr_t Slice)
{
/* Just update the interval, this is always followed by a context switch */
_RMP_Tim_Elapse(RMP_MSP430_Tim_Get());
/* The context switch hook will help us to turn it back on very soon */
RMP_MSP430_Tim_Off();
}
/* End Function:RMP_Dly_Hook *************************************************/
/* Function:RMP_Start_Hook ****************************************************
Description : RMP start hook. Puts all unused GPIO port input pull-down.
Input : None.
Output : None.
Return : None.
******************************************************************************/
void RMP_Start_Hook(void)
{
/* P1.2 - UART TXD */
P1DIR=1U<<2;
P1OUT=0x00U;
P1REN=0xFBU;
/* P2.6 - XIN, P2.7 - XOUT */
P2DIR=0x80U;
P2OUT=0x00U;
P2REN=0x3FU;
P3DIR=0x00U;
P3REN=0xFFU;
P3OUT=0x00U;
/* Timer turned off initially */
RMP_MSP430_Tim_Off();
}
/* End Function:RMP_Start_Hook ***********************************************/
/* Function:RMP_Init_Idle *****************************************************
Description : RMP idle hook.
Input : None.
Output : None.
Return : None.
******************************************************************************/
void RMP_Init_Idle(void)
{
/* Put CPU in LPM3 if there are timer pending, LPM4 if nothing */
RMP_INT_MASK();
if(_RMP_Tim_Idle()==0)
{
RMP_DBG_S("> Init: entering LPM4 idle mode.\r\n");
__bis_SR_register(LPM4_bits|GIE);
}
else
{
RMP_DBG_S("> Init: entering LPM3 idle mode.\r\n");
__bis_SR_register(LPM3_bits|GIE);
}
}
/* End Function:RMP_Init_Idle ************************************************/
/* End Of File ***************************************************************/
/* Copyright (C) Evo-Devo Instrum. All rights reserved ***********************/