This repository has been archived by the owner on Oct 11, 2019. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathservo.c
150 lines (115 loc) · 3.93 KB
/
servo.c
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
/* Copyright (C) 2010 Richard Barlow
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/
#include <io.h>
#include <signal.h>
#include <stdbool.h>
#include "servo.h"
/* Pulse width in us */
#define MIN_PULSE 600
#define MAX_PULSE 2400
/* Tick width in us */
/* This is actually the inverse of the tick width (tick width = 0.125us. */
#define TICK 8
/* Min/Max number of ticks per pulse */
#define MIN_TICK (MIN_PULSE*TICK)
#define MAX_TICK (MAX_PULSE*TICK)
#define TICK_PER_STEP ((uint8_t)((MAX_TICK-MIN_TICK)/SERVO_STEPS))
#define step_to_tick(step) (MIN_TICK+(TICK_PER_STEP*step))
#define tick_to_step(tick) ((tick-MIN_TICK)/TICK_PER_STEP)
/* Current servo being controlled */
static uint8_t curr_servo;
/* Current positions of the servos, stored as ticks */
static uint16_t position[8];
/* Whether each output is enabled or disabled */
static bool enabled[8];
/* Combine the value from position[n] and enabled[n]
so that the timer waits MIN_TICKs between these
*/
#define servo_get_ticks(n) ( enabled[n] ? position[n] : MIN_TICK )
/* Disable the servo output -- take control of it away from the timer */
#define output_disable() do { \
P4OUT &= ~1; \
P4SEL &= ~1; \
} while (0)
/* Enable the servo output -- give control of the the output to the timer */
#define output_enable() do { \
P4SEL |= 1; \
} while (0)
interrupt (TIMERB0_VECTOR) servo_timer_isr(void) {
/* Stop timer while we fiddle with it */
TBCTL &= ~MC_3;
/* Move onto the next servo */
curr_servo++;
if (curr_servo == 8)
curr_servo = 0;
/* Set the decoder output to the correct servo */
P4OUT &= ~(7<<2);
P4OUT |= curr_servo << 2;
if( enabled[curr_servo] )
/* Give the timer control of the output pin */
output_enable();
else
/* Keep the output pin low */
output_disable();
/* Enable the decoder, start of the pulse */
TBCCTL0 &= ~OUTMOD_7;
TBCCTL0 |= OUTMOD_OUT;
TBCCTL0 |= OUT;
/* Get everything setup to disable the decoder in a bit*/
TBCCTL0 |= OUTMOD_RESET;
TBCCR0 = servo_get_ticks(curr_servo);
/* Start timer running again */
TBR = 0;
TBCTL |= MC_CONT;
}
void servo_init(void) {
uint8_t i;
/* Disable all servos to start with */
for( i=0; i<8; i++ ) {
enabled[i] = false;
}
curr_servo = 0;
/* Configure the timer/servo output */
P4DIR |= 1; /* P4.0 is an output */
/* Start with it disabled as no servos are configured yet */
output_disable();
/* Setup the output selection pins */
P4OUT &= ~(7<<2);
P4DIR |= (7<<2); /* P4.{2,3,4} are outputs */
/* Setup Timer B */
TBCTL |= TBCLR; /* Clear timer settings */
TBCTL |= TBSSEL_SMCLK /* Source clock from SMCLK (16MHz) */
| ID_DIV2; /* Divide SMCLK by 2 */
TBCCTL0 |= OUTMOD_OUT; /* Set output mode to 'OUT bit value' */
TBCCTL0 |= OUT; /* Set CCR0 ouput high, start of a pulse */
TBCCTL0 |= OUTMOD_RESET /* Set oupput mode to 'Reset' */
| CCIE; /* Enable interrupt on CCR0 */
TBCCR0 = servo_get_ticks(curr_servo);
/* and off we go */
TBR = 0;
TBCTL |= MC_CONT;
}
void servo_set(uint8_t servo, uint16_t pos) {
if (servo > 7)
return;
if (pos > SERVO_STEPS)
pos = SERVO_STEPS;
position[servo] = step_to_tick(pos);
enabled[servo] = true;
}
uint16_t servo_get(uint8_t servo) {
if (servo > 7)
return SERVO_ERR;
return tick_to_step(position[servo]);
}