Dec 17, 2024 |
4 min read

Servo Control with MPU6050 and NodeMcu

In this tutorial, we will explore how to control a servo motor using the MPU6050 gyroscope and accelerometer sensor with a NodeMCU ESP8266. This project is perfect for beginners who want to dive into sensor-based control systems, especially in robotics. By the end, you’ll learn how to map the tilt of the MPU6050 sensor to control a servo motor.

Materials Required

To follow along, ensure you have the following components:

  1. NodeMCU ESP8266

  2. MPU6050 Gyroscope and Accelerometer Sensor

  3. Servo Motor

  4. Breadboard

  5. Jumper Wires

  6. External 5V Power Supply (for the servo motor)

How It Works

The MPU6050 sensor detects motion and orientation by measuring acceleration and angular velocity. We will use its gyroscope data (specifically the X-axis) to control the angle of the servo motor. The NodeMCU will read the sensor data via I2C communication and adjust the servo’s position accordingly.

Circuit Diagram

blog-image

Connections

  1. MPU6050 to NodeMCU:

    • VCC → 3.3V

    • GND → GND

    • SCL → D1 (GPIO 5)

    • SDA → D2 (GPIO 4)

  2. Servo Motor to NodeMCU:

    • VCC → External 5V Power Supply

    • GND → Common GND

    • Signal → D4 (GPIO 2)

Note: Since the servo motor requires 5V, it’s important to power it externally while ensuring all grounds are connected.

Arduino Code

Here is the complete code to control the servo motor:

#include <Adafruit_MPU6050.h>
#include <Adafruit_Sensor.h>
#include <Wire.h>
#include <Servo.h>

Adafruit_MPU6050 mpu;
Servo myServo;

const int servoPin = D4; // Define the pin for the servo signal
int servoAngle = 90; // Initialize the servo at 90 degrees (neutral position)

float gyroX_offset = 0; // Offset for gyro X-axis

void setup() {
Serial.begin(115200);
mpu.begin();

// Set ranges for sensor sensitivity
mpu.setAccelerometerRange(MPU6050_RANGE_8_G);
mpu.setGyroRange(MPU6050_RANGE_500_DEG);
mpu.setFilterBandwidth(MPU6050_BAND_21_HZ);

// Attach the servo
myServo.attach(servoPin);
myServo.write(servoAngle);

// Calibrate gyroscope
gyroX_offset = calibrateGyro();
Serial.print("Calibration complete. Gyro X offset: ");
Serial.println(gyroX_offset);
}

void loop() {
sensors_event_t a, g, temp;
mpu.getEvent(&a, &g, &temp);

// Adjust gyro X reading
float gyroX = g.gyro.x - gyroX_offset;

// Update servo angle based on gyro reading
servoAngle += gyroX * 10; // Adjust sensitivity with multiplier
servoAngle = constrain(servoAngle, 0, 180);

// Move servo
myServo.write(servoAngle);

// Debugging information
Serial.print("Gyro X: ");
Serial.print(gyroX);
Serial.print(" rad/s | Servo Angle: ");
Serial.println(servoAngle);

delay(50); // Small delay for stability
}

float calibrateGyro() {
const int calibrationTime = 5000; // Calibration time in milliseconds
int sampleCount = 0;
float gyroX_sum = 0;

unsigned long startTime = millis();
while (millis() - startTime < calibrationTime) {
sensors_event_t a, g, temp;
mpu.getEvent(&a, &g, &temp);

gyroX_sum += g.gyro.x;
sampleCount++;
delay(10);
}

return gyroX_sum / sampleCount;
}

Code Explanation

1. Importing Libraries

#include <Adafruit_MPU6050.h>
#include <Adafruit_Sensor.h>
#include <Wire.h>
#include <Servo.h>

These libraries handle communication with the MPU6050 sensor and control of the servo motor.

2. Object Initialization

Adafruit_MPU6050 mpu;
Servo myServo;

  • mpu: Initializes the MPU6050 object to read sensor data.

  • myServo: Creates an object to control the servo motor.

3. Variables and Constants

const int servoPin = D4;
int servoAngle = 90;
float gyroX_offset = 0;

  • servoPin: GPIO pin for controlling the servo motor.

  • servoAngle: Stores the current position of the servo (initially 90°).

  • gyroX_offset: Used for calibrating the gyroscope to remove drift.

4. Setup Function

void setup() {
Serial.begin(115200);
mpu.begin();
mpu.setAccelerometerRange(MPU6050_RANGE_8_G);
mpu.setGyroRange(MPU6050_RANGE_500_DEG);
mpu.setFilterBandwidth(MPU6050_BAND_21_HZ);
myServo.attach(servoPin);
myServo.write(servoAngle);
gyroX_offset = calibrateGyro();
}

  • Initializes the sensor and servo motor.

  • Configures sensitivity ranges for the accelerometer and gyroscope.

  • Calibrates the gyroscope to calculate the gyroX_offset.

5. Main Loop

void loop() {
sensors_event_t a, g, temp;
mpu.getEvent(&a, &g, &temp);
float gyroX = g.gyro.x - gyroX_offset;
servoAngle += gyroX * 10;
servoAngle = constrain(servoAngle, 0, 180);
myServo.write(servoAngle);
Serial.print("Gyro X: ");
Serial.print(gyroX);
Serial.print(" rad/s | Servo Angle: ");
Serial.println(servoAngle);
delay(50);
}

  • Reads data from the MPU6050.

  • Adjusts the servo angle based on the gyroscope X-axis reading.

  • Ensures the servo angle stays within valid bounds (0° to 180°).

  • Moves the servo motor to the new position.

6. Calibration Function

float calibrateGyro() {
const int calibrationTime = 5000;
int sampleCount = 0;
float gyroX_sum = 0;
unsigned long startTime = millis();
while (millis() - startTime < calibrationTime) {
sensors_event_t a, g, temp;
mpu.getEvent(&a, &g, &temp);
gyroX_sum += g.gyro.x;
sampleCount++;
delay(10);
}
return gyroX_sum / sampleCount;
}

  • Collects gyroscope readings over 5 seconds.

  • Computes the average X-axis reading to remove drift.

Testing the Project

  1. Connect the NodeMCU to your computer and upload the code using the Arduino IDE.

  2. Open the Serial Monitor at a baud rate of 115200 to view the gyroscope readings and servo angle.

  3. Tilt the MPU6050 sensor along the X-axis and observe the servo motor moving accordingly.

Troubleshooting Tips

  1. Check your wiring: Ensure all connections are secure and correct.

  2. External Power: Use a reliable 5V power supply for the servo motor.

  3. Library Installation: Verify that the Adafruit libraries are installed in the Arduino IDE.

  4. Gyroscope Calibration: Restart the system if the servo behaves erratically.

Support

Thank you for reading! If you enjoyed this post and want to support my work, consider supporting me by leaving a comment or sharing this post with a friend.