2024-01-28 17:11:38 +08:00
|
|
|
|
// lcm related headfile
|
|
|
|
|
#include <lcm/lcm-cpp.hpp>
|
|
|
|
|
#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 <iostream>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <math.h>
|
|
|
|
|
#include <cmath>
|
|
|
|
|
// unitree_sdk2 related headfile
|
|
|
|
|
#include <unitree/robot/channel/channel_publisher.hpp>
|
|
|
|
|
#include <unitree/robot/channel/channel_subscriber.hpp>
|
|
|
|
|
#include <unitree/idl/go2/LowState_.hpp>
|
|
|
|
|
#include <unitree/idl/go2/LowCmd_.hpp>
|
|
|
|
|
#include <unitree/idl/go2/WirelessController_.hpp>
|
|
|
|
|
#include <unitree/robot/client/client.hpp>
|
|
|
|
|
#include <unitree/common/thread/thread.hpp>
|
|
|
|
|
#include <unitree/common/time/time_tool.hpp>
|
|
|
|
|
#include <unitree/robot/go2/robot_state/robot_state_client.hpp>
|
|
|
|
|
|
|
|
|
|
#define TOPIC_LOWCMD "rt/lowcmd"
|
|
|
|
|
#define TOPIC_LOWSTATE "rt/lowstate"
|
|
|
|
|
#define TOPIC_JOYSTICK "rt/wirelesscontroller"
|
|
|
|
|
|
|
|
|
|
constexpr double PosStopF = (2.146E+9f);
|
|
|
|
|
constexpr double VelStopF = (16000.0f);
|
|
|
|
|
|
|
|
|
|
uint32_t crc32_core(uint32_t* ptr, uint32_t len)
|
|
|
|
|
{ // 无需更改:Unitree 提供的电机校验函数
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// 遥控器键值联合体
|
|
|
|
|
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<unitree_go::msg::dds_::LowCmd_> lowcmd_publisher;
|
|
|
|
|
unitree::robot::ChannelSubscriberPtr<unitree_go::msg::dds_::LowState_> lowstate_subscriber;
|
|
|
|
|
unitree::robot::ChannelSubscriberPtr<unitree_go::msg::dds_::WirelessController_> 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(10.0f);
|
|
|
|
|
rsc.Init();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int Custom::queryServiceStatus(const std::string& serviceName)
|
|
|
|
|
{
|
|
|
|
|
std::vector<unitree::robot::go2::ServiceState> serviceStateList;
|
|
|
|
|
int ret,serviceStatus;
|
|
|
|
|
ret = rsc.ServiceList(serviceStateList);
|
|
|
|
|
size_t i, count=serviceStateList.size();
|
|
|
|
|
for (i=0; i<count; i++)
|
|
|
|
|
{
|
|
|
|
|
const unitree::robot::go2::ServiceState& serviceState = serviceStateList[i];
|
|
|
|
|
if(serviceState.name == serviceName)
|
|
|
|
|
{
|
|
|
|
|
if(serviceState.status == 0)
|
|
|
|
|
{
|
|
|
|
|
std::cout << "name: " << serviceState.name <<" is activate"<<std::endl;
|
|
|
|
|
serviceStatus = 1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
std::cout << "name:" << serviceState.name <<" is deactivate"<<std::endl;
|
|
|
|
|
serviceStatus = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return serviceStatus;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Custom::activateService(const std::string& serviceName,int activate)
|
|
|
|
|
{
|
|
|
|
|
rsc.ServiceSwitch(serviceName, activate);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Custom::LowStateMessageHandler(const void* message)
|
|
|
|
|
{
|
|
|
|
|
// 用sdk2读取的底层state
|
|
|
|
|
low_state = *(unitree_go::msg::dds_::LowState_*)message;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Custom::JoystickHandler(const void *message)
|
|
|
|
|
{
|
|
|
|
|
// 遥控器信号
|
|
|
|
|
joystick = *(unitree_go::msg::dds_::WirelessController_ *)message;
|
|
|
|
|
key.value = joystick.keys();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------------------
|
|
|
|
|
// 线程 1 : lcm send 线程
|
|
|
|
|
void Custom::lcm_send(){
|
|
|
|
|
// leg_control_lcm_data
|
|
|
|
|
for (int i = 0; i < 12; i++)
|
|
|
|
|
{
|
|
|
|
|
leg_control_lcm_data.q[i] = low_state.motor_state()[i].q();
|
|
|
|
|
leg_control_lcm_data.qd[i] = low_state.motor_state()[i].dq();
|
|
|
|
|
leg_control_lcm_data.tau_est[i] = low_state.motor_state()[i].tau_est();
|
|
|
|
|
}
|
|
|
|
|
// 从IMU读取姿态信息
|
|
|
|
|
for(int i = 0; i < 4; i++){
|
|
|
|
|
// 四元数
|
|
|
|
|
body_state_simple.quat[i] = low_state.imu_state().quaternion()[i];
|
|
|
|
|
}
|
|
|
|
|
for(int i = 0; i < 3; i++){
|
|
|
|
|
// roll pitch yaw
|
|
|
|
|
body_state_simple.rpy[i] = low_state.imu_state().rpy()[i];
|
|
|
|
|
// IMU 三轴加速度
|
|
|
|
|
body_state_simple.aBody[i] = low_state.imu_state().accelerometer()[i];
|
|
|
|
|
// IMU 三轴线性加速度
|
|
|
|
|
body_state_simple.omegaBody[i] = low_state.imu_state().gyroscope()[i];
|
|
|
|
|
}
|
|
|
|
|
for(int i = 0; i < 4; i++){
|
|
|
|
|
// 足端触地力
|
|
|
|
|
body_state_simple.contact_estimate[i] = low_state.foot_force()[i];
|
|
|
|
|
}
|
|
|
|
|
// 遥控器按键值和摇杆数值
|
|
|
|
|
rc_command.left_stick[0] = joystick.lx();
|
|
|
|
|
rc_command.left_stick[1] = joystick.ly();
|
|
|
|
|
rc_command.right_stick[0] = joystick.rx();
|
|
|
|
|
rc_command.right_stick[1] = joystick.ry();
|
|
|
|
|
rc_command.right_lower_right_switch = key.components.R2;
|
|
|
|
|
rc_command.right_upper_switch = key.components.R1;
|
|
|
|
|
rc_command.left_lower_left_switch = key.components.L2;
|
|
|
|
|
rc_command.left_upper_switch = key.components.L1;
|
|
|
|
|
|
|
|
|
|
// 这里的mode是用来控制啥的?
|
|
|
|
|
if(key.components.A > 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;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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; //接收神经网络输出的关节信号
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------------------
|
|
|
|
|
// 线程 2 : lcm receive 线程
|
|
|
|
|
void Custom::lcm_receive(){
|
|
|
|
|
// 循环处理,接受lcm消息
|
|
|
|
|
while(true){
|
|
|
|
|
lc.handle();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
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] = 60; // 关节PD参数有待调整 60
|
|
|
|
|
joint_command_simple.kd[i] = 5; // 关节PD参数有待调整 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"<<std::endl;
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// -------------------------------------------------------------------------------
|
|
|
|
|
// 线程 3 : unitree_sdk2 send 线程
|
|
|
|
|
void Custom::LowCmdWrite(){
|
|
|
|
|
motiontime++;
|
|
|
|
|
|
|
|
|
|
if(_firstRun && leg_control_lcm_data.q[0] != 0){
|
|
|
|
|
for(int i = 0; i < 12; i++){
|
|
|
|
|
// 程序首次运行至此的时候
|
|
|
|
|
// 将当前各关节角度设置为目标角度
|
|
|
|
|
joint_command_simple.q_des[i] = leg_control_lcm_data.q[i];
|
|
|
|
|
// 初始化L2+B,防止damping被误触发
|
|
|
|
|
key.components.B = 0;
|
|
|
|
|
key.components.L2 = 0;
|
|
|
|
|
}
|
|
|
|
|
_firstRun = false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// 写了一段安全冗余代码
|
|
|
|
|
// 当roll角超过限制,或pitch角超过限制,或遥控器按下L2+B键
|
|
|
|
|
// if ( low_state.imu_state().rpy()[0] > 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.5 || std::abs(low_state.imu_state().rpy()[1]) > 0.5 || ((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;
|
|
|
|
|
}
|
2024-03-10 21:53:11 +08:00
|
|
|
|
std::cout << "======= Switched to Damping Mode, and the thread is sleeping ========"<<std::endl;
|
|
|
|
|
sleep(0.5);
|
|
|
|
|
|
|
|
|
|
while (true)
|
|
|
|
|
{
|
|
|
|
|
sleep(0.5);
|
|
|
|
|
if ((int)key.components.B==1 && (int)key.components.L2==1) // [L2+B] is pressed again
|
|
|
|
|
{
|
|
|
|
|
exit(0);
|
|
|
|
|
}else if ((int)key.components.A==1 && (int)key.components.L2==1)
|
|
|
|
|
{
|
|
|
|
|
std::cout << "======= activate sport_mode service ========"<<std::endl;
|
|
|
|
|
rsc.ServiceSwitch("sport_mode", 1);
|
|
|
|
|
sleep(1);
|
|
|
|
|
exit(0);
|
|
|
|
|
}else
|
|
|
|
|
{ std::cout << "======= Press [L2+B] again to exit ========"<<std::endl;
|
|
|
|
|
std::cout << "======= If the robot is set to nominal pose manually, Press [L2+A] to activate sport_mode service ========"<<std::endl;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-28 17:11:38 +08:00
|
|
|
|
}
|
|
|
|
|
else{
|
|
|
|
|
for (int i = 0; i < 12; i++)
|
|
|
|
|
{
|
|
|
|
|
// 在确保安全的前提下,才执行神经网络模型的输出
|
|
|
|
|
low_cmd.motor_cmd()[i].q() = joint_command_simple.q_des[i];
|
|
|
|
|
low_cmd.motor_cmd()[i].dq() = joint_command_simple.qd_des[i];
|
|
|
|
|
low_cmd.motor_cmd()[i].kp() = joint_command_simple.kp[i];
|
|
|
|
|
low_cmd.motor_cmd()[i].kd() = joint_command_simple.kd[i];
|
|
|
|
|
low_cmd.motor_cmd()[i].tau() = joint_command_simple.tau_ff[i];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*此段代码中第一行首先计算了 CRC 校验码。
|
|
|
|
|
最后一行代码表示调用 lowcmd_publisher的Write()函数将控制命令发送给 Go2 机器人。*/
|
|
|
|
|
low_cmd.crc() = crc32_core((uint32_t *)&low_cmd, (sizeof(unitree_go::msg::dds_::LowCmd_)>>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<unitree_go::msg::dds_::LowCmd_>(TOPIC_LOWCMD));
|
|
|
|
|
lowcmd_publisher->InitChannel();
|
|
|
|
|
/*create low_state dds subscriber*/
|
|
|
|
|
lowstate_subscriber.reset(new unitree::robot::ChannelSubscriber<unitree_go::msg::dds_::LowState_>(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<unitree_go::msg::dds_::WirelessController_>(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);
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-10 21:53:11 +08:00
|
|
|
|
std::cout << "Communication level is set to LOW-level." << std::endl
|
2024-01-28 17:11:38 +08:00
|
|
|
|
<< "WARNING: Make sure the robot is hung up." << std::endl
|
2024-03-10 21:53:11 +08:00
|
|
|
|
<< "Caution: The scripts is about to shutdown Unitree sport_mode Service." << std::endl
|
2024-01-28 17:11:38 +08:00
|
|
|
|
<< "Press Enter to continue..." << std::endl;
|
|
|
|
|
std::cin.ignore();
|
|
|
|
|
|
2024-03-10 21:53:11 +08:00
|
|
|
|
unitree::robot::ChannelFactory::Instance()->Init(0, argv[1]); // 传入本机的网卡地址(PC or Jetson Orin)
|
2024-01-28 17:11:38 +08:00
|
|
|
|
|
|
|
|
|
Custom custom;
|
|
|
|
|
|
|
|
|
|
custom.InitRobotStateClient();
|
2024-03-10 21:53:11 +08:00
|
|
|
|
if(custom.queryServiceStatus("sport_mode"))
|
2024-01-28 17:11:38 +08:00
|
|
|
|
{
|
|
|
|
|
std::cout<<"Trying to deactivate the service: " << "sport_mode" << std::endl;
|
|
|
|
|
custom.activateService("sport_mode",0);
|
|
|
|
|
sleep(1);
|
2024-03-10 21:53:11 +08:00
|
|
|
|
} 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();
|
2024-01-28 17:11:38 +08:00
|
|
|
|
}
|
2024-03-10 21:53:11 +08:00
|
|
|
|
|
2024-01-28 17:11:38 +08:00
|
|
|
|
|
|
|
|
|
custom.Init();
|
|
|
|
|
|
|
|
|
|
std::cout<<"Communicatino is set up successfully" << std::endl;
|
|
|
|
|
std::cout<<"LCM <<<------------>>> Unitree SDK2" << std::endl;
|
2024-03-10 21:53:11 +08:00
|
|
|
|
std::cout<<"------------------------------------" << std::endl;
|
|
|
|
|
std::cout<<"------------------------------------" << std::endl;
|
|
|
|
|
std::cout<<"Press L2+B if any unexpected error occurs" << std::endl;
|
2024-01-28 17:11:38 +08:00
|
|
|
|
|
|
|
|
|
custom.Loop();
|
|
|
|
|
|
2024-03-10 21:53:11 +08:00
|
|
|
|
while (true)
|
2024-01-28 17:11:38 +08:00
|
|
|
|
{
|
|
|
|
|
sleep(10);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|