ROS 2 Distro | Branch | Build status |
---|---|---|
Rolling | rolling |
|
Humble | humble |
|
Galactic | galactic |
|
Foxy | foxy |
All software mentioned in this guide must be installed on the robot computer.
It is assumed that you already have ROS2 and linorobot2 package installed. If you haven't, go to linorobot2 package for installation guide.
cd $HOME
git clone https://github.com/linorobot/linorobot2_hardware -b $ROS_DISTRO
Download and install platformio. Platformio allows you to develop, configure, and upload the firmware without the Arduino IDE. This means that you can upload the firmware remotely which is ideal on headless setup especially when all components have already been fixed.
curl -fsSL -o get-platformio.py https://raw.githubusercontent.com/platformio/platformio-core-installer/master/get-platformio.py
python3 get-platformio.py
Add platformio to your $PATH:
echo "PATH=\"\$PATH:\$HOME/.platformio/penv/bin\"" >> $HOME/.bashrc
source $HOME/.bashrc
Download the udev rules from Teensy's website:
wget https://www.pjrc.com/teensy/00-teensy.rules
and copy the file to /etc/udev/rules.d :
sudo cp 00-teensy.rules /etc/udev/rules.d/
sudo apt install screen
Robot Orientation:
-------------FRONT-------------
WHEEL1 WHEEL2 (2WD)
WHEEL3 WHEEL4 (4WD)
--------------BACK--------------
If you're building a 2 wheel drive robot, assign MOTOR1
and MOTOR2
to the left and right motors respectively.
For mecanum robots, follow the wheels' orientation below.
Supported Motor Drivers:
-
GENERIC_2_IN_MOTOR_DRIVER - Motor drivers that have EN (pwm) pin, and 2 direction pins (usually DIRA, DIRB pins). Example: L298 Breakout boards.
-
GENERIC_1_IN_MOTOR_DRIVER - Motor drivers that have EN (pwm) pin, and 1 direction pin (usual DIR pin). These drivers usually have logic gates included to lessen the pins required in controlling the driver. Example: Pololu MC33926 Motor Driver Shield.
-
BTS7960_MOTOR_DRIVER - BTS7960 motor driver.
-
ESC_MOTOR_DRIVER - Bi-directional (forward/reverse) electronic speed controllers.
The motor drivers are configurable from the config file explained in the later part of this document.
Supported IMUs:
- GY-85
- MPU6050
- MPU9150
- MPU9250
- QMI8658
Supported MAGs:
- HMC5883L
- AK8963
- AK8975
- AK09918
- QMC5883L
Below are connection diagrams you can follow for each supported motor driver and IMU. For simplicity, only one motor connection is provided but the same diagram can be used to connect the rest of the motors. You are free to decide which microcontroller pin to use just ensure that the following are met:
-
Reserve SCL0 and SDA0 (pins 18 and 19 on Teensy boards) for IMU.
-
When connecting the motor driver's EN/PWM pin, ensure that the microcontroller pin used is PWM enabled. You can check out PJRC's pinout page for more info.
Alternatively, you can also use the pre-defined pin assignments in lino_base_config.h. Teensy 3.x and 4.x have different mapping of PWM pins, read the notes beside each pin assignment in lino_base_config.h carefully to avoid connecting your driver's PWM pin to a non PWM pin on Teensy.
All diagrams below are based on Teensy 4.0 microcontroller and GY85 IMU. Click the images for higher resolution.
Take note of the IMU's correct orientation when mounted on the robot. Ensure that the IMU's axes are facing the correct direction:
- X - Front
- Y - Left
- Z - Up
Reference designs you can follow in building your robot.
A minimal setup with a 5V powered robot computer.
A more advanced setup with a 19V powered computer and USB hub connected to sensors.
For bigger robots, you can add an emergency switch in between the motor drivers' power supply and motor drivers.
Go to the config folder and open lino_base_config.h. Uncomment the base, motor driver and IMU you want to use for your robot. For example:
#define LINO_BASE DIFFERENTIAL_DRIVE
#define USE_GENERIC_2_IN_MOTOR_DRIVER
#define USE_GY85_IMU
Constants' Meaning:
ROBOT TYPE (LINO_BASE)
-
DIFFERENTIAL_DRIVE - 2 wheel drive or tracked robots w/ 2 motors.
-
SKID_STEER - 4 wheel drive robots.
-
MECANUM - 4 wheel drive robots using mecanum wheels.
MOTOR DRIVERS
-
USE_GENERIC_2_IN_MOTOR_DRIVER - Motor drivers that have EN (pwm) pin, and 2 direction pins (usually DIRA, DIRB pins).
-
USE_GENERIC_1_IN_MOTOR_DRIVER - Motor drivers that have EN (pwm) pin, and 1 direction pin (usual DIR pin). These drivers usually have logic gates included to lessen the pins required in controlling the driver.
-
USE_BTS7960_MOTOR_DRIVER - BTS7960 motor driver.
-
USE_ESC_MOTOR_DRIVER - Bi-directional (forward/reverse) electronic speed controllers.
INERTIAL MEASUREMENT UNIT (IMU)
-
USE_GY85_IMU - GY-85 IMUs.
-
USE_MPU6050_IMU - MPU6060 IMUs.
-
USE_MPU9150_IMU - MPU9150 IMUs.
-
USE_MPU9250_IMU - MPU9250 IMUs.
-
USE_QMI8658_IMU - QMI8658 IMUs.
-
USE_HMC5883L_MAG - HMC5883L MAGs.
-
USE_AK8963_MAG - AK8963 MAGs.
-
USE_AK8975_MAG - AK8975 MAGs.
-
USE_AK09918_MAG - AK09918 MAGs.
-
USE_QMC5883L_MAG - QMC5883L MAGs.
-
MAG_BIAS - Magnetometer calibration, eg { -352, -382, -10 }.
If you enable magnetometer device, you will need to enable Magdwick filter by adding "madgwick:=true" in robot bringup. The topic from imu will be named as imu/data_raw. The Madgwick filter will fuse imu/data_raw and imu/mag to imu/data.
ros2 launch linorobot2_bringup bringup.launch.py madgwick:=true
Magnetometer calibration should be taken on board with all hardware installed, inlcuding all connectors, battery and motors. The calibration package will rotate the robot slowly for 60 sec. And compute the hard iron bias. More info here.
sudo apt-get install ros-humble-robot-calibration -y
rm -rf /tmp/magnetometer_calibration.bag
ros2 run robot_calibration magnetometer_calibration
...
mag_bias_x: -7.94713e-06
mag_bias_y: 3.49388e-05
mag_bias_z: -5.40286e-05
Set the MAG_BIAS to these values in robot configuration file.
#define MAG_BIAS { -7.94713e-06, 3.49388e-05, -5.40286e-05 }
Next, fill in the robot settings accordingly:
#define K_P 0.6
#define K_I 0.8
#define K_D 0.5
#define MOTOR_MAX_RPM 100
#define MAX_RPM_RATIO 0.85
#define MOTOR_OPERATING_VOLTAGE 24
#define MOTOR_POWER_MAX_VOLTAGE 12
#define MOTOR_POWER_MEASURED_VOLTAGE 11.7
#define COUNTS_PER_REV1 2200
#define COUNTS_PER_REV2 2200
#define COUNTS_PER_REV3 2200
#define COUNTS_PER_REV4 2200
#define WHEEL_DIAMETER 0.09
#define LR_WHEELS_DISTANCE 0.2
#define PWM_BITS 10
#define PWM_FREQUENCY 20000
Constants' Meaning:
-
K_P, K_I, K_D - PID constants used to translate the robot's target velocity to motor speed. These values would likely work on your build, change these only if you experience jittery motions from the robot or you'd want to fine-tune it further.
-
MOTOR_MAX_RPM - Motor's maximum number of rotations it can do in a minute specified by the manufacturer.
-
MAX_RPM_RATIO - Percentage of the motor's maximum RPM that the robot is allowed to move. This parameter ensures that the user-defined velocity will not be more than or equal the motor's max RPM, allowing the PID to have ample space to add/subtract RPM values to reach the target velocity. For instance, if your motor's maximum velocity is 0.5 m/s with
MAX_RPM_RATIO
set to 0.85, and you asked the robot to move at 0.5 m/s, the robot's maximum velocity will be capped at 0.425 m/s (0.85 * 0.5m/s). You can set this parameter to 1.0 if your wheels can spin way more than your operational speed.Wheel velocity can be computed as: MAX_WHEEL_VELOCITY = (
MOTOR_MAX_RPM
/ 60.0) * PI *WHEEL_DIAMETER
-
MOTOR_OPERATING_VOLTAGE - Motor's operating voltage specified by the manufacturer (usually 5V/6V, 12V, 24V, 48V). This parameter is used to calculate the motor encoder's
COUNTS_PER_REV
constant during calibration and actual maximum RPM of the motors. For instance, a robot withMOTOR_OPERATING_VOLTAGE
of 24V with aMOTOR_POWER_MAX_VOLTAGE
of 12V, will only have half of the manufacturer's specified maximum RPM ((MOTOR_POWER_MAX_VOLTAGE
/MOTOR_OPERATING_VOLTAGE
) *MOTOR_MAX_RPM
). -
MOTOR_POWER_MAX_VOLTAGE - Maximum voltage of the motor's power source. This parameter is used to calculate the actual maximum RPM of the motors.
-
MOTOR_POWER_MEASURED_VOLTAGE - Measured voltage of the motor's power source. If you don't have a multimeter, it's best to fully charge your battery and set this parameter to your motor's operating voltage (
MOTOR_OPERATING_VOLTAGE
). This parameter is used to calculate the motor encoder'sCOUNTS_PER_REV
constant. You can ignore this if you're using the manufacturer's specified counts per rev. -
COUNTS_PER_REVX - The total number of pulses the encoder has to read to be considered as one revolution. You can either use the manufacturer's specification or the calibrated value in the next step. If you're planning to use the calibrated value, ensure that you have defined the correct values for
MOTOR_OPERATING_VOLTAGE
andMOTOR_POWER_MEASURED_VOLTAGE
. -
WHEEL_DIAMETER - Diameter of the wheels in meters.
-
LR_WHEELS_DISTANCE - The distance between the center of left and right wheels in meters.
-
PWM_BITS - Number of bits in generating the PWM signal. You can use the default value if you're unsure what to put here. More info here.
-
PWM_FREQUENCY - Frequency of the PWM signals used to control the motor drivers. You can use the default value if you're unsure what to put here. More info here.
Only modify the pin assignments under the motor driver constant that you are using ie. #ifdef USE_GENERIC_2_IN_MOTOR_DRIVER
. You can check out PJRC's pinout page for each board's pin layout.
The pin assignments found in lino_base_config.h are based on Linorobot's PCB board. You can wire up your electronic components based on the default pin assignments but you're also free to modify it depending on your setup. Just ensure that you're connecting MOTORX_PWM pins to a PWM enabled pin on the microcontroller and reserve SCL and SDA pins for the IMU, and pin 13 (built-in LED) for debugging.
// INVERT ENCODER COUNTS
#define MOTOR1_ENCODER_INV false
#define MOTOR2_ENCODER_INV false
#define MOTOR3_ENCODER_INV false
#define MOTOR4_ENCODER_INV false
// INVERT MOTOR DIRECTIONS
#define MOTOR1_INV false
#define MOTOR2_INV false
#define MOTOR3_INV false
#define MOTOR4_INV false
// ENCODER PINS
#define MOTOR1_ENCODER_A 14
#define MOTOR1_ENCODER_B 15
#define MOTOR2_ENCODER_A 11
#define MOTOR2_ENCODER_B 12
#define MOTOR3_ENCODER_A 17
#define MOTOR3_ENCODER_B 16
#define MOTOR4_ENCODER_A 9
#define MOTOR4_ENCODER_B 10
// MOTOR PINS
#ifdef USE_GENERIC_2_IN_MOTOR_DRIVER
#define MOTOR1_PWM 21 //Pin no 21 is not a PWM pin on Teensy 4.x, you can swap it with pin no 1 instead.
#define MOTOR1_IN_A 20
#define MOTOR1_IN_B 1
#define MOTOR2_PWM 5
#define MOTOR2_IN_A 6
#define MOTOR2_IN_B 8
#define MOTOR3_PWM 22
#define MOTOR3_IN_A 23
#define MOTOR3_IN_B 0
#define MOTOR4_PWM 4
#define MOTOR4_IN_A 3
#define MOTOR4_IN_B 2
#define PWM_MAX pow(2, PWM_BITS) - 1
#define PWM_MIN -PWM_MAX
#endif
Constants' Meaning:
-
MOTORX_ENCODER_A - Microcontroller pin that is connected to the first read pin of the motor encoder. This pin is usually labelled as A pin on the motor encoder board.
-
MOTORX_ENCODER_B - Microcontroller pin that is connected to the second read pin of the motor encoder. This pin is usually labelled as B pin on the motor encoder board.
-
MOTORX_ENCODER_INV - Flag used to change the sign of the encoder value. More on that later.
-
MOTORX_PWM - Microcontroller pin that is connected to the PWM pin of the motor driver. This pin is usually labelled as EN or ENABLE pin on the motor driver board.
-
MOTORX_IN_A - Microcontroller pin that is connected to one of the motor driver's direction pins. This pin is usually labelled as DIRA or DIR1 pin on the motor driver board. On BTS7960 driver, this is one of the two PWM pins connected to the driver (RPWM/LPWM).
-
MOTORX_IN_B - Microcontroller pin that is connected to one of the motor driver's direction pins. This pin is usually labelled as DIRB or DIR2 pin on the motor driver board. On BTS7960 driver, this is one of the two PWM pins connected to the driver (RPWM/LPWM).
-
MOTORX_INV - Flag used to invert the direction of the motor. More on that later.
WIFI related settings
-
AGENT_IP - micro-ROS agent IP. eg. host IP, { 192, 168, 1, 100 }
-
AGENT_PORT - micro-ROS agent port. default 8888
-
WIFI_SSID
-
WIFI_PASSWORD
The mirco-ROS wifi transport is selected with a setting in firmare/platformio.ini.
board_microros_transport = wifi
Before proceeding, ensure that your robot is elevated and the wheels aren't touching the ground. 5.1
Go to calibration folder and upload the firmware:
cd linorobot2_hardware/calibration
pio run --target upload -e <your_teensy_board>
Available Teensy boards:
- teensy31
- teensy35
- teensy36
- teensy40
- teensy41
Some Linux machines might encounter a problem related to libusb. If so, install libusb-dev:
sudo apt install libusb-dev
Start spinning the motors by running:
screen /dev/ttyACM0
On the terminal type spin
and press the enter key.
The wheels will spin one by one for 10 seconds from Motor1 to Motor4. Check if each wheel's direction is spinning forward and take note of the motors that are spinning in the opposite direction. Set MOTORX_INV constant in lino_base_config.h to true
to invert the motor's direction. Reupload the calibration firmware once you're done. Press Ctrl
+ a
+ d
to exit the screen terminal.
cd linorobot2_hardware/calibration
pio run --target upload -e <your_teensy_board>
Open your terminal and run:
screen /dev/ttyACM0
Type sample
and press the enter key. Verify if all the wheels are spinning forward. Redo the previous step if there are motors still spinning in the opposite direction.
You'll see a summary of the total encoder readings and counts per revolution after the motors have been sampled. If you see any negative number in the MOTOR ENCODER READINGS section, invert the encoder pin by setting MOTORX_ENCODER_INV
in lino_base_config.h to true
. Reupload the calibration firmware to check if the encoder pins have been reconfigured properly:
cd linorobot2_hardware/calibration
pio run --target upload -e <your_teensy_board>
screen /dev/ttyACM0
Type sample
and press the enter key. Verify if all encoder values are now positive. Redo this step if you missed out any.
On the previous instruction where you check the encoder reads for each motor, you'll see that there's also COUNTS PER REVOLUTION values printed on the screen. If you have defined MOTOR_OPERATING_VOLTAGE
and MOTOR_POWER_MEASURED_VOLTAGE
, you can assign these values to COUNTS_PER_REVX
constants in lino_base_config.h to have a more accurate model of the encoder.
Ensure that the robot pass all the requirements before uploading the firmware:
- Defined the correct motor rpm.
- Motors' IDs are correct.
- Motors spin in the right direction.
- Encoders' signs are correct.
- Defined the correct encoder's COUNTS_PER_REV.
- Defined the correct robot type.
- Defined the correct motor driver.
- Defined the correct IMU.
- Defined the correct wheel diameter.
- Defined the correct distance between wheels.
Run:
cd linorobot2_hardware/firmware
pio run --target upload -e <your_teensy_board>
This will allow the robot to receive Twist messages to control the robot, and publish odometry and IMU data straight from the microcontroller. Compared to Linorobot's ROS1 version, the odometry and IMU data published from the microcontroller use standard ROS2 messages and do not require any relay nodes to reconstruct the data to complete sensor_msgs/Imu and nav_msgs/Odometry messages.
Run the serial transprot agent: (the serial device used for esp32 is /dev/ttyUSB0)
ros2 run micro_ros_agent micro_ros_agent serial --dev /dev/ttyACM0
Or run the wifi transport agent:
ros2 run micro_ros_agent micro_ros_agent udp4 --port 8888
Run teleop_twist_keyboard package and follow the instructions on the terminal on how to drive the robot:
ros2 run teleop_twist_keyboard teleop_twist_keyboard
Check if the odom and IMU data are published:
ros2 topic list
Now you should see the following topics:
/cmd_vel
/imu/data
/odom/unfiltered
/parameter_events
/rosout
Echo odometry data:
ros2 topic echo /odom/unfiltered
Echo IMU data:
ros2 topic echo /imu/data
Once the hardware is done, you can go back to linorobot2 package and start defining the robot's URDF.
- Check if the motors are powered.
- Check if you have bad wiring.
- Check if you have misconfigured the motor's pin assignment in lino_base_config.h.
- Check if you uncommented the correct motor driver (ie.
USE_GENERIC_2_IN_MOTOR_DRIVER
) - Check if you assigned the motor driver pins under the correct motor driver constant. For instance, if you uncommented
USE_GENERIC_2_IN_MOTOR_DRIVER
, all the pins you assigned must be inside theifdef USE_GENERIC_2_IN_MOTOR_DRIVER
macro.
- Check if the motor drivers have been connected to the correct microcontroller pin.
- Check if you have misconfigured the motor's pin assignment in lino_base_config.h.
- Check if the encoders are powered.
- Check if you have bad wiring.
- Check if you have misconfigured the encoder's pin assignment in lino_base_config.h.
- Check if the Teensy's GND pin is connected to the motor driver's GND pin.
- Check if the Teensy's GND pin is connected to the motor driver's GND pin.
-
Check if you're passing the correct serial port. Run:
ls /dev/ttyACM*
and ensure that the available serial port matches the port you're passing to the screen app.
-
Check if you forgot to copy the udev rule:
ls /etc/udev/rules.d/00-teensy.rules
Remember to restart your computer if you just copied the udev rule.
- Check if you're assigning the correct Teensy board when uploading the firmware. If you're unsure which Teensy board you're using, take a look at the label on the biggest chip found in your Teensy board and compare it with the boards shown on PJRC's website.
- This happens when the target velocity is more than or equal the motor's RPM (usually happens on low RPM motors). To fix this, set the
MAX_RPM_RATIO
lower to allow the PID controller to compensate for errors.
- This happens due to the same reason as 7. When the motor hits its maximum rpm and fails to reach the target velocity, the PID controller's error continously increases. The abrupt turning motion is due to the PID controller's attempt to further compensate the accumulated error. To fix this, set the
MAX_RPM_RATIO
lower to allow the PID controller to compensate for errors while moving to avoid huge accumulative errors when the robot stops.
To add a new distro to the CI tests, modify the rolling
(default) branch. Inside of .github/workflows
, duplicate an existing distro workflow YAML file. For example, to add ROS2 Iron support, one could copy humble-firmware-build.yml
to iron-firmware-build.yml
. Assuming that an iron
branch exists (if not one could create one using the humble
branch as a base and modify as necessary), inside of iron-firmware-build.yml
, rename all instances of the word humble
with iron
. It would be as simple as using 'find and replace' in many IDEs. Commit these changes to a feature branch, create a PR to merge into the rolling
branch, and then backport the PR to other branches. It is only necessary to have iron-firmware-build.yml
on the rolling
and iron
branch, however it may be simpler to keep the branches in sync by having every workflow file on all branches.
Lastly, the new branch must be added to the CI table written in Markdown at the top of README.md that displays the status of each branch using badges. This table is organized with the most current ROS2 branch at the top, which is always rolling
, and then in descending chronological order. Adding a new distro can be done by copying an existing row of the table, pasting in the appropriate position, and changing the titles and branch names in the relative paths.