allegroNode.cpp
Go to the documentation of this file.
00001 /*
00002  * allegroNode.cpp
00003  *
00004  *  Created on: Feb 1, 2013
00005  *  Authors: Alex ALSPACH, Seungsu KIM
00006  */
00007  
00008 // JOINT SPACE POSITION CONTROL
00009 // Using  timer callback 
00010  
00011 #include <iostream>
00012 #include <boost/thread/thread.hpp>
00013 #include <boost/date_time.hpp>
00014 #include <boost/thread/locks.hpp>
00015 
00016 #include "ros/ros.h"
00017 #include "ros/service.h"
00018 #include "ros/service_server.h"
00019 #include "sensor_msgs/JointState.h"
00020 #include "std_msgs/String.h"
00021 #include "std_msgs/Float32.h"
00022 
00023 #include <stdio.h>
00024 #include <math.h>
00025 #include <iostream>
00026 #include <string>
00027 
00028 #include "controlAllegroHand.h"
00029 
00030 
00031 #define RADIANS_TO_DEGREES(radians) ((radians) * (180.0 / M_PI))
00032 #define DEGREES_TO_RADIANS(angle) ((angle) / 180.0 * M_PI)
00033 
00034 
00035 // Topics
00036 #define JOINT_STATE_TOPIC "/allegroHand/joint_states"
00037 #define JOINT_CMD_TOPIC "/allegroHand/joint_cmd"
00038 #define EXT_CMD_TOPIC "/allegroHand/external_cmd"
00039 
00040 double current_position[DOF_JOINTS]                     = {0.0};
00041 double previous_position[DOF_JOINTS]                    = {0.0};
00042 
00043 double current_position_filtered[DOF_JOINTS]    = {0.0};
00044 double previous_position_filtered[DOF_JOINTS]   = {0.0};
00045 
00046 double current_velocity[DOF_JOINTS]                     = {0.0};
00047 double previous_velocity[DOF_JOINTS]                    = {0.0};
00048 double current_velocity_filtered[DOF_JOINTS]    = {0.0};
00049 
00050 double desired_position[DOF_JOINTS]                             = {0.0};
00051 double desired_torque[DOF_JOINTS]                               = {0.0};
00052 
00053 double k_p[DOF_JOINTS]                          = { 600.0,  600.0,  600.0, 1000.0,  // Default P Gains for PD Controller
00054                                                                                 600.0,  600.0,  600.0, 1000.0,  // These gains are loaded if the 'gains_pd.yaml' file is not loaded
00055                                                                                 600.0,  600.0,  600.0, 1000.0,
00056                                                                            1000.0, 1000.0, 1000.0,  600.0 };
00057 
00058 double k_d[DOF_JOINTS]                          = {  15.0,   20.0,   15.0,   15.0,  // Default D Gains for PD Controller
00059                                                                                  15.0,   20.0,   15.0,   15.0,  // These gains are loaded if the 'gains_pd.yaml' file is not loaded
00060                                                                                  15.0,   20.0,   15.0,   15.0,
00061                                                                                  30.0,   20.0,   20.0,   15.0 };
00062                 
00063                                                                         
00064 double home_pose[DOF_JOINTS]            = {   0.0,  -10.0,   45.0,   45.0,  // Default (HOME) position (degrees)
00065                                                                                   0.0,  -10.0,   45.0,   45.0,  // This position is loaded and set upon system start
00066                                                                                   5.0,   -5.0,   50.0,   45.0,  // if no 'initial_position.yaml' parameter is loaded.
00067                                                                              60.0,   25.0,   15.0,   45.0 };
00068 
00069 
00070 std::string pGainParams[DOF_JOINTS] = { "~gains_pd/p/j00", "~gains_pd/p/j01", "~gains_pd/p/j02", "~gains_pd/p/j03", 
00071                                                                                 "~gains_pd/p/j10", "~gains_pd/p/j11", "~gains_pd/p/j12", "~gains_pd/p/j13",
00072                                                                                 "~gains_pd/p/j20", "~gains_pd/p/j21", "~gains_pd/p/j22", "~gains_pd/p/j23", 
00073                                                                                 "~gains_pd/p/j30", "~gains_pd/p/j31", "~gains_pd/p/j32", "~gains_pd/p/j33"};
00074 
00075 std::string dGainParams[DOF_JOINTS] = { "~gains_pd/d/j00", "~gains_pd/d/j01", "~gains_pd/d/j02", "~gains_pd/d/j03", 
00076                                                                                 "~gains_pd/d/j10", "~gains_pd/d/j11", "~gains_pd/d/j12", "~gains_pd/d/j13",
00077                                                                                 "~gains_pd/d/j20", "~gains_pd/d/j21", "~gains_pd/d/j22", "~gains_pd/d/j23", 
00078                                                                                 "~gains_pd/d/j30", "~gains_pd/d/j31", "~gains_pd/d/j32", "~gains_pd/d/j33"};
00079                                                                                 
00080 std::string initialPosition[DOF_JOINTS] = {     "~initial_position/j00", "~initial_position/j01", "~initial_position/j02", "~initial_position/j03", 
00081                                                                                         "~initial_position/j10", "~initial_position/j11", "~initial_position/j12", "~initial_position/j13",
00082                                                                                         "~initial_position/j20", "~initial_position/j21", "~initial_position/j22", "~initial_position/j23", 
00083                                                                                         "~initial_position/j30", "~initial_position/j31", "~initial_position/j32", "~initial_position/j33"};                                                                            
00084 
00085 
00086 std::string jointNames[DOF_JOINTS]      = {    "joint_0.0",    "joint_1.0",    "joint_2.0",   "joint_3.0" , 
00087                                                                                    "joint_4.0",    "joint_5.0",    "joint_6.0",   "joint_7.0" , 
00088                                                                                    "joint_8.0",    "joint_9.0",    "joint_10.0",  "joint_11.0", 
00089                                                                                    "joint_12.0",   "joint_13.0",   "joint_14.0",  "joint_15.0" };
00090 
00091 
00092 int frame = 0;
00093 
00094 // Flags
00095 int lEmergencyStop = 0;
00096 
00097 boost::mutex *mutex;
00098 
00099 // ROS Messages
00100 ros::Publisher joint_state_pub;
00101 ros::Subscriber joint_cmd_sub;          // handles external joint command (eg. sensor_msgs/JointState)
00102 ros::Subscriber external_cmd_sub;       // handles any other type of eternal command (eg. std_msgs/String)
00103 sensor_msgs::JointState msgJoint;
00104 std::string  ext_cmd;
00105 
00106 // ROS Time
00107 ros::Time tstart;
00108 ros::Time tnow;
00109 double secs;
00110 double dt;
00111 
00112 // Initialize CAN device                
00113 controlAllegroHand *canDevice;
00114 
00115 
00116 
00117 
00118 
00119 
00120 
00121 // Called when a desired joint position message is received
00122 void SetjointCallback(const sensor_msgs::JointState& msg)
00123 {
00124         mutex->lock();
00125         for(int i=0;i<DOF_JOINTS;i++) desired_position[i] = msg.position[i];
00126         mutex->unlock();        
00127 }
00128 
00129 
00130 // Called when an external (string) message is received
00131 void extCmdCallback(const std_msgs::String::ConstPtr& msg)
00132 {
00133         ROS_INFO("CTRL: Processing external command");
00134         ext_cmd = msg->data.c_str();
00135 
00136         // Compare the message received to an expected input
00137         if (ext_cmd.compare("comparisonString") == 0)
00138         {
00139                 // Do something
00140         }    
00141         else
00142         {
00143                 // Do something else
00144         }       
00145 }
00146 
00147 
00148 
00149 
00150 
00151 
00152 // In case of the Allegro Hand, this callback is processed every 0.003 seconds
00153 void timerCallback(const ros::TimerEvent& event)
00154 {
00155 
00156         // Calculate loop time;
00157         tnow = ros::Time::now();
00158         dt = 1e-9*(tnow - tstart).nsec;
00159         tstart = tnow;
00160         
00161         //if ((1/dt)>400)
00162         //      printf("%f\n",(1/dt-333.33333));
00163         
00164                 
00165         // save last iteration info
00166         for(int i=0; i<DOF_JOINTS; i++)
00167         {
00168                 previous_position[i] = current_position[i];
00169                 previous_position_filtered[i] = current_position_filtered[i];
00170                 previous_velocity[i] = current_velocity[i];
00171         }
00172                 
00173                 
00174 /*  ================================= 
00175     =       CAN COMMUNICATION       = 
00176     ================================= */        
00177         canDevice->setTorque(desired_torque);           // WRITE joint torque
00178         lEmergencyStop = canDevice->update();           // Returns -1 in case of an error
00179         canDevice->getJointInfo(current_position);      // READ current joint positions
00180 
00181                 
00182                 
00183                                 
00184         // Stop program and shutdown node when Allegro Hand is switched off
00185         if( lEmergencyStop < 0 )
00186         {
00187                 ROS_ERROR("\n\nAllegro Hand Node is Shutting Down! (Emergency Stop)\n");
00188                 ros::shutdown();
00189         }
00190 
00191 
00192 
00193 /*  ================================= 
00194     =       LOWPASS FILTERING       =   
00195     ================================= */
00196         for(int i=0; i<DOF_JOINTS; i++)    
00197         {
00198         current_position_filtered[i] = (0.6*current_position_filtered[i]) + (0.198*previous_position[i]) + (0.198*current_position[i]);
00199                 current_velocity[i] = (current_position_filtered[i] - previous_position_filtered[i]) / dt;
00200                 current_velocity_filtered[i] = (0.6*current_velocity_filtered[i]) + (0.198*previous_velocity[i]) + (0.198*current_velocity[i]);
00201         }       
00202         
00203         
00204         
00205 /*  ================================= 
00206     =        POSITION CONTROL       =   
00207     ================================= */
00208     if(frame>100) // give the low pass filters 100 iterations (0.03s) to build up good data.
00209     {   
00210                 for(int i=0; i<DOF_JOINTS; i++)    
00211                 {
00212                         desired_torque[i] = k_p[i]*(desired_position[i]-current_position_filtered[i]) - k_d[i]*current_velocity_filtered[i];
00213                         desired_torque[i] = desired_torque[i]/800.0;
00214                 }
00215         }               
00216 
00217 
00218         // PUBLISH current position, velocity and effort (torque)
00219         msgJoint.header.stamp           = tnow; 
00220         for(int i=0; i<DOF_JOINTS; i++)
00221         {
00222                 msgJoint.position[i]    = current_position_filtered[i];
00223                 msgJoint.velocity[i]    = current_velocity_filtered[i];
00224                 msgJoint.effort[i]              = desired_torque[i];
00225         }
00226         joint_state_pub.publish(msgJoint);
00227                 
00228         frame++;
00229         
00230 
00231 } // end timerCallback
00232 
00233 
00234 
00235 
00236 
00237 
00238 
00239 
00240 int main(int argc, char** argv)
00241 {
00242 
00243         using namespace std;
00244         
00245         ros::init(argc, argv, "allegro_hand_core");
00246         ros::Time::init();
00247         
00248         ros::NodeHandle nh;
00249         
00250         
00251         // Setup timer callback (ALLEGRO_CONTROL_TIME_INTERVAL = 0.003)
00252         ros::Timer timer = nh.createTimer(ros::Duration(0.003), timerCallback);
00253 
00254         mutex = new boost::mutex();
00255 
00256         // Publisher and Subscribers
00257         joint_state_pub = nh.advertise<sensor_msgs::JointState>(JOINT_STATE_TOPIC, 3);
00258         joint_cmd_sub = nh.subscribe(JOINT_CMD_TOPIC, 3, SetjointCallback);
00259         external_cmd_sub = nh.subscribe(EXT_CMD_TOPIC, 1, extCmdCallback);
00260         
00261         // Create arrays 16 long for each of the four joint state components
00262         msgJoint.position.resize(DOF_JOINTS);
00263         msgJoint.velocity.resize(DOF_JOINTS);
00264         msgJoint.effort.resize(DOF_JOINTS);
00265         msgJoint.name.resize(DOF_JOINTS);
00266 
00267 
00268         // Joint names (for use with joint_state_publisher GUI - matches URDF)
00269         for(int i=0; i<DOF_JOINTS; i++) msgJoint.name[i] = jointNames[i];       
00270         
00271         
00272         // Get Allegro Hand information from parameter server
00273         // This information is found in the Hand-specific "zero.yaml" file from the allegro_hand_description package    
00274         string robot_name, whichHand, manufacturer, origin, serial;
00275         double version;
00276         ros::param::get("~hand_info/robot_name",robot_name);
00277         ros::param::get("~hand_info/which_hand",whichHand);
00278         ros::param::get("~hand_info/manufacturer",manufacturer);
00279         ros::param::get("~hand_info/origin",origin);
00280         ros::param::get("~hand_info/serial",serial);
00281         ros::param::get("~hand_info/version",version);
00282 
00283 
00284         // set gains_pd via gains_pd.yaml or to defaul values
00285         if (ros::param::has("~gains_pd"))
00286         {
00287                 ROS_INFO("\n\nCTRL: PD gains loaded from param server.\n");
00288                 for(int i=0; i<DOF_JOINTS; i++)
00289                 {
00290                         ros::param::get(pGainParams[i], k_p[i]);
00291                         ros::param::get(dGainParams[i], k_d[i]);
00292                         //printf("%f ", k_p[i]);
00293                 }
00294                 //printf("\n");
00295         }
00296         else
00297         {
00298                 // gains will be loaded every control iteration
00299                 ROS_WARN("\n\nCTRL: PD gains not loaded.\nCheck launch file is loading /parameters/gains_pd.yaml\nloading default PD gains...\n");
00300         }
00301 
00302 
00303         // set initial position via initial_position.yaml or to defaul values
00304         if (ros::param::has("~initial_position"))
00305         {
00306                 ROS_INFO("\n\nCTRL: Initial Pose loaded from param server.\n");
00307                 for(int i=0; i<DOF_JOINTS; i++)
00308                 {
00309                         ros::param::get(initialPosition[i], desired_position[i]);
00310                         desired_position[i] = DEGREES_TO_RADIANS(desired_position[i]);
00311                 }
00312         }
00313         else
00314         {
00315                 ROS_WARN("\n\nCTRL: Initial postion not loaded.\nCheck launch file is loading /parameters/initial_position.yaml\nloading Home position instead...\n");
00316                 // Home position
00317                 for(int i=0; i<DOF_JOINTS; i++) desired_position[i] = DEGREES_TO_RADIANS(home_pose[i]);                                                                         
00318         }
00319 
00320 
00321         // Initialize CAN device
00322         canDevice = new controlAllegroHand();
00323         canDevice->init();
00324         usleep(3000);
00325         
00326         // Dump Allegro Hand information to the terminal        
00327         cout << endl << endl << robot_name << " v" << version << endl << serial << " (" << whichHand << ")" << endl << manufacturer << endl << origin << endl << endl;
00328 
00329         // Start ROS time
00330         tstart = ros::Time::now();
00331 
00332         // Starts control loop, message pub/subs and all other callbacks
00333         ros::spin();                    
00334 
00335         // Clean shutdown: shutdown node, shutdown can, be polite.
00336         nh.shutdown();
00337         delete canDevice;
00338         printf("\nAllegro Hand Node has been shut down. Bye!\n\n");
00339         return 0;
00340         
00341 }
00342 
00343 


allegro_hand_core_pd
Author(s): Alex Alspach (SimLab), Seungsu Kim (EPFL)
autogenerated on Mon Jan 6 2014 11:02:50