This project develops an advanced inverse kinematics solution for the Maira robot using Denavit-Hartenberg (DH) parameters. It calculates orientation errors with quaternions, enforces joint limits (±180° for all axes, except A2: ±120° and A4: ±150°), computes the Jacobian matrix, and handles redundancy using pseudo-inverse, null-space projection, and a secondary objective enforcing the joints to stay close to mean of its ranges. The implementation is object-oriented, efficient, and includes condition number calculation for the Jacobian. A test function estimates the success rate in terms of Forbenius norm of the difference of target and converged transformations, that encapsulates the proximity of the solution.
- Prerequisites
- Project Structure
- Building the Project
- Testing the Project
- Running the Project
- Solution Approach
Before you begin, ensure you have met the following requirements:
- You are working in a Linux environment (here using Ubuntu 22.04.3 LTS).
- You have installed C++ compiler (e.g., g++, clang++).
- You have installed CMake, preferably the latest version.
- You have installed Make, preferably GNU Make 4.3 or latest.
- Additional dependencies of Eigen and Gtest will be added through the
CMakeLists.txt
.
Open up your command line and first clone the repository using ssh preferably. If you do not have an ssh key then click here to generate o ne and click here to add to your github profile. Change the directory to the project.
git clone git@github.com:Shubham1965/InverseKinematics7dof.git
cd InverseKinematics7dof
You'll see the following directory structure.
├── CMakeLists.txt
├── doc
│ └── Inverse_Kinematics.pdf
├── include
│ ├── InverseKinematics.h
│ └── Robot.h
├── ReadMe.md
├── src
│ ├── CMakeLists.txt
│ ├── InverseKinematics.cpp
│ ├── main.cpp
│ └── Robot.cpp
└── tests
├── CMakeLists.txt
└── test_inverse_kinematics.cpp
Before building the project check whether you have cmake and make installed by checking their versions using cmake --version
and make --version
. Next, in order to build the project, execute the following commands.
mkdir build
cd build
cmake ..
make
After building, the directory structure should look like this:
├── CMakeLists.txt
├── build/
│ ├── src/
│ │ └── bin_main.exe
│ ├── tests/
│ │ └── test_ik.exe
│ └── ... # other build files
├── doc
│ └── Inverse_Kinematics.pdf
├── include
│ ├── InverseKinematics.h
│ └── Robot.h
├── ReadMe.md
├── src
│ ├── CMakeLists.txt
│ ├── InverseKinematics.cpp
│ ├── main.cpp
│ └── Robot.cpp
└── tests
├── CMakeLists.txt
└── test_inverse_kinematics.cpp
Before running the project, it is important to test whether all functions work. Ideally there should have been tests for each function definitions but here i made it jut for one. So, in order to test the project, navigate to the build/tests
directory and execute
./tests/test_ik
This is a test routine to check whether InverseKinematics::solve
function always works and gives an output. This is just a sanity check.
To run the main executable after testing, navigate to the build/src
directory and execute
./src/bin_main
This is the executable of main.cpp
where you can change the desired position and orientation,
// Target position and orientation
// Example target position describing the target position is at (0.5m, 0.3m, 0.6m) in the 3D space
Eigen::Vector3d target_position(1.0, 0.3, 0.6);
// Example quaternion (w, x, y, z) describing the target orientation as a 90-degree rotation around the z-axis
Eigen::Quaterniond target_orientation(-0.7071, 0, 0, 0.7071);
and the initial configuration,
Eigen::VectorXd initial_theta(7);
initial_theta << 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0; // Zero initial joint angles
and rebuild, test and run in order to check the success rate of convergence within 50 iterations.
The main crux of solving inverse kinematics numerically is handling the redundancy when you have more number of degrees of freedom than the dimensionality of the operational space. By incorporating the following modifications, you can handle redundancy more effectively and ensure that the joint limits are respected during convergence.
-
Damped Pseudo-Inverse: A damping factor is added to the pseudo-inverse calculation to improve numerical stability, especially to avoid singularities.
-
Null-Space Projection: The null-space matrix 𝑁 allows for movements that do not affect the primary task. This is used to optimize secondary objectives, such as keeping joint angles away from limits.
-
Secondary Objective: A simple gradient descent towards the middle of the joint range helps avoid joint limits. The secondary objective term
$\alpha q_{diff}$ can be adjusted according to the specific requirements of your application. -
Joint Limit Enforcement: After each iteration, the joint angles are clamped to ensure they stay within the specified limits.
View the complete algorithm here - Inverse Kinematics Algorithm
Initial
$q_{seed}$ , the damping factor$\lambda$ , and the secondary objective gain$\alpha$ plays a crucial role in convergence. They act as a parameter set that needs to be further tuned in order to converge within stipulated amount of iterations. Further refinement would be on how to change these parameters to find the optimal deviation on angles and quickly converge to the optimal configuration within the iteration limits.