// lcm related headfile #include #include "leg_control_data_lcmt.hpp" #include "state_estimator_lcmt.hpp" #include "rc_command_lcmt.hpp" #include "pd_tau_targets_lcmt.hpp" // standard headfile #include #include #include #include #include // unitree_sdk2 related headfile #include #include #include #include #include #include #include #include #include #define TOPIC_LOWCMD "rt/lowcmd" #define TOPIC_LOWSTATE "rt/lowstate" #define TOPIC_JOYSTICK "rt/wirelesscontroller" // 为保证项目代码的稳定性和易理解,没有采用unitree_sdk2中采用的using namespace语句 constexpr double PosStopF = (2.146E+9f); constexpr double VelStopF = (16000.0f); // 无需更改:Unitree 提供的电机校验函数 uint32_t crc32_core(uint32_t* ptr, uint32_t len) { unsigned int xbit = 0; unsigned int data = 0; unsigned int CRC32 = 0xFFFFFFFF; const unsigned int dwPolynomial = 0x04c11db7; for (unsigned int i = 0; i < len; i++) { xbit = 1 << 31; data = ptr[i]; for (unsigned int bits = 0; bits < 32; bits++) { if (CRC32 & 0x80000000) { CRC32 <<= 1; CRC32 ^= dwPolynomial; } else { CRC32 <<= 1; } if (data & xbit) CRC32 ^= dwPolynomial; xbit >>= 1; } } return CRC32; } // 遥控器键值联合体,摘自unitree_sdk2,无需更改 typedef union { struct { uint8_t R1 : 1; uint8_t L1 : 1; uint8_t start : 1; uint8_t select : 1; uint8_t R2 : 1; uint8_t L2 : 1; uint8_t F1 : 1; uint8_t F2 : 1; uint8_t A : 1; uint8_t B : 1; uint8_t X : 1; uint8_t Y : 1; uint8_t up : 1; uint8_t right : 1; uint8_t down : 1; uint8_t left : 1; } components; uint16_t value; } xKeySwitchUnion; class Custom { public: explicit Custom(){} ~Custom(){} void Init(); void InitLowCmd(); void Loop(); void LowStateMessageHandler(const void* messages); void JoystickHandler(const void *message); void InitRobotStateClient(); void activateService(const std::string& serviceName,int activate); void lcm_send(); void lcm_receive(); void lcm_receive_Handler(const lcm::ReceiveBuffer *rbuf, const std::string & chan, const pd_tau_targets_lcmt* msg); void LowCmdWrite(); void SetNominalPose(); int queryServiceStatus(const std::string& serviceName); leg_control_data_lcmt leg_control_lcm_data = {0}; state_estimator_lcmt body_state_simple = {0}; pd_tau_targets_lcmt joint_command_simple = {0}; rc_command_lcmt rc_command = {0}; unitree_go::msg::dds_::LowState_ low_state{}; unitree_go::msg::dds_::LowCmd_ low_cmd{}; unitree_go::msg::dds_::WirelessController_ joystick{}; unitree::robot::go2::RobotStateClient rsc; unitree::robot::ChannelPublisherPtr lowcmd_publisher; unitree::robot::ChannelSubscriberPtr lowstate_subscriber; unitree::robot::ChannelSubscriberPtr joystick_suber; lcm::LCM lc; xKeySwitchUnion key; int mode = 0; int motiontime = 0; float dt = 0.002; // unit [second] bool _firstRun; /*LowCmd write thread*/ // DDS相关的底层命令发送线程指针 unitree::common::ThreadPtr LcmSendThreadPtr; unitree::common::ThreadPtr LcmRecevThreadPtr; unitree::common::ThreadPtr lowCmdWriteThreadPtr; }; void Custom::InitRobotStateClient() { rsc.SetTimeout(5.0f); rsc.Init(); } int Custom::queryServiceStatus(const std::string& serviceName) { std::vector serviceStateList; int ret,serviceStatus; ret = rsc.ServiceList(serviceStateList); size_t i, count=serviceStateList.size(); for (i=0; i 0){ mode = 0; } else if(key.components.B > 0){ mode = 1; }else if(key.components.X > 0){ mode = 2; }else if(key.components.Y > 0){ mode = 3; }else if(key.components.up > 0){ mode = 4; }else if(key.components.right > 0){ mode = 5; }else if(key.components.down > 0){ mode = 6; }else if(key.components.left > 0){ mode = 7; } rc_command.mode = mode; lc.publish("leg_control_data", &leg_control_lcm_data); lc.publish("state_estimator_data", &body_state_simple); lc.publish("rc_command", &rc_command); // std::cout << "loop: messsages are sending ......" << std::endl; } // ------------------------------------------------------------------------------- // 线程 2 : lcm receive 线程 // 此线程作用:实时通过lcm中间件读取pytorch神经网络输出的期望关节控制信号(q, qd, kp, kd, tau_ff) // 查看 go2_gym_deploy/envs/lcm_agent.py 文件,可以知道: // 神经网络只输出期望的q,而kp,kd是可以自定义设置的, qd 和 tau_ff 被设置为0 void Custom::lcm_receive_Handler(const lcm::ReceiveBuffer *rbuf, const std::string & chan, const pd_tau_targets_lcmt* msg){ (void) rbuf; (void) chan; joint_command_simple = *msg; //接收神经网络输出的关节信号 } // 此处参考lcm推荐的标准格式,循环处理,接受lcm消息 void Custom::lcm_receive(){ while(true){ lc.handle(); } } // ------------------------------------------------------------------------------- // 线程 3 : unitree_sdk2 command write 线程 // 此线程作用:初始化low_cmd,经过合理的状态机后,电机将执行神经网络的输出 void Custom::InitLowCmd() { //LowCmd 类型中的 head 成员 表示帧头, //此帧头将用于 CRC 校验。head 、levelFlag、gpio 等按例程所示设置为默认值即可。 low_cmd.head()[0] = 0xFE; low_cmd.head()[1] = 0xEF; low_cmd.level_flag() = 0xFF; low_cmd.gpio() = 0; /*LowCmd 类型中有 20 个 motorCmd 成员, 每一个成员的命令用于控制 Go2 机器人上相对应的一个电机, 但 Go2 机器人上只有 12 个电机, 故仅有前 12 个有效,剩余的8个起保留作用。*/ for(int i=0; i<20; i++) { /*此行命令中将 motorCmd 成员的 mode 变量设置为 0x01, 0x01 表示将电机设置为伺服模式。 如果用户在调试过程中发现无法控制 Go2 机器人的关节电机, 请检查变量的值是否为0x01。*/ low_cmd.motor_cmd()[i].mode() = (0x01); // motor switch to servo (PMSM) mode low_cmd.motor_cmd()[i].q() = (PosStopF); low_cmd.motor_cmd()[i].dq() = (VelStopF); low_cmd.motor_cmd()[i].kp() = (0); low_cmd.motor_cmd()[i].kd() = (0); low_cmd.motor_cmd()[i].tau() = (0); } } void Custom::SetNominalPose(){ // 运行此cpp文件后,不仅是初始化通信 // 同样会在趴下时的初始化关节角度 // 将各个电机都设置为位置模式 for(int i = 0; i < 12; i++){ joint_command_simple.qd_des[i] = 0; joint_command_simple.tau_ff[i] = 0; joint_command_simple.kp[i] = 20; joint_command_simple.kd[i] = 0.5; } // 趴下时的关节角度 joint_command_simple.q_des[0] = -0.3; joint_command_simple.q_des[1] = 1.2; joint_command_simple.q_des[2] = -2.721; joint_command_simple.q_des[3] = 0.3; joint_command_simple.q_des[4] = 1.2; joint_command_simple.q_des[5] = -2.721; joint_command_simple.q_des[6] = -0.3; joint_command_simple.q_des[7] = 1.2; joint_command_simple.q_des[8] = -2.721; joint_command_simple.q_des[9] = 0.3; joint_command_simple.q_des[10] = 1.2; joint_command_simple.q_des[11] = -2.721; std::cout<<"SET NOMINAL POSE"< 0.5 || low_state.imu_state().rpy()[1] > 0.5 || ((int)key.components.B==1 && (int)key.components.L2==1)) if ( std::abs(low_state.imu_state().rpy()[0]) > 0.8 || std::abs(low_state.imu_state().rpy()[1]) > 0.8 || ((int)key.components.B==1 && (int)key.components.L2==1)) { for (int i = 0; i < 12; i++){ // 进入damping模式 low_cmd.motor_cmd()[i].q() = 0; low_cmd.motor_cmd()[i].dq() = 0; low_cmd.motor_cmd()[i].kp() = 0; low_cmd.motor_cmd()[i].kd() = 5; low_cmd.motor_cmd()[i].tau() = 0; } std::cout << "======= Switched to Damping Mode, and the thread is sleeping ========"<>> Unitree SDK2" << std::endl; std::cout<<"------------------------------------" << std::endl; std::cout<<"------------------------------------" << std::endl; std::cout<<"Press L2+B if any unexpected error occurs" << std::endl; break; }else{ std::cout << "======= Press [L2+B] again to exit ========"<>2)-1); lowcmd_publisher->Write(low_cmd); } // // 与循环工作的线程相关的函数定义已完结 //---------------------------------------------------------------------- void Custom::Init(){ _firstRun = true; InitLowCmd(); SetNominalPose(); // 这里决定了调用lc.handle()的时候,订阅什么消息,进行什么操作 // 订阅什么消息:"pd_plustau_targets" // 进行什么操作: lcm_receive_Handler lc.subscribe("pd_plustau_targets", &Custom::lcm_receive_Handler, this); /*create low_cmd publisher*/ lowcmd_publisher.reset(new unitree::robot::ChannelPublisher(TOPIC_LOWCMD)); lowcmd_publisher->InitChannel(); /*create low_state dds subscriber*/ lowstate_subscriber.reset(new unitree::robot::ChannelSubscriber(TOPIC_LOWSTATE)); lowstate_subscriber->InitChannel(std::bind(&Custom::LowStateMessageHandler, this, std::placeholders::_1), 1); /*create joystick dds subscriber*/ joystick_suber.reset(new unitree::robot::ChannelSubscriber(TOPIC_JOYSTICK)); joystick_suber->InitChannel(std::bind(&Custom::JoystickHandler, this, std::placeholders::_1), 1); } void Custom::Loop(){ // 新增线程可以实现loop function的功能 // intervalMicrosec : 1微秒 = 0.000001秒 // 当dt=0.002s // ntervalMicrosec = 2000us /*lcm send thread*/ LcmSendThreadPtr = unitree::common::CreateRecurrentThreadEx("lcm_send_thread", UT_CPU_ID_NONE, dt*1e6, &Custom::lcm_send, this); /*lcm receive thread*/ LcmRecevThreadPtr = unitree::common::CreateRecurrentThreadEx("lcm_recev_thread", UT_CPU_ID_NONE, dt*1e6, &Custom::lcm_receive, this); /*low command write thread*/ lowCmdWriteThreadPtr = unitree::common::CreateRecurrentThreadEx("dds_write_thread", UT_CPU_ID_NONE, dt*1e6, &Custom::LowCmdWrite, this); } int main(int argc, char **argv) { if (argc < 2) { std::cout << "Usage: " << argv[0] << " networkInterface" << std::endl; exit(-1); } std::cout << "Communication level is set to LOW-level." << std::endl << "WARNING: Make sure the robot is hung up." << std::endl << "Caution: The scripts is about to shutdown Unitree sport_mode Service." << std::endl << "Press Enter to continue..." << std::endl; std::cin.ignore(); unitree::robot::ChannelFactory::Instance()->Init(0, argv[1]); // 传入本机的网卡地址(PC or Jetson Orin) Custom custom; custom.InitRobotStateClient(); if(custom.queryServiceStatus("sport_mode")) { std::cout<<"Trying to deactivate the service: " << "sport_mode" << std::endl; custom.activateService("sport_mode",0); sleep(0.5); if(!custom.queryServiceStatus("sport_mode")){ std::cout<<"Trying to deactivate the service: " << "sport_mode" << std::endl; } } else{ std::cout <<"sportd_mode is already deactivated now" << std::endl <<"next step is setting up communication" << std::endl << "Press Enter to continue..." << std::endl; std::cin.ignore(); } custom.Init(); std::cout<<"Communicatino is set up successfully" << std::endl; std::cout<<"LCM <<<------------>>> Unitree SDK2" << std::endl; std::cout<<"------------------------------------" << std::endl; std::cout<<"------------------------------------" << std::endl; std::cout<<"Press L2+B if any unexpected error occurs" << std::endl; custom.Loop(); while (true) { sleep(10); } return 0; }