## Changes  
1. Fixed `observations_history` order to ensure consistency in data processing.  
2. Removed row-column dependencies in `config.yaml` and explicitly specified joint order.  
3. Simplified launch files: Now, different models for single robot can be launched specified using `cfg:=`. Example:  
   ```bash
   roslaunch rl_sar gazebo_go2.launch cfg:=himloco
   ```  
4. Moved `command_mapping` and `state_mapping` from *.hpp to `config.yaml` for better configuration management.  
5. Added simulation and deployment SDK for `GoldenRetriever-l4w4` robot.  
6. Added support for pretrained models of `go2`, `go2w`, `b2`, `b2w`, `a1`.
7. Reorganized the project structure:  
   - Moved SDK to `thirdparty/`.  
   - Moved core libraries to `core/`.  
8. Fixed duplicate `#ifndef` issues in hpp files.
This commit is contained in:
robotsfan 2025-03-13 19:06:16 +08:00 committed by GitHub
parent f13aac008a
commit 87be546a0c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
784 changed files with 1734712 additions and 377 deletions

View File

@ -1,28 +0,0 @@
name: Send submodule updates to parent repo
on:
push:
branches:
- main
jobs:
update:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v2
with:
repository: fan-ziqi/BlindDog
token: ${{ secrets.PRIVATE_TOKEN_GITHUB }}
- name: Pull & update submodules recursively
run: |
git submodule update --init --recursive
git submodule update --recursive --remote
- name: Commit
run: |
git config user.email "actions@github.com"
git config user.name "GitHub Actions - update submodules"
git add --all
git commit -m "Update submodules" || echo "No changes to commit"
git push

View File

