-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathMRSC_Adapter_3Pin_Servo.ino
208 lines (166 loc) · 8.06 KB
/
MRSC_Adapter_3Pin_Servo.ino
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
/* Standalone MRSC "Micro Rc Stability Control" converter for 3 pin RC steering servo
MPU: Atmega 328P 3.3V, 8MHz
Board: Pro Mini
MPU-6050 board: GY-521
Pins:
- Steering input 4
- Remote gain input 5 (standard servo pulses from spare channel)
- MRSC Gain potentiometer input A1 (if no remote gain is wired)
- MRSC direction inversion switch A2
- Steering output A0
- MPU-6050 SDA A4
- MPU-6050 SCL A5
*/
const float codeVersion = 0.3; // Software revision
//
// =======================================================================================================
// LIBRARIES
// =======================================================================================================
//
#include <Wire.h> // I2C library (for the MPU-6050 gyro /accelerometer)
#include <Servo.h> // Servo library
#include "mpu.h" // MPU-6050 handling (.h file in sketch folder)
//
// =======================================================================================================
// PIN ASSIGNMENTS & GLOBAL VARIABLES
// =======================================================================================================
//
volatile uint8_t prev; // remembers state of input bits from previous interrupt
volatile uint32_t risingEdge[2]; // time of last rising edge for each channel
volatile uint32_t uSec[2]; // the latest measured pulse width for each channel
// Create Servo objects
Servo servoSteering;
// Servo limits (initial value = center position = 90° / 1500uSec)
// Limits are adjusted during the first steering operation!
// Always move the steering wheel inside its entire range after switching on the car
byte limSteeringL = 90, limSteeringR = 90; // usually 45 - 135° (90° is the servo center position)
int limuSecL = 1500, limuSecR = 1500; // usually 1000 - 2000 uSec (1500 uSec is in the servo center position)
// MRSC gain
byte mrscGain = 80; // This MRSC gain applies, if you don't have an adjustment pot and don't call readInputs()
float headingMultiplier = 2.0; // adjust until front wheels stay in parallel with the ground, if the car is swiveled around
// Switch states
boolean mpuInversed = false;
// Pin definition (don't change servo input pins, interrupt routine is hardcoded)
#define INPUT_STEERING 4
#define INPUT_GAIN 5
#define OUTPUT_STEERING A0
#define GAIN_POT A1
#define INVERSE_MPU_DIRECTION A2
//
// =======================================================================================================
// SERVO INPUT PIN CHANGE INTERRUPT ROUTINE
// =======================================================================================================
// Based on: http://ceptimus.co.uk/?p=66
ISR(PCINT2_vect) { // one or more of pins 4~5 have changed state
uint32_t now = micros();
uint8_t curr = PIND; // current state of the 2 input bits
uint8_t changed = curr ^ prev; // bitwise XOR (= bit has changed)
uint8_t channel = 0;
for (uint8_t mask = B00010000; mask <= B00100000 ; mask <<= 1) { // do it with bit 4 and 5 masked
if (changed & mask) { // this pin has changed state
if (curr & mask) { // Rising edge, so remember time
risingEdge[channel] = now;
}
else { // Falling edge, so store pulse width
uSec[channel] = now - risingEdge[channel];
}
}
channel++;
}
prev = curr;
}
//
// =======================================================================================================
// MAIN ARDUINO SETUP (1x during startup)
// =======================================================================================================
//
void setup() {
// Configure inputs
pinMode(INVERSE_MPU_DIRECTION, INPUT_PULLUP);
pinMode(GAIN_POT, INPUT);
// Activate servo signal input pullup resistors
pinMode(INPUT_STEERING, INPUT_PULLUP);
pinMode(INPUT_GAIN, INPUT_PULLUP);
// Interrupt settings. See: http://www.atmel.com/Images/Atmel-42735-8-bit-AVR-Microcontroller-ATmega328-328P_Datasheet.pdf
PCMSK2 |= B00110000; // PinChangeMaskRegister: set the mask to allow pins 4-5 to generate interrupts (see page 94)
PCICR |= B00000100; // PinChangeInterruptControlRegister: enable interupt for port D (Interrupt Enable 2, see page 92)
// Servo pins
servoSteering.attach(OUTPUT_STEERING);
servoSteering.write((limSteeringL + limSteeringR) / 2); // Servo to center position
// MPU 6050 accelerometer / gyro setup
setupMpu6050();
}
//
// =======================================================================================================
// DETECT STEERING RANGE
// =======================================================================================================
//
void detectSteeringRange() {
// input signal calibration (for center point only)
int steeringuSec = uSec[0];
if (steeringuSec > 500 && steeringuSec < limuSecL) limuSecL = steeringuSec; // around 1000uS
if (steeringuSec < 2500 && steeringuSec > limuSecR) limuSecR = steeringuSec; // around 2000uS
// output signal calibration
int servoAngle = map(uSec[0], 1000, 2000, 45, 135); // The range usually is 45 to 135° (+/- 45°)
if (servoAngle > 20 && servoAngle < limSteeringL) limSteeringL = servoAngle;
if (servoAngle < 160 && servoAngle > limSteeringR) limSteeringR = servoAngle;
}
//
// =======================================================================================================
// READ INPUTS
// =======================================================================================================
//
void readInputs() {
mrscGain = map(analogRead(GAIN_POT), 0, 255, 0, 100);
mpuInversed = !digitalRead(INVERSE_MPU_DIRECTION);
}
//
// =======================================================================================================
// SIGNAL VALIDITY CHECK
// =======================================================================================================
//
void checkValidity() {
for (int i = 0; i <= 1; i++) {
if (uSec[i] < 800 || uSec[i] > 2200) uSec[i] = 1500; // go to neutral, if an invalid servo signal arrives
}
}
//
// =======================================================================================================
// MRSC (MICRO RC STABILITY CONTROL) CALCULATIONS
// =======================================================================================================
// For cars with stability control (steering overlay depending on gyro yaw rate)
void mrsc() {
int steeringAngle;
long gyroFeedback;
// Read remote gain signal (standard servo pulses)
mrscGain = map(uSec[1], 1000, 2000, 0, 100);
// Read sensor data
readMpu6050Data();
// Compute steering compensation overlay
int turnRateSetPoint = map(uSec[0], limuSecL, limuSecR, -50, 50); // turnRateSetPoint = steering angle (1000 to 2000us) = -50 to 50
int steering = abs(turnRateSetPoint); // this value is required to compute the gain later on and is always positive
int gain = map(steering, 0, 50, mrscGain, (mrscGain / 5)); // MRSC gain around center position is 5 times more!
if (steering < 5 && mrscGain > 85) { // Straight run @ high gain, "heading hold" mode -------------
gyroFeedback = yaw_angle * headingMultiplier; // degrees
}
else { // cornering or low gain, correction depending on yaw rate in °/s --------------------------
gyroFeedback = yaw_rate * 50; // degrees/s * speed (always 50%)
yaw_angle = 0; // reset yaw angle (heading direction)
}
if (mpuInversed) steeringAngle = turnRateSetPoint + (gyroFeedback * gain / 100); // Compensation depending on the pot value
else steeringAngle = turnRateSetPoint - (gyroFeedback * gain / 100);
steeringAngle = constrain (steeringAngle, -50, 50); // range = -50 to 50
// Control steering servo
servoSteering.write(map(steeringAngle, -50, 50, limSteeringL, limSteeringR) ); // 45 - 135°
}
//
// =======================================================================================================
// MAIN LOOP
// =======================================================================================================
//
void loop() {
//readInputs(); // Read pots and switches
checkValidity(); // Signal valid?
detectSteeringRange(); // Detect the steering input signal range
mrsc(); // Do stability control calculations
}