-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathTransmitter_with_trim external int.ino
500 lines (430 loc) · 21.4 KB
/
Transmitter_with_trim external int.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
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
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
// Arduino Transmitter with Trim
#include <SPI.h>
#include <nRF24L01.h>
#include <RF24.h>
#include <EEPROM.h>
#include <Wire.h>
#include <avr/sleep.h>
#include <avr/power.h>
#include <Adafruit_GFX.h>
#include <Adafruit_SSD1306.h>
#define SCREEN_WIDTH 128
#define SCREEN_HEIGHT 64
Adafruit_SSD1306 display(SCREEN_WIDTH, SCREEN_HEIGHT, &Wire, -1);
// Define a boolean flag to control the OLED display state
bool isDisplayOn = true;
#define OLED_RESET #
// Define the time (in milliseconds) after which the Arduino should go to sleep
const unsigned long INACTIVITY_TIMEOUT = 600000; // 10 minutes
const int externalButtonPin = 0; // Pin for the external button
unsigned long lastActivityTime = 0; // Variable to store the time of the last joystick activity
// Millis for updating eeprom values
unsigned long lastButtonPressTime = 0;
unsigned long buttonPressInterval = 10; // Adjust this interval as needed (in milliseconds)
// communication pipe is defined for the NRF24L01 module. The radio object is created to interact with the NRF24L01 module using the pins 9 (CE) and 10 (CSN).
const uint64_t pipeOut = 000322; // NOTE: The same as in the receiver 000322
RF24 radio(9, 10); // nRF24L01 (CE, CSN)
/*Defines the analog pins for reading joystick X and Y axis values.
tempRaw stores the raw temperature reading from a sensor, and joystickActive keeps track of whether the joystick is actively being used. */
const int joystickPinX = A0; // Joystick X-axis pin
const int joystickPinY = A1; // Joystick Y-axis pin
int16_t tempRaw; //16 bit integer used to store raw temp data from MPU6050
//boolean (true/false) value.
bool joystickActive = false; // Initializing that the joystick is not active
//Defines the pin numbers for buttons used to adjust trim values.
#define trimbut_1 1 // Trim button 1 / Pin D1
#define trimbut_2 2 // Trim button 2 / Pin D2
#define trimbut_3 3 // Trim button 3 / Pin D3
#define trimbut_4 4 // Trim button 4 / Pin D4
#define trimbut_5 5 // Trim button 5 / Pin D5
#define trimbut_6 6 // Trim button 6 / Pin D6
const int MPU = 0x68; // MPU6050 I2C address
float AccX, AccY, AccZ; //These variables are used to store the accelerometer readings for the X, Y, and Z axes.
float GyroX, GyroY, GyroZ; //These variables are used to store the gyroscope readings for the X, Y, and Z axes.
float accAngleX, accAngleY, gyroAngleX, gyroAngleY; //These variables are used to store calculated angle values based on accelerometer and gyroscope data.
float angleX, angleY;
float AccErrorX, AccErrorY, GyroErrorX, GyroErrorY; //Variables store error values that are calculated during setup to correct for errors in accelerometer and gyroscope readings.
// Define variables to keep track of button state and count the number of button presses
const int buttonPin = 0; // Define the digital pin for the button
bool buttonPressed = false; // Flag to indicate if the button is pressed
bool useIMU = false; // Flag to indicate whether to use IMU for control
/* ElapsedTime stores the time duration between successive gyro readings. It's of type float and represents time in seconds.
CurrentTime and previousTime store the current and previous time in milliseconds. They help calculate elapsedTime.
c is an integer variable that is used as a counter in certain sections of the code, like during error calculation. */
float elapsedTime, currentTime, previousTime;
int c = 0;
int tvalue1 = EEPROM.read(1) * 4; // Reading trim values from Eeprom
int tvalue2 = EEPROM.read(3) * 4;
int tvalue3 = EEPROM.read(5) * 4;
// Max size of this struct is 32 bytes - NRF24L01 buffer limit
struct Signal { //Defines a structure named Signal that represents the control signals to be transmitted
byte throttle;
byte pitch;
byte roll;
byte yaw;
byte aux1;
byte aux2;
byte aux3;
byte aux4;
byte aux5;
byte aux6 ;
//The byte data type is an 8-bit unsigned integer that can represent values from 0 to 255.
};
Signal data; //The line creates an instance of the Signal structure of above named data
void ResetData() { // If signal lost then the default position of all axis
data.throttle = 12;
data.pitch = 127;
data.roll = 127;
data.yaw = 127;
data.aux1 = 0;
data.aux2 = 0;
data.aux3 = 0;
data.aux4 = 0;
data.aux5 = 0;
data.aux6 = 0;
}
void setup() {
Serial.begin(9600);
if (!display.begin(0x3D)) {
Serial.println(F("SSD1306 allocation failed"));
// for (;;);
}
display.clearDisplay();
display.setTextSize(1);
display.setTextColor(SSD1306_WHITE);
display.setCursor(0, 0);
// Initialize interface to the MPU6050
initialize_MPU6050();
// Call this function if you need to get the IMU error values for your module
calculate_IMU_error();
lastActivityTime = millis(); // Initialize lastActivityTime
// Define the radio communication
radio.begin(); //initializes the nRF24L01 radio module
Serial.println("Radio COM Started");
radio.openWritingPipe(pipeOut); //This sets the pipe address that the radio module will use to transmit data
radio.setAutoAck(false); //This disables the automatic acknowledgment feature.
radio.setDataRate(RF24_250KBPS); // The lowest data rate value for more stable communication
radio.setPALevel(RF24_PA_MAX); // This sets the power amplifier (PA) level to maximum.
radio.stopListening(); // This line instructs the radio module to stop listening for incoming data.
ResetData(); // Used to reset all values if the connnection is lost
// Activate the internal pull-up resistors
pinMode(trimbut_1, INPUT_PULLUP);
pinMode(trimbut_2, INPUT_PULLUP);
pinMode(trimbut_3, INPUT_PULLUP);
pinMode(trimbut_4, INPUT_PULLUP);
pinMode(trimbut_5, INPUT_PULLUP);
pinMode(trimbut_6, INPUT_PULLUP);
//Defines analog pin as input
pinMode(joystickPinX, INPUT);
pinMode(joystickPinY, INPUT);
// IMU Pin
pinMode(buttonPin, INPUT_PULLUP);
tvalue1= EEPROM.read(1) * 4;
tvalue2= EEPROM.read(3) * 4;
tvalue3= EEPROM.read(5) * 4;
//These line reads a value from the EEPROM memory from respected address and stores at respected variables
}
void wakeUp() {
// This function is called when the external interrupt occurs
}
void enterSleepMode() {
sleep_enable(); // Enable sleep
attachInterrupt(digitalPinToInterrupt(externalButtonPin), wakeUp, LOW); // Attach interrupt to wake up on LOW level
set_sleep_mode(SLEEP_MODE_PWR_DOWN); // Deep Sleep Mode
sleep_mode(); // Enter sleep mode
sleep_disable(); // Disable sleep after waking up
detachInterrupt(digitalPinToInterrupt(externalButtonPin)); // Detach interrupt
}
void checkJoystickActivity() {
int joystickX = analogRead(joystickPinX); // Represents X-Axis of Joystick
int joystickY = analogRead(joystickPinY); // Represents Y-Axis of Joystick
// Define a threshold to account for errors or fluctuations
int joystickThreshold = 10;
// This condition checks if either the X-axis or the Y-axis deviates from the center position (512) by more than the threshold
if (abs(joystickX - 512) > joystickThreshold || abs(joystickY - 512) > joystickThreshold) {
joystickActive = true; // If the condition is met, the Joystick is ACTIVE
} else {
joystickActive = false; // If not, it's NOT ACTIVE
}
}
/* Border_Map that takes five integer arguments:
* val (the input value to be mapped)
* lower (the lower bound of the input range)
* middle (a value that defines the split point between two mapping ranges)
* upper (the upper bound of the input range)
* reverse (a boolean indicating whether to reverse the output range)
*/
int Border_Map(int val, int lower, int middle, int upper, bool reverse)
{
val = constrain(val, lower, upper);
//The constrain function is used here to ensure that the input value val is within the specified range [lower, upper]. If val is outside this range, it is clipped to the closest bound.
if ( val < middle )
//If val is less than middle, it means that the input value is in the lower range. In this case, the map function is used to linearly map the input value from the lower to middle range to an output range of [0, 128]
val = map(val, lower, middle, 0, 128);
//If val is greater than or equal to middle, it means that the input value is in the upper range. In this case, the map function is used to linearly map the input value from the middle to upper range to an output range of [128, 255].
else
val = map(val, middle, upper, 128, 255);
// If the reverse argument is true, the output value is further transformed by subtracting it from 255.
//This effectively reverses the output range, giving a higher output value for lower input values and vice versa.
return ( reverse ? 255 - val : val );
}
void loop()
{
display.clearDisplay();
checkJoystickActivity();
unsigned long currentTime = millis();
if (joystickActive) {
lastActivityTime = currentTime; // Update last activity time
} else {
// If joystick is inactive for INACTIVITY_TIMEOUT, enter sleep mode
if (currentTime - lastActivityTime >= INACTIVITY_TIMEOUT) {
Serial.println("Going to sleep due to Inactivity");
display.clearDisplay();
display.println("Going to sleep due to Inactivity");
display.display();
delay(2000);
display.clearDisplay();
// Check if the display was on before turning it off
if (isDisplayOn) {
display.ssd1306_command(SSD1306_DISPLAYOFF);
isDisplayOn = false;
}
digitalWrite(13, LOW); // Turn off the onboard LED
enterSleepMode();
} else {
// If the joystick is inactive but not for too long, keep the display on
if (!isDisplayOn) {
display.ssd1306_command(SSD1306_DISPLAYON);
isDisplayOn = true;
display.begin(0x3D); // Reinitialize the display if necessary
}
}
}
readTemp();
// Read trim values from EEPROM
int yawTrim = EEPROM.read(1) * 4;
int pitchTrim = EEPROM.read(3) * 4;
int rollTrim = EEPROM.read(5) * 4;
// Display trim values on the left side of the OLED display
display.setCursor(0, 0);
display.print("Yaw: ");
display.println(yawTrim);
display.print("Pitch: ");
display.println(pitchTrim);
display.print("Roll: ");
display.println(rollTrim);
// Read and display values of auxiliary channels on the right side
int aux1Value = analogRead(A4);
int aux2Value = analogRead(A5);
int aux3Value = analogRead(A6);
int aux4Value = analogRead(A7);
display.setCursor(SCREEN_WIDTH / 2, 0); // Move cursor to the right side
display.print("Aux1: ");
display.println(aux1Value);
display.print("Aux2: ");
display.println(aux2Value);
display.print("Aux3: ");
display.println(aux3Value);
display.print("Aux4: ");
display.println(aux4Value);
// Read and display values of auxiliary channels as high/low
int aux5Value = digitalRead(7);
int aux6Value = digitalRead(8);
display.setCursor(0, SCREEN_HEIGHT - 16); // Move cursor to the bottom left
display.print("Aux5: ");
display.println(aux5Value == HIGH ? "HIGH" : "LOW");
display.setCursor(SCREEN_WIDTH / 2, SCREEN_HEIGHT - 16); // Move cursor to the bottom right
display.print("Aux6: ");
display.println(aux6Value == HIGH ? "HIGH" : "LOW");
// Check and handle trim buttons
handleTrimButton(trimbut_1, tvalue1, 630, 280, 1);
handleTrimButton(trimbut_2, tvalue1, 280, 630, 1);
handleTrimButton(trimbut_3, tvalue2, 630, 280, 3);
handleTrimButton(trimbut_4, tvalue2, 280, 630, 3);
handleTrimButton(trimbut_5, tvalue3, 630, 280, 5);
handleTrimButton(trimbut_6, tvalue3, 280, 630, 5);
// Control Stick Calibration for channels
data.roll = Border_Map( analogRead(A3), 0, tvalue1, 1023, true ); // "true" or "false" for signal direction | Alieron
data.pitch = Border_Map( analogRead(A2), 0, tvalue2, 1023, true ); // Elevator
data.throttle = Border_Map( analogRead(A1),570, 800, 1023, false ); // For Single side ESC |
// data.throttle = Border_Map( analogRead(A1),0, 512, 1023, false ); // For Bidirectional ESC |
data.yaw = Border_Map( analogRead(A0), 0, tvalue3, 1023, true ); // Rudder
data.aux1 = Border_Map( analogRead(A4), 0, 512, 1023, true );
data.aux2 = Border_Map( analogRead(A5), 0, 512, 1023, true );
data.aux3 = Border_Map( analogRead(A6), 0, 512, 1023, true );
data.aux4 = Border_Map( analogRead(A7), 0, 512, 1023, true );
data.aux5 = digitalRead(7);
data.aux6 = digitalRead(8);
// Read the state of the button
int buttonState = digitalRead(buttonPin);
static int buttonPressCount = 0; // Track the number of button presses
// Check if the button is pressed (LOW) and wasn't pressed before
if (buttonState == LOW && !buttonPressed) {
// Button is pressed for the first time
buttonPressed = true;
delay(50); // Debounce delay to avoid multiple presses
// Increment the buttonPressCount
buttonPressCount++;
// Toggle the useIMU flag when the button is pressed once
if (buttonPressCount == 1) {
useIMU = !useIMU;
if (useIMU) {
// IMU control is enabled, you can read and use IMU data for control
read_IMU();
Serial.print("IMU");
} else {
// IMU control is disabled, you can stop reading IMU data here
// Add any code here to handle stopping IMU reading
}
} else if (buttonPressCount == 2) {
// Button is pressed twice, reset the buttonPressCount
buttonPressCount = 0;
}
}
// Check if the button is released (HIGH)
if (buttonState == HIGH) {
buttonPressed = false; // Button is released, reset the buttonPressed flag
}
radio.write(&data, sizeof(Signal));
display.display();
}
void handleTrimButton(int trimButton, int &tvalue, int lowerLimit, int upperLimit, int eepromAddress) {
int trimbuttonState = digitalRead(trimButton);
unsigned long currentTime = millis();
if (trimbuttonState == LOW) {
// Button is pressed
if (currentTime - lastButtonPressTime >= buttonPressInterval) {
// Sufficient time has passed since the last button press
if (tvalue < upperLimit) {
tvalue += 15;
EEPROM.write(eepromAddress, tvalue / 4);
} else if (tvalue > lowerLimit) {
tvalue -= 15;
EEPROM.write(eepromAddress, tvalue / 4);
}
lastButtonPressTime = currentTime;
}
}
}
void initialize_MPU6050() {
Wire.begin(); // Initialize comunication
Wire.beginTransmission(MPU); // Start communication with MPU6050 // MPU=0x68
Wire.write(0x6B); // Talk to the register 6B
Wire.write(0x00); // Make reset - place a 0 into the 6B register
Wire.endTransmission(true); //end the transmission
// Configure Accelerometer
Wire.beginTransmission(MPU);
Wire.write(0x1C); //Talk to the ACCEL_CONFIG register
Wire.write(0x10); //Set the register bits as 00010000 (+/- 8g full scale range)
Wire.endTransmission(true);
// Configure Gyro
Wire.beginTransmission(MPU);
Wire.write(0x1B); // Talk to the GYRO_CONFIG register (1B hex)
Wire.write(0x10); // Set the register bits as 00010000 (1000dps full scale)
Wire.endTransmission(true);
// Configure TEMP
Wire.beginTransmission(MPU);
Wire.write(0x6B); // PWR_MGMT_1 register
Wire.write(0); // set to zero (wakes up the MPU-6050)
Wire.endTransmission(true);
}
void calculate_IMU_error() {
// We can call this funtion in the setup section to calculate the accelerometer and gury data error. From here we will get the error values used in the above equations printed on the Serial Monitor.
// Note that we should place the IMU flat in order to get the proper values, so that we then can the correct values
// Read accelerometer values 200 times
while (c < 200) {
Wire.beginTransmission(MPU);
Wire.write(0x3B);
Wire.endTransmission(false);
Wire.requestFrom(MPU, 6, true);
// For each axis (X, Y, and Z), two bytes are read from the sensor and then combined using bit-shifting (<< 8) and bitwise OR operations to form a 16-bit value.
// The combined 16-bit value is divided by 4096.0 to convert it into an acceleration value in g-units (gravity units).
AccX = (Wire.read() << 8 | Wire.read()) / 4096.0 ;
AccY = (Wire.read() << 8 | Wire.read()) / 4096.0 ;
AccZ = (Wire.read() << 8 | Wire.read()) / 4096.0 ;
// Measure the average error or bias present in the accelerometer readings due to sensor imperfections.
// These error values can be used later to compensate for inaccuracies when calculating the orientation of the sensor based on accelerometer data.
AccErrorX = AccErrorX + ((atan((AccY) / sqrt(pow((AccX), 2) + pow((AccZ), 2))) * 180 / PI));
AccErrorY = AccErrorY + ((atan(-1 * (AccX) / sqrt(pow((AccY), 2) + pow((AccZ), 2))) * 180 / PI));
c++;
}
//Divide the sum by 200 to get the error value
AccErrorX = AccErrorX / 200;
AccErrorY = AccErrorY / 200;
c = 0;
// Read gyro values 200 times
while (c < 200) {
Wire.beginTransmission(MPU);
Wire.write(0x43);
Wire.endTransmission(false);
Wire.requestFrom(MPU, 4, true);
GyroX = Wire.read() << 8 | Wire.read();
GyroY = Wire.read() << 8 | Wire.read();
// Sum all readings
GyroErrorX = GyroErrorX + (GyroX / 32.8);
GyroErrorY = GyroErrorY + (GyroY / 32.8);
c++;
}
//Divide the sum by 200 to get the error value
GyroErrorX = GyroErrorX / 200;
GyroErrorY = GyroErrorY / 200;
// Print the error values on the Serial Monitor
Serial.print("AccErrorX: ");
Serial.println(AccErrorX);
Serial.print("AccErrorY: ");
Serial.println(AccErrorY);
Serial.print("GyroErrorX: ");
Serial.println(GyroErrorX);
Serial.print("GyroErrorY: ");
Serial.println(GyroErrorY);
}
void read_IMU() {
// === Read acceleromter data === //
Wire.beginTransmission(MPU);
Wire.write(0x3B); // Start with register 0x3B (ACCEL_XOUT_H)
Wire.endTransmission(false);
Wire.requestFrom(MPU, 6, true); // Read 6 registers total, each axis value is stored in 2 registers
//For a range of +-8g, we need to divide the raw values by 4096, according to the datasheet
AccX = (Wire.read() << 8 | Wire.read()) / 4096.0; // X-axis value
AccY = (Wire.read() << 8 | Wire.read()) / 4096.0; // Y-axis value
AccZ = (Wire.read() << 8 | Wire.read()) / 4096.0; // Z-axis value
// Calculating angle values using
accAngleX = (atan(AccY / sqrt(pow(AccX, 2) + pow(AccZ, 2))) * 180 / PI) + 1.15; // AccErrorX ~(-1.15) See the calculate_IMU_error()custom function for more details
accAngleY = (atan(-1 * AccX / sqrt(pow(AccY, 2) + pow(AccZ, 2))) * 180 / PI) - 0.52; // AccErrorX ~(0.5)
// === Read gyro data === //
previousTime = currentTime; // Previous time is stored before the actual time read
currentTime = millis(); // Current time actual time read
elapsedTime = (currentTime - previousTime) / 1000; // Divide by 1000 to get seconds
Wire.beginTransmission(MPU);
Wire.write(0x43); // Gyro data first register address 0x43
Wire.endTransmission(false);
Wire.requestFrom(MPU, 4, true); // Read 4 registers total, each axis value is stored in 2 registers
GyroX = (Wire.read() << 8 | Wire.read()) / 32.8; // For a 1000dps range we have to divide first the raw value by 32.8, according to the datasheet
GyroY = (Wire.read() << 8 | Wire.read()) / 32.8;
GyroX = GyroX + 1.85; //// GyroErrorX ~(-1.85)
GyroY = GyroY - 0.15; // GyroErrorY ~(0.15)
// Currently the raw values are in degrees per seconds, deg/s, so we need to multiply by sendonds (s) to get the angle in degrees
gyroAngleX = GyroX * elapsedTime;
gyroAngleY = GyroY * elapsedTime;
// Complementary filter - combine acceleromter and gyro angle values
angleX = 0.98 * (angleX + gyroAngleX) + 0.02 * accAngleX;
angleY = 0.98 * (angleY + gyroAngleY) + 0.02 * accAngleY;
// Map the angle values from -90deg to +90 deg into values from 0 to 255, like the values we are getting from the Joystick
data.throttle = map(angleX, -90, +90, 255, 0);
data.roll = map(angleY, -90, +90, 0, 255);
}
void readTemp() {
Wire.beginTransmission(MPU);
Wire.write(0x41); // Temp register address
Wire.endTransmission(false);
Wire.requestFrom(MPU, 2, true);
tempRaw = Wire.read() << 8 | Wire.read(); // Combine the two bytes
float temperature = (tempRaw / 340.0) + 36.53; // Conversion formula from datasheet
Serial.print("Temperature: ");
Serial.print(temperature);
Serial.println(" °C");
if (temperature >= 100.0) {
Serial.println("Entering sleep mode.");
enterSleepMode();
}
}