Skip to content

6D Cartesian space hybrid force-velocity control using positional inner loop and wrist mounted FT sensor.

Notifications You must be signed in to change notification settings

yifan-hou/force_control

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

77 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

force_control

Implementations of 6D Cartesian space admittance control. Supports hybrid force-velocity control.

The algorithm creates a virtual spring-mass-damper system using a position controlled robot. You can specify the following parameters:

  • 6x6 stiffness matrix, inertia matrix, damper matrix.

You can update the following online:

  • Direction and dimension of force (soft) and position (rigid) control axes

Hardware requirements:

  • A robot arm with high accuracy (high stiffness) position control interface.
  • A wrist mounted FT sensor.

Author: Yifan Hou yifanhou at stanford dot edu

SAFETY WARNING

Force control is a high rate, high order control scheme that can go very wrong very quickly. Make sure you understand what you are doing before using this code. If the compliance parameters are not suitable, e.g. the robot is configured to be too soft and light while force feedback is not well calibrated, the robot will drift away very fast, which can be dangerous.

For your own safety, the following steps are recommended before launching a force-controlled robot:

  1. Start from enabling only one translational compliance axis (using setForceControlledAxis). Get the compliance control to work, get a feeling of what parameters make sense for your robot before enabling more axes. Common mistakes to pay attention to:
  • Force feedback transformation is wrong. This could cause a positive feedback loop.
  • Force sensor is badly calibrated.
  • Compliance parameters are set to be too senstitive (unstable motion) or too insensitive (no response to external force).
  1. Safe parameters to use when testing for the first time:
  • Set the inertia value to the same magnitude as the actual robot mass. For example, 2~5kg is reasonable for a typical table top robot arm like ABB120, UR5e.
  • Set damping to a small value, e.g. 0.1
  • Set stiffness to a reasonable value.
  • direct_force_control_gains and direct_force_control_I_limit should be all zero.
  1. Make sure the robot stays clear from any potential collisions.
  2. Make sure you have the emergency stop button at your thumb.
  3. Start to run the controller.
  • Stop immediately if there is any sudden/unstable motion.
  • If the robot appears to be stable, gently push the robot in the direction where you enabled compliance. Check if the robot can be dragged as expected. If not, check your sensor sign/transformation/robot tool frame setting, etc.
  • If the signs/direction seems fine but the robot is just shaking a bit, graduately increase damping.
  1. Now you have a one-axis compliance control working. You can play with the parameters as you wish, e.g. graduately reduce the inertia values and damping to get a more "soft" feeling.
  2. Enable all three translational axes.
  3. Redo the above for rotational axes. Note the order of magnitude of parameters are quiet different between rotational and translational axes.

Install

Dependency

Please install the following packages:

Build

cd force_control
mkdir build && cd build
cmake ..
make -j
make install

How to use

Use with cmake

# replace ${CMAKE_INSTALL_PREFIX} with your install location
find_library(FORCE_CONTROLLERS FORCE_CONTROLLERS HINTS ${CMAKE_INSTALL_PREFIX}/lib/)
find_library(RUT Utilities HINTS ${CMAKE_INSTALL_PREFIX}/lib/RobotUtilities)

# your executable
add_executable(force_control_demo src/main.cc)
target_link_libraries(force_control_demo
  ${RUT}
  ${FORCE_CONTROLLERS}
)

config example

Save the following as config.yaml:

admittance_controller:
  dt: 0.002
  log_to_file: false
  log_file_path: "/tmp/admittance_controller.log"
  alert_overrun: false
  compliance6d:
    stiffness: [100, 100, 100, 1, 1, 1]
    damping: [2, 2, 2, 0.2, 0.2, 0.2]
    inertia: [5, 5, 5, 0.005, 0.005, 0.005]
    stiction: [0, 0, 0, 0, 0, 0]
  max_spring_force_magnitude: 50
  direct_force_control_gains:
    P_trans: 0
    I_trans: 0
    D_trans: 0
    P_rot: 0
    I_rot: 0
    D_rot: 0
  direct_force_control_I_limit: [0, 0, 0, 0, 0, 0]

c++ code example

Headers:

#include <RobotUtilities/spatial_utilities.h>
#include <force_control/admittance_controller.h>

typedef Eigen::Matrix<double, 6, 1> Vector6d;

Create the controller config, initialize controller:

// load config
AdmittanceController::AdmittanceControllerConfig admittance_config;
const std::string CONFIG_PATH = "path_to/config.yaml";
YAML::Node config{};
try {
  config = YAML::LoadFile(CONFIG_PATH);
  deserialize(config["admittance_controller"], admittance_config);
} catch (const std::exception& e) {
  std::cerr << "Failed to load the config file: " << e.what() << std::endl;
  return -1;
}

AdmittanceController controller;

RUT::Timer timer;
RUT::TimePoint time0 = timer.tic();
RUT::Vector7d pose, pose_ref, pose_cmd;
RUT::Vector6d wrench, wrench_WTr;

controller.init(time0, admittance_config, pose);

Set force control axes and dimension. There are a couple options:

// Regular admittance control, all 6 axes are force dimensions:
RUT::Matrix6d Tr = RUT::Matrix6d::Identity();
int n_af = 6;
controller.setForceControlledAxis(Tr, n_af);

// HFVC, compliant translational motion, rigid rotation motion 
RUT::Matrix6d Tr = RUT::Matrix6d::Identity();
int n_af = 3;
controller.setForceControlledAxis(Tr, n_af);

// HFVC, compliant rotational motion, rigid translational motion
RUT::Matrix6d Tr;
Tr << 0, 0, 0, 1, 0, 0,
    0, 0, 0, 0, 1, 0,
    0, 0, 0, 0, 0, 1,
    1, 0, 0, 0, 0, 0,
    0, 1, 0, 0, 0, 0,
    0, 0, 1, 0, 0, 0;
int n_af = 3;
controller.setForceControlledAxis(Tr, n_af);

// n_af = 0 disables compliance. All axes uses rigid motion
RUT::Matrix6d Tr = RUT::Matrix6d::Identity();
int n_af = 0;
controller.setForceControlledAxis(Tr, n_af);

Now we are ready to start the control loop. Assuming we have access to a robot_ptr object that can provides pose and wrench feedback.

pose_ref = pose;
wrench_WTr.setZero();

timer.tic();

while (true) {
    // Update robot status
    robot_ptr->getCartesian(pose);
    robot_ptr->getWrenchTool(wrench);
    controller.setRobotStatus(pose, wrench);

    // Update robot reference
    controller.setRobotReference(pose_ref, wrench_WTr);

    // Compute the control output
    controller.step(pose_cmd);

    // send action to robot
    robot_ptr->setCartesian(pose_cmd);
    
    // sleep till next iteration
    spin();
}

Reference

This package was implemented as a part of Y. Hou and M. T. Mason, "Robust Execution of Contact-Rich Motion Plans by Hybrid Force-Velocity Control," 2019 International Conference on Robotics and Automation (ICRA), Montreal, QC, Canada, 2019, pp. 1933-1939

The implementation was initially based on James A. Maples and Joseph J. Becker, "Experience in Force Control of Robotic Manipulators", Then a lot more functionalities were added. Please contact yifan for questions.

About

6D Cartesian space hybrid force-velocity control using positional inner loop and wrist mounted FT sensor.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published