@ -1,5 +1,10 @@
# rl_sar
[![Ubuntu 20.04/22.04](https://img.shields.io/badge/Ubuntu-20.04/22.04-blue.svg?logo=ubuntu)](https://ubuntu.com/)
[![ROS Noetic](https://img.shields.io/badge/ros-noetic-brightgreen.svg?logo=ros)](https://wiki.ros.org/noetic)
[![ROS2 Foxy/Humble](https://img.shields.io/badge/ros2-foxy/humble-brightgreen.svg?logo=ros)](https://wiki.ros.org/foxy)
[![License](https://img.shields.io/badge/license-Apache2.0-yellow.svg?logo=apache)](https://opensource.org/license/apache-2-0)
[中文文档](README_CN.md)
**Version Select: [ROS-Noetic](https://github.com/fan-ziqi/rl_sar/tree/main) | [ROS2-Foxy/Humble](https://github.com/fan-ziqi/rl_sar/tree/ros2)**
@ -7,14 +12,17 @@
This repository provides a framework for simulation verification and physical deployment of robot reinforcement learning algorithms, suitable for quadruped robots, wheeled robots, and humanoid robots. "sar" stands for "simulation and real"
feature:
- Support legged_gym based on IsaacGym and IsaacLab based on IsaacSim. Use `framework` to distinguish.
- The code has two versions: **ROS-Noetic** and **ROS2-Foxy/Humble**
- The code supports both cpp and python, you can find python version in `src/rl_sar/scripts`
- Built-in pre-trained models for multiple robot simulations, including `Unitree-A1`, `Unitree-Go2`, `Unitree-Go2W`, `Unitree-B2`, `Unitree-B2W`, `FFTAI-GR1T1`, `FFTAI-GR1T2`, `GoldenRetriever-L4W0`, `GoldenRetriever-L4W4`;
- The training framework supports **IsaacGym** and **IsaacSim**, distinguished by `framework`;
- The code has **ROS-Noetic** and **ROS2-Foxy/Humble** versions;
- The code has **cpp** and **python** versions, with the python version located in `src/rl_sar/scripts`;
> [!NOTE]
> If you want to train policy using IsaacLab(IsaacSim), please use [robot_lab](https://github.com/fan-ziqi/robot_lab) project.
>
> [Click to discuss on Discord](https://discord.gg/vmVjkhVugU)
> The order of joints in robot_lab cfg file `joint_names` is the same as that defined in `xxx/robot_lab/config.yaml` in this project.
>
> Discuss in [Github Discussion](https://github.com/fan-ziqi/rl_sar/discussions) or [Discord](https://discord.gg/vmVjkhVugU).
## Preparation
@ -32,13 +40,13 @@ This project uses `ros-noetic` (Ubuntu 20.04) and requires the installation of t
sudo apt install ros-noetic-teleop-twist-keyboard ros-noetic-controller-interface ros-noetic-gazebo-ros-control ros-noetic-joint-state-controller ros-noetic-effort-controllers ros-noetic-joint-trajectory-controller ros-noetic-joy
```
Download and deploy `libtorch` at any location
Download and deploy `libtorch` at any location (Please modify **\<YOUR_PATH\>** below to the actual path)
```bash
cd /path/to/your/libtorch
cd <YOUR_PATH>/libtorch
wget https://download.pytorch.org/libtorch/cpu/libtorch-cxx11-abi-shared-with-deps-2.0.1%2Bcpu.zip
unzip libtorch-cxx11-abi-shared-with-deps-2.0.1+cpu.zip -d ./
echo 'export Torch_DIR=/path/to/your/libtorch' >> ~/.bashrc
echo 'export Torch_DIR=<YOUR_PATH>/libtorch' >> ~/.bashrc
source ~/.bashrc
```
@ -56,7 +64,7 @@ sudo apt install libtbb-dev
<details>
<summary>You can also use source code installation, click to expand</summary>
<summary>You can also use source code installation (Click to expand)</summary>
Install yaml-cpp
@ -92,9 +100,9 @@ catkin build
## Running
In the following text, **\<ROBOT\>_\<PLATFORM\>** is used to represent different environments, which can be `a1_isaacgym`, `a1_isaacsim`, `go2_isaacgym`, `gr1t1_isaacgym`, or `gr1t2_isaacgym`.
In the following text, **\<ROBOT\>/\<CONFIG\>** is used to represent different environments, such as `a1/isaacgym` and `go2/himloco`.
Before running, copy the trained pt model file to `rl_sar/src/rl_sar/models/<ROBOT>_<PLATFORM>`, and configure the parameters in `config.yaml`.
Before running, copy the trained pt model file to `rl_sar/src/rl_sar/models/<ROBOT>/<CONFIG>`, and configure the parameters in `config.yaml`.
### Simulation
@ -102,7 +110,7 @@ Open a terminal, launch the gazebo simulation environment
```bash
source devel/setup.bash
roslaunch rl_sar gazebo_<ROBOT>_<PLATFORM>.launch
roslaunch rl_sar gazebo_<ROBOT>.launch cfg:=<CONFIG>
```
Open a new terminal, launch the control program
@ -129,12 +137,14 @@ Gamepad Controls
### Real Robots
#### Unitree A1
<details>
<summary>Unitree A1 (Click to expand)</summary>
Unitree A1 can be connected using both wireless and wired methods:
* Wireless: Connect to the Unitree starting with WIFI broadcasted by the robot **(Note: Wireless connection may lead to packet loss, disconnection, or even loss of control, please ensure safety)**
* Wired: Use an Ethernet cable to connect any port on the computer and the robot, configure the computer IP as 192.168.123.162, and the gateway as 255.255.255.0
* Wired: Use an Ethernet cable to connect any port on the computer and the robot, configure the computer IP as 192.168.123.162, and the netmask as 255.255.255.0
Open a new terminal and start the control program
@ -147,7 +157,11 @@ Press the **R2** button on the controller to switch the robot to the default sta
Or press **0** on the keyboard to switch the robot to the default standing position, press **P** to switch to RL control mode, and press **1** in any state to switch to the initial lying position. WS controls x-axis, AD controls yaw, and JL controls y-axis.
#### Unitree Go2
</details>
<details>
<summary>Unitree Go2 (Click to expand)</summary>
1. Connect one end of the Ethernet cable to the Go2 robot and the other end to the user's computer. Then, enable USB Ethernet on the computer and configure it. The IP address of the onboard computer on the Go2 robot is 192.168.123.161, so the computer's USB Ethernet address should be set to the same network segment as the robot. For example, enter 192.168.123.222 in the "Address" field ("222" can be replaced with another number).
2. Use the `ifconfig` command to find the name of the network interface for the 123 network segment, such as `enxf8e43b808e06`. In the following steps, replace `<YOUR_NETWORK_INTERFACE>` with the actual network interface name.
@ -158,6 +172,8 @@ Or press **0** on the keyboard to switch the robot to the default standing posit
```
4. Go2 supports both joy and keyboard control, using the same method as mentioned above for A1.
</details>
### Train the actuator network
Take A1 as an example below
@ -175,10 +191,10 @@ Take A1 as an example below
## Add Your Robot
In the following text, **\<ROBOT\>_\<PLATFORM\>** is used to represent your robot environment.
In the following text, **\<ROBOT\>/\<CONFIG\>** is used to represent your robot environment.
1. Create a model package named `<ROBOT>_description` in the `rl_sar/src/robots` directory. Place the robot's URDF file in the `rl_sar/src/robots/<ROBOT>_description/urdf` directory and name it `<ROBOT>.urdf`. Additionally, create a joint configuration file with the namespace `<ROBOT>_gazebo` in the `rl_sar/src/robots/<ROBOT>_description/config` directory.
2. Place the trained RL model files in the `rl_sar/src/rl_sar/models/<ROBOT>_<PLATFORM>` directory, and create a new `config.yaml` file in this path. Refer to the `rl_sar/src/rl_sar/models/a1_isaacgym/config.yaml` file to modify the parameters.
2. Place the trained RL model files in the `rl_sar/src/rl_sar/models/<ROBOT>/<CONFIG>` directory, and create a new `config.yaml` file in this path. Refer to the `rl_sar/src/rl_sar/models/a1/isaacgym/config.yaml` file to modify the parameters.
3. Modify the `forward()` function in the code as needed to adapt to different models.
4. If you need to run simulations, modify the launch files as needed by referring to those in the `rl_sar/src/rl_sar/launch` directory.
5. If you need to run on the physical robot, modify the file `rl_sar/src/rl_sar/src/rl_real_a1.cpp` as needed.

View File

@ -1,5 +1,10 @@
# rl_sar
[![Ubuntu 20.04/22.04](https://img.shields.io/badge/Ubuntu-20.04/22.04-blue.svg?logo=ubuntu)](https://ubuntu.com/)
[![ROS Noetic](https://img.shields.io/badge/ros-noetic-brightgreen.svg?logo=ros)](https://wiki.ros.org/noetic)
[![ROS2 Foxy/Humble](https://img.shields.io/badge/ros2-foxy/humble-brightgreen.svg?logo=ros)](https://wiki.ros.org/foxy)
[![License](https://img.shields.io/badge/license-Apache2.0-yellow.svg?logo=apache)](https://opensource.org/license/apache-2-0)
[English document](README.md)
**版本选择: [ROS-Noetic](https://github.com/fan-ziqi/rl_sar/tree/main) | [ROS2-Foxy/Humble](https://github.com/fan-ziqi/rl_sar/tree/ros2)**
@ -7,14 +12,17 @@
本仓库提供了机器人强化学习算法的仿真验证与实物部署框架,适配四足机器人、轮足机器人、人形机器人。"sar"代表"simulation and real"
特性:
- 支持基于IsaacGym的legged_gym也支持基于IsaacSim的IsaacLab用`framework`加以区分。
- 代码有**ROS-Noetic**和**ROS2-Foxy/Humble**两个版本
- 代码有python和cpp两个版本python版本可以在`src/rl_sar/scripts`中找到
- 内置多种机器人仿真的预训练模型,包括 `Unitree-A1`、`Unitree-Go2`、`Unitree-Go2W`、`Unitree-B2`、`Unitree-B2W`、`FFTAI-GR1T1`、`FFTAI-GR1T2`、`GoldenRetriever-L4W0`、`GoldenRetriever-L4W4`
- 训练框架支持**IsaacGym**和**IsaacSim**,用`framework`加以区分;
- 代码有**ROS-Noetic**和**ROS2-Foxy/Humble**两个版本;
- 代码有**python**和**cpp**两个版本其中python版本在`src/rl_sar/scripts`内;
> [!NOTE]
> 如果你想使用IsaacLabIsaacSim训练策略请使用[robot_lab](https://github.com/fan-ziqi/robot_lab)项目。
> 如果你想使用IsaacLabIsaacSim训练策略请使用 [robot_lab](https://github.com/fan-ziqi/robot_lab) 项目。
>
> [点击在Discord上讨论](https://discord.gg/MC9KguQHtt)
> robot_lab配置文件中的关节顺序 `joint_names` 与本项目代码中 `xxx/robot_lab/config.yaml` 中定义的相同。
>
> 在 [Github Discussion](https://github.com/fan-ziqi/rl_sar/discussions) 或 [Discord](https://discord.gg/MC9KguQHtt) 中讨论
## 准备
@ -32,13 +40,13 @@ git clone https://github.com/fan-ziqi/rl_sar.git
sudo apt install ros-noetic-teleop-twist-keyboard ros-noetic-controller-interface ros-noetic-gazebo-ros-control ros-noetic-joint-state-controller ros-noetic-effort-controllers ros-noetic-joint-trajectory-controller ros-noetic-joy
```
在任意位置下载并部署`libtorch`
在任意位置下载并部署`libtorch`(请修改下面的 **\<YOUR_PATH\>** 为实际路径)
```bash
cd /path/to/your/libtorch
cd <YOUR_PATH>/libtorch
wget https://download.pytorch.org/libtorch/cpu/libtorch-cxx11-abi-shared-with-deps-2.0.1%2Bcpu.zip
unzip libtorch-cxx11-abi-shared-with-deps-2.0.1+cpu.zip -d ./
echo 'export Torch_DIR=/path/to/your/libtorch' >> ~/.bashrc
echo 'export Torch_DIR=<YOUR_PATH>/libtorch' >> ~/.bashrc
source ~/.bashrc
```
@ -56,7 +64,7 @@ sudo apt install libtbb-dev
<details>
<summary>也可以使用源码安装,点击展开</summary>
<summary>也可使用源码安装(点击展开)</summary>
安装yaml-cpp
@ -93,9 +101,9 @@ catkin build
## 运行
下文中使用 **\<ROBOT\>_\<PLATFORM\>** 代替表示不同的环境,可以是 `a1_isaacgym``a1_isaacsim``go2_isaacgym``gr1t1_isaacgym``gr1t2_isaacgym`
下文中使用 **\<ROBOT\>/\<CONFIG\>** 代替表示不同的环境,如 `a1/isaacgym``go2/himloco`
运行前请将训练好的pt模型文件拷贝到`rl_sar/src/rl_sar/models/<ROBOT>_<PLATFORM>`中,并配置`config.yaml`中的参数。
运行前请将训练好的pt模型文件拷贝到`rl_sar/src/rl_sar/models/<ROBOT>/<CONFIG>`中,并配置`config.yaml`中的参数。
### 仿真
@ -103,7 +111,7 @@ catkin build
```bash
source devel/setup.bash
roslaunch rl_sar gazebo_<ROBOT>_<PLATFORM>.launch
roslaunch rl_sar gazebo_<ROBOT>.launch cfg:=<CONFIG>
```
打开一个新终端,启动控制程序
@ -130,12 +138,14 @@ source devel/setup.bash
### 真实机器人
#### Unitree A1
<details>
<summary>Unitree A1点击展开</summary>
与Unitree A1连接可以使用无线与有线两种方式
* 无线连接机器人发出的Unitree开头的WIFI **(注意:无线连接可能会出现丢包断联甚至失控,请注意安全)**
* 有线:用网线连接计算机和机器人的任意网口,配置计算机ip为192.168.123.162,网关255.255.255.0
* 有线:用网线连接计算机和机器人的任意网口,配置计算机地址为192.168.123.162,子网掩码255.255.255.0
新建终端,启动控制程序
@ -148,7 +158,11 @@ rosrun rl_sar rl_real_a1
或者按下键盘上的**0**键让机器人切换到默认站起姿态,按下**P**键切换到RL控制模式任意状态按下**1**键切换到最初的趴下姿态。WS控制xAD控制yawJL控制y。
#### Unitree Go2
</details>
<details>
<summary>Unitree Go2点击展开</summary>
1. 用网线的一端连接Go2机器人另一端连接用户电脑并开启电脑的 USB Ethernet 后进行配置。机器狗机载电脑的 IP 地地址为 192.168.123.161,故需将电脑 USB Ethernet 地址设置为与机器狗同一网段,如在 Address 中输入 192.168.123.222 (“222”可以改成其他)。
2. 通过`ifconfig`命令查看123网段的网卡名字如`enxf8e43b808e06`,下文用 \<YOUR_NETWORK_INTERFACE\> 代替
@ -159,6 +173,8 @@ rosrun rl_sar rl_real_a1
```
4. Go2支持手柄与键盘控制方法与上面a1相同
</details>
### 训练执行器网络
下面拿A1举例
@ -176,10 +192,10 @@ rosrun rl_sar rl_real_a1
## 添加你的机器人
下文中使用 **\<ROBOT\>_\<PLATFORM\>** 代替表示你的机器人环境
下文中使用 **\<ROBOT\>/\<CONFIG\>** 代替表示你的机器人环境
1. 在`rl_sar/src/robots`路径下创建名为`<ROBOT>_description`的模型包将模型的urdf放到`rl_sar/src/robots/<ROBOT>_description/urdf`路径下并命名为`<ROBOT>.urdf`,并在`rl_sar/src/robots/<ROBOT>_description/config`路径下创建命名空间为`<ROBOT>_gazebo`的关节配置文件
2. 将训练好的RL模型文件放到`rl_sar/src/rl_sar/models/<ROBOT>_<PLATFORM>`路径下并在此路径中新建config.yaml文件参考`rl_sar/src/rl_sar/models/a1_isaacgym/config.yaml`文件修改其中参数
2. 将训练好的RL模型文件放到`rl_sar/src/rl_sar/models/<ROBOT>/<CONFIG>`路径下并在此路径中新建config.yaml文件参考`rl_sar/src/rl_sar/models/a1/isaacgym/config.yaml`文件修改其中参数
3. 按需修改代码中的`forward()`函数,以适配不同的模型
4. 若需要运行仿真,则参考`rl_sar/src/rl_sar/launch`路径下的launch文件自行修改
5. 若需要运行实物,则参考`rl_sar/src/rl_sar/src/rl_real_a1.cpp`文件自行修改

View File

@ -43,28 +43,28 @@ link_directories(/usr/local/lib)
include_directories(${YAML_CPP_INCLUDE_DIR})
# Unitree A1
include_directories(library/unitree_legged_sdk_3.2/include)
link_directories(library/unitree_legged_sdk_3.2/lib)
include_directories(library/thirdparty/unitree_legged_sdk_3.2/include)
link_directories(library/thirdparty/unitree_legged_sdk_3.2/lib)
set(UNITREE_A1_LIBS -pthread unitree_legged_sdk_amd64 lcm)
# Unitree Go2
include_directories(library/unitree_sdk2/include)
link_directories(library/unitree_sdk2/lib/x86_64)
include_directories(library/unitree_sdk2/thirdparty/include)
include_directories(library/unitree_sdk2/thirdparty/include/ddscxx)
link_directories(library/unitree_sdk2/thirdparty/lib/x86_64)
include_directories(library/thirdparty/unitree_sdk2/include)
link_directories(library/thirdparty/unitree_sdk2/lib/x86_64)
include_directories(library/thirdparty/unitree_sdk2/thirdparty/include)
include_directories(library/thirdparty/unitree_sdk2/thirdparty/include/ddscxx)
link_directories(library/thirdparty/unitree_sdk2/thirdparty/lib/x86_64)
set(UNITREE_GO2_LIBS -pthread unitree_sdk2 ddsc ddscxx)
include_directories(
include
${catkin_INCLUDE_DIRS}
library/matplotlibcpp
library/observation_buffer
library/rl_sdk
library/loop
library/core/matplotlibcpp
library/core/observation_buffer
library/core/rl_sdk
library/core/loop
)
add_library(rl_sdk library/rl_sdk/rl_sdk.cpp)
add_library(rl_sdk library/core/rl_sdk/rl_sdk.cpp)
target_link_libraries(rl_sdk "${TORCH_LIBRARIES}" Python3::Python Python3::Module TBB::tbb)
set_property(TARGET rl_sdk PROPERTY CXX_STANDARD 14)
find_package(Python3 COMPONENTS NumPy)
@ -74,7 +74,7 @@ else()
target_compile_definitions(rl_sdk WITHOUT_NUMPY)
endif()
add_library(observation_buffer library/observation_buffer/observation_buffer.cpp)
add_library(observation_buffer library/core/observation_buffer/observation_buffer.cpp)
target_link_libraries(observation_buffer "${TORCH_LIBRARIES}")
set_property(TARGET observation_buffer PROPERTY CXX_STANDARD 14)
@ -96,6 +96,13 @@ target_link_libraries(rl_real_go2
rl_sdk observation_buffer yaml-cpp
)
include_directories(library/thirdparty/l4w4_sdk)
add_executable(rl_real_l4w4 src/rl_real_l4w4.cpp )
target_link_libraries(rl_real_l4w4
${catkin_LIBRARIES} -pthread
rl_sdk observation_buffer yaml-cpp
)
catkin_install_python(PROGRAMS
scripts/rl_sim.py
scripts/actuator_net.py

View File

@ -3,8 +3,8 @@
* SPDX-License-Identifier: Apache-2.0
*/
#ifndef RL_REAL_HPP
#define RL_REAL_HPP
#ifndef RL_REAL_A1_HPP
#define RL_REAL_A1_HPP
#include "rl_sdk.hpp"
#include "observation_buffer.hpp"
@ -61,8 +61,6 @@ private:
int motiontime = 0;
std::vector<double> mapped_joint_positions;
std::vector<double> mapped_joint_velocities;
int command_mapping[12] = {3, 4, 5, 0, 1, 2, 9, 10, 11, 6, 7, 8};
int state_mapping[12] = {3, 4, 5, 0, 1, 2, 9, 10, 11, 6, 7, 8};
};
#endif // RL_REAL_HPP
#endif // RL_REAL_A1_HPP

View File

@ -3,8 +3,8 @@
* SPDX-License-Identifier: Apache-2.0
*/
#ifndef RL_REAL_HPP
#define RL_REAL_HPP
#ifndef RL_REAL_GO2_HPP
#define RL_REAL_GO2_HPP
#include "rl_sdk.hpp"
#include "observation_buffer.hpp"
@ -107,8 +107,6 @@ private:
int motiontime = 0;
std::vector<double> mapped_joint_positions;
std::vector<double> mapped_joint_velocities;
int command_mapping[12] = {3, 4, 5, 0, 1, 2, 9, 10, 11, 6, 7, 8};
int state_mapping[12] = {3, 4, 5, 0, 1, 2, 9, 10, 11, 6, 7, 8};
};
#endif // RL_REAL_HPP
#endif // RL_REAL_GO2_HPP

View File

@ -0,0 +1,60 @@
/*
* Copyright (c) 2024-2025 Ziqi Fan
* SPDX-License-Identifier: Apache-2.0
*/
#ifndef RL_REAL_L4W4_HPP
#define RL_REAL_L4W4_HPP
#include "rl_sdk.hpp"
#include "observation_buffer.hpp"
#include "loop.hpp"
#include "l4w4_sdk.hpp"
#include <csignal>
#include "matplotlibcpp.h"
namespace plt = matplotlibcpp;
class RL_Real : public RL
{
public:
RL_Real();
~RL_Real();
private:
// rl functions
torch::Tensor Forward() override;
void GetState(RobotState<double> *state) override;
void SetCommand(const RobotCommand<double> *command) override;
void RunModel();
void RobotControl();
// history buffer
ObservationBuffer history_obs_buf;
torch::Tensor history_obs;
// loop
std::shared_ptr<LoopFunc> loop_keyboard;
std::shared_ptr<LoopFunc> loop_control;
std::shared_ptr<LoopFunc> loop_rl;
std::shared_ptr<LoopFunc> loop_plot;
// plot
const int plot_size = 100;
std::vector<int> plot_t;
std::vector<std::vector<double>> plot_real_joint_pos, plot_target_joint_pos;
void Plot();
// l4w4 interface
L4W4SDK l4w4_sdk;
LowCmd l4w4_low_command = {0};
LowState l4w4_low_state = {0};
xRockerBtnDataStruct unitree_joy;
// others
int motiontime = 0;
std::vector<double> mapped_joint_positions;
std::vector<double> mapped_joint_velocities;
};
#endif // RL_REAL_L4W4_HPP

View File

@ -1,7 +1,9 @@
<launch>
<arg name="wname" default="stairs"/>
<arg name="rname" default="a1"/>
<param name="robot_name" type="str" value="$(arg rname)_isaacgym"/>
<arg name="cfg" default=""/>
<param name="robot_name" type="str" value="$(arg rname)"/>
<param name="config_name" type="str" value="$(arg cfg)"/>
<param name="ros_namespace" type="str" value="/$(arg rname)_gazebo/"/>
<param name="gazebo_model_name" type="str" value="$(arg rname)_gazebo"/>
<arg name="robot_path" value="(find $(arg rname)_description)"/>

View File

@ -1,7 +1,9 @@
<launch>
<arg name="wname" default="stairs"/>
<arg name="rname" default="a1"/>
<param name="robot_name" type="str" value="$(arg rname)_isaacsim"/>
<arg name="rname" default="b2"/>
<arg name="cfg" default=""/>
<param name="robot_name" type="str" value="$(arg rname)"/>
<param name="config_name" type="str" value="$(arg cfg)"/>
<param name="ros_namespace" type="str" value="/$(arg rname)_gazebo/"/>
<param name="gazebo_model_name" type="str" value="$(arg rname)_gazebo"/>
<arg name="robot_path" value="(find $(arg rname)_description)"/>
@ -32,7 +34,7 @@
<!-- Run a python script to the send a service call to gazebo_ros to spawn a URDF robot -->
<!-- Set trunk and joint positions at startup -->
<node pkg="gazebo_ros" type="spawn_model" name="urdf_spawner" respawn="false" output="screen"
args="-urdf -z 0.6 -model $(arg rname)_gazebo -param robot_description"/>
args="-urdf -z 1.0 -model $(arg rname)_gazebo -param robot_description"/>
<!-- Load joint controller configurations from YAML file to parameter server -->
<rosparam file="$(arg dollar)$(arg robot_path)/config/robot_control.yaml" command="load"/>

View File

@ -0,0 +1,57 @@
<launch>
<arg name="wname" default="stairs"/>
<arg name="rname" default="b2w"/>
<arg name="cfg" default=""/>
<param name="robot_name" type="str" value="$(arg rname)"/>
<param name="config_name" type="str" value="$(arg cfg)"/>
<param name="ros_namespace" type="str" value="/$(arg rname)_gazebo/"/>
<param name="gazebo_model_name" type="str" value="$(arg rname)_gazebo"/>
<arg name="robot_path" value="(find $(arg rname)_description)"/>
<arg name="dollar" value="$"/>
<arg name="paused" default="true"/>
<arg name="use_sim_time" default="true"/>
<arg name="gui" default="true"/>
<arg name="headless" default="false"/>
<arg name="debug" default="false"/>
<!-- Debug mode will hung up the robot, use "true" or "false" to switch it. -->
<arg name="user_debug" default="false"/>
<include file="$(find gazebo_ros)/launch/empty_world.launch">
<arg name="world_name" value="$(find rl_sar)/worlds/$(arg wname).world"/>
<arg name="debug" value="$(arg debug)"/>
<arg name="gui" value="$(arg gui)"/>
<arg name="paused" value="$(arg paused)"/>
<arg name="use_sim_time" value="$(arg use_sim_time)"/>
<arg name="headless" value="$(arg headless)"/>
</include>
<!-- Load the URDF into the ROS Parameter Server -->
<param name="robot_description" textfile="$(find b2w_description)/urdf/b2w_description.urdf"/>
<!-- Run a python script to the send a service call to gazebo_ros to spawn a URDF robot -->
<!-- Set trunk and joint positions at startup -->
<node pkg="gazebo_ros" type="spawn_model" name="urdf_spawner" respawn="false" output="screen"
args="-urdf -z 1.2 -model $(arg rname)_gazebo -param robot_description"/>
<!-- Load joint controller configurations from YAML file to parameter server -->
<rosparam file="$(arg dollar)$(arg robot_path)/config/robot_control.yaml" command="load"/>
<!-- load the controllers -->
<node pkg="controller_manager" type="spawner" name="controller_spawner" respawn="false"
output="screen" ns="/$(arg rname)_gazebo" args="joint_state_controller
FL_hip_controller FL_thigh_controller FL_calf_controller FL_foot_controller
FR_hip_controller FR_thigh_controller FR_calf_controller FR_foot_controller
RL_hip_controller RL_thigh_controller RL_calf_controller RL_foot_controller
RR_hip_controller RR_thigh_controller RR_calf_controller RR_foot_controller "/>
<!-- convert joint states to TF transforms for rviz, etc -->
<node pkg="robot_state_publisher" type="robot_state_publisher" name="robot_state_publisher"
respawn="false" output="screen">
<remap from="/joint_states" to="/$(arg rname)_gazebo/joint_states"/>
</node>
<!-- Load joy node -->
<node pkg="joy" type="joy_node" name="joy_node" output="screen"/>
</launch>

View File

@ -1,7 +1,9 @@
<launch>
<arg name="wname" default="stairs"/>
<arg name="rname" default="go2"/>
<param name="robot_name" type="str" value="$(arg rname)_isaacgym"/>
<arg name="cfg" default=""/>
<param name="robot_name" type="str" value="$(arg rname)"/>
<param name="config_name" type="str" value="$(arg cfg)"/>
<param name="ros_namespace" type="str" value="/$(arg rname)_gazebo/"/>
<param name="gazebo_model_name" type="str" value="$(arg rname)_gazebo"/>
<arg name="robot_path" value="(find $(arg rname)_description)"/>

View File

@ -0,0 +1,57 @@
<launch>
<arg name="wname" default="stairs"/>
<arg name="rname" default="go2w"/>
<arg name="cfg" default=""/>
<param name="robot_name" type="str" value="$(arg rname)"/>
<param name="config_name" type="str" value="$(arg cfg)"/>
<param name="ros_namespace" type="str" value="/$(arg rname)_gazebo/"/>
<param name="gazebo_model_name" type="str" value="$(arg rname)_gazebo"/>
<arg name="robot_path" value="(find $(arg rname)_description)"/>
<arg name="dollar" value="$"/>
<arg name="paused" default="true"/>
<arg name="use_sim_time" default="true"/>
<arg name="gui" default="true"/>
<arg name="headless" default="false"/>
<arg name="debug" default="false"/>
<!-- Debug mode will hung up the robot, use "true" or "false" to switch it. -->
<arg name="user_debug" default="false"/>
<include file="$(find gazebo_ros)/launch/empty_world.launch">
<arg name="world_name" value="$(find rl_sar)/worlds/$(arg wname).world"/>
<arg name="debug" value="$(arg debug)"/>
<arg name="gui" value="$(arg gui)"/>
<arg name="paused" value="$(arg paused)"/>
<arg name="use_sim_time" value="$(arg use_sim_time)"/>
<arg name="headless" value="$(arg headless)"/>
</include>
<!-- Load the URDF into the ROS Parameter Server -->
<param name="robot_description" textfile="$(find go2w_description)/urdf/go2w_description.urdf"/>
<!-- Run a python script to the send a service call to gazebo_ros to spawn a URDF robot -->
<!-- Set trunk and joint positions at startup -->
<node pkg="gazebo_ros" type="spawn_model" name="urdf_spawner" respawn="false" output="screen"
args="-urdf -z 0.8 -model $(arg rname)_gazebo -param robot_description"/>
<!-- Load joint controller configurations from YAML file to parameter server -->
<rosparam file="$(arg dollar)$(arg robot_path)/config/robot_control.yaml" command="load"/>
<!-- load the controllers -->
<node pkg="controller_manager" type="spawner" name="controller_spawner" respawn="false"
output="screen" ns="/$(arg rname)_gazebo" args="joint_state_controller
FL_hip_controller FL_thigh_controller FL_calf_controller FL_foot_controller
FR_hip_controller FR_thigh_controller FR_calf_controller FR_foot_controller
RL_hip_controller RL_thigh_controller RL_calf_controller RL_foot_controller
RR_hip_controller RR_thigh_controller RR_calf_controller RR_foot_controller "/>
<!-- convert joint states to TF transforms for rviz, etc -->
<node pkg="robot_state_publisher" type="robot_state_publisher" name="robot_state_publisher"
respawn="false" output="screen">
<remap from="/joint_states" to="/$(arg rname)_gazebo/joint_states"/>
</node>
<!-- Load joy node -->
<node pkg="joy" type="joy_node" name="joy_node" output="screen"/>
</launch>

View File

@ -1,7 +1,9 @@
<launch>
<arg name="wname" default="stairs"/>
<arg name="rname" default="gr1t1"/>
<param name="robot_name" type="str" value="$(arg rname)_isaacgym"/>
<arg name="cfg" default=""/>
<param name="robot_name" type="str" value="$(arg rname)"/>
<param name="config_name" type="str" value="$(arg cfg)"/>
<param name="ros_namespace" type="str" value="/$(arg rname)_gazebo/"/>
<param name="gazebo_model_name" type="str" value="$(arg rname)_gazebo"/>
<arg name="robot_path" value="(find $(arg rname)_description)"/>

View File

@ -1,7 +1,9 @@
<launch>
<arg name="wname" default="stairs"/>
<arg name="rname" default="gr1t2"/>
<param name="robot_name" type="str" value="$(arg rname)_isaacgym"/>
<arg name="cfg" default=""/>
<param name="robot_name" type="str" value="$(arg rname)"/>
<param name="config_name" type="str" value="$(arg cfg)"/>
<param name="ros_namespace" type="str" value="/$(arg rname)_gazebo/"/>
<param name="gazebo_model_name" type="str" value="$(arg rname)_gazebo"/>
<arg name="robot_path" value="(find $(arg rname)_description)"/>

View File

@ -0,0 +1,57 @@
<launch>
<arg name="wname" default="stairs"/>
<arg name="rname" default="l4w4"/>
<arg name="cfg" default=""/>
<param name="robot_name" type="str" value="$(arg rname)"/>
<param name="config_name" type="str" value="$(arg cfg)"/>
<param name="ros_namespace" type="str" value="/$(arg rname)_gazebo/"/>
<param name="gazebo_model_name" type="str" value="$(arg rname)_gazebo"/>
<arg name="robot_path" value="(find $(arg rname)_description)"/>
<arg name="dollar" value="$"/>
<arg name="paused" default="true"/>
<arg name="use_sim_time" default="true"/>
<arg name="gui" default="true"/>
<arg name="headless" default="false"/>
<arg name="debug" default="false"/>
<!-- Debug mode will hung up the robot, use "true" or "false" to switch it. -->
<arg name="user_debug" default="false"/>
<include file="$(find gazebo_ros)/launch/empty_world.launch">
<arg name="world_name" value="$(find rl_sar)/worlds/$(arg wname).world"/>
<arg name="debug" value="$(arg debug)"/>
<arg name="gui" value="$(arg gui)"/>
<arg name="paused" value="$(arg paused)"/>
<arg name="use_sim_time" value="$(arg use_sim_time)"/>
<arg name="headless" value="$(arg headless)"/>
</include>
<!-- Load the URDF into the ROS Parameter Server -->
<param name="robot_description" textfile="$(find l4w4_description)/urdf/l4w4.urdf"/>
<!-- Run a python script to the send a service call to gazebo_ros to spawn a URDF robot -->
<!-- Set trunk and joint positions at startup -->
<node pkg="gazebo_ros" type="spawn_model" name="urdf_spawner" respawn="false" output="screen"
args="-urdf -z 1.0 -model $(arg rname)_gazebo -param robot_description"/>
<!-- Load joint controller configurations from YAML file to parameter server -->
<rosparam file="$(arg dollar)$(arg robot_path)/config/robot_control.yaml" command="load"/>
<!-- load the controllers -->
<node pkg="controller_manager" type="spawner" name="controller_spawner" respawn="false"
output="screen" ns="/$(arg rname)_gazebo" args="joint_state_controller
FL_hip_controller FL_thigh_controller FL_calf_controller FL_foot_controller
FR_hip_controller FR_thigh_controller FR_calf_controller FR_foot_controller
RL_hip_controller RL_thigh_controller RL_calf_controller RL_foot_controller
RR_hip_controller RR_thigh_controller RR_calf_controller RR_foot_controller "/>
<!-- convert joint states to TF transforms for rviz, etc -->
<node pkg="robot_state_publisher" type="robot_state_publisher" name="robot_state_publisher"
respawn="false" output="screen">
<remap from="/joint_states" to="/$(arg rname)_gazebo/joint_states"/>
</node>
<!-- Load joy node -->
<node pkg="joy" type="joy_node" name="joy_node" output="screen"/>
</launch>

View File

@ -38,10 +38,18 @@ void ObservationBuffer::insert(torch::Tensor new_obs)
obs_buf.index({torch::indexing::Slice(torch::indexing::None), torch::indexing::Slice(-num_obs, torch::indexing::None)}) = new_obs;
}
/**
* @brief Gets history of observations indexed by obs_ids.
*
* @param obs_ids An array of integers with which to index the desired
* observations, where 0 is the latest observation and
* include_history_steps - 1 is the oldest observation.
* @return A torch::Tensor containing the concatenated observations.
*/
torch::Tensor ObservationBuffer::get_obs_vec(std::vector<int> obs_ids)
{
std::vector<torch::Tensor> obs;
for (int i = obs_ids.size() - 1; i >= 0; --i)
for (int i = 0; i < obs_ids.size(); ++i)
{
int obs_id = obs_ids[i];
int slice_idx = include_history_steps - obs_id - 1;

View File

@ -102,17 +102,18 @@ void RL::InitControl()
void RL::ComputeOutput(const torch::Tensor &actions, torch::Tensor &output_dof_pos, torch::Tensor &output_dof_vel, torch::Tensor &output_dof_tau)
{
torch::Tensor joint_actions_scaled = actions * this->params.action_scale;
torch::Tensor wheel_actions_scaled = torch::zeros({1, this->params.num_of_dofs});
torch::Tensor actions_scaled = actions * this->params.action_scale;
torch::Tensor pos_actions_scaled = actions_scaled;
torch::Tensor vel_actions_scaled = torch::zeros_like(actions);
for (int i : this->params.wheel_indices)
{
joint_actions_scaled[0][i] = 0.0;
wheel_actions_scaled[0][i] = actions[0][i] * this->params.action_scale_wheel;
pos_actions_scaled[0][i] = 0.0;
vel_actions_scaled[0][i] = actions[0][i];
}
torch::Tensor actions_scaled = joint_actions_scaled + wheel_actions_scaled;
output_dof_pos = joint_actions_scaled + this->params.default_dof_pos;
output_dof_vel = wheel_actions_scaled;
output_dof_tau = this->params.rl_kp * (actions_scaled + this->params.default_dof_pos - this->obs.dof_pos) - this->params.rl_kd * this->obs.dof_vel;
torch::Tensor all_actions_scaled = pos_actions_scaled + vel_actions_scaled;
output_dof_pos = pos_actions_scaled + this->params.default_dof_pos;
output_dof_vel = vel_actions_scaled;
output_dof_tau = this->params.rl_kp * (all_actions_scaled + this->params.default_dof_pos - this->obs.dof_pos) - this->params.rl_kd * this->obs.dof_vel;
output_dof_tau = torch::clamp(output_dof_tau, -(this->params.torque_limits), this->params.torque_limits);
}
@ -458,45 +459,14 @@ std::vector<T> ReadVectorFromYaml(const YAML::Node &node)
return values;
}
template <typename T>
std::vector<T> ReadVectorFromYaml(const YAML::Node &node, const std::string &framework, const int &rows, const int &cols)
void RL::ReadYaml(std::string robot_path)
{
std::vector<T> values;
for (const auto &val : node)
{
values.push_back(val.as<T>());
}
if (framework == "isaacsim")
{
std::vector<T> transposed_values(cols * rows);
for (int r = 0; r < rows; ++r)
{
for (int c = 0; c < cols; ++c)
{
transposed_values[c * rows + r] = values[r * cols + c];
}
}
return transposed_values;
}
else if (framework == "isaacgym")
{
return values;
}
else
{
throw std::invalid_argument("Unsupported framework: " + framework);
}
}
void RL::ReadYaml(std::string robot_name)
{
// The config file is located at "rl_sar/src/rl_sar/models/<robot_name>/config.yaml"
std::string config_path = std::string(CMAKE_CURRENT_SOURCE_DIR) + "/models/" + robot_name + "/config.yaml";
// The config file is located at "rl_sar/src/rl_sar/models/<robot_path/config.yaml"
std::string config_path = std::string(CMAKE_CURRENT_SOURCE_DIR) + "/models/" + robot_path + "/config.yaml";
YAML::Node config;
try
{
config = YAML::LoadFile(config_path)[robot_name];
config = YAML::LoadFile(config_path)[robot_path];
}
catch (YAML::BadFile &e)
{
@ -506,8 +476,6 @@ void RL::ReadYaml(std::string robot_name)
this->params.model_name = config["model_name"].as<std::string>();
this->params.framework = config["framework"].as<std::string>();
int rows = config["rows"].as<int>();
int cols = config["cols"].as<int>();
this->params.dt = config["dt"].as<double>();
this->params.decimation = config["decimation"].as<int>();
this->params.num_observations = config["num_observations"].as<int>();
@ -528,13 +496,10 @@ void RL::ReadYaml(std::string robot_name)
}
else
{
this->params.clip_actions_upper = torch::tensor(ReadVectorFromYaml<double>(config["clip_actions_upper"], this->params.framework, rows, cols)).view({1, -1});
this->params.clip_actions_lower = torch::tensor(ReadVectorFromYaml<double>(config["clip_actions_lower"], this->params.framework, rows, cols)).view({1, -1});
this->params.clip_actions_upper = torch::tensor(ReadVectorFromYaml<double>(config["clip_actions_upper"])).view({1, -1});
this->params.clip_actions_lower = torch::tensor(ReadVectorFromYaml<double>(config["clip_actions_lower"])).view({1, -1});
}
this->params.action_scale = config["action_scale"].as<double>();
this->params.hip_scale_reduction = config["hip_scale_reduction"].as<double>();
this->params.hip_scale_reduction_indices = ReadVectorFromYaml<int>(config["hip_scale_reduction_indices"]);
this->params.action_scale_wheel = config["action_scale_wheel"].as<double>();
this->params.action_scale = torch::tensor(ReadVectorFromYaml<double>(config["action_scale"])).view({1, -1});
this->params.wheel_indices = ReadVectorFromYaml<int>(config["wheel_indices"]);
this->params.num_of_dofs = config["num_of_dofs"].as<int>();
this->params.lin_vel_scale = config["lin_vel_scale"].as<double>();
@ -543,13 +508,15 @@ void RL::ReadYaml(std::string robot_name)
this->params.dof_vel_scale = config["dof_vel_scale"].as<double>();
this->params.commands_scale = torch::tensor(ReadVectorFromYaml<double>(config["commands_scale"])).view({1, -1});
// this->params.commands_scale = torch::tensor({this->params.lin_vel_scale, this->params.lin_vel_scale, this->params.ang_vel_scale});
this->params.rl_kp = torch::tensor(ReadVectorFromYaml<double>(config["rl_kp"], this->params.framework, rows, cols)).view({1, -1});
this->params.rl_kd = torch::tensor(ReadVectorFromYaml<double>(config["rl_kd"], this->params.framework, rows, cols)).view({1, -1});
this->params.fixed_kp = torch::tensor(ReadVectorFromYaml<double>(config["fixed_kp"], this->params.framework, rows, cols)).view({1, -1});
this->params.fixed_kd = torch::tensor(ReadVectorFromYaml<double>(config["fixed_kd"], this->params.framework, rows, cols)).view({1, -1});
this->params.torque_limits = torch::tensor(ReadVectorFromYaml<double>(config["torque_limits"], this->params.framework, rows, cols)).view({1, -1});
this->params.default_dof_pos = torch::tensor(ReadVectorFromYaml<double>(config["default_dof_pos"], this->params.framework, rows, cols)).view({1, -1});
this->params.joint_controller_names = ReadVectorFromYaml<std::string>(config["joint_controller_names"], this->params.framework, rows, cols);
this->params.rl_kp = torch::tensor(ReadVectorFromYaml<double>(config["rl_kp"])).view({1, -1});
this->params.rl_kd = torch::tensor(ReadVectorFromYaml<double>(config["rl_kd"])).view({1, -1});
this->params.fixed_kp = torch::tensor(ReadVectorFromYaml<double>(config["fixed_kp"])).view({1, -1});
this->params.fixed_kd = torch::tensor(ReadVectorFromYaml<double>(config["fixed_kd"])).view({1, -1});
this->params.torque_limits = torch::tensor(ReadVectorFromYaml<double>(config["torque_limits"])).view({1, -1});
this->params.default_dof_pos = torch::tensor(ReadVectorFromYaml<double>(config["default_dof_pos"])).view({1, -1});
this->params.joint_controller_names = ReadVectorFromYaml<std::string>(config["joint_controller_names"]);
this->params.command_mapping = ReadVectorFromYaml<int>(config["command_mapping"]);
this->params.state_mapping = ReadVectorFromYaml<int>(config["state_mapping"]);
}
void RL::CSVInit(std::string robot_name)

View File

@ -87,10 +87,7 @@ struct ModelParams
std::vector<int> observations_history;
double damping;
double stiffness;
double action_scale;
double hip_scale_reduction;
std::vector<int> hip_scale_reduction_indices;
double action_scale_wheel;
torch::Tensor action_scale;
std::vector<int> wheel_indices;
int num_of_dofs;
double lin_vel_scale;
@ -108,6 +105,8 @@ struct ModelParams
torch::Tensor commands_scale;
torch::Tensor default_dof_pos;
std::vector<std::string> joint_controller_names;
std::vector<int> command_mapping;
std::vector<int> state_mapping;
};
struct Observations
@ -152,7 +151,7 @@ public:
torch::Tensor QuatRotateInverse(torch::Tensor q, torch::Tensor v, const std::string &framework);
// yaml params
void ReadYaml(std::string robot_name);
void ReadYaml(std::string robot_path);
// csv logger
std::string csv_filename;
@ -164,7 +163,7 @@ public:
void KeyboardInterface();
// others
std::string robot_name;
std::string robot_name, config_name;
STATE running_state = STATE_RL_RUNNING; // default running_state set to STATE_RL_RUNNING
bool simulation_running = false;

View File

@ -0,0 +1,60 @@
#ifndef L4W4_SDK_COMM_H
#define L4W4_SDK_COMM_H
#include <stdint.h>
typedef struct
{
float x;
float y;
float z;
} Vector3;
typedef struct
{
float quaternion[4]; // quaternion, normalized, (w,x,y,z)
float gyroscope[3]; // angular velocity unit: rad/s)
float accelerometer[3]; // m/(s2)
float rpy[3]; // euler angleunit: rad)
int8_t temperature;
} IMU; // when under accelerated motion, the attitude of the robot calculated by IMU will drift.
typedef struct
{
uint8_t mode; // motor working mode
float q; // current angle (unit: radian)
float dq; // current velocity (unit: radian/second)
float ddq; // current acc (unit: radian/second*second)
float tauEst; // current estimated output torque (unit: N.m)
float q_raw; // current angle (unit: radian)
float dq_raw; // current velocity (unit: radian/second)
float ddq_raw;
int8_t temperature; // current temperature (temperature conduction is slow that leads to lag)
uint32_t reserve[2];
} MotorState; // motor feedback
typedef struct
{
uint8_t mode; // desired working mode
float q; // desired angle (unit: radian)
float dq; // desired velocity (unit: radian/second)
float tau; // desired output torque (unit: N.m)
float Kp; // desired position stiffness (unit: N.m/rad )
float Kd; // desired velocity stiffness (unit: N.m/(rad/s) )
uint32_t reserve[3];
} MotorCmd; // motor control
typedef struct
{
IMU imu;
MotorState motorState[20];
uint8_t wirelessRemote[40]; // wireless commands
} LowState; // low level feedback
typedef struct
{
MotorCmd motorCmd[20];
uint8_t wirelessRemote[40];
} LowCmd; // low level control
#endif // L4W4_SDK_COMM_H

View File

@ -0,0 +1,41 @@
#ifndef L4W4_SDK_JOYSTICK_H
#define L4W4_SDK_JOYSTICK_H
#include <stdint.h>
// 16b
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;
// 40 Byte (now used 24B)
typedef struct {
uint8_t head[2];
xKeySwitchUnion btn;
float lx;
float rx;
float ry;
float L2;
float ly;
uint8_t idle[16];
} xRockerBtnDataStruct;
#endif // L4W4_SDK_JOYSTICK_H

View File

@ -0,0 +1,534 @@
#ifndef L4W4_SDK_HPP
#define L4W4_SDK_HPP
#include <iostream>
#include <netinet/in.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <sys/stat.h>
#include <string.h>
#include <arpa/inet.h>
#include "joystick.h"
#include "comm.h"
class L4W4SDK
{
private:
float tmp_time_from_mcu = -1;
int show_rc = 0;
int ctr = 0;
float t0 = -1;
int n_cpu = 0;
int n_run = 0;
float read_float(unsigned char *buff, int *idx);
float read_from1byte(unsigned char *buff, int *idx, float v_min, float v_max);
float read_from2bytes(unsigned char *buff, int *idx, float v_min, float v_max);
void write_into2bytes(float value, unsigned char *buff, int *idx, float v_min, float v_max);
float read_byte(unsigned char *buff, int *idx);
float Vector3_Dot(Vector3 v1, Vector3 v2);
Vector3 Vector3_Cross(Vector3 u, Vector3 v);
float Sign(float value);
float Vector3_invSqrt(Vector3 v);
Vector3 Vector3_Direction(Vector3 v);
float Clamp(float value, float min, float max);
float Angle_vA_2_vB(Vector3 vA, Vector3 vB, Vector3 axle);
void Vector3_Normalize(Vector3 *v);
Vector3 Quaternion_Transform(float vx, float vy, float vz, float qx, float qy, float qz, float qw);
public:
L4W4SDK() {};
~L4W4SDK() {};
int client_socket;
struct sockaddr_in server_addr;
int recv_len = 0;
int ex_send_recv = -1;
unsigned char sent_buff[256];
unsigned char recv_buff[512];
void InitUDP();
void AnalyzeUDP(unsigned char *recv_buff, LowState &lowState);
void SendUDP(LowCmd &lowCmd);
void PrintMCU(int running_state);
void InitCmdData(LowCmd& cmd);
};
void L4W4SDK::InitCmdData(LowCmd& cmd)
{
for (int i = 0; i < 20; ++i) {
cmd.motorCmd[i].mode = 0;
cmd.motorCmd[i].q = 0.0;
cmd.motorCmd[i].dq = 0.0;
cmd.motorCmd[i].tau = 0.0;
cmd.motorCmd[i].Kp = 0.0;
cmd.motorCmd[i].Kd = 0.0;
}
memset(cmd.wirelessRemote, 0, sizeof(cmd.wirelessRemote));
}
void L4W4SDK::PrintMCU(int running_state)
{
float t1 = tmp_time_from_mcu;
if(t1 < t0)
t0 = t1;
float ms = (t1 - t0) * 1000.0;
n_cpu++;
n_run++;
if(ms > 1000)
{
std::cout<<"mcu time = "<< tmp_time_from_mcu<<", n_cpu = "<<n_cpu<<", runningState= "<< running_state<<std::endl;
t0 = t1;
n_cpu = 0;
n_run = 0;
}
else
{
if(n_run > 50)
{
std::cout<<" dull ms= "<<ms<<", n_cpu = "<<n_cpu<<", runningState= "<< running_state<<std::endl;
n_run = 0;
}
}
}
void L4W4SDK::SendUDP(LowCmd &lowCmd)
{
int idx = 0;
// front right
write_into2bytes(lowCmd.motorCmd[4].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[4].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[4].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[5].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[5].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[5].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[6].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[6].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[6].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[7].dq, sent_buff, &idx, -200, 200 );
write_into2bytes(lowCmd.motorCmd[7].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[7].Kd, sent_buff, &idx, 0, 1000 );
// rear right
write_into2bytes(lowCmd.motorCmd[12].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[12].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[12].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[13].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[13].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[13].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[14].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[14].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[14].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[15].dq, sent_buff, &idx, -200, 200 );
write_into2bytes(lowCmd.motorCmd[15].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[15].Kd, sent_buff, &idx, 0, 1000 );
// rear left
write_into2bytes(lowCmd.motorCmd[8].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[8].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[8].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[9].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[9].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[9].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[10].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[10].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[10].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[11].dq, sent_buff, &idx, -200, 200 );
write_into2bytes(lowCmd.motorCmd[11].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[11].Kd, sent_buff, &idx, 0, 1000 );
// front left
write_into2bytes(lowCmd.motorCmd[0].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[0].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[0].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[1].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[1].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[1].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[2].q, sent_buff, &idx, -3.2, 3.2 );
write_into2bytes(lowCmd.motorCmd[2].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[2].Kd, sent_buff, &idx, 0, 1000 );
write_into2bytes(-lowCmd.motorCmd[3].dq, sent_buff, &idx, -200, 200 );
write_into2bytes(lowCmd.motorCmd[3].Kp, sent_buff, &idx, 0, 1000 );
write_into2bytes(lowCmd.motorCmd[3].Kd, sent_buff, &idx, 0, 1000 );
int sss = sendto(client_socket, sent_buff, idx, 0, (const sockaddr*)& server_addr, sizeof(server_addr));
}
void L4W4SDK::InitUDP()
{
struct sockaddr_in client_addr;
client_addr.sin_family = AF_INET;
client_addr.sin_addr.s_addr = htons(INADDR_ANY);//INADDR_ANY表示自动获取本机地址 inet_addr("192.168.1.102");
client_addr.sin_port = htons(0); //0表示让系统自动分配一个空闲端口
client_socket = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP);
if(client_socket < 0)
{
std::cout <<"socket error" << std::endl;
perror("socket error");
exit(1);
}
else
{
std::cout<<"socket created ("<<client_socket<<")"<<std::endl;
}
int ret = bind(client_socket, (struct sockaddr*)&client_addr, sizeof(client_addr));
if(ret >= 0)
{
std::cout << "udp bind success (" << ret << ")" <<std::endl;
std::cout << "my ip = " << inet_ntoa(client_addr.sin_addr) << std::endl;
std::cout << "my port = " << client_addr.sin_port << std::endl;
}
else
std::cout << "udp bind failed (" << ret << ")" <<std::endl;
server_addr.sin_family = AF_INET;
server_addr.sin_port = htons(777);
server_addr.sin_addr.s_addr = inet_addr("192.168.1.101");
}
float L4W4SDK::read_float(unsigned char *buff, int *idx)
{
float v = 0;
*((unsigned char*)&v + 0) = buff[*idx]; (*idx)++;
*((unsigned char*)&v + 1) = buff[*idx]; (*idx)++;
*((unsigned char*)&v + 2) = buff[*idx]; (*idx)++;
*((unsigned char*)&v + 3) = buff[*idx]; (*idx)++;
return v;
}
float L4W4SDK::read_from1byte(unsigned char *buff, int *idx, float v_min, float v_max)
{
float v = (float)buff[*idx] / 255.0 * (v_max - v_min) + v_min;
(*idx)++;
return v;
}
float L4W4SDK::read_from2bytes(unsigned char *buff, int *idx, float v_min, float v_max)
{
unsigned int v = 0;
*((unsigned char*)&v + 0) = buff[*idx]; (*idx)++;
*((unsigned char*)&v + 1) = buff[*idx]; (*idx)++;
return v / 65535.0 * (v_max - v_min) + v_min;
}
void L4W4SDK::write_into2bytes(float value, unsigned char *buff, int *idx, float v_min, float v_max)
{
ushort v16 = (value - v_min) / (v_max - v_min) * 65535;
buff[*idx] = (v16>>8) & 0xff;
(*idx)++;
buff[*idx] = v16 & 0xff;
(*idx)++;
}
float L4W4SDK::read_byte(unsigned char *buff, int *idx)
{
float v = (float)buff[*idx];
(*idx)++;
return v;
}
float L4W4SDK::Vector3_Dot(Vector3 v1, Vector3 v2)
{
return v1.x * v2.x + v1.y * v2.y + v1.z * v2.z;
}
Vector3 L4W4SDK::Vector3_Cross(Vector3 u, Vector3 v)
{
Vector3 vn;
vn.x = u.y * v.z - u.z * v.y;
vn.y = u.z * v.x - u.x * v.z;
vn.z = u.x * v.y - u.y * v.x;
return vn;
}
float L4W4SDK::Sign(float value)
{
if(value >= 0)
return 1;
else
return -1;
}
float L4W4SDK::Vector3_invSqrt(Vector3 v)
{
return 1.0 / sqrtf(v.x * v.x + v.y * v.y + v.z * v.z);
}
Vector3 L4W4SDK::Vector3_Direction(Vector3 v)
{
float invSqrt = Vector3_invSqrt(v);
Vector3 dv = v;
dv.x *= invSqrt;
dv.y *= invSqrt;
dv.z *= invSqrt;
return dv;
}
float L4W4SDK::Clamp(float value, float min, float max)
{
if(min > max)
{
float tmp = min;
min = max;
max = tmp;
}
if(value < min) return min;
else if(value > max) return max;
else return value;
}
float L4W4SDK::Angle_vA_2_vB(Vector3 vA, Vector3 vB, Vector3 axle)
{
Vector3 vA_dir = Vector3_Direction(vA);
Vector3 vB_dir = Vector3_Direction(vB);
Vector3 axle_dir = Vector3_Direction(axle);
float sinA = Clamp( Vector3_Dot(axle_dir, Vector3_Cross(vA_dir, vB_dir) ), -1, 1);
float angle_raw = asinf(sinA);
float angle = Vector3_Dot(vA_dir, vB_dir) > 0 ? angle_raw : Sign(angle_raw) * M_PI - angle_raw;
return angle;
}
void L4W4SDK::Vector3_Normalize(Vector3 *v)
{
float invSqrt = Vector3_invSqrt(*v);
v->x *= invSqrt;
v->y *= invSqrt;
v->z *= invSqrt;
}
Vector3 L4W4SDK::Quaternion_Transform(float vx, float vy, float vz, float qx, float qy, float qz, float qw)
{
float dot_u_v = qx * vx + qy * vy + qz * vz;
float dot_u_u = qx * qx + qy * qy + qz * qz;
float x_of_uXv = qy * vz - qz * vy;
float y_of_uXv = qz * vx - qx * vz;
float z_of_uXv = qx * vy - qy * vx;
float k1 = 2.0 * dot_u_v;
float k2 = qw * qw - dot_u_u;
float k3 = 2.0 * qw;
Vector3 vout;
vout.x = k1 * qx + k2 * vx + k3 * x_of_uXv;
vout.y = k1 * qy + k2 * vy + k3 * y_of_uXv;
vout.z = k1 * qz + k2 * vz + k3 * z_of_uXv;
return vout;
}
void L4W4SDK::AnalyzeUDP(unsigned char *recv_buff, LowState &lowState)
{
int idx = 2;
tmp_time_from_mcu = read_float(recv_buff, &idx);
float BatteryVoltage = read_from1byte(recv_buff, &idx, 20, 60);
float MCUTemperature = read_byte(recv_buff, &idx);
float acc_lx = read_float(recv_buff, &idx);
float acc_ly = read_float(recv_buff, &idx);
float acc_lz = read_float(recv_buff, &idx);
float omega_lx = read_float(recv_buff, &idx);
float omega_ly = read_float(recv_buff, &idx);
float omega_lz = read_float(recv_buff, &idx);
float orientation_x = read_float(recv_buff, &idx);
float orientation_y = read_float(recv_buff, &idx);
float orientation_z = read_float(recv_buff, &idx);
float orientation_w = read_float(recv_buff, &idx);
Vector3 robot_up_w = Quaternion_Transform(0, 1, 0, orientation_x, orientation_y, orientation_z, orientation_w);
Vector3 current_trunk_front = Quaternion_Transform(1, 0, 0, orientation_x, orientation_y, orientation_z, orientation_w);
Vector3 current_trunk_right = Quaternion_Transform(0, 0, 1, orientation_x, orientation_y, orientation_z, orientation_w);
Vector3 trunk_hori_front = current_trunk_front;
trunk_hori_front.y = 0;
Vector3_Normalize(&trunk_hori_front);
Vector3 trunk_hori_right = Vector3_Cross(trunk_hori_front, {0,1,0});
float r2d = 180.0f / M_PI;
float robot_yaw_deg = Angle_vA_2_vB({1,0,0}, trunk_hori_front, {0,1,0}) * r2d;
float robot_pitch_deg = Angle_vA_2_vB(trunk_hori_front, current_trunk_front, trunk_hori_right) * r2d;
float robot_roll_deg = Angle_vA_2_vB(trunk_hori_right, current_trunk_right, current_trunk_front) * r2d;
read_byte(recv_buff, &idx);
read_byte(recv_buff, &idx);
xRockerBtnDataStruct *rockerBtn = (xRockerBtnDataStruct*)(&(lowState.wirelessRemote));
unsigned char key1 = recv_buff[idx]; idx++;
rockerBtn->btn.components.R1 = (key1 & 0x80) >> 7;
rockerBtn->btn.components.L1 = (key1 & 0x40) >> 6;
rockerBtn->btn.components.start = (key1 & 0x20) >> 5;
rockerBtn->btn.components.select = (key1 & 0x10) >> 4;
rockerBtn->btn.components.R2 = (key1 & 0x08) >> 3;
rockerBtn->btn.components.L2 = (key1 & 0x04) >> 2;
rockerBtn->btn.components.F1 = (key1 & 0x02) >> 1;
rockerBtn->btn.components.F2 = (key1 & 0x01) >> 0;
unsigned char key2 = recv_buff[idx]; idx++;
rockerBtn->btn.components.A = (key2 & 0x80) >> 7;
rockerBtn->btn.components.B = (key2 & 0x40) >> 6;
rockerBtn->btn.components.X = (key2 & 0x20) >> 5;
rockerBtn->btn.components.Y = (key2 & 0x10) >> 4;
rockerBtn->btn.components.up = (key2 & 0x08) >> 3;
rockerBtn->btn.components.right = (key2 & 0x04) >> 2;
rockerBtn->btn.components.down = (key2 & 0x02) >> 1;
rockerBtn->btn.components.left = (key2 & 0x01) >> 0;
rockerBtn->lx = read_float(recv_buff, &idx);
rockerBtn->rx = read_float(recv_buff, &idx);
rockerBtn->ly = read_float(recv_buff, &idx);
rockerBtn->L2 = read_float(recv_buff, &idx);
rockerBtn->ry = read_float(recv_buff, &idx);
if(rockerBtn->btn.components.F1)
show_rc = 1;
else
show_rc = 0;
idx+=4*4;
float motor_leg1_j0 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg1_j0_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg1_j1 = read_from2bytes(recv_buff, &idx, -1.3*M_PI, 0.7*M_PI);
float motor_leg1_j1_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg1_j2 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg1_j2_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg2_j0 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg2_j0_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg2_j1 = read_from2bytes(recv_buff, &idx, -1.3*M_PI, 0.7*M_PI);
float motor_leg2_j1_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg2_j2 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg2_j2_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg3_j0 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg3_j0_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg3_j1 = read_from2bytes(recv_buff, &idx, -1.3*M_PI, 0.7*M_PI);
float motor_leg3_j1_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg3_j2 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg3_j2_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg4_j0 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg4_j0_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg4_j1 = read_from2bytes(recv_buff, &idx, -1.3*M_PI, 0.7*M_PI);
float motor_leg4_j1_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg4_j2 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg4_j2_dot = read_from2bytes(recv_buff, &idx, -33, 33);
float motor_leg1_j3 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg1_j3_dot = read_from2bytes(recv_buff, &idx, -200, 200);
float motor_leg2_j3 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg2_j3_dot = read_from2bytes(recv_buff, &idx, -200, 200);
float motor_leg3_j3 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg3_j3_dot = read_from2bytes(recv_buff, &idx, -200, 200);
float motor_leg4_j3 = read_from2bytes(recv_buff, &idx, -M_PI, M_PI);
float motor_leg4_j3_dot = read_from2bytes(recv_buff, &idx, -200, 200);
lowState.imu.quaternion[0] = orientation_w;
lowState.imu.quaternion[1] = orientation_x;
lowState.imu.quaternion[2] = -orientation_z;
lowState.imu.quaternion[3] = orientation_y;
lowState.imu.gyroscope[0] = omega_lx;
lowState.imu.gyroscope[1] = -omega_lz;
lowState.imu.gyroscope[2] = omega_ly;
lowState.imu.accelerometer[0] = acc_lx;
lowState.imu.accelerometer[1] = -acc_lz;
lowState.imu.accelerometer[2] = acc_ly;
// leg4 leg1 leg3 leg2
// hip thigh shank wheel
lowState.motorState[0].q = lowState.motorState[0].q_raw = motor_leg4_j0;
lowState.motorState[0].dq = lowState.motorState[0].dq_raw = motor_leg4_j0_dot;
lowState.motorState[0].ddq = lowState.motorState[0].ddq_raw = 0;
lowState.motorState[1].q = lowState.motorState[1].q_raw = -motor_leg4_j1;
lowState.motorState[1].dq = lowState.motorState[1].dq_raw = -motor_leg4_j1_dot;
lowState.motorState[1].ddq = lowState.motorState[1].ddq_raw = 0;
lowState.motorState[2].q = lowState.motorState[2].q_raw = -motor_leg4_j2;
lowState.motorState[2].dq = lowState.motorState[2].dq_raw = -motor_leg4_j2_dot;
lowState.motorState[2].ddq = lowState.motorState[2].ddq_raw = 0;
lowState.motorState[3].q = lowState.motorState[3].q_raw = -motor_leg4_j3;
lowState.motorState[3].dq = lowState.motorState[3].dq_raw = -motor_leg4_j3_dot;
lowState.motorState[3].ddq = lowState.motorState[3].ddq_raw = 0;
lowState.motorState[4].q = lowState.motorState[4].q_raw = motor_leg1_j0;
lowState.motorState[4].dq = lowState.motorState[4].dq_raw = motor_leg1_j0_dot;
lowState.motorState[4].ddq = lowState.motorState[4].ddq_raw = 0;
lowState.motorState[5].q = lowState.motorState[5].q_raw = -motor_leg1_j1;
lowState.motorState[5].dq = lowState.motorState[5].dq_raw = -motor_leg1_j1_dot;
lowState.motorState[5].ddq = lowState.motorState[5].ddq_raw = 0;
lowState.motorState[6].q = lowState.motorState[6].q_raw = -motor_leg1_j2;
lowState.motorState[6].dq = lowState.motorState[6].dq_raw = -motor_leg1_j2_dot;
lowState.motorState[6].ddq = lowState.motorState[6].ddq_raw = 0;
lowState.motorState[7].q = lowState.motorState[7].q_raw = -motor_leg1_j3;
lowState.motorState[7].dq = lowState.motorState[7].dq_raw = -motor_leg1_j3_dot;
lowState.motorState[7].ddq = lowState.motorState[7].ddq_raw = 0;
lowState.motorState[8].q = lowState.motorState[8].q_raw = motor_leg3_j0;
lowState.motorState[8].dq = lowState.motorState[8].dq_raw = motor_leg3_j0_dot;
lowState.motorState[8].ddq = lowState.motorState[8].ddq_raw = 0;
lowState.motorState[9].q = lowState.motorState[9].q_raw = -motor_leg3_j1;
lowState.motorState[9].dq = lowState.motorState[9].dq_raw = -motor_leg3_j1_dot;
lowState.motorState[9].ddq = lowState.motorState[9].ddq_raw = 0;
lowState.motorState[10].q = lowState.motorState[10].q_raw = -motor_leg3_j2;
lowState.motorState[10].dq = lowState.motorState[10].dq_raw = -motor_leg3_j2_dot;
lowState.motorState[10].ddq = lowState.motorState[10].ddq_raw = 0;
lowState.motorState[11].q = lowState.motorState[11].q_raw = -motor_leg3_j3;
lowState.motorState[11].dq = lowState.motorState[11].dq_raw = -motor_leg3_j3_dot;
lowState.motorState[11].ddq = lowState.motorState[11].ddq_raw = 0;
lowState.motorState[12].q = lowState.motorState[12].q_raw = motor_leg2_j0;
lowState.motorState[12].dq = lowState.motorState[12].dq_raw = motor_leg2_j0_dot;
lowState.motorState[12].ddq = lowState.motorState[12].ddq_raw = 0;
lowState.motorState[13].q = lowState.motorState[13].q_raw = -motor_leg2_j1;
lowState.motorState[13].dq = lowState.motorState[13].dq_raw = -motor_leg2_j1_dot;
lowState.motorState[13].ddq = lowState.motorState[13].ddq_raw = 0;
lowState.motorState[14].q = lowState.motorState[14].q_raw = -motor_leg2_j2;
lowState.motorState[14].dq = lowState.motorState[14].dq_raw = -motor_leg2_j2_dot;
lowState.motorState[14].ddq = lowState.motorState[14].ddq_raw = 0;
lowState.motorState[15].q = lowState.motorState[15].q_raw = -motor_leg2_j3;
lowState.motorState[15].dq = lowState.motorState[15].dq_raw = -motor_leg2_j3_dot;
lowState.motorState[15].ddq = lowState.motorState[15].ddq_raw = 0;
ctr++;
if(ctr>=100)
{
//float r2d = 180.0/M_PI;
//std::cout<<"mcu time = "<<tmp_time_from_mcu<<",\tbattery = "<< BatteryVoltage<<std::endl;
//std::cout<<"lx = "<<rockerBtn->lx<<",\try = "<<rockerBtn->ry<<",\trx = "<<rockerBtn->rx<<std::endl;
//if(show_rc)
if(0)
{
std::cout<< " rc: " << ((int)(rockerBtn->ly * 100) / 100.0) << "\t" << ((int)(rockerBtn->lx * 100) / 100.0);
std::cout<< "\t" << ((int)(rockerBtn->ry * 100) / 100.0) << "\t" << ((int)(rockerBtn->rx * 100) / 100.0)<<std::endl;
}
{
std::cout<<"yaw pitch roll = "<<robot_yaw_deg<<"\t"<<robot_pitch_deg<<"\t"<<robot_roll_deg<<std::endl;
}
ctr = 0;
}
}
#endif // L4W4_SDK_HPP

Some files were not shown because too many files have changed in this diff Show